#include "world.h"
#include <stdlib.h>
#include "brender.h"
#include "brucetrk.h"
#include "car.h"
#include "depth.h"
#include "displays.h"
#include "drmem.h"
#include "errors.h"
#include "finteray.h"
#include "flicplay.h"
#include "formats.h"
#include "globvars.h"
#include "globvrpb.h"
#include "graphics.h"
#include "harness/trace.h"
#include "input.h"
#include "loading.h"
#include "opponent.h"
#include "pd/sys.h"
#include "pedestrn.h"
#include "piping.h"
#include "replay.h"
#include "spark.h"
#include "trig.h"
#include "utility.h"
#include <float.h>
#include <string.h>
static const br_scalar viewdistance_surmultiplier = 2.5; // Pierre-Marie Baty -- view distance surmultiplier addition
int gFunkotronics_array_size;
int gGroovidelics_array_size;
int gSpec_vol_mode;
tFunkotronic_spec* gFunkotronics_array;
tGroovidelic_spec* gGroovidelics_array;
char* gFunk_type_names[] = { "spin", "rock", "throb", "slither", "roll" };
char* gFunk_anim_names[] = { "frames", "flic" };
char* gTime_mode_names[] = { "approximate", "accurate" };
char* gFunk_move_names[] = { "linear", "harmonic", "flash", "controlled", "absolute", "continuous" };
char* gFunk_nature_names[] = { "constant", "distance", "lastlap", "otherlaps" };
char* gGroove_nature_names[] = { "constant", "distance" };
char* gAxis_names[] = { "x", "y", "z" };
char* gLollipop_names[] = { "xlollipop", "ylollipop", "zlollipop" };
char* gGroove_path_names[] = { "straight", "circular" };
char* gGroove_object_names[] = { "spin", "rock", "throb", "shear" };
char* gDepth_effect_names[] = { "dark", "fog" };
br_actor* gGroove_by_proxy_actor;
tRotate_mode gCurrent_rotate_mode = eRotate_mode_y;
tScale_mode gCurrent_scale_mode = eScale_mode_all;
int gNumber_of_additional_models;
tRoad_texturing_level gRoad_texturing_level = eRTL_full;
tWall_texturing_level gWall_texturing_level = eWTL_full;
tCar_texturing_level gCar_texturing_level = eCTL_full;
int gRendering_accessories;
int gNumber_of_actors;
int gNumber_of_lights;
br_actor* gActor_array[500];
float* gGroove_funk_bindings[960];
br_actor* gDelete_list[500];
br_actor* gLight_array[50];
br_model* gAdditional_models[1000];
br_actor* gSpec_vol_actors[100];
tPath_name gAdditional_actor_path;
tPath_name gAdditional_model_path;
tU32 gPrevious_groove_times[2];
int gRace_file_version;
br_vector3 gActor_centre;
float gTemp;
br_actor* gLast_actor;
br_actor* gKnown_actor;
br_actor* gAdditional_actors;
int gDelete_count;
br_scalar gNearest_distance;
br_actor* gNearest_actor;
br_actor* gStandard_lamp;
br_scalar gSight_distance_squared;
// IDA: float __cdecl MapSawToTriangle(float pNumber)
float MapSawToTriangle(float pNumber) {
LOG_TRACE("(%f)", pNumber);
if (pNumber >= 0.5) {
return 3.0 - pNumber * 4.0;
} else {
return pNumber * 4.0 - 1.0;
}
}
// IDA: void __cdecl SetSightDistance(br_scalar pYon)
void SetSightDistance(br_scalar pYon) {
gSight_distance_squared = pYon * 1.02f * (pYon * 1.02f);
}
// IDA: br_actor* __usercall FindActorInArray@<EAX>(char *pThe_name@<EAX>)
br_actor* FindActorInArray(char* pThe_name) {
int i;
LOG_TRACE("(\"%s\")", pThe_name);
for (i = 0; i < gNumber_of_actors; i++) {
if (strcmp(gActor_array
[i
]->identifier
, pThe_name
) == 0) {
return gActor_array[i];
}
}
return NULL;
}
// IDA: br_actor* __usercall FindLightInArray@<EAX>(char *pThe_name@<EAX>)
br_actor* FindLightInArray(char* pThe_name) {
int i;
LOG_TRACE("(\"%s\")", pThe_name);
for (i = 0; i < gNumber_of_lights; i++) {
if (strcmp(gLight_array
[i
]->identifier
, pThe_name
) == 0) {
return gLight_array[i];
}
}
return NULL;
}
// IDA: br_actor* __usercall CloneActor@<EAX>(br_actor *pSource_actor@<EAX>)
br_actor* CloneActor(br_actor* pSource_actor) {
br_actor* new_actor;
br_actor* child_actor;
br_actor* new_child_actor;
LOG_TRACE("(%p)", pSource_actor);
new_actor = BrActorAllocate(pSource_actor->type, pSource_actor->type_data);
new_actor->model = pSource_actor->model;
new_actor->material = pSource_actor->material;
if (pSource_actor->identifier != NULL) {
if (new_actor->identifier != NULL) {
BrResFree(new_actor->identifier);
}
new_actor->identifier = BrResStrDup(new_actor, pSource_actor->identifier);
}
new_actor->t = pSource_actor->t;
for (child_actor = pSource_actor->children; child_actor != NULL; child_actor = child_actor->next) {
new_child_actor = CloneActor(child_actor);
BrActorAdd(new_actor, new_child_actor);
}
return new_actor;
}
// IDA: void __usercall InitialiseStorageSpace(tBrender_storage *pStorage_space@<EAX>, int pMax_pixelmaps@<EDX>, int pMax_shade_tables@<EBX>, int pMax_materials@<ECX>, int pMax_models)
void InitialiseStorageSpace(tBrender_storage* pStorage_space, int pMax_pixelmaps, int pMax_shade_tables, int pMax_materials, int pMax_models) {
LOG_TRACE("(%p, %d, %d, %d, %d)", pStorage_space, pMax_pixelmaps, pMax_shade_tables, pMax_materials, pMax_models);
pStorage_space->pixelmaps_count = 0;
pStorage_space->shade_tables_count = 0;
pStorage_space->materials_count = 0;
pStorage_space->models_count = 0;
pStorage_space->max_pixelmaps = pMax_pixelmaps;
pStorage_space->max_shade_tables = pMax_shade_tables;
pStorage_space->max_materials = pMax_materials;
pStorage_space->max_models = pMax_models;
pStorage_space->pixelmaps = BrMemCalloc(pMax_pixelmaps, sizeof(br_pixelmap*), kMem_stor_space_pix);
pStorage_space->shade_tables = BrMemCalloc(pMax_shade_tables, sizeof(br_pixelmap*), kMem_stor_space_tab);
pStorage_space->materials = BrMemCalloc(pMax_materials, sizeof(br_material*), kMem_stor_space_mat);
pStorage_space->models = BrMemCalloc(pMax_models, sizeof(br_model*), kMem_stor_space_mod);
pStorage_space->saved_colour_maps = BrMemCalloc(pMax_materials, sizeof(br_pixelmap*), kMem_stor_space_save);
}
// IDA: void __usercall DisposeStorageSpace(tBrender_storage *pStorage_space@<EAX>)
void DisposeStorageSpace(tBrender_storage* pStorage_space) {
BrMemFree(pStorage_space->pixelmaps);
BrMemFree(pStorage_space->shade_tables);
BrMemFree(pStorage_space->materials);
BrMemFree(pStorage_space->models);
BrMemFree(pStorage_space->saved_colour_maps);
}
// IDA: void __usercall ClearOutStorageSpace(tBrender_storage *pStorage_space@<EAX>)
void ClearOutStorageSpace(tBrender_storage* pStorage_space) {
int i;
LOG_TRACE("(%p)", pStorage_space);
for (i = 0; pStorage_space->pixelmaps_count > i; ++i) {
if (pStorage_space->pixelmaps[i] != NULL) {
BrMapRemove(pStorage_space->pixelmaps[i]);
BrPixelmapFree(pStorage_space->pixelmaps[i]);
}
}
pStorage_space->pixelmaps_count = 0;
for (i = 0; pStorage_space->shade_tables_count > i; ++i) {
if (pStorage_space->shade_tables[i] != NULL) {
BrTableRemove(pStorage_space->shade_tables[i]);
BrPixelmapFree(pStorage_space->shade_tables[i]);
}
}
pStorage_space->shade_tables_count = 0;
for (i = 0; pStorage_space->materials_count > i; ++i) {
if (pStorage_space->materials[i] != NULL) {
BrMaterialRemove(pStorage_space->materials[i]);
BrMaterialFree(pStorage_space->materials[i]);
}
}
pStorage_space->materials_count = 0;
for (i = 0; pStorage_space->models_count > i; ++i) {
if (pStorage_space->models[i] != NULL) {
BrModelRemove(pStorage_space->models[i]);
BrModelFree(pStorage_space->models[i]);
}
}
pStorage_space->models_count = 0;
}
// IDA: tAdd_to_storage_result __usercall AddPixelmapToStorage@<EAX>(tBrender_storage *pStorage_space@<EAX>, br_pixelmap **pThe_pm@<EDX>)
// This seems like the signature should be `br_pixelmap* pThe_pm`
tAdd_to_storage_result AddPixelmapToStorage(tBrender_storage* pStorage_space, br_pixelmap** pThe_pm) {
int i;
LOG_TRACE("(%p, %p)", pStorage_space, pThe_pm);
if (pStorage_space->pixelmaps_count >= pStorage_space->max_pixelmaps) {
return eStorage_not_enough_room;
}
for (i = 0; i < pStorage_space->pixelmaps_count; i++) {
if (pStorage_space->pixelmaps[i]->identifier
&& ((br_pixelmap*)pThe_pm)->identifier
&& strcmp(pStorage_space
->pixelmaps
[i
]->identifier
, ((br_pixelmap
*)pThe_pm
)->identifier
) == 0) {
return eStorage_duplicate;
}
}
pStorage_space->pixelmaps[pStorage_space->pixelmaps_count] = (br_pixelmap*)pThe_pm;
pStorage_space->pixelmaps_count++;
return eStorage_allocated;
}
// IDA: tAdd_to_storage_result __usercall AddShadeTableToStorage@<EAX>(tBrender_storage *pStorage_space@<EAX>, br_pixelmap *pThe_st@<EDX>)
tAdd_to_storage_result AddShadeTableToStorage(tBrender_storage* pStorage_space, br_pixelmap* pThe_st) {
int i;
LOG_TRACE("(%p, %p)", pStorage_space, pThe_st);
if (pStorage_space->shade_tables_count >= pStorage_space->max_shade_tables) {
return eStorage_not_enough_room;
}
for (i = 0; i < pStorage_space->shade_tables_count; i++) {
if (pStorage_space->shade_tables[i]->identifier
&& pThe_st->identifier
&& !strcmp(pStorage_space
->shade_tables
[i
]->identifier
, pThe_st
->identifier
)) {
return eStorage_duplicate;
}
}
pStorage_space->shade_tables[pStorage_space->shade_tables_count++] = pThe_st;
return eStorage_allocated;
}
// IDA: tAdd_to_storage_result __usercall AddMaterialToStorage@<EAX>(tBrender_storage *pStorage_space@<EAX>, br_material *pThe_mat@<EDX>)
tAdd_to_storage_result AddMaterialToStorage(tBrender_storage* pStorage_space, br_material* pThe_mat) {
int i;
LOG_TRACE("(%p, %p)", pStorage_space, pThe_mat);
if (pStorage_space->materials_count >= pStorage_space->max_materials) {
return eStorage_not_enough_room;
}
for (i = 0; i < pStorage_space->materials_count; i++) {
if (pStorage_space->materials[i]->identifier
&& pThe_mat->identifier
&& !strcmp(pStorage_space
->materials
[i
]->identifier
, pThe_mat
->identifier
)) {
return eStorage_duplicate;
}
}
pStorage_space->saved_colour_maps[pStorage_space->materials_count] = 0;
pStorage_space->materials[pStorage_space->materials_count++] = pThe_mat;
return eStorage_allocated;
}
// IDA: tAdd_to_storage_result __usercall AddModelToStorage@<EAX>(tBrender_storage *pStorage_space@<EAX>, br_model *pThe_mod@<EDX>)
tAdd_to_storage_result AddModelToStorage(tBrender_storage* pStorage_space, br_model* pThe_mod) {
int i;
LOG_TRACE("(%p, %p)", pStorage_space, pThe_mod);
if (pStorage_space->materials_count >= pStorage_space->max_models) {
return eStorage_not_enough_room;
}
for (i = 0; i < pStorage_space->models_count; i++) {
if (pStorage_space->models[i]
&& pStorage_space->models[i]->identifier
&& pThe_mod->identifier
&& !strcmp(pStorage_space
->models
[i
]->identifier
, pThe_mod
->identifier
)) {
return eStorage_duplicate;
}
}
pStorage_space->models[pStorage_space->models_count] = pThe_mod;
pStorage_space->models_count++;
return eStorage_allocated;
}
// IDA: int __usercall LoadNPixelmaps@<EAX>(tBrender_storage *pStorage_space@<EAX>, FILE *pF@<EDX>, int pCount@<EBX>)
int LoadNPixelmaps(tBrender_storage* pStorage_space, FILE* pF, int pCount) {
tPath_name the_path;
int i;
int j;
int new_ones;
int total;
char s[256];
char* str;
br_pixelmap* temp_array[200];
LOG_TRACE("(%p, %p, %d)", pStorage_space, pF, pCount);
new_ones = 0;
for (i = 0; i < pCount; ++i) {
PossibleService();
GetALineAndDontArgue(pF, s);
PathCat(the_path, gApplication_path, gGraf_specs[gGraf_spec_index].data_dir_name);
PathCat(the_path, the_path, "PIXELMAP");
PathCat(the_path, the_path, str);
AllowOpenToFail();
total = DRPixelmapLoadMany(the_path, temp_array, COUNT_OF(temp_array));
if (total == 0) {
PathCat(the_path, gApplication_path, "PIXELMAP");
PathCat(the_path, the_path, str);
total = DRPixelmapLoadMany(the_path, temp_array, COUNT_OF(temp_array));
if (total == 0) {
FatalError(kFatalError_LoadPixelmapFile_S, str);
}
}
for (j = 0; j < total; j++) {
if (temp_array[j] != NULL) {
switch (AddPixelmapToStorage(pStorage_space, (br_pixelmap**)temp_array[j])) {
case eStorage_not_enough_room:
FatalError(kFatalError_InsufficientPixelmapSlots);
break;
case eStorage_duplicate:
BrPixelmapFree(temp_array[j]);
break;
case eStorage_allocated:
BrMapAdd(temp_array[j]);
++new_ones;
break;
}
}
}
}
return new_ones;
}
// IDA: br_pixelmap* __usercall LoadSinglePixelmap@<EAX>(tBrender_storage *pStorage_space@<EAX>, char *pName@<EDX>)
br_pixelmap* LoadSinglePixelmap(tBrender_storage* pStorage_space, char* pName) {
br_pixelmap* temp;
LOG_TRACE("(%p, \"%s\")", pStorage_space, pName);
temp = LoadPixelmap(pName);
if (!temp) {
return BrMapFind(pName);
}
switch (AddPixelmapToStorage(pStorage_space, (br_pixelmap**)temp)) {
case eStorage_not_enough_room:
FatalError(kFatalError_InsufficientPixelmapSlots);
break;
case eStorage_duplicate:
BrPixelmapFree(temp);
return BrMapFind(pName);
case eStorage_allocated:
BrMapAdd(temp);
return temp;
}
return NULL;
}
// IDA: br_material* __usercall LoadSingleMaterial@<EAX>(tBrender_storage *pStorage_space@<EAX>, char *pName@<EDX>)
br_material* LoadSingleMaterial(tBrender_storage* pStorage_space, char* pName) {
br_material* temp;
LOG_TRACE("(%p, \"%s\")", pStorage_space, pName);
temp = LoadMaterial(pName);
if (!temp) {
return BrMaterialFind(pName);
}
switch (AddMaterialToStorage(pStorage_space, temp)) {
case eStorage_not_enough_room:
FatalError(kFatalError_InsufficientMaterialSlots);
break;
case eStorage_duplicate:
BrMaterialFree(temp);
return BrMaterialFind(pName);
case eStorage_allocated:
BrMaterialAdd(temp);
return temp;
}
return NULL;
}
// IDA: int __usercall LoadNShadeTables@<EAX>(tBrender_storage *pStorage_space@<EAX>, FILE *pF@<EDX>, int pCount@<EBX>)
int LoadNShadeTables(tBrender_storage* pStorage_space, FILE* pF, int pCount) {
tPath_name the_path;
int i;
int j;
int new_ones;
int total;
char s[256];
char* str;
br_pixelmap* temp_array[50];
LOG_TRACE("(%p, %p, %d)", pStorage_space, pF, pCount);
new_ones = 0;
for (i = 0; i < pCount; i++) {
PossibleService();
GetALineAndDontArgue(pF, s);
PathCat(the_path, gApplication_path, "SHADETAB");
PathCat(the_path, the_path, str);
total = DRPixelmapLoadMany(the_path, temp_array, 50);
if (total == 0) {
FatalError(kFatalError_LoadShadeTableFile_S, str);
}
for (j = 0; j < total; j++) {
if (temp_array[j]) {
switch (AddShadeTableToStorage(pStorage_space, temp_array[j])) {
case eStorage_not_enough_room:
FatalError(kFatalError_InsufficientShadeTableSlots);
break;
case eStorage_duplicate:
BrPixelmapFree(temp_array[j]);
break;
case eStorage_allocated:
BrTableAdd(temp_array[j]);
new_ones++;
break;
}
}
}
}
return new_ones;
}
// IDA: br_pixelmap* __usercall LoadSingleShadeTable@<EAX>(tBrender_storage *pStorage_space@<EAX>, char *pName@<EDX>)
br_pixelmap* LoadSingleShadeTable(tBrender_storage* pStorage_space, char* pName) {
br_pixelmap* temp;
LOG_TRACE("(%p, \"%s\")", pStorage_space, pName);
temp = LoadShadeTable(pName);
if (!temp) {
return BrTableFind(pName);
}
switch (AddShadeTableToStorage(pStorage_space, temp)) {
case eStorage_not_enough_room:
FatalError(kFatalError_InsufficientShadeTableSlots);
break;
case eStorage_duplicate:
BrPixelmapFree(temp);
return BrTableFind(pName);
case eStorage_allocated:
BrTableAdd(temp);
return temp;
}
return NULL;
}
// IDA: int __usercall LoadNMaterials@<EAX>(tBrender_storage *pStorage_space@<EAX>, FILE *pF@<EDX>, int pCount@<EBX>)
int LoadNMaterials(tBrender_storage* pStorage_space, FILE* pF, int pCount) {
tPath_name the_path;
int i;
int j;
int new_ones;
int total;
char s[256];
char* str;
br_material* temp_array[200];
LOG_TRACE("(%p, %p, %d)", pStorage_space, pF, pCount);
new_ones = 0;
for (i = 0; i < pCount; ++i) {
PossibleService();
GetALineAndDontArgue(pF, s);
PathCat(the_path, gApplication_path, "MATERIAL");
PathCat(the_path, the_path, str);
total = BrMaterialLoadMany(the_path, temp_array, 200);
if (total == 0) {
FatalError(kFatalError_LoadMaterialFile_S, str);
}
for (j = 0; j < total; j++) {
if (temp_array[j]) {
switch (AddMaterialToStorage(pStorage_space, temp_array[j])) {
case eStorage_not_enough_room:
FatalError(kFatalError_InsufficientMaterialSlots);
break;
case eStorage_duplicate:
BrMaterialFree(temp_array[j]);
break;
case eStorage_allocated:
BrMaterialAdd(temp_array[j]);
new_ones++;
}
}
}
}
return new_ones;
}
// IDA: int __usercall LoadNModels@<EAX>(tBrender_storage *pStorage_space@<EAX>, FILE *pF@<EDX>, int pCount@<EBX>)
int LoadNModels(tBrender_storage* pStorage_space, FILE* pF, int pCount) {
tPath_name the_path;
int i;
int j;
int new_ones;
int total;
char s[256];
char* str;
br_model* temp_array[2000];
//v11model* prepared; // Pierre-Marie Baty -- unused variable
//int group; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, %p, %d)", pStorage_space, pF, pCount);
new_ones = 0;
for (i = 0; i < pCount; i++) {
PossibleService();
GetALineAndDontArgue(pF, s);
PathCat(the_path, gApplication_path, "MODELS");
PathCat(the_path, the_path, str);
total = BrModelLoadMany(the_path, temp_array, 2000);
if (total == 0) {
FatalError(kFatalError_LoadModelFile_S, str);
}
for (j = 0; j < total; j++) {
if (temp_array[j]) {
switch (AddModelToStorage(pStorage_space, temp_array[j])) {
case eStorage_not_enough_room:
FatalError(kFatalError_InsufficientModelSlots);
break;
case eStorage_duplicate:
BrModelFree(temp_array[j]);
break;
case eStorage_allocated:
temp_array[j]->flags |= 0x80u;
RemoveDoubleSided(temp_array[j]);
BrModelAdd(temp_array[j]);
++new_ones;
break;
}
}
}
}
return new_ones;
}
// IDA: void __usercall DodgyModelUpdate(br_model *pM@<EAX>)
void DodgyModelUpdate(br_model* pM) {
LOG_TRACE("(%p)", pM);
BrResFree(pM->faces);
BrResFree(pM->vertices);
pM->nfaces = 0;
pM->nvertices = 0;
pM->faces = NULL;
pM->vertices = NULL;
}
// IDA: br_material* __usercall SuffixedMaterial@<EAX>(br_material *pOld@<EAX>, char *pSuffix@<EDX>)
br_material* SuffixedMaterial(br_material* pOld, char* pSuffix) {
br_material* new_mat;
char* new_id;
LOG_TRACE("(%p, \"%s\")", pOld, pSuffix);
new_id
= BrMemAllocate
(strlen(pOld
->identifier
) + strlen(pSuffix
) + 1, kMem_new_mat_id
);
sprintf(new_id
, "%s%s", pOld
->identifier
, pSuffix
);
new_mat = BrMaterialFind(new_id);
if (new_mat == NULL) {
new_mat = BrMaterialAllocate(NULL);
MaterialCopy(new_mat, pOld);
new_mat->identifier = new_id;
BrMaterialAdd(new_mat);
} else {
BrMemFree(new_id);
}
return new_mat;
}
// IDA: int __usercall FaceIsRoad@<EAX>(br_model *pModel@<EAX>, tU16 pFace@<EDX>)
int FaceIsRoad(br_model* pModel, tU16 pFace) {
br_vector3 v0;
br_vector3 v1;
br_vector3 cross;
LOG_TRACE("(%p, %d)", pModel, pFace);
BrVector3Sub(&v0, &pModel->vertices[pModel->faces[pFace].vertices[0]].p, &pModel->vertices[pModel->faces[pFace].vertices[1]].p);
BrVector3Sub(&v1, &pModel->vertices[pModel->faces[pFace].vertices[1]].p, &pModel->vertices[pModel->faces[pFace].vertices[2]].p);
BrVector3Cross(&cross, &v0, &v1);
return sqrtf(cross.v[0] * cross.v[0] + cross.v[2] * cross.v[2]) < 0.7f * cross.v[1];
}
// IDA: br_material* __usercall RoadPerspToUntex@<EAX>(br_model *pModel@<EAX>, tU16 pFace@<EDX>)
br_material* RoadPerspToUntex(br_model* pModel, tU16 pFace) {
br_material* old_mat;
br_material* new_mat;
LOG_TRACE("(%p, %d)", pModel, pFace);
old_mat = pModel->faces[pFace].material;
if (old_mat->colour_map == NULL || !FaceIsRoad(pModel, pFace)) {
return NULL;
}
new_mat = SuffixedMaterial(old_mat, ".road");
if (new_mat->colour_map != NULL) {
new_mat->colour_map = NULL;
BrMaterialUpdate(new_mat, BR_MATU_ALL);
}
return new_mat;
}
// IDA: br_material* __usercall WallPerspToLinear@<EAX>(br_model *pModel@<EAX>, tU16 pFace@<EDX>)
br_material* WallPerspToLinear(br_model* pModel, tU16 pFace) {
br_material* old_mat;
br_material* new_mat;
LOG_TRACE("(%p, %d)", pModel, pFace);
old_mat = pModel->faces[pFace].material;
if (old_mat->colour_map == NULL || !(old_mat->flags & BR_MATF_PERSPECTIVE) || FaceIsRoad(pModel, pFace)) {
return NULL;
}
new_mat = SuffixedMaterial(old_mat, ".pwall");
if (new_mat->flags & BR_MATF_PERSPECTIVE) {
new_mat->flags &= ~BR_MATF_PERSPECTIVE;
BrMaterialUpdate(new_mat, BR_MATU_ALL);
}
return new_mat;
}
// IDA: br_material* __usercall WallPerspToUntex@<EAX>(br_model *pModel@<EAX>, tU16 pFace@<EDX>)
br_material* WallPerspToUntex(br_model* pModel, tU16 pFace) {
br_material* old_mat;
br_material* new_mat;
LOG_TRACE("(%p, %d)", pModel, pFace);
old_mat = pModel->faces[pFace].material;
if (old_mat->colour_map == NULL || FaceIsRoad(pModel, pFace)) {
return NULL;
}
if (old_mat->flags & BR_MATF_PERSPECTIVE) {
new_mat = SuffixedMaterial(old_mat, ".pwall");
} else {
new_mat = SuffixedMaterial(old_mat, ".lwall");
}
if (new_mat->colour_map != NULL) {
new_mat->colour_map = NULL;
BrMaterialUpdate(new_mat, BR_MATU_ALL);
}
return new_mat;
}
// IDA: void __usercall ProcessModelFaceMaterials2(br_model *pModel@<EAX>, tPMFM2CB pCallback@<EDX>)
void ProcessModelFaceMaterials2(br_model* pModel, tPMFM2CB pCallback) {
//tU16 f; // Pierre-Marie Baty -- unused variable
//tU16 group; // Pierre-Marie Baty -- unused variable
//br_material* old_mat; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, %d)", pModel, pCallback);
NOT_IMPLEMENTED();
}
// IDA: void __usercall ProcessModelFaceMaterials(br_model *pModel@<EAX>, tPMFMCB pCallback@<EDX>)
void ProcessModelFaceMaterials(br_model* pModel, tPMFMCB pCallback) {
tU16 f;
br_material* possible_mat;
br_material* new_mat;
LOG_TRACE("(%p, %d)", pModel, pCallback);
new_mat = NULL;
for (f = 0; f < pModel->nfaces; f++) {
if (pModel->faces[f].material != NULL) {
possible_mat = (*pCallback)(pModel, f);
if (possible_mat != NULL) {
pModel->faces[f].material = possible_mat;
new_mat = possible_mat;
}
}
}
if (new_mat != NULL) {
BrModelUpdate(pModel, BR_MATU_ALL);
}
}
// IDA: int __usercall LoadNTrackModels@<EAX>(tBrender_storage *pStorage_space@<EAX>, FILE *pF@<EDX>, int pCount@<EBX>)
int LoadNTrackModels(tBrender_storage* pStorage_space, FILE* pF, int pCount) {
tPath_name the_path;
int i;
int j;
int new_ones;
int total;
//int group; // Pierre-Marie Baty -- unused variable
char s[256];
char* str;
br_model* temp_array[2000];
//v11model* prepared; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, %p, %d)", pStorage_space, pF, pCount);
new_ones = 0;
for (i = 0; i < pCount; i++) {
GetALineAndDontArgue(pF, s);
PathCat(the_path, gApplication_path, "MODELS");
PathCat(the_path, the_path, str);
total = BrModelLoadMany(the_path, temp_array, 2000);
if (total == 0) {
FatalError(kFatalError_LoadModelFile_S, str);
}
for (j = 0; j < total; j++) {
if (temp_array[j]) {
switch (AddModelToStorage(pStorage_space, temp_array[j])) {
case eStorage_not_enough_room:
FatalError(kFatalError_InsufficientModelSlots);
break;
case eStorage_duplicate:
BrModelFree(temp_array[j]);
break;
case eStorage_allocated:
temp_array[j]->flags |= BR_MODF_UPDATEABLE;
if (gRoad_texturing_level == eRTL_none) {
ProcessModelFaceMaterials(temp_array[j], RoadPerspToUntex);
}
switch (gWall_texturing_level) {
case eWTL_none:
ProcessModelFaceMaterials(temp_array[j], WallPerspToUntex);
break;
case eWTL_linear:
ProcessModelFaceMaterials(temp_array[j], WallPerspToLinear);
break;
default:
break;
}
RemoveDoubleSided(temp_array[j]);
BrModelAdd(temp_array[j]);
new_ones++;
}
}
}
}
return new_ones;
}
// IDA: void __usercall LoadSomePixelmaps(tBrender_storage *pStorage_space@<EAX>, FILE *pF@<EDX>)
void LoadSomePixelmaps(tBrender_storage* pStorage_space, FILE* pF) {
//tPath_name the_path; // Pierre-Marie Baty -- unused variable
//int i; // Pierre-Marie Baty -- unused variable
//int j; // Pierre-Marie Baty -- unused variable
int count;
//int new_ones; // Pierre-Marie Baty -- unused variable
char s[256];
char* str;
//br_pixelmap* temp_array[200]; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, %p)", pStorage_space, pF);
GetALineAndDontArgue(pF, s);
LoadNPixelmaps(pStorage_space, pF, count);
}
// IDA: void __usercall LoadSomeShadeTables(tBrender_storage *pStorage_space@<EAX>, FILE *pF@<EDX>)
void LoadSomeShadeTables(tBrender_storage* pStorage_space, FILE* pF) {
//tPath_name the_path; // Pierre-Marie Baty -- unused variable
//int i; // Pierre-Marie Baty -- unused variable
//int j; // Pierre-Marie Baty -- unused variable
int count;
//int new_ones; // Pierre-Marie Baty -- unused variable
char s[256];
char* str;
//br_pixelmap* temp_array[50]; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, %p)", pStorage_space, pF);
GetALineAndDontArgue(pF, s);
LoadNShadeTables(pStorage_space, pF, count);
}
// IDA: void __usercall LoadSomeMaterials(tBrender_storage *pStorage_space@<EAX>, FILE *pF@<EDX>)
void LoadSomeMaterials(tBrender_storage* pStorage_space, FILE* pF) {
//tPath_name the_path; // Pierre-Marie Baty -- unused variable
//int i; // Pierre-Marie Baty -- unused variable
//int j; // Pierre-Marie Baty -- unused variable
int count;
//int new_ones; // Pierre-Marie Baty -- unused variable
char s[256];
char* str;
//br_material* temp_array[200]; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, %p)", pStorage_space, pF);
GetALineAndDontArgue(pF, s);
LoadNMaterials(pStorage_space, pF, count);
}
// IDA: void __usercall LoadSomeModels(tBrender_storage *pStorage_space@<EAX>, FILE *pF@<EDX>)
void LoadSomeModels(tBrender_storage* pStorage_space, FILE* pF) {
//tPath_name the_path; // Pierre-Marie Baty -- unused variable
//int i; // Pierre-Marie Baty -- unused variable
//int j; // Pierre-Marie Baty -- unused variable
int count;
//int new_ones; // Pierre-Marie Baty -- unused variable
char s[256];
char* str;
//br_model* temp_array[2000]; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, %p)", pStorage_space, pF);
GetALineAndDontArgue(pF, s);
LoadNModels(pStorage_space, pF, count);
}
// IDA: void __usercall LoadSomeTrackModels(tBrender_storage *pStorage_space@<EAX>, FILE *pF@<EDX>)
void LoadSomeTrackModels(tBrender_storage* pStorage_space, FILE* pF) {
//tPath_name the_path; // Pierre-Marie Baty -- unused variable
//int i; // Pierre-Marie Baty -- unused variable
//int j; // Pierre-Marie Baty -- unused variable
int count;
//int new_ones; // Pierre-Marie Baty -- unused variable
char s[256];
char* str;
//br_model* temp_array[2000]; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, %p)", pStorage_space, pF);
GetALineAndDontArgue(pF, s);
LoadNTrackModels(pStorage_space, pF, count);
}
// IDA: void __usercall AddFunkGrooveBinding(int pSlot_number@<EAX>, float *pPeriod_address@<EDX>)
void AddFunkGrooveBinding(int pSlot_number, float* pPeriod_address) {
LOG_TRACE("(%d, %p)", pSlot_number, pPeriod_address);
if (pSlot_number < 0 || pSlot_number >= COUNT_OF(gGroove_funk_bindings)) {
FatalError(kFatalError_DefinedRefNumGrooveFunkOutOfRange);
}
gGroove_funk_bindings[pSlot_number] = pPeriod_address;
*pPeriod_address = 0.0f;
}
// IDA: void __usercall ControlBoundFunkGroove(int pSlot_number@<EAX>, float pValue)
void ControlBoundFunkGroove(int pSlot_number, float pValue) {
LOG_TRACE("(%d, %f)", pSlot_number, pValue);
if (pSlot_number < 0) {
return;
}
if (pSlot_number >= COUNT_OF(gGroove_funk_bindings)) {
FatalError(kFatalError_UsedRefNumGrooveFunkOutOfRange);
}
*gGroove_funk_bindings[pSlot_number] = pValue;
}
// IDA: float __usercall ControlBoundFunkGroovePlus@<ST0>(int pSlot_number@<EAX>, float pValue)
float ControlBoundFunkGroovePlus(int pSlot_number, float pValue) {
LOG_TRACE("(%d, %f)", pSlot_number, pValue);
if (pSlot_number < 0) {
return 0.f;
}
if (pSlot_number >= COUNT_OF(gGroove_funk_bindings)) {
FatalError(kFatalError_UsedRefNumGrooveFunkOutOfRange);
}
*gGroove_funk_bindings[pSlot_number] = fmodf(*gGroove_funk_bindings[pSlot_number] + pValue, 1.f);
return *gGroove_funk_bindings[pSlot_number];
}
// IDA: void __usercall ShiftBoundGrooveFunks(char *pStart@<EAX>, char *pEnd@<EDX>, int pDelta@<EBX>)
void ShiftBoundGrooveFunks(char* pStart, char* pEnd, ptrdiff_t pDelta) {
int i;
LOG_TRACE("(\"%s\", \"%s\", %d)", pStart, pEnd, pDelta);
for (i = 0; i < COUNT_OF(gGroove_funk_bindings); i++) {
if (pStart <= (char*)gGroove_funk_bindings[i] && (char*)gGroove_funk_bindings[i] < pEnd) {
gGroove_funk_bindings[i] = (float*)((char*)gGroove_funk_bindings[i] + (pDelta & ~(sizeof(void*) - 1))); // original code is (pDelta & 0xFFFFFFFC) but this caused problems;
}
}
}
// IDA: tFunkotronic_spec* __cdecl AddNewFunkotronic()
tFunkotronic_spec* AddNewFunkotronic(void) {
void* new_array;
int i;
LOG_TRACE("()");
for (i = 0; i < gFunkotronics_array_size; i++) {
if (gFunkotronics_array[i].owner == -999) {
memset(&gFunkotronics_array
[i
], 0, sizeof(tFunkotronic_spec
));
return &gFunkotronics_array[i];
}
}
gFunkotronics_array_size += 16;
new_array = BrMemCalloc(gFunkotronics_array_size, sizeof(tFunkotronic_spec), kMem_funk_spec);
if (gFunkotronics_array != NULL) {
memcpy(new_array
, gFunkotronics_array
, (gFunkotronics_array_size
- 16) * sizeof(tFunkotronic_spec
));
ShiftBoundGrooveFunks(
(char*)gFunkotronics_array,
(char*)&gFunkotronics_array[gFunkotronics_array_size - 16],
(char*)new_array - (char*)gFunkotronics_array);
BrMemFree(gFunkotronics_array);
}
gFunkotronics_array = new_array;
for (i = 0; i < 16; i++) {
gFunkotronics_array[gFunkotronics_array_size - 16 + i].owner = -999;
}
return &gFunkotronics_array[gFunkotronics_array_size - 16];
}
// IDA: void __usercall DisposeFunkotronics(int pOwner@<EAX>)
void DisposeFunkotronics(int pOwner) {
int i;
tFunkotronic_spec* the_funk;
LOG_TRACE("(%d)", pOwner);
if (gFunkotronics_array == NULL) {
return;
}
for (i = 0; i < gFunkotronics_array_size; i++) {
the_funk = &gFunkotronics_array[i];
PossibleService();
if (the_funk->owner == pOwner) {
the_funk->owner = -999;
if (the_funk->proximity_array != NULL) {
BrMemFree(the_funk->proximity_array);
}
if (the_funk->texture_animation_type == eTexture_animation_flic) {
BrMemFree(the_funk->texture_animation_data.flic_info.flic_data);
EndFlic(&the_funk->texture_animation_data.flic_info.flic_descriptor);
BrMemFree(the_funk->material->colour_map->pixels);
the_funk->material->colour_map->pixels = NULL;
BrPixelmapFree(the_funk->material->colour_map);
the_funk->material->colour_map = NULL;
}
}
}
}
// IDA: void __usercall AddProximityVertex(br_vector3 *pV@<EAX>, tFunkotronic_spec *pThe_funk@<EDX>)
void AddProximityVertex(br_vector3* pV, tFunkotronic_spec* pThe_funk) {
LOG_TRACE("(%p, %p)", pV, pThe_funk);
pThe_funk->proximity_array[pThe_funk->proximity_count] = *pV;
pThe_funk->proximity_count++;
}
// IDA: void __cdecl AddProximityVertexXYZ(br_scalar pX, br_scalar pY, br_scalar pZ, tFunkotronic_spec *pThe_funk)
void AddProximityVertexXYZ(br_scalar pX, br_scalar pY, br_scalar pZ, tFunkotronic_spec* pThe_funk) {
br_vector3 v;
LOG_TRACE("(%f, %f, %f, %p)", pX, pY, pZ, pThe_funk);
v.v[0] = pX;
v.v[1] = pY;
v.v[2] = pZ;
AddProximityVertex(&v, pThe_funk);
}
// IDA: br_uint_32 __usercall CalcProximities@<EAX>(br_actor *pActor@<EAX>, br_material *pMat@<EDX>, tFunkotronic_spec *pThe_funk@<EBX>)
br_uint_32 CalcProximities(br_actor* pActor, br_material* pMat, tFunkotronic_spec* pThe_funk) {
br_face* the_face;
int i;
LOG_TRACE("(%p, %p, %p)", pActor, pMat, pThe_funk);
if (pActor->model != NULL) {
if (pThe_funk->material == pMat) {
pThe_funk->proximity_count += 8;
} else {
for (i = 0; i < pActor->model->nfaces; i++) {
the_face = &pActor->model->faces[i];
if (pThe_funk->material == the_face->material) {
pThe_funk->proximity_count += 3;
}
}
}
}
return 0;
}
// IDA: br_uint_32 __usercall AddProximities@<EAX>(br_actor *pActor@<EAX>, br_material *pMat@<EDX>, tFunkotronic_spec *pThe_funk@<EBX>)
br_uint_32 AddProximities(br_actor* pActor, br_material* pMat, tFunkotronic_spec* pThe_funk) {
br_face* the_face;
int i;
LOG_TRACE("(%p, %p, %p)", pActor, pMat, pThe_funk);
if (pActor->model != NULL) {
if (pThe_funk->material == pMat) {
AddProximityVertexXYZ(
pActor->model->bounds.min.v[0],
pActor->model->bounds.min.v[1],
pActor->model->bounds.min.v[2],
pThe_funk);
AddProximityVertexXYZ(
pActor->model->bounds.min.v[0],
pActor->model->bounds.min.v[1],
pActor->model->bounds.max.v[2],
pThe_funk);
AddProximityVertexXYZ(
pActor->model->bounds.min.v[0],
pActor->model->bounds.max.v[1],
pActor->model->bounds.min.v[2],
pThe_funk);
AddProximityVertexXYZ(
pActor->model->bounds.min.v[0],
pActor->model->bounds.max.v[1],
pActor->model->bounds.max.v[2],
pThe_funk);
AddProximityVertexXYZ(
pActor->model->bounds.max.v[0],
pActor->model->bounds.min.v[1],
pActor->model->bounds.min.v[2],
pThe_funk);
AddProximityVertexXYZ(
pActor->model->bounds.max.v[0],
pActor->model->bounds.min.v[1],
pActor->model->bounds.max.v[2],
pThe_funk);
AddProximityVertexXYZ(
pActor->model->bounds.max.v[0],
pActor->model->bounds.max.v[1],
pActor->model->bounds.min.v[2],
pThe_funk);
AddProximityVertexXYZ(
pActor->model->bounds.max.v[0],
pActor->model->bounds.max.v[1],
pActor->model->bounds.max.v[2],
pThe_funk);
} else {
for (i = 0; i < pActor->model->nfaces; i++) {
the_face = &pActor->model->faces[i];
if (pThe_funk->material == the_face->material) {
AddProximityVertex(&pActor->model->vertices[the_face->vertices[0]].p, pThe_funk);
AddProximityVertex(&pActor->model->vertices[the_face->vertices[1]].p, pThe_funk);
AddProximityVertex(&pActor->model->vertices[the_face->vertices[2]].p, pThe_funk);
}
}
}
}
return 0;
}
// IDA: void __usercall Adjust2FloatsForExceptions(float *pVictim1@<EAX>, float *pVictim2@<EDX>, br_pixelmap *pCulprit@<EBX>)
void Adjust2FloatsForExceptions(float* pVictim1, float* pVictim2, br_pixelmap* pCulprit) {
//tException_list e; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, %p, %p)", pVictim1, pVictim2, pCulprit);
NOT_IMPLEMENTED();
}
// IDA: void __usercall AddFunkotronics(FILE *pF@<EAX>, int pOwner@<EDX>, int pRef_offset@<EBX>)
void AddFunkotronics(FILE* pF, int pOwner, int pRef_offset) {
char s[256];
char* str;
int first_time;
int i;
int j;
tFunkotronic_spec* the_funk;
float speed1;
float speed2;
float a_min;
float d_min;
float s_min;
float a_max;
float d_max;
float s_max;
void* the_pixels;
br_pixelmap* the_pixelmap;
float x_0;
float x_1;
int d_0;
int d_1;
LOG_TRACE("(%p, %d, %d)", pF, pOwner, pRef_offset);
first_time = 1;
PossibleService();
GetALineAndDontArgue(pF, s);
if (strcmp(s
, "END OF FUNK") == 0) {
break;
}
if (!first_time) {
if (strcmp(s
, "NEXT FUNK") != 0) {
FatalError(kFatalError_FunkotronicFile);
}
GetALineAndDontArgue(pF, s);
}
first_time = 0;
the_funk = AddNewFunkotronic();
the_funk->owner = pOwner;
the_funk->material = BrMaterialFind(str);
if (the_funk->material == NULL) {
FatalError(kFatalError_FindMaterialUsedByFunkotronicFile_S, str);
}
the_funk->mode = GetALineAndInterpretCommand(pF, gFunk_nature_names, COUNT_OF(gFunk_nature_names));
the_funk->matrix_mod_type = GetALineAndInterpretCommand(pF, gFunk_type_names, COUNT_OF(gFunk_type_names));
if (the_funk->matrix_mod_type != eMatrix_mod_none) {
the_funk->matrix_mode = GetALineAndInterpretCommand(pF, gFunk_move_names, COUNT_OF(gFunk_move_names));
}
switch (the_funk->matrix_mod_type) {
case eMatrix_mod_spin:
if (the_funk->matrix_mode == eMove_controlled || the_funk->matrix_mode == eMove_absolute) {
i = GetAnInt(pF);
AddFunkGrooveBinding(i + pRef_offset, &the_funk->matrix_mod_data.spin_info.period);
} else {
x_0 = GetAFloat(pF);
the_funk->matrix_mod_data.spin_info.period = (x_0 == 0.0f) ? 0.0f : 1000.0f / x_0;
}
break;
case eMatrix_mod_rock: // rock
if (the_funk->matrix_mode == eMove_controlled || the_funk->matrix_mode == eMove_absolute) {
d_0 = GetAnInt(pF);
AddFunkGrooveBinding(d_0 + pRef_offset, &the_funk->matrix_mod_data.rock_info.period);
} else {
x_0 = GetAFloat(pF);
the_funk->matrix_mod_data.rock_info.period = (x_0 == 0.0f) ? 0.0f : 1000.0f / x_0;
}
the_funk->matrix_mod_data.rock_info.rock_angle = GetAFloat(pF);
GetPairOfFloats(pF, &x_0, &x_1);
the_funk->matrix_mod_data.rock_info.x_centre = x_0 / 100.0f;
the_funk->matrix_mod_data.rock_info.y_centre = x_1 / 100.0f;
break;
case eMatrix_mod_throb: // throb
if (the_funk->matrix_mode == eMove_controlled || the_funk->matrix_mode == eMove_absolute) {
GetPairOfInts(pF, &d_0, &d_1);
if (d_0 >= 0) {
AddFunkGrooveBinding(d_0 + pRef_offset, &the_funk->matrix_mod_data.throb_info.x_period);
}
if (d_1 >= 0) {
AddFunkGrooveBinding(d_1 + pRef_offset, &the_funk->matrix_mod_data.throb_info.y_period);
}
} else {
GetPairOfFloats(pF, &speed1, &speed2);
the_funk->matrix_mod_data.throb_info.x_period = (speed1 == 0.0f) ? 0.0f : 1000.0f / speed1;
the_funk->matrix_mod_data.throb_info.y_period = (speed2 == 0.0f) ? 0.0f : 1000.0f / speed2;
}
GetPairOfFloatPercents(
pF,
&the_funk->matrix_mod_data.throb_info.x_magnitude,
&the_funk->matrix_mod_data.throb_info.y_magnitude);
GetPairOfFloats(pF, &x_0, &x_1);
the_funk->matrix_mod_data.throb_info.x_centre = x_0 / 100.0f;
the_funk->matrix_mod_data.throb_info.y_centre = x_1 / 100.0f;
if (the_funk->matrix_mode != eMove_controlled) {
if (the_funk->matrix_mod_data.throb_info.x_period == 0.0f) {
the_funk->matrix_mod_data.throb_info.x_period = 1.0f;
the_funk->matrix_mod_data.throb_info.x_magnitude = 0.0f;
}
if (the_funk->matrix_mod_data.throb_info.y_period == 0.0f) {
the_funk->matrix_mod_data.throb_info.y_period = 1.0f;
the_funk->matrix_mod_data.throb_info.y_magnitude = 0.0f;
}
}
break;
case eMatrix_mod_slither: // slither
if (the_funk->matrix_mode == eMove_controlled || the_funk->matrix_mode == eMove_absolute) {
GetPairOfInts(pF, &d_0, &d_1);
if (d_0 >= 0) {
AddFunkGrooveBinding(d_0 + pRef_offset, &the_funk->matrix_mod_data.slither_info.x_period);
}
if (d_1 >= 0) {
AddFunkGrooveBinding(d_1 + pRef_offset, &the_funk->matrix_mod_data.slither_info.y_period);
}
} else {
GetPairOfFloats(pF, &speed1, &speed2);
the_funk->matrix_mod_data.slither_info.x_period = (speed1 == 0.0f) ? 0.0f : 1000.0f / speed1;
the_funk->matrix_mod_data.slither_info.y_period = (speed2 == 0.0f) ? 0.0f : 1000.0f / speed2;
}
GetPairOfFloatPercents(
pF,
&the_funk->matrix_mod_data.slither_info.x_magnitude,
&the_funk->matrix_mod_data.slither_info.y_magnitude);
if (the_funk->matrix_mode != eMove_controlled) {
if (the_funk->matrix_mod_data.slither_info.x_period == 0.0f) {
the_funk->matrix_mod_data.slither_info.x_period = 1.0f;
the_funk->matrix_mod_data.slither_info.x_magnitude = 0.0f;
}
if (the_funk->matrix_mod_data.slither_info.y_period == 0.0f) {
the_funk->matrix_mod_data.slither_info.y_period = 1.0f;
the_funk->matrix_mod_data.slither_info.y_magnitude = 0.0f;
}
}
break;
case eMatrix_mod_roll: // roll
if (the_funk->matrix_mode == eMove_controlled || the_funk->matrix_mode == eMove_absolute) {
GetPairOfInts(pF, &d_0, &d_1);
if (d_0 >= 0) {
AddFunkGrooveBinding(d_0 + pRef_offset, &the_funk->matrix_mod_data.roll_info.x_period);
}
if (d_1 >= 0) {
AddFunkGrooveBinding(d_1 + pRef_offset, &the_funk->matrix_mod_data.roll_info.y_period);
}
} else {
GetPairOfFloats(pF, &speed1, &speed2);
the_funk->matrix_mod_data.roll_info.x_period = speed1 == 0.0f ? 0.0f : 1000.0f / speed1;
the_funk->matrix_mod_data.roll_info.y_period = speed2 == 0.0f ? 0.0f : 1000.0f / speed2;
}
break;
default:
break;
}
the_funk->lighting_animation_type = GetALineAndInterpretCommand(pF, gFunk_move_names, COUNT_OF(gFunk_move_names));
if (the_funk->lighting_animation_type != eMove_none) {
if (the_funk->lighting_animation_type == eMove_controlled || the_funk->lighting_animation_type == eMove_absolute) {
d_0 = GetAnInt(pF);
AddFunkGrooveBinding(d_0 + pRef_offset, &the_funk->lighting_animation_period);
} else {
x_0 = GetAFloat(pF);
the_funk->lighting_animation_period = (x_0 == 0.0f) ? 0.0f : 1000.0f / x_0;
}
GetThreeFloatPercents(pF, &a_min, &d_min, &s_min);
GetThreeFloatPercents(pF, &a_max, &d_max, &s_max);
the_funk->ambient_base = (a_min + a_max) / 2.0f;
the_funk->direct_base = (d_min + d_max) / 2.0f;
the_funk->specular_base = (s_min + s_max) / 2.0f;
the_funk->ambient_delta = (a_max - a_min) / 2.0f;
the_funk->direct_delta = (d_max - d_min) / 2.0f;
the_funk->specular_delta = (s_max - s_min) / 2.0f;
}
the_funk->texture_animation_type = GetALineAndInterpretCommand(pF, gFunk_anim_names, COUNT_OF(gFunk_anim_names));
if (the_funk->texture_animation_type != eTexture_animation_none) {
the_funk->time_mode = GetALineAndInterpretCommand(pF, gTime_mode_names, COUNT_OF(gTime_mode_names));
}
if (the_funk->texture_animation_type == eTexture_animation_flic && gAusterity_mode) {
the_funk->texture_animation_type = eTexture_animation_none;
GetALineAndDontArgue(pF, s);
}
the_funk->last_frame = 0.0f;
if (the_funk->texture_animation_type == eTexture_animation_flic) {
GetAString(pF, s);
the_funk->texture_animation_data.flic_info.flic_data = 0;
if (LoadFlicData(
s,
&the_funk->texture_animation_data.flic_info.flic_data,
&the_funk->texture_animation_data.flic_info.flic_data_length)) {
the_funk->texture_animation_data.flic_info.flic_descriptor.data_start = NULL;
StartFlic(
s,
-1,
&the_funk->texture_animation_data.flic_info.flic_descriptor,
the_funk->texture_animation_data.flic_info.flic_data_length,
(tS8*)the_funk->texture_animation_data.flic_info.flic_data,
0,
0,
0,
0);
the_funk->last_frame = 0.0f;
the_pixels = BrMemAllocate(
the_funk->texture_animation_data.flic_info.flic_descriptor.height
* ((the_funk->texture_animation_data.flic_info.flic_descriptor.width + 3) & ~3),
kMem_video_pixels);
if (gScreen->row_bytes < 0) {
BrFatal(
"C:\\Msdev\\Projects\\DethRace\\World.c",
1729,
"Bruce bug at line %d, file C:\\Msdev\\Projects\\DethRace\\World.c",
193);
}
the_pixelmap = DRPixelmapAllocate(
gScreen->type,
the_funk->texture_animation_data.flic_info.flic_descriptor.width,
the_funk->texture_animation_data.flic_info.flic_descriptor.height,
the_pixels,
0);
AssertFlicPixelmap(&the_funk->texture_animation_data.flic_info.flic_descriptor, the_pixelmap);
the_funk->material->colour_map = the_pixelmap;
BrMaterialUpdate(the_funk->material, BR_MATU_ALL);
} else {
the_funk->texture_animation_type = eTexture_animation_none;
}
} else if (the_funk->texture_animation_type == eTexture_animation_frames) {
i = GetALineAndInterpretCommand(pF, gFunk_move_names, COUNT_OF(gFunk_move_names));
the_funk->texture_animation_data.frames_info.mode = i;
if (the_funk->texture_animation_data.frames_info.mode == eMove_controlled
|| the_funk->texture_animation_data.frames_info.mode == eMove_absolute) {
d_0 = GetAnInt(pF);
AddFunkGrooveBinding(d_0 + pRef_offset, &the_funk->texture_animation_data.frames_info.period);
} else {
x_0 = GetAFloat(pF);
the_funk->texture_animation_data.frames_info.period = (x_0 == 0.0f) ? 0.0f : 1000.0F / x_0;
}
the_funk->texture_animation_data.frames_info.texture_count = (int)GetAFloat(pF);
for (i = 0; i < the_funk->texture_animation_data.frames_info.texture_count; i++) {
GetAString(pF, s);
the_funk->texture_animation_data.frames_info.textures[i] = BrMapFind(s);
if (the_funk->texture_animation_data.frames_info.textures[i] == NULL) {
FatalError(kFatalError_AnimationFramePixelmapUsedByFunkotronicFile);
}
}
}
the_funk->proximity_count = 0;
the_funk->proximity_array = 0;
if (the_funk->mode == eFunk_mode_distance) {
DRActorEnumRecurseWithMat(gUniverse_actor, NULL, (recurse_with_mat_cbfn*)CalcProximities, the_funk);
the_funk->proximity_array = BrMemAllocate(sizeof(br_vector3) * the_funk->proximity_count, kMem_funk_prox_array);
the_funk->proximity_count = 0;
DRActorEnumRecurseWithMat(gUniverse_actor, 0, (recurse_with_mat_cbfn*)AddProximities, the_funk);
for (i = 0; i < the_funk->proximity_count; i++) {
for (j = i + 1; j < the_funk->proximity_count; j++) {
if (the_funk->proximity_array[j].v[0] == the_funk->proximity_array[i].v[0]
&& the_funk->proximity_array[j].v[1] == the_funk->proximity_array[i].v[1]
&& the_funk->proximity_array[j].v[2] == the_funk->proximity_array[i].v[2]) {
&the_funk->proximity_array[j],
&the_funk->proximity_array[j + 1],
sizeof(br_vector3) * (the_funk->proximity_count - j - 1));
the_funk->proximity_count--;
j--;
}
}
}
}
}
}
// IDA: void __usercall DisposeGroovidelics(int pOwner@<EAX>)
void DisposeGroovidelics(int pOwner) {
int i;
tGroovidelic_spec* the_groove;
LOG_TRACE("(%d)", pOwner);
if (gGroovidelics_array == NULL) {
return;
}
for (i = 0; i < gGroovidelics_array_size; i++) {
the_groove = &gGroovidelics_array[i];
PossibleService();
if (the_groove->owner == pOwner) {
the_groove->owner = -999;
}
}
}
// IDA: tGroovidelic_spec* __cdecl AddNewGroovidelic()
tGroovidelic_spec* AddNewGroovidelic(void) {
void* new_array;
int i;
LOG_TRACE("()");
for (i = 0; i < gGroovidelics_array_size; i++) {
if (gGroovidelics_array[i].owner == -999) {
memset(&gGroovidelics_array
[i
], 0, sizeof(tGroovidelic_spec
));
return &gGroovidelics_array[i];
}
}
gGroovidelics_array_size += 16;
new_array = BrMemCalloc(gGroovidelics_array_size, sizeof(tGroovidelic_spec), kMem_groove_spec);
if (gGroovidelics_array != NULL) {
memcpy(new_array
, gGroovidelics_array
, (gGroovidelics_array_size
- 16) * sizeof(tGroovidelic_spec
));
ShiftBoundGrooveFunks(
(char*)gGroovidelics_array,
(char*)&gGroovidelics_array[gGroovidelics_array_size - 16],
(char*)new_array - (char*)gGroovidelics_array);
BrMemFree(gGroovidelics_array);
}
gGroovidelics_array = new_array;
for (i = 0; i < 16; i++) {
gGroovidelics_array[i + gGroovidelics_array_size - 16].owner = -999;
}
return &gGroovidelics_array[gGroovidelics_array_size - 16];
}
// IDA: void __usercall AddGroovidelics(FILE *pF@<EAX>, int pOwner@<EDX>, br_actor *pParent_actor@<EBX>, int pRef_offset@<ECX>, int pAllowed_to_be_absent)
void AddGroovidelics(FILE* pF, int pOwner, br_actor* pParent_actor, int pRef_offset, int pAllowed_to_be_absent) {
char s[256];
char* str;
int first_time;
//int i; // Pierre-Marie Baty -- unused variable
//int j; // Pierre-Marie Baty -- unused variable
tGroovidelic_spec* the_groove;
float x_0;
float x_1;
float x_2;
int d_0;
int d_1;
int d_2;
LOG_TRACE("(%p, %d, %p, %d, %d)", pF, pOwner, pParent_actor, pRef_offset, pAllowed_to_be_absent);
first_time = 1;
PossibleService();
GetALineAndDontArgue(pF, s);
if (strcmp(s
, "END OF GROOVE") == 0) {
break;
}
if (!first_time) {
if (strcmp(s
, "NEXT GROOVE") != 0) {
FatalError(kFatalError_GroovidelicFile);
}
GetALineAndDontArgue(pF, s);
}
first_time = 0;
the_groove = AddNewGroovidelic();
the_groove->owner = pOwner;
the_groove->actor = DRActorFindRecurse(pParent_actor, str);
if (the_groove->actor == NULL) {
if (!pAllowed_to_be_absent && !gAusterity_mode) {
FatalError(kFatalError_FindActorUsedByGroovidelicFile_S, str);
}
if (gGroove_by_proxy_actor == NULL) {
gGroove_by_proxy_actor = BrActorAllocate(BR_ACTOR_MODEL, NULL);
gGroove_by_proxy_actor->model = LoadModel("PROXY.DAT");
BrModelAdd(gGroove_by_proxy_actor->model);
BrActorAdd(gDont_render_actor, gGroove_by_proxy_actor);
}
the_groove->actor = gGroove_by_proxy_actor;
}
the_groove->lollipop_mode = GetALineAndInterpretCommand(pF, gLollipop_names, COUNT_OF(gLollipop_names));
the_groove->mode = GetALineAndInterpretCommand(pF, gGroove_nature_names, COUNT_OF(gGroove_nature_names));
the_groove->path_type = GetALineAndInterpretCommand(pF, gGroove_path_names, COUNT_OF(gGroove_path_names));
the_groove->path_interrupt_status = eInterrupt_none;
the_groove->object_interrupt_status = eInterrupt_none;
if (the_groove->path_type != eGroove_path_none) {
the_groove->path_mode = GetALineAndInterpretCommand(pF, gFunk_move_names, COUNT_OF(gFunk_move_names));
}
if (the_groove->path_type == eGroove_path_circular) {
TELL_ME_IF_WE_PASS_THIS_WAY();
// GetThreeFloats(pF, &v90, &v89, &v88);
// the_groove->path_data.circular_info.centre.v[0] = v90;
// the_groove->path_data.circular_info.centre.v[1] = v89;
// the_groove->path_data.circular_info.centre.v[2] = v88;
// if (the_groove->path_data.circular_info.centre.v[2] == 0.0
// && *(float*)&the_groove->path_data.circular_info.axis == 0.0
// && *((float*)&the_groove->path_data.circular_info + 6) == 0.0) {
// v25 = (_DWORD*)the_groove->actor->t.t.translate.t.v;
// v26 = (_DWORD*)&the_groove->path_data.circular_info.centre.v[2];
// *v26 = *v25;
// v26[1] = v25[1];
// v26[2] = v25[2];
// }
// if (the_groove->path_mode != 3 && the_groove->path_mode != 4) {
// v29 = GetAFloat(pF);
// x_0 = v29;
// if (v31) {
// v62 = 0.0;
// } else {
// v62 = 1000.0 / x_0;
// }
// the_groove->path_data.circular_info.period = v62;
// } else {
// v27 = &the_groove->path_data.circular_info.period;
// v28 = GetAnInt(pF);
// AddFunkGrooveBinding(v28 + pRef_offset, v27);
// }
// v32 = GetAFloat(pF);
// the_groove->path_data.circular_info.radius = v32;
// v33 = GetALineAndInterpretCommand(pF, gAxis_names, 3);
// the_groove->path_data.circular_info.axis = v33;
} else if (the_groove->path_type == eGroove_path_straight) {
GetThreeFloats(pF,
&the_groove->path_data.straight_info.centre.v[0],
&the_groove->path_data.straight_info.centre.v[1],
&the_groove->path_data.straight_info.centre.v[2]);
if (Vector3IsZero(&the_groove->path_data.straight_info.centre)) {
BrVector3Copy(&the_groove->path_data.straight_info.centre,
&the_groove->actor->t.t.translate.t);
}
if (the_groove->path_mode == eMove_controlled || the_groove->path_mode == eMove_absolute) {
AddFunkGrooveBinding(pRef_offset + GetAnInt(pF), &the_groove->path_data.straight_info.period);
} else {
x_0 = GetAFloat(pF);
the_groove->path_data.straight_info.period = x_0 == 0.0f ? 0.0f : 1000.0 / x_0;
}
GetThreeFloats(
pF,
&the_groove->path_data.straight_info.x_delta,
&the_groove->path_data.straight_info.y_delta,
&the_groove->path_data.straight_info.z_delta);
}
the_groove->object_type = GetALineAndInterpretCommand(pF, gGroove_object_names, COUNT_OF(gGroove_object_names));
BrVector3Copy(&the_groove->object_position, &the_groove->actor->t.t.translate.t);
if (the_groove->object_type != eGroove_object_none) {
the_groove->object_mode = GetALineAndInterpretCommand(pF, gFunk_move_names, COUNT_OF(gFunk_move_names));
}
switch (the_groove->object_type) {
case eGroove_object_spin:
if (the_groove->object_mode == eMove_controlled || the_groove->object_mode == eMove_absolute) {
AddFunkGrooveBinding(pRef_offset + GetAnInt(pF), &the_groove->object_data.spin_info.period);
} else {
x_0 = GetAFloat(pF);
the_groove->object_data.spin_info.period = (x_0 == 0.0f) ? 0.0f : (1000.0f / x_0);
}
GetThreeFloats(pF,
&the_groove->object_centre.v[0],
&the_groove->object_centre.v[1],
&the_groove->object_centre.v[2]);
the_groove->object_data.spin_info.axis = GetALineAndInterpretCommand(pF, gAxis_names, COUNT_OF(gAxis_names));
break;
case eGroove_object_rock:
if (the_groove->object_mode == eMove_controlled || the_groove->object_mode == eMove_absolute) {
AddFunkGrooveBinding(pRef_offset + GetAnInt(pF), &the_groove->object_data.rock_info.period);
} else {
x_0 = GetAFloat(pF);
the_groove->object_data.rock_info.period = (x_0 == 0.0f) ? 0.0f : (1000.0f / x_0);
}
GetThreeFloats(pF,
&the_groove->object_centre.v[0],
&the_groove->object_centre.v[1],
&the_groove->object_centre.v[2]);
the_groove->object_data.rock_info.axis = GetALineAndInterpretCommand(pF, gAxis_names, COUNT_OF(gAxis_names));
the_groove->object_data.rock_info.max_angle = GetAFloat(pF);
break;
case eGroove_object_throb:
if (the_groove->object_mode == eMove_controlled || the_groove->object_mode == eMove_absolute) {
GetThreeInts(pF, &d_0, &d_1, &d_2);
if (d_0 >= 0) {
AddFunkGrooveBinding(pRef_offset + d_0, &the_groove->object_data.throb_info.x_period);
}
if (d_1 >= 0) {
AddFunkGrooveBinding(pRef_offset + d_1, &the_groove->object_data.throb_info.y_period);
}
if (d_2 >= 0) {
AddFunkGrooveBinding(pRef_offset + d_2, &the_groove->object_data.throb_info.z_period);
}
} else {
GetThreeFloats(pF, &x_0, &x_1, &x_2);
the_groove->object_data.throb_info.x_period = (x_0 == 0.0f) ? 0.0f : (1000.0f / x_0);
the_groove->object_data.throb_info.y_period = (x_1 == 0.0f) ? 0.0f : (1000.0f / x_1);
the_groove->object_data.throb_info.z_period = (x_2 == 0.0f) ? 0.0f : (1000.0f / x_2);
}
GetThreeFloats(pF,
&the_groove->object_centre.v[0],
&the_groove->object_centre.v[1],
&the_groove->object_centre.v[2]);
GetThreeFloatPercents(
pF,
&the_groove->object_data.throb_info.x_magnitude,
&the_groove->object_data.throb_info.y_magnitude,
&the_groove->object_data.throb_info.z_magnitude);
break;
case eGroove_object_shear:
if (the_groove->object_mode == eMove_controlled || the_groove->object_mode == eMove_absolute) {
GetThreeInts(pF, &d_0, &d_1, &d_2);
if (d_0 >= 0) {
AddFunkGrooveBinding(pRef_offset + d_0, &the_groove->object_data.shear_info.x_period);
}
if (d_1 >= 0) {
AddFunkGrooveBinding(pRef_offset + d_1, &the_groove->object_data.shear_info.y_period);
}
if (d_2 >= 0) {
AddFunkGrooveBinding(pRef_offset + d_2, &the_groove->object_data.shear_info.z_period);
}
} else {
GetThreeFloats(pF, &x_0, &x_1, &x_2);
the_groove->object_data.shear_info.x_period = x_0 == 0.0f ? 0.0f : 1000.0 / x_0;
the_groove->object_data.shear_info.y_period = x_1 == 0.0f ? 0.0f : 1000.0 / x_1;
the_groove->object_data.shear_info.z_period = x_2 == 0.0f ? 0.0f : 1000.0 / x_2;
}
GetThreeFloats(pF,
&the_groove->object_centre.v[0],
&the_groove->object_centre.v[1],
&the_groove->object_centre.v[2]);
GetThreeFloatPercents(
pF,
&the_groove->object_data.shear_info.x_magnitude,
&the_groove->object_data.shear_info.y_magnitude,
&the_groove->object_data.shear_info.z_magnitude);
break;
default:
break;
}
}
}
// IDA: void __usercall KillGroovadelic(int pOwner@<EAX>)
void KillGroovadelic(int pOwner) {
int i;
tGroovidelic_spec* the_groove;
LOG_TRACE("(%d)", pOwner);
if (gGroovidelics_array == NULL) {
return;
}
for (i = 0; i < gGroovidelics_array_size; i++) {
the_groove = &gGroovidelics_array[i];
if (the_groove->owner != pOwner) {
continue;
}
if (the_groove->path_mode == eMove_controlled) {
continue;
}
if (the_groove->path_mode == eMove_absolute) {
continue;
}
if (the_groove->object_mode == eMove_controlled) {
continue;
}
if (the_groove->object_mode == eMove_absolute) {
continue;
}
the_groove->owner = -999;
}
}
// IDA: void __usercall KillFunkotronic(int pOwner@<EAX>)
void KillFunkotronic(int pOwner) {
int i;
tFunkotronic_spec* the_funk;
LOG_TRACE("(%d)", pOwner);
if (gFunkotronics_array == NULL) {
return;
}
for (i = 0; i < gFunkotronics_array_size; i++) {
the_funk = &gFunkotronics_array[i];
if (the_funk->owner != pOwner) {
continue;
}
if (the_funk->matrix_mode == eMove_controlled) {
continue;
}
if (the_funk->matrix_mode == eMove_absolute) {
continue;
}
if (the_funk->lighting_animation_type == eMove_controlled) {
continue;
}
if (the_funk->lighting_animation_type == eMove_absolute) {
continue;
}
if (the_funk->texture_animation_data.frames_info.mode == eMove_controlled && the_funk->texture_animation_type == eTexture_animation_frames) {
continue;
}
the_funk->owner = -999;
}
}
// IDA: br_uint_32 __usercall DeleteBastards@<EAX>(br_actor *pActor@<EAX>, br_matrix34 *pMatrix@<EDX>, void *pArg@<EBX>)
br_uint_32 DeleteBastards(br_actor* pActor, br_matrix34* pMatrix, void* pArg) {
int i;
int parent_already_doomed;
LOG_TRACE("(%p, %p, %p)", pActor, pMatrix, pArg);
if ((gAdditional_actors != pActor && (pActor->identifier == NULL || pActor->identifier[0] == '&') && Vector3IsZero((br_vector3*)pMatrix->m[3])) || (pActor->model == NULL && pActor->type == BR_ACTOR_MODEL)) {
parent_already_doomed = 0;
for (i = 0; i < gDelete_count; i++) {
if (gDelete_list[i] == pActor) {
parent_already_doomed = 1;
break;
}
}
if (!parent_already_doomed) {
gDelete_list[gDelete_count] = pActor;
gDelete_count++;
}
}
return 0;
}
// IDA: void __cdecl DeleteAnyZeroBastards()
void DeleteAnyZeroBastards(void) {
int i;
LOG_TRACE("()");
gDelete_count = 0;
DRActorEnumRecurseWithTrans(gAdditional_actors, NULL, DeleteBastards, NULL);
for (i = 0; i < gDelete_count; i++) {
BrActorRemove(gDelete_list[i]);
}
}
// IDA: br_uint_32 __usercall ApplyTransToModels@<EAX>(br_actor *pActor@<EAX>, br_matrix34 *pMatrix@<EDX>, void *pArg@<EBX>)
br_uint_32 ApplyTransToModels(br_actor* pActor, br_matrix34* pMatrix, void* pArg) {
int i;
br_vector3 temp_point;
LOG_TRACE("(%p, %p, %p)", pActor, pMatrix, pArg);
if (pActor->identifier == NULL || pActor->identifier[0] == '&') {
return 0;
}
if (pActor->model != NULL) {
for (i = 0; i < pActor->model->nvertices; i++) {
BrVector3Copy(&temp_point, &pActor->model->vertices[i].p);
BrMatrix34ApplyP(&pActor->model->vertices[i].p, &temp_point, pMatrix);
}
BrModelUpdate(pActor->model, BR_MATU_ALL);
}
BrMatrix34Identity(&pActor->t.t.mat);
pActor->t.type = BR_TRANSFORM_IDENTITY;
return 0;
}
// IDA: int __usercall FindSpecVolIndex@<EAX>(br_actor *pActor@<EAX>)
int FindSpecVolIndex(br_actor* pActor) {
int i;
//tSpecial_volume* v; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p)", pActor);
for (i = 0; i < gProgram_state.special_volume_count; i++) {
if (gSpec_vol_actors[i] == pActor) {
return i;
}
}
return -1;
}
// IDA: void __usercall MungeMaterial(br_matrix34 *pMat@<EAX>, br_material *pMat_1@<EDX>, br_material *pMat_2@<EBX>, int pAxis_0@<ECX>, int pAxis_1)
void MungeMaterial(br_matrix34* pMat, br_material* pMat_1, br_material* pMat_2, int pAxis_0, int pAxis_1) {
LOG_TRACE("(%p, %p, %p, %d, %d)", pMat, pMat_1, pMat_2, pAxis_0, pAxis_1);
pMat_1->map_transform.m[0][0] = 6.f * BrVector3Length((br_vector3*)pMat->m[pAxis_0]);
pMat_1->map_transform.m[1][1] = 6.f * BrVector3Length((br_vector3*)pMat->m[pAxis_1]);
BrMatrix23Copy(&pMat_2->map_transform, &pMat_1->map_transform);
}
// IDA: void __usercall SetSpecVolMatSize(br_actor *pActor@<EAX>)
void SetSpecVolMatSize(br_actor* pActor) {
br_model* model;
LOG_TRACE("(%p)", pActor);
model = pActor->model;
MungeMaterial(&pActor->t.t.mat, model->faces[5].material, model->faces[17].material, 0, 1);
MungeMaterial(&pActor->t.t.mat, model->faces[11].material, model->faces[23].material, 1, 2);
MungeMaterial(&pActor->t.t.mat, model->faces[7].material, model->faces[19].material, 0, 2);
}
// IDA: void __usercall FindInverseAndWorldBox(tSpecial_volume *pSpec@<EAX>)
void FindInverseAndWorldBox(tSpecial_volume* pSpec) {
br_bounds bnds;
LOG_TRACE("(%p)", pSpec);
bnds.min.v[0] = -1.0;
bnds.min.v[1] = -1.0;
bnds.min.v[2] = -1.0;
bnds.max.v[0] = 1.0;
bnds.max.v[1] = 1.0;
bnds.max.v[2] = 1.0;
GetNewBoundingBox(&pSpec->bounds, &bnds, &pSpec->mat);
BrMatrix34Inverse(&pSpec->inv_mat, &pSpec->mat);
}
// IDA: void __cdecl UpdateSpecVol()
void UpdateSpecVol(void) {
int index;
tSpecial_volume* v;
LOG_TRACE("()");
index = FindSpecVolIndex(gLast_actor);
if (index >= 0) {
v = &gProgram_state.special_volumes[index];
BrMatrix34Copy(&v->mat, &gLast_actor->t.t.mat);
FindInverseAndWorldBox(v);
SetSpecVolMatSize(gLast_actor);
}
}
// IDA: void __cdecl SaveSpecialVolumes()
void SaveSpecialVolumes(void) {
tPath_name the_path;
FILE* f;
int i;
tSpecial_volume* v;
LOG_TRACE("()");
PathCat(the_path, gApplication_path, "SPECSAVE.TXT");
f = DRfopen(the_path, "wt");
if (f == NULL) {
return;
}
fprintf(f
, "// SPECIAL EFFECTS VOLUMES\n\n");
fprintf(f
, "%d\t\t\t\t// # special effects volumes\n\n", gProgram_state.
special_volume_count);
for (i = 0; i < gProgram_state.special_volume_count; i++) {
v = &gProgram_state.special_volumes[i];
if (v->no_mat) {
fprintf(f
, "%s\n", "DEFAULT WATER");
} else {
fprintf(f
, "%.3f, %.3f, %.3f\n", v
->mat.
m[0][0], v
->mat.
m[0][1], v
->mat.
m[0][2]);
fprintf(f
, "%.3f, %.3f, %.3f\n", v
->mat.
m[1][0], v
->mat.
m[1][1], v
->mat.
m[1][2]);
fprintf(f
, "%.3f, %.3f, %.3f\n", v
->mat.
m[2][0], v
->mat.
m[2][1], v
->mat.
m[2][2]);
fprintf(f
, "%.3f, %.3f, %.3f\n", v
->mat.
m[3][0], v
->mat.
m[3][1], v
->mat.
m[3][2]);
}
fprintf(f
, "%.0f\t\t\t\t// gravity multiplier\n", v
->gravity_multiplier
);
fprintf(f
, "%.0f\t\t\t\t// viscosity multiplier\n", v
->viscosity_multiplier
);
fprintf(f
, "%.0f\t\t\t\t// Car damage per millisecond\n", v
->car_damage_per_ms
);
fprintf(f
, "%.0f\t\t\t\t// Pedestrian damage per millisecond\n", v
->ped_damage_per_ms
);
fprintf(f
, "%d\t\t\t\t\t// camera effect index\n", v
->camera_special_effect_index
);
fprintf(f
, "%d\t\t\t\t\t// sky colour\n", v
->sky_col
);
fprintf(f
, "%s\t\t\t\t// Windscreen material to use\n", (v
->screen_material
!= NULL
) ? v
->screen_material
->identifier
: "none");
fprintf(f
, "%d\t\t\t\t\t// Sound ID of entry noise\n", v
->entry_noise
);
fprintf(f
, "%d\t\t\t\t\t// Sound ID of exit noise\n", v
->exit_noise
);
fprintf(f
, "%d\t\t\t\t\t// Engine noise index\n", v
->engine_noise_index
);
fprintf(f
, "%d\t\t\t\t\t// material index\n", v
->material_modifier_index
);
}
}
// IDA: void __cdecl SaveAdditionalStuff()
void SaveAdditionalStuff(void) {
LOG_TRACE("()");
if (gSpec_vol_mode) {
UpdateSpecVol();
SaveSpecialVolumes();
} else {
DeleteAnyZeroBastards();
if (gLast_actor != NULL) {
DRActorEnumRecurseWithTrans(gLast_actor, NULL, ApplyTransToModels, NULL);
}
BrActorSave(gAdditional_actor_path, gAdditional_actors);
BrModelSaveMany(gAdditional_model_path, gAdditional_models, gNumber_of_additional_models);
}
}
// IDA: br_uint_32 __cdecl ProcessMaterials(br_actor *pActor, tPMFM2CB pCallback)
br_uint_32 ProcessMaterials(br_actor* pActor, tPMFM2CB pCallback) {
LOG_TRACE("(%p, %d)", pActor, pCallback);
NOT_IMPLEMENTED();
}
// IDA: br_uint_32 __cdecl ProcessFaceMaterials2(br_actor *pActor, tPMFM2CB pCallback)
br_uint_32 ProcessFaceMaterials2(br_actor* pActor, tPMFM2CB pCallback) {
LOG_TRACE("(%p, %d)", pActor, pCallback);
NOT_IMPLEMENTED();
}
// IDA: void __usercall ChangePerspToSubdivCB(br_material *pMaterial@<EAX>)
void ChangePerspToSubdivCB(br_material* pMaterial) {
LOG_TRACE("(%p)", pMaterial);
NOT_IMPLEMENTED();
}
// IDA: void __cdecl ChangePerspToSubdiv()
void ChangePerspToSubdiv(void) {
LOG_TRACE("()");
NOT_IMPLEMENTED();
}
// IDA: void __usercall ChangeSubdivToPerspCB(br_material *pMaterial@<EAX>)
void ChangeSubdivToPerspCB(br_material* pMaterial) {
LOG_TRACE("(%p)", pMaterial);
NOT_IMPLEMENTED();
}
// IDA: void __cdecl ChangeSubdivToPersp()
void ChangeSubdivToPersp(void) {
LOG_TRACE("()");
NOT_IMPLEMENTED();
}
// IDA: br_uint_32 __cdecl ProcessFaceMaterials(br_actor *pActor, tPMFMCB pCallback)
br_uintptr_t ProcessFaceMaterials(br_actor* pActor, tPMFMCB pCallback) {
LOG_TRACE("(%p, %d)", pActor, pCallback);
if (pActor->identifier == NULL || pActor->identifier[0] != '&') {
if (pActor->type == BR_ACTOR_MODEL && pActor->model != NULL) {
ProcessModelFaceMaterials(pActor->model, pCallback);
}
return BrActorEnum(pActor, (br_actor_enum_cbfn*)ProcessFaceMaterials, pCallback);
} else {
return 0;
}
}
// IDA: int __usercall DRPixelmapHasZeros@<EAX>(br_pixelmap *pm@<EAX>)
int DRPixelmapHasZeros(br_pixelmap* pm) {
int x;
int y;
char* row_ptr;
char* pp;
//int i; // Pierre-Marie Baty -- unused variable
if (pm->flags & BR_PMF_NO_ACCESS) {
return 1;
}
row_ptr = (char*)pm->pixels + (pm->row_bytes * pm->base_y) + pm->base_x;
for (y = 0; y < pm->height; y++) {
pp = row_ptr;
for (x = 0; x < pm->width; x++) {
if (!pp)
return 1;
pp++;
}
row_ptr += pm->row_bytes;
}
return 0;
}
// IDA: int __usercall StorageContainsPixelmap@<EAX>(tBrender_storage *pStorage@<EAX>, br_pixelmap *pMap@<EDX>)
int StorageContainsPixelmap(tBrender_storage* pStorage, br_pixelmap* pMap) {
int i;
for (i = 0; i < pStorage->pixelmaps_count; i++) {
if (pMap == pStorage->pixelmaps[i]) {
return 1;
}
}
return 0;
}
// IDA: void __usercall HideStoredOpaqueTextures(tBrender_storage *pStorage@<EAX>)
void HideStoredOpaqueTextures(tBrender_storage* pStorage) {
int i;
for (i = 0; i < pStorage->materials_count; i++) {
if (pStorage->materials[i]->colour_map && StorageContainsPixelmap(pStorage, pStorage->materials[i]->colour_map)) {
if (!DRPixelmapHasZeros(pStorage->materials[i]->colour_map)) {
pStorage->saved_colour_maps[i] = pStorage->materials[i]->colour_map;
pStorage->materials[i]->colour_map = NULL;
pStorage->materials[i]->flags &= 0xFDu;
BrMaterialUpdate(pStorage->materials[i], BR_MATU_ALL);
}
}
}
}
// IDA: void __usercall RevealStoredTransparentTextures(tBrender_storage *pStorage@<EAX>)
void RevealStoredTransparentTextures(tBrender_storage* pStorage) {
int i;
for (i = 0; i < pStorage->materials_count; i++) {
if (pStorage->saved_colour_maps[i]) {
if (DRPixelmapHasZeros(pStorage->saved_colour_maps[i])) {
pStorage->materials[i]->colour_map = pStorage->saved_colour_maps[i];
pStorage->saved_colour_maps[i] = NULL;
pStorage->materials[i]->flags |= 2u;
BrMaterialUpdate(pStorage->materials[i], BR_MATU_ALL);
}
}
}
}
// IDA: void __usercall HideStoredTextures(tBrender_storage *pStorage@<EAX>)
void HideStoredTextures(tBrender_storage* pStorage) {
int i;
for (i = 0; i < pStorage->materials_count; i++) {
if (pStorage->materials[i]->colour_map) {
if (StorageContainsPixelmap(pStorage, pStorage->materials[i]->colour_map)) {
pStorage->saved_colour_maps[i] = pStorage->materials[i]->colour_map;
pStorage->materials[i]->colour_map = NULL;
pStorage->materials[i]->flags &= 0xFDu;
BrMaterialUpdate(pStorage->materials[i], BR_MATU_ALL);
}
}
}
}
// IDA: void __usercall RevealStoredTextures(tBrender_storage *pStorage@<EAX>)
void RevealStoredTextures(tBrender_storage* pStorage) {
int i;
for (i = 0; i < pStorage->materials_count; i++) {
if (pStorage->saved_colour_maps[i]) {
pStorage->materials[i]->colour_map = pStorage->saved_colour_maps[i];
pStorage->saved_colour_maps[i] = NULL;
pStorage->materials[i]->flags |= BR_MATF_PRELIT;
BrMaterialUpdate(pStorage->materials[i], BR_MATU_ALL);
}
}
}
// IDA: void __usercall SetCarStorageTexturingLevel(tBrender_storage *pStorage@<EAX>, tCar_texturing_level pNew@<EDX>, tCar_texturing_level pOld@<EBX>)
void SetCarStorageTexturingLevel(tBrender_storage* pStorage, tCar_texturing_level pNew, tCar_texturing_level pOld) {
LOG_TRACE("(%p, %d, %d)", pStorage, pNew, pOld);
switch (pNew) {
case eCTL_none:
HideStoredTextures(pStorage);
break;
case eCTL_transparent:
switch (pOld) {
case eCTL_none:
RevealStoredTransparentTextures(pStorage);
break;
case eCTL_full:
HideStoredOpaqueTextures(pStorage);
break;
default:
break;
}
break;
case eCTL_full:
RevealStoredTextures(pStorage);
break;
default:
break;
}
}
// IDA: tCar_texturing_level __cdecl GetCarTexturingLevel()
tCar_texturing_level GetCarTexturingLevel(void) {
LOG_TRACE("()");
return gCar_texturing_level;
}
// IDA: void __usercall SetCarTexturingLevel(tCar_texturing_level pLevel@<EAX>)
void SetCarTexturingLevel(tCar_texturing_level pLevel) {
LOG_TRACE("(%d)", pLevel);
if (pLevel != gCar_texturing_level) {
if (gOur_car_storage_space.models_count != 0) {
SetCarStorageTexturingLevel(&gOur_car_storage_space, pLevel, gCar_texturing_level);
}
if (gTheir_cars_storage_space.models_count != 0) {
SetCarStorageTexturingLevel(&gTheir_cars_storage_space, pLevel, gCar_texturing_level);
}
if (gNet_cars_storage_space.models_count != 0) {
SetCarStorageTexturingLevel(&gTheir_cars_storage_space, pLevel, gCar_texturing_level);
}
}
gCar_texturing_level = pLevel;
}
// IDA: int __usercall HasThisSuffix@<EAX>(char *pIdent@<EAX>, char *pSuffix@<EDX>)
int HasThisSuffix(char* pIdent, char* pSuffix) {
size_t len_ident;
size_t len_suffix;
LOG_TRACE("(\"%s\", \"%s\")", pIdent, pSuffix);
if (pIdent == NULL) {
return 0;
}
if (pIdent < pSuffix) {
return 0;
}
return strcmp(pIdent
+ len_ident
- len_suffix
, pSuffix
) == 0;
}
// IDA: br_material* __usercall UnsuffixedMaterial@<EAX>(char *pOld_ident@<EAX>, char *pSuffix@<EDX>)
br_material* UnsuffixedMaterial(char* pOld_ident, char* pSuffix) {
br_material* result;
int unsuffixed_len;
char* new_id;
LOG_TRACE("(\"%s\", \"%s\")", pOld_ident, pSuffix);
new_id = BrMemAllocate(unsuffixed_len + 1, kMem_new_mat_id_2);
sprintf(new_id
, "%.*s", unsuffixed_len
, pOld_ident
);
result = BrMaterialFind(new_id);
BrMemFree(new_id);
return result;
}
// IDA: br_material* __usercall RoadUntexToPersp@<EAX>(br_model *pModel@<EAX>, tU16 pFace@<EDX>)
br_material* RoadUntexToPersp(br_model* pModel, tU16 pFace) {
br_material* old_mat;
br_material* new_mat;
LOG_TRACE("(%p, %d)", pModel, pFace);
old_mat = pModel->faces[pFace].material;
if (HasThisSuffix(old_mat->identifier, ".road")) {
new_mat = UnsuffixedMaterial(old_mat->identifier, ".road");
} else {
new_mat = NULL;
}
return new_mat;
}
// IDA: br_material* __usercall WallLinearToUntex@<EAX>(br_model *pModel@<EAX>, tU16 pFace@<EDX>)
br_material* WallLinearToUntex(br_model* pModel, tU16 pFace) {
br_material* old_mat;
//br_material* new_mat; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, %d)", pModel, pFace);
old_mat = pModel->faces[pFace].material;
if (HasThisSuffix(old_mat->identifier, ".pwall")) {
if (old_mat->colour_map != NULL) {
old_mat->colour_map = NULL;
BrMaterialUpdate(old_mat, BR_MATU_ALL);
}
} else {
if (!FaceIsRoad(pModel, pFace) && old_mat->identifier != NULL && old_mat->colour_map != NULL) {
old_mat = SuffixedMaterial(old_mat, ".lwall");
if (old_mat->colour_map != NULL) {
old_mat->colour_map = NULL;
BrMaterialUpdate(old_mat, BR_MATU_ALL);
}
} else {
old_mat = NULL;
}
}
return old_mat;
}
// IDA: br_material* __usercall WallUntexToLinear@<EAX>(br_model *pModel@<EAX>, tU16 pFace@<EDX>)
br_material* WallUntexToLinear(br_model* pModel, tU16 pFace) {
br_material* old_mat;
br_material* new_mat;
LOG_TRACE("(%p, %d)", pModel, pFace);
old_mat = pModel->faces[pFace].material;
if (HasThisSuffix(old_mat->identifier, ".lwall")) {
new_mat = UnsuffixedMaterial(old_mat->identifier, ".lwall");
} else if (HasThisSuffix(old_mat->identifier, ".pwall")) {
old_mat->colour_map = UnsuffixedMaterial(old_mat->identifier, ".pwall")->colour_map;
old_mat->flags &= ~BR_MATF_PERSPECTIVE;
BrMaterialUpdate(old_mat, BR_MATU_ALL);
new_mat = NULL;
} else {
new_mat = NULL;
}
return new_mat;
}
// IDA: br_material* __usercall WallUntexToPersp@<EAX>(br_model *pModel@<EAX>, tU16 pFace@<EDX>)
br_material* WallUntexToPersp(br_model* pModel, tU16 pFace) {
br_material* old_mat;
br_material* new_mat;
LOG_TRACE("(%p, %d)", pModel, pFace);
old_mat = pModel->faces[pFace].material;
if (HasThisSuffix(old_mat->identifier, ".lwall")) {
new_mat = UnsuffixedMaterial(old_mat->identifier, ".lwall");
} else if (HasThisSuffix(old_mat->identifier, ".pwall")) {
new_mat = UnsuffixedMaterial(old_mat->identifier, ".pwall");
} else {
new_mat = NULL;
}
return new_mat;
}
// IDA: br_material* __usercall WallLinearToPersp@<EAX>(br_model *pModel@<EAX>, tU16 pFace@<EDX>)
br_material* WallLinearToPersp(br_model* pModel, tU16 pFace) {
br_material* old_mat;
br_material* new_mat;
LOG_TRACE("(%p, %d)", pModel, pFace);
old_mat = pModel->faces[pFace].material;
if (HasThisSuffix(old_mat->identifier, ".pwall")) {
new_mat = UnsuffixedMaterial(old_mat->identifier, ".pwall");
} else {
new_mat = NULL;
}
return new_mat;
}
// IDA: tRoad_texturing_level __cdecl GetRoadTexturingLevel()
tRoad_texturing_level GetRoadTexturingLevel(void) {
return gRoad_texturing_level;
}
// IDA: void __usercall SetRoadTexturingLevel(tRoad_texturing_level pLevel@<EAX>)
void SetRoadTexturingLevel(tRoad_texturing_level pLevel) {
LOG_TRACE("(%d)", pLevel);
gRoad_texturing_level = pLevel;
}
// IDA: void __usercall ReallySetRoadTexturingLevel(tRoad_texturing_level pLevel@<EAX>)
void ReallySetRoadTexturingLevel(tRoad_texturing_level pLevel) {
LOG_TRACE("(%d)", pLevel);
if (pLevel != gRoad_texturing_level) {
ProcessFaceMaterials(gProgram_state.track_spec.the_actor, (pLevel == eRTL_none) ? RoadUntexToPersp : RoadPerspToUntex);
}
}
// IDA: tWall_texturing_level __cdecl GetWallTexturingLevel()
tWall_texturing_level GetWallTexturingLevel(void) {
LOG_TRACE("()");
return gWall_texturing_level;
}
// IDA: void __usercall SetWallTexturingLevel(tWall_texturing_level pLevel@<EAX>)
void SetWallTexturingLevel(tWall_texturing_level pLevel) {
LOG_TRACE("(%d)", pLevel);
gWall_texturing_level = pLevel;
}
// IDA: void __usercall ReallySetWallTexturingLevel(tWall_texturing_level pLevel@<EAX>)
void ReallySetWallTexturingLevel(tWall_texturing_level pLevel) {
static tPMFMCB* tweaker[3][3] = {
{
NULL,
WallUntexToLinear,
WallUntexToPersp,
},
{
WallLinearToUntex,
NULL,
WallLinearToPersp,
},
{
WallPerspToUntex,
WallPerspToLinear,
NULL,
},
};
LOG_TRACE("(%d)", pLevel);
if (gWall_texturing_level != pLevel) {
ProcessFaceMaterials(gProgram_state.track_spec.the_actor, tweaker[gWall_texturing_level][pLevel]);
}
}
// IDA: br_material* __usercall DisposeSuffixedMaterials@<EAX>(br_model *pModel@<EAX>, tU16 pFace@<EDX>)
br_material* DisposeSuffixedMaterials(br_model* pModel, tU16 pFace) {
size_t max_suffix_len;
br_material* mat;
br_material* victim;
static char* suffixes[3] = { ".road", ".pwall", ".lwall" };
int s;
char* id;
LOG_TRACE("(%p, %d)", pModel, pFace);
mat = pModel->faces[pFace].material;
if (mat->identifier == NULL) {
return NULL;
}
max_suffix_len = 0;
for (s = 0; s < COUNT_OF(suffixes); s++) {
if (max_suffix_len
< strlen(suffixes
[s
])) {
max_suffix_len
= strlen(suffixes
[s
]);
}
}
id
= BrMemAllocate
(strlen(mat
->identifier
) + max_suffix_len
+ 1, kMem_new_mat_id_3
);
for (s = 0; s < COUNT_OF(suffixes); s++) {
sprintf(id
, "%s%s", mat
->identifier
, suffixes
[s
]);
victim = BrMaterialFind(id);
if (victim != NULL) {
BrMaterialRemove(victim);
BrMaterialFree(victim);
}
}
return NULL;
}
// IDA: void __cdecl DisposeTexturingMaterials()
void DisposeTexturingMaterials(void) {
LOG_TRACE("()");
switch (gWall_texturing_level) {
case eWTL_linear:
ProcessFaceMaterials(gProgram_state.track_spec.the_actor, WallLinearToPersp);
break;
case eWTL_none:
ProcessFaceMaterials(gProgram_state.track_spec.the_actor, WallUntexToPersp);
break;
default:
break;
}
switch (gRoad_texturing_level) {
case eRTL_none:
ProcessFaceMaterials(gProgram_state.track_spec.the_actor, RoadUntexToPersp);
break;
default:
break;
}
if (gWall_texturing_level != eWTL_full || gRoad_texturing_level != eRTL_full) {
ProcessFaceMaterials(gProgram_state.track_spec.the_actor, DisposeSuffixedMaterials);
}
}
// IDA: br_uint_32 __cdecl SetAccessoryRenderingCB(br_actor *pActor, void *pFlag)
br_uintptr_t SetAccessoryRenderingCB(br_actor* pActor, void* pFlag) {
if (pActor->identifier && *pActor->identifier == '&') {
pActor->render_style = *(br_uint_8*)pFlag;
}
return 0;
}
// IDA: void __usercall SetAccessoryRendering(int pOn@<EAX>)
void SetAccessoryRendering(int pOn) {
int style;
LOG_TRACE("(%d)", pOn);
if (gTrack_actor) {
if (pOn) {
style = BR_RSTYLE_FACES;
} else {
style = BR_RSTYLE_NONE;
}
DRActorEnumRecurse(gTrack_actor, (br_actor_enum_cbfn*)SetAccessoryRenderingCB, &style);
}
gRendering_accessories = pOn;
}
// IDA: int __cdecl GetAccessoryRendering()
int GetAccessoryRendering(void) {
LOG_TRACE("()");
return gRendering_accessories;
}
// IDA: void __usercall SetCarSimplificationLevel(int pLevel@<EAX>)
void SetCarSimplificationLevel(int pLevel) {
LOG_TRACE("(%d)", pLevel);
gCar_simplification_level = pLevel;
}
// IDA: int __cdecl GetCarSimplificationLevel()
int GetCarSimplificationLevel(void) {
LOG_TRACE("()");
return gCar_simplification_level;
}
// IDA: void __usercall ParseSpecialVolume(FILE *pF@<EAX>, tSpecial_volume *pSpec@<EDX>, char *pScreen_name_str@<EBX>)
void ParseSpecialVolume(FILE* pF, tSpecial_volume* pSpec, char* pScreen_name_str) {
char s[256];
pSpec->gravity_multiplier = GetAScalar(pF);
pSpec->viscosity_multiplier = GetAScalar(pF);
pSpec->car_damage_per_ms = GetAScalar(pF);
pSpec->ped_damage_per_ms = GetAScalar(pF);
pSpec->camera_special_effect_index = GetAnInt(pF);
pSpec->sky_col = GetAnInt(pF);
GetAString(pF, s);
if (pScreen_name_str) {
} else {
pSpec->screen_material = BrMaterialFind(s);
}
pSpec->entry_noise = GetAnInt(pF);
pSpec->exit_noise = GetAnInt(pF);
pSpec->engine_noise_index = GetAnInt(pF);
pSpec->material_modifier_index = GetAnInt(pF);
}
// IDA: void __usercall AddExceptionToList(tException_list *pDst@<EAX>, tException_list pNew@<EDX>)
void AddExceptionToList(tException_list* pDst, tException_list pNew) {
LOG_TRACE("(%p, %d)", pDst, pNew);
NOT_IMPLEMENTED();
}
// IDA: void __usercall LoadExceptionsFile(char *pName@<EAX>)
void LoadExceptionsFile(char* pName) {
//FILE* f; // Pierre-Marie Baty -- unused variable
//char line[256]; // Pierre-Marie Baty -- unused variable
//char* tok; // Pierre-Marie Baty -- unused variable
//int file_version; // Pierre-Marie Baty -- unused variable
//tException_list e; // Pierre-Marie Baty -- unused variable
//char delimiters[4]; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(\"%s\")", pName);
NOT_IMPLEMENTED();
}
// IDA: void __usercall LoadExceptionsFileForTrack(char *pTrack_file_name@<EAX>)
void LoadExceptionsFileForTrack(char* pTrack_file_name) {
//tPath_name exceptions_file_name; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(\"%s\")", pTrack_file_name);
NOT_IMPLEMENTED();
}
// IDA: void __cdecl FreeExceptions()
void FreeExceptions(void) {
//tException_list list; // Pierre-Marie Baty -- unused variable
//tException_list next; // Pierre-Marie Baty -- unused variable
LOG_TRACE("()");
NOT_IMPLEMENTED();
}
// IDA: void __usercall LoadTrack(char *pFile_name@<EAX>, tTrack_spec *pTrack_spec@<EDX>, tRace_info *pRace_info@<EBX>)
void LoadTrack(char* pFile_name, tTrack_spec* pTrack_spec, tRace_info* pRace_info) {
//char temp_name[14]; // Pierre-Marie Baty -- unused variable
FILE* f;
FILE* non_car_f;
FILE* g;
int i;
int j;
int k;
int group;
//int needs_updating; // Pierre-Marie Baty -- unused variable
int killed_sky;
int line_count;
int ped_subs_index;
int sl;
int num_materials;
int count;
int num_non_cars;
int style;
int cp_rect_w[2];
int cp_rect_h[2];
#if defined(DETHRACE_FIX_BUGS)
int skid_mark_cnt = 0;
#endif
tPath_name the_path;
//tPath_name general_file_path; // Pierre-Marie Baty -- unused variable
char s[256];
char* str;
float temp_float;
//br_actor* new_root; // Pierre-Marie Baty -- unused variable
//br_vector3 a; // Pierre-Marie Baty -- unused variable
//br_vector3 b; // Pierre-Marie Baty -- unused variable
tU16 sky_pixels_high;
tNon_car_spec* non_car;
tSpecial_volume* spec;
br_vector3 p[3];
//br_vector3 v1; // Pierre-Marie Baty -- unused variable
//br_vector3 v2; // Pierre-Marie Baty -- unused variable
//br_vector3 temp_v; // Pierre-Marie Baty -- unused variable
//br_bounds temp_bounds; // Pierre-Marie Baty -- unused variable
tPed_subs* ped_subs;
br_pixelmap* sky;
br_material* material;
LOG_TRACE("(\"%s\", %p, %p)", pFile_name, pTrack_spec, pRace_info);
killed_sky = 0;
PathCat(the_path, gApplication_path, "RACES");
PathCat(the_path, the_path, pFile_name);
f = DRfopen(the_path, "rt");
if (f == NULL) {
FatalError(kFatalError_OpenRacesFile);
}
GetALineAndDontArgue(f, s);
if (strcmp(str
, "VERSION") == 0) {
sscanf(str
, "%d", &gRace_file_version
);
GetALineAndDontArgue(f, s);
} else {
gRace_file_version = 0;
}
sscanf(str
, "%f", &temp_float
);
pRace_info->initial_position.v[0] = temp_float;
sscanf(str
, "%f", &temp_float
);
pRace_info->initial_position.v[1] = temp_float;
sscanf(str
, "%f", &temp_float
);
pRace_info->initial_position.v[2] = temp_float;
PossibleService();
GetALineAndDontArgue(f, s);
sscanf(str
, "%f", &temp_float
);
pRace_info->initial_yaw = temp_float;
GetThreeInts(f, pRace_info->initial_timer, &pRace_info->initial_timer[1], &pRace_info->initial_timer[2]);
pRace_info->total_laps = GetAnInt(f);
GetThreeInts(f, &pRace_info->bonus_score[0][0], &pRace_info->bonus_score[0][1], &pRace_info->bonus_score[0][2]);
GetThreeInts(f, &pRace_info->bonus_score[1][0], &pRace_info->bonus_score[1][1], &pRace_info->bonus_score[1][2]);
GetThreeInts(f, &pRace_info->bonus_score[2][0], &pRace_info->bonus_score[2][1], &pRace_info->bonus_score[2][2]);
if (gRace_file_version > 1) {
GetPairOfInts(f, &cp_rect_w[0], &cp_rect_w[1]);
GetPairOfInts(f, &cp_rect_h[0], &cp_rect_h[1]);
}
PossibleService();
pRace_info->check_point_count = GetAnInt(f);
for (i = 0; i < pRace_info->check_point_count; i++) {
PossibleService();
if (gProgram_state.sausage_eater_mode) {
GetALineAndDontArgue(f, s);
GetThreeInts(
f,
&pRace_info->checkpoints[i].time_value[0],
&pRace_info->checkpoints[i].time_value[1],
&pRace_info->checkpoints[i].time_value[2]);
} else {
GetThreeInts(
f,
&pRace_info->checkpoints[i].time_value[0],
&pRace_info->checkpoints[i].time_value[1],
&pRace_info->checkpoints[i].time_value[2]);
GetALineAndDontArgue(f, s);
}
pRace_info->checkpoints[i].quad_count = GetAnInt(f);
for (j = 0; j < pRace_info->checkpoints[i].quad_count; j++) {
for (k = 0; k < 4; ++k) {
GetThreeScalars(
f,
&pRace_info->checkpoints[i].vertices[j][k].v[0],
&pRace_info->checkpoints[i].vertices[j][k].v[1],
&pRace_info->checkpoints[i].vertices[j][k].v[2]);
}
p[0] = pRace_info->checkpoints[i].vertices[j][0];
p[1] = pRace_info->checkpoints[i].vertices[j][1];
p[2] = pRace_info->checkpoints[i].vertices[j][2];
pRace_info->checkpoints[i].normal[j].v[0] = (p[2].v[2] - p[0].v[2]) * (p[1].v[1] - p[0].v[1]) - (p[1].v[2] - p[0].v[2]) * (p[2].v[1] - p[0].v[1]);
pRace_info->checkpoints[i].normal[j].v[1] = (p[1].v[2] - p[0].v[2]) * (p[2].v[0] - p[0].v[0]) - (p[2].v[2] - p[0].v[2]) * (p[1].v[0] - p[0].v[0]);
pRace_info->checkpoints[i].normal[j].v[2] = (p[2].v[1] - p[0].v[1]) * (p[1].v[0] - p[0].v[0]) - (p[1].v[1] - p[0].v[1]) * (p[2].v[0] - p[0].v[0]);
}
if (gRace_file_version > 0) {
if (gRace_file_version == 1) {
GetPairOfInts(f, pRace_info->checkpoints[i].map_left, &pRace_info->checkpoints[i].map_left[1]);
GetPairOfInts(f, pRace_info->checkpoints[i].map_top, &pRace_info->checkpoints[i].map_top[1]);
GetPairOfInts(f, pRace_info->checkpoints[i].map_right, &pRace_info->checkpoints[i].map_right[1]);
GetPairOfInts(f, pRace_info->checkpoints[i].map_bottom, &pRace_info->checkpoints[i].map_bottom[1]);
} else {
GetPairOfInts(f, pRace_info->checkpoints[i].map_left, &pRace_info->checkpoints[i].map_left[1]);
GetPairOfInts(f, pRace_info->checkpoints[i].map_top, &pRace_info->checkpoints[i].map_top[1]);
pRace_info->checkpoints[i].map_right[0] = cp_rect_w[0] + pRace_info->checkpoints[i].map_left[0];
pRace_info->checkpoints[i].map_right[1] = cp_rect_w[0] + pRace_info->checkpoints[i].map_left[1];
pRace_info->checkpoints[i].map_bottom[0] = cp_rect_h[0] + pRace_info->checkpoints[i].map_top[0];
pRace_info->checkpoints[i].map_bottom[1] = cp_rect_h[0] + pRace_info->checkpoints[i].map_top[1];
}
}
}
if (gRace_file_version <= 3) {
LoadSomePixelmaps(&gTrack_storage_space, f);
} else if (gAusterity_mode) {
SkipNLines(f);
LoadSomePixelmaps(&gTrack_storage_space, f);
} else {
LoadSomePixelmaps(&gTrack_storage_space, f);
SkipNLines(f);
}
LoadSomeShadeTables(&gTrack_storage_space, f);
if (gRace_file_version <= 3) {
LoadSomeMaterials(&gTrack_storage_space, f);
} else if (gAusterity_mode) {
SkipNLines(f);
LoadSomeMaterials(&gTrack_storage_space, f);
} else {
LoadSomeMaterials(&gTrack_storage_space, f);
SkipNLines(f);
}
for (i = 0; gTrack_storage_space.materials_count > i; ++i) {
PossibleService();
if (gTrack_storage_space.materials[i]->flags & (BR_MATF_LIGHT | BR_MATF_PRELIT | BR_MATF_SMOOTH)) {
gTrack_storage_space.materials[i]->flags &= ~(BR_MATF_LIGHT | BR_MATF_PRELIT | BR_MATF_SMOOTH);
if (gTrack_storage_space.materials[i]->flags & BR_MATF_TWO_SIDED) {
gTrack_storage_space.materials[i]->user = DOUBLESIDED_USER_FLAG;
gTrack_storage_space.materials[i]->flags &= ~BR_MATF_TWO_SIDED;
}
BrMaterialUpdate(gTrack_storage_space.materials[i], BR_MATU_RENDERING);
}
}
if (gRace_file_version <= 5) {
LoadSomeTrackModels(&gTrack_storage_space, f);
} else if (gAusterity_mode) {
SkipNLines(f);
LoadSomeTrackModels(&gTrack_storage_space, f);
} else {
LoadSomeTrackModels(&gTrack_storage_space, f);
SkipNLines(f);
}
PrintMemoryDump(0, "JUST LOADED IN TEXTURES/MATS/MODELS FOR TRACK");
if (gRace_file_version <= 5) {
GetALineAndDontArgue(f, s);
PathCat(the_path, gApplication_path, "ACTORS");
PathCat(the_path, the_path, str);
} else if (gAusterity_mode) {
GetALineAndDontArgue(f, s);
GetALineAndDontArgue(f, s);
PathCat(the_path, gApplication_path, "ACTORS");
PathCat(the_path, the_path, str);
} else {
GetALineAndDontArgue(f, s);
PathCat(the_path, gApplication_path, "ACTORS");
PathCat(the_path, the_path, str);
GetALineAndDontArgue(f, s);
}
pTrack_spec->the_actor = BrActorLoad(the_path);
if (gRace_file_version <= 6) {
gDefault_blend_pc = 75;
} else {
GetAString(f, s);
if (!sscanf(s
, "%d", &gDefault_blend_pc
) || gDefault_blend_pc
< 0 || gDefault_blend_pc
> 100) {
PDFatalError("Wanted a default blend percentage. Didn't get one. Old data file?");
}
}
PossibleService();
ExtractColumns(pTrack_spec);
for (i = 0; gTrack_storage_space.models_count > i; ++i) {
PossibleService();
if (gTrack_storage_space.models[i] && gTrack_storage_space.models[i]->flags & 0x82) {
gTrack_storage_space.models[i]->flags &= 0xFF7Du;
for (group = 0; group < V11MODEL(gTrack_storage_space.models[i])->ngroups; group++) {
int f = V11MODEL(gTrack_storage_space.models[i])->groups[group].face_user[0];
material = gTrack_storage_space.models[i]->faces[f].material;
V11MODEL(gTrack_storage_space.models[i])->groups[group].user = material;
if (material && !material->index_shade) {
material->index_shade = BrTableFind("DRRENDER.TAB");
BrMaterialUpdate(material, 0x7FFFu);
}
}
DodgyModelUpdate(gTrack_storage_space.models[i]);
}
}
PrintMemoryDump(0, "JUST LOADED IN TRACK ACTOR AND PROCESSED COLUMNS");
gTrack_actor = pTrack_spec->the_actor;
if (!gRendering_accessories && !gNet_mode) {
PossibleService();
style = BR_RSTYLE_NONE;
DRActorEnumRecurse(gTrack_actor, (br_actor_enum_cbfn*)SetAccessoryRenderingCB, &style);
}
BrActorAdd(gUniverse_actor, pTrack_spec->the_actor);
GetALineAndDontArgue(f, s);
PathCat(gAdditional_model_path, gApplication_path, "MODELS");
PathCat(gAdditional_model_path, gAdditional_model_path, str);
gNumber_of_additional_models = 0;
PossibleService();
PathCat(gAdditional_actor_path, gApplication_path, "ACTORS");
PathCat(gAdditional_actor_path, gAdditional_actor_path, str);
gAdditional_actors = BrActorAllocate(BR_ACTOR_NONE, NULL);
BrActorAdd(gUniverse_actor, gAdditional_actors);
gLast_actor = NULL;
SaveAdditionalStuff();
GetAString(f, s);
sky = BrMapFind(s);
if (gAusterity_mode && sky) {
for (i = 0; gTrack_storage_space.pixelmaps_count > i; ++i) {
if (gTrack_storage_space.pixelmaps[i] == sky) {
BrMapRemove(gTrack_storage_space.pixelmaps[i]);
BrPixelmapFree(gTrack_storage_space.pixelmaps[i]);
gTrack_storage_space.pixelmaps[i] = gTrack_storage_space.pixelmaps[gTrack_storage_space.pixelmaps_count - 1];
gTrack_storage_space.pixelmaps_count--;
break;
}
}
sky = 0;
killed_sky = 1;
}
gProgram_state.default_depth_effect.sky_texture = sky;
if (sky) {
sky_pixels_high = gProgram_state.default_depth_effect.sky_texture->height;
} else {
sky_pixels_high = 100;
}
PossibleService();
gSky_image_width = BrDegreeToAngle(360.0 / GetAnInt(f));
gSky_image_height = BrDegreeToAngle(GetAScalar(f));
gSky_image_underground = gSky_image_height * (sky_pixels_high - GetAnInt(f)) / sky_pixels_high;
MungeRearviewSky();
PossibleService();
gProgram_state.default_depth_effect.type = GetALineAndInterpretCommand(f, gDepth_effect_names, 2);
GetPairOfInts(f, &gProgram_state.default_depth_effect.start, &gProgram_state.default_depth_effect.end);
if (killed_sky && gProgram_state.default_depth_effect.type != eDepth_effect_fog) {
gProgram_state.default_depth_effect.type = eDepth_effect_fog;
gProgram_state.default_depth_effect.start = 7;
gProgram_state.default_depth_effect.end = 0;
}
InstantDepthChange(
gProgram_state.default_depth_effect.type,
gProgram_state.default_depth_effect.sky_texture,
gProgram_state.default_depth_effect.start,
gProgram_state.default_depth_effect.end);
gSwap_sky_texture = 0;
if (!GetSkyTextureOn()) {
ToggleSkyQuietly();
}
gSwap_depth_effect_type = -1;
if (!GetDepthCueingOn()) {
ToggleDepthCueingQuietly();
}
PossibleService();
gDefault_engine_noise_index = GetAnInt(f);
gDefault_water_spec_vol = &gDefault_default_water_spec_vol;
gProgram_state.special_volume_count = GetAnInt(f);
if (gProgram_state.special_volume_count) {
gProgram_state.special_volumes = BrMemAllocate(sizeof(tSpecial_volume) * gProgram_state.special_volume_count, kMem_special_volume);
i = 0;
spec = gProgram_state.special_volumes;
for (i = 0; i < gProgram_state.special_volume_count; i++) {
PossibleService();
spec->no_mat = 0;
GetALineAndDontArgue(f, s);
if (strcmp(s
, "NEW IMPROVED!") == 0) {
GetThreeScalars(f, &spec->mat.m[0][0], &spec->mat.m[0][1], &spec->mat.m[0][2]);
GetThreeScalars(f, &spec->mat.m[1][0], &spec->mat.m[1][1], &spec->mat.m[1][2]);
GetThreeScalars(f, &spec->mat.m[2][0], &spec->mat.m[2][1], &spec->mat.m[2][2]);
GetThreeScalars(f, &spec->mat.m[3][0], &spec->mat.m[3][1], &spec->mat.m[3][2]);
FindInverseAndWorldBox(spec);
ParseSpecialVolume(f, spec, NULL);
} else if (strcmp(s
, "DEFAULT WATER") == 0) {
spec->bounds.min.v[0] = 0.0;
spec->bounds.min.v[1] = 0.0;
spec->bounds.min.v[2] = 0.0;
spec->bounds.max.v[0] = 0.0;
spec->bounds.max.v[1] = 0.0;
spec->bounds.max.v[2] = 0.0;
ParseSpecialVolume(f, spec, NULL);
gDefault_water_spec_vol = spec;
spec->no_mat = 1;
} else {
TELL_ME_IF_WE_PASS_THIS_WAY();
spec->no_mat = 0;
sscanf(str
, "%f", &spec
->bounds.
min.
v[0]);
sscanf(str
, "%f", &spec
->bounds.
min.
v[1]);
sscanf(str
, "%f", &spec
->bounds.
min.
v[2]);
GetThreeScalars(f, &spec->bounds.max.v[0], &spec->bounds.max.v[1], &spec->bounds.max.v[2]);
BrMatrix34Identity(&spec->mat);
for (k = 0; k < 3; ++k) {
// FIXME: not 100% sure this is correct
spec->mat.m[3][k] = (spec->bounds.max.v[k] + spec->bounds.min.v[k]) / 2.f;
spec->mat.m[k][k] = spec->bounds.max.v[k] - spec->bounds.min.v[k];
}
ParseSpecialVolume(f, spec, NULL);
}
spec++;
}
}
GetAString(f, s);
gProgram_state.standard_screen = BrMaterialFind(s);
GetAString(f, s);
gProgram_state.standard_screen_dark = BrMaterialFind(s);
GetAString(f, s);
gProgram_state.standard_screen_fog = BrMaterialFind(s);
gProgram_state.special_screens_count = GetAnInt(f);
if (gProgram_state.special_screens_count) {
gProgram_state.special_screens = BrMemAllocate(sizeof(tSpecial_screen) * gProgram_state.special_screens_count, kMem_special_screen);
for (i = 0; i < gProgram_state.special_screens_count; i++) {
GetFourScalars(
f,
&gProgram_state.special_screens[i].min_x,
&gProgram_state.special_screens[i].min_z,
&gProgram_state.special_screens[i].max_x,
&gProgram_state.special_screens[i].max_z);
GetAString(f, s);
material = BrMaterialFind(s);
gProgram_state.special_screens[i].material = material;
}
}
PossibleService();
GetAString(f, s);
pRace_info->map_image = LoadPixelmap(s);
PrintMemoryDump(0, "JUST LOADING SKY/SPEC VOLS/SCREENS/MAP");
for (i = 0; i < 4; ++i) {
GetThreeScalars(
f,
&pRace_info->map_transformation.m[i][0],
&pRace_info->map_transformation.m[i][1],
&pRace_info->map_transformation.m[i][2]);
}
GetALineAndDontArgue(f, s);
AddFunkotronics(f, -2, 720);
GetALineAndDontArgue(f, s);
AddGroovidelics(f, -2, gUniverse_actor, 720, 0);
PossibleService();
PrintMemoryDump(0, "JUST LOADING IN FUNKS AND GROOVES");
ped_subs = NULL;
count = 0;
if (gRace_file_version > 3) {
line_count = GetAnInt(f);
if (gAusterity_mode) {
if (line_count >= 0) {
PathCat(the_path, gApplication_path, "PEDSUBS.TXT");
g = DRfopen(the_path, "rt");
if (g == NULL) {
FatalError(kFatalError_OpenRacesFile);
}
for (i = 0; i < line_count; ++i) {
SkipNLines(g);
}
count = GetAnInt(g);
ped_subs = BrMemAllocate(sizeof(tPed_subs) * count, kMem_misc);
for (ped_subs_index = 0; ped_subs_index < count; ped_subs_index++) {
GetPairOfInts(g, &ped_subs[i].orig, &ped_subs[i].subs);
}
}
}
}
PossibleService();
LoadInPedestrians(f, count, ped_subs);
if (ped_subs != NULL) {
BrMemFree(ped_subs);
}
PrintMemoryDump(0, "JUST LOADED IN PEDS");
LoadInOppoPaths(f);
PrintMemoryDump(0, "JUST LOADED IN OPPO PATHS");
num_materials = GetAnInt(f);
for (i = 0; i < num_materials; i++) {
PossibleService();
pRace_info->material_modifiers[i].car_wall_friction = GetAScalar(f);
pRace_info->material_modifiers[i].tyre_road_friction = GetAScalar(f);
pRace_info->material_modifiers[i].down_force = GetAScalar(f);
pRace_info->material_modifiers[i].bumpiness = GetAScalar(f);
pRace_info->material_modifiers[i].tyre_noise_index = GetAnInt(f);
pRace_info->material_modifiers[i].crash_noise_index = GetAnInt(f);
pRace_info->material_modifiers[i].scrape_noise_index = GetAnInt(f);
pRace_info->material_modifiers[i].sparkiness = GetAScalar(f);
pRace_info->material_modifiers[i].smoke_type = GetAnInt(f);
GetAString(f, s);
pRace_info->material_modifiers[i].skid_mark_material = NULL;
} else {
LoadSinglePixelmap(&gTrack_storage_space, str);
str[sl] = 0;
material = LoadSingleMaterial(&gTrack_storage_space, str);
pRace_info->material_modifiers[i].skid_mark_material = material;
#if defined(DETHRACE_FIX_BUGS)
skid_mark_cnt++;
#endif
}
}
#if defined(DETHRACE_FIX_BUGS)
/* Display skidmarks even if the race has no specified skidmark material. */
if (!skid_mark_cnt && num_materials) {
LOG_WARN("Track %s has no valid skid mark material, setting the default one",
pRace_info->track_file_name);
LoadSinglePixelmap(&gTrack_storage_space, "SKIDMARK.PIX");
material = LoadSingleMaterial(&gTrack_storage_space, "SKIDMARK.MAT");
pRace_info->material_modifiers[0].skid_mark_material = material;
}
#endif
for (i = num_materials; i < 10; ++i) {
pRace_info->material_modifiers[i].car_wall_friction = 1.0;
pRace_info->material_modifiers[i].tyre_road_friction = 1.0;
pRace_info->material_modifiers[i].down_force = 1.0;
pRace_info->material_modifiers[i].bumpiness = 0.0;
pRace_info->material_modifiers[i].tyre_noise_index = 0;
pRace_info->material_modifiers[i].crash_noise_index = 0;
pRace_info->material_modifiers[i].scrape_noise_index = 0;
pRace_info->material_modifiers[i].sparkiness = 1.0;
pRace_info->material_modifiers[i].smoke_type = 1;
pRace_info->material_modifiers[i].skid_mark_material = 0;
}
i = 10;
pRace_info->material_modifiers[i].car_wall_friction = 1.0;
pRace_info->material_modifiers[i].tyre_road_friction = 1.0;
pRace_info->material_modifiers[i].down_force = 0.0;
pRace_info->material_modifiers[i].bumpiness = 0.0;
pRace_info->material_modifiers[i].tyre_noise_index = -1;
pRace_info->material_modifiers[i].crash_noise_index = 0;
pRace_info->material_modifiers[i].scrape_noise_index = 0;
pRace_info->material_modifiers[i].sparkiness = 0.0;
pRace_info->material_modifiers[i].smoke_type = 1;
pRace_info->material_modifiers[i].skid_mark_material = 0;
gDefault_water_spec_vol->material_modifier_index = i;
num_non_cars = GetAnInt(f);
non_car = BrMemCalloc(num_non_cars + 5, sizeof(tNon_car_spec), kMem_non_car_spec);
if (!non_car && num_non_cars) {
FatalError(kFatalError_OpenRacesFile);
}
gProgram_state.non_cars = non_car;
gProgram_state.num_non_car_spaces = num_non_cars + NONCAR_UNUSED_SLOTS;
for (i = 0; i < COUNT_OF(gNon_car_spec_list); ++i) {
gNon_car_spec_list[i] = '\0';
}
for (i = 0; i < NONCAR_UNUSED_SLOTS; i++) {
non_car->collision_info.driver = eDriver_non_car_unused_slot;
non_car++;
}
i = 0;
for (i = 0; i < num_non_cars; i++) {
PossibleService();
GetAString(f, s);
PathCat(the_path, gApplication_path, "NONCARS");
PathCat(the_path, the_path, s);
non_car_f = DRfopen(the_path, "rt");
if (non_car_f == NULL) {
FatalError(kFatalError_Open_S, the_path);
}
ReadNonCarMechanicsData(non_car_f, non_car);
PossibleService();
ReadShrapnelMaterials(non_car_f, &non_car->collision_info);
gNon_car_spec_list[non_car->collision_info.index] = i + 1;
non_car++;
}
GetSmokeShadeTables(f);
pRace_info->number_of_net_start_points = GetAnInt(f);
for (i = 0; i < pRace_info->number_of_net_start_points; i++) {
GetThreeScalars(
f,
&pRace_info->net_starts[i].pos.v[0],
&pRace_info->net_starts[i].pos.v[1],
&pRace_info->net_starts[i].pos.v[2]);
pRace_info->net_starts[i].yaw = GetAScalar(f);
}
if (gRace_file_version <= 2) {
LoadInKevStuff(NULL);
} else {
LoadInKevStuff(f);
}
if (gRace_file_version < 5) {
gYon_multiplier = viewdistance_surmultiplier * 1.0; // Pierre-Marie Baty -- view distance surmultiplier addition
} else {
gYon_multiplier = viewdistance_surmultiplier * GetAScalar(f); // Pierre-Marie Baty -- view distance surmultiplier addition
}
GetAString(f, s);
if (strcmp(s
, pFile_name
) != 0) {
FatalError(kFatalError_FileCorrupt_S, pFile_name);
}
}
// IDA: br_uint_32 __cdecl RemoveBounds(br_actor *pActor, void *pArg)
br_uintptr_t RemoveBounds(br_actor* pActor, void* pArg) {
LOG_TRACE("(%p, %p)", pActor, pArg);
if (pActor->type == BR_ACTOR_BOUNDS || pActor->type == BR_ACTOR_BOUNDS_CORRECT) {
BrResFree(pActor->type_data);
pActor->type_data = NULL;
}
return 0;
}
// IDA: void __usercall RemoveBoundsStructures(br_actor *pActor@<EAX>)
void RemoveBoundsStructures(br_actor* pActor) {
LOG_TRACE("(%p)", pActor);
DRActorEnumRecurse(pActor, RemoveBounds, NULL);
}
// IDA: void __usercall FreeTrack(tTrack_spec *pTrack_spec@<EAX>)
void FreeTrack(tTrack_spec* pTrack_spec) {
int i;
tNon_car_spec* non_car;
LOG_TRACE("(%p)", pTrack_spec);
if (gAdditional_actors != NULL) {
BrActorRemove(gAdditional_actors);
BrActorFree(gAdditional_actors);
}
PossibleService();
DisposeTexturingMaterials();
PossibleService();
DisposeColumns(pTrack_spec);
PossibleService();
RemoveBoundsStructures(pTrack_spec->the_actor);
BrActorRemove(pTrack_spec->the_actor);
BrActorFree(pTrack_spec->the_actor);
pTrack_spec->the_actor = NULL;
gTrack_actor = NULL;
PossibleService();
DisposeFunkotronics(-2);
PossibleService();
ClearOutStorageSpace(&gTrack_storage_space);
PossibleService();
DisposeGroovidelics(-2);
PossibleService();
DisposeOpponentPaths();
PossibleService();
DisposeKevStuff();
PossibleService();
if (gCurrent_race.map_image != NULL) {
BrPixelmapFree(gCurrent_race.map_image);
}
if (gProgram_state.special_screens_count != 0) {
BrMemFree(gProgram_state.special_screens);
}
PossibleService();
for (i = 0, non_car = gProgram_state.non_cars; i < gProgram_state.num_non_car_spaces; i++, non_car++) {
PossibleService();
if (non_car->collision_info.driver == eDriver_non_car && non_car->collision_info.car_master_actor != NULL) {
BrActorRemove(non_car->collision_info.car_master_actor);
BrActorFree(non_car->collision_info.car_master_actor);
}
}
if (gProgram_state.non_cars != NULL) {
BrMemFree(gProgram_state.non_cars);
}
FreeSmokeShadeTables();
}
// IDA: void __usercall ProcessTrack(br_actor *pWorld@<EAX>, tTrack_spec *pTrack_spec@<EDX>, br_actor *pCamera@<EBX>, br_matrix34 *pCamera_to_world_transform@<ECX>, int pRender_blends)
void ProcessTrack(br_actor* pWorld, tTrack_spec* pTrack_spec, br_actor* pCamera, br_matrix34* pCamera_to_world_transform, int pRender_blends) {
LOG_TRACE("(%p, %p, %p, %p, %d)", pWorld, pTrack_spec, pCamera, pCamera_to_world_transform, pRender_blends);
RenderTrack(pWorld, pTrack_spec, pCamera, pCamera_to_world_transform, pRender_blends);
if (gAdditional_actors) {
if (!pRender_blends) {
BrZbSceneRenderAdd(gAdditional_actors);
}
}
}
// IDA: br_scalar __cdecl NormaliseDegreeAngle(br_scalar pAngle)
br_scalar NormaliseDegreeAngle(br_scalar pAngle) {
LOG_TRACE("(%f)", pAngle);
while (pAngle < .0f) {
pAngle += 360.f;
}
return pAngle;
}
#define SAW(T, PERIOD) (fmodf((T), (PERIOD)) / (PERIOD))
#define MOVE_FUNK_PARAMETER(DEST, MODE, PERIOD, AMPLITUDE, FLASH_VALUE) \
do { \
switch (MODE) { \
case eMove_continuous: \
if ((PERIOD) == 0.f) { \
DEST = 0.f; \
} else { \
DEST = (AMPLITUDE)*SAW(f_the_time, (PERIOD)); \
} \
break; \
case eMove_controlled: \
DEST = (PERIOD) * (AMPLITUDE); \
break; \
case eMove_absolute: \
DEST = (PERIOD); \
break; \
case eMove_linear: \
if ((PERIOD) == 0.f) { \
DEST = 0.f; \
} else { \
DEST = (AMPLITUDE)*MapSawToTriangle(SAW(f_the_time, (PERIOD))); \
} \
break; \
case eMove_flash: \
if (2 * fmodf(f_the_time, (PERIOD)) > (PERIOD)) { \
DEST = (FLASH_VALUE); \
} else { \
DEST = -(FLASH_VALUE); \
} \
break; \
case eMove_harmonic: \
if ((PERIOD) == 0.f) { \
DEST = 0.f; \
} else { \
DEST = (AMPLITUDE)*BR_SIN(BR_ANGLE_DEG(SAW(f_the_time, (PERIOD)) * 360)); \
} \
break; \
default: \
TELL_ME_IF_WE_PASS_THIS_WAY(); \
} \
} while (0)
// IDA: void __cdecl FunkThoseTronics()
void FunkThoseTronics(void) {
int i;
int j;
int iteration_count;
int finished;
tFunkotronic_spec* the_funk;
//br_vector3* the_proximity; // Pierre-Marie Baty -- unused variable
tS32 the_time;
//tS32 old_last_time; // Pierre-Marie Baty -- unused variable
//tS32 time_diff; // Pierre-Marie Baty -- unused variable
br_matrix23* mat_matrix;
br_material* the_material;
float f_the_time;
float rot_amount;
float f_time_diff;
br_vector2 tmp_v2;
br_pixelmap* old_colour_map;
LOG_TRACE("()");
if (gFunkotronics_array == NULL) {
return;
}
DontLetFlicFuckWithPalettes();
the_time = GetTotalTime();
f_the_time = (float)the_time;
for (i = 0; i < gFunkotronics_array_size; i++) {
the_funk = &gFunkotronics_array[i];
if (the_funk->owner == -999) {
continue;
}
j = 0;
if (the_funk->mode == eFunk_mode_distance && the_funk->proximity_array != NULL) {
j = -2;
for (j = 0; j < the_funk->proximity_count; j++) {
if (Vector3DistanceSquared(&the_funk->proximity_array[j], gOur_pos) <= gSight_distance_squared) {
j = -1;
break;
}
}
}
if (j == -1 || (j != -2 && (the_funk->mode != eFunk_mode_last_lap_only || gLap >= gTotal_laps) && (the_funk->mode != eFunk_mode_all_laps_but_last || gLap < gTotal_laps))) {
the_material = the_funk->material;
mat_matrix = &the_material->map_transform;
if (!gAction_replay_mode || !ReplayIsPaused() || the_funk->matrix_mode == eMove_controlled || the_funk->matrix_mode == eMove_absolute) {
switch (the_funk->matrix_mod_type) {
case eMatrix_mod_spin:
BrMatrix23Identity(mat_matrix);
the_material->map_transform.m[2][0] -= .5f;
the_material->map_transform.m[2][1] -= .5f;
if (the_funk->matrix_mod_data.spin_info.period > 0.f) {
f_time_diff = 1.f - fmodf(the_funk->matrix_mod_data.spin_info.period, 1.f);
} else {
f_time_diff = fmodf(-the_funk->matrix_mod_data.spin_info.period, 1.f);
}
MOVE_FUNK_PARAMETER(rot_amount, the_funk->matrix_mode, f_time_diff, 65536.f, 7.5f);
DRMatrix23PostRotate(mat_matrix, (br_angle)rot_amount);
the_material->map_transform.m[2][0] += .5f;
the_material->map_transform.m[2][1] += .5f;
break;
case eMatrix_mod_rock:
BrMatrix23Identity(mat_matrix);
the_material->map_transform.m[2][0] -= the_funk->matrix_mod_data.rock_info.x_centre;
the_material->map_transform.m[2][1] -= the_funk->matrix_mod_data.rock_info.y_centre;
MOVE_FUNK_PARAMETER(rot_amount, the_funk->matrix_mode, the_funk->matrix_mod_data.rock_info.period, the_funk->matrix_mod_data.rock_info.rock_angle, the_funk->matrix_mod_data.rock_info.rock_angle);
DRMatrix23PostRotate(mat_matrix, BrDegreeToAngle(NormaliseDegreeAngle(rot_amount)));
the_material->map_transform.m[2][0] += the_funk->matrix_mod_data.rock_info.x_centre;
the_material->map_transform.m[2][1] += the_funk->matrix_mod_data.rock_info.y_centre;
break;
case eMatrix_mod_throb:
BrMatrix23Identity(mat_matrix);
the_material->map_transform.m[2][0] -= the_funk->matrix_mod_data.throb_info.x_centre;
the_material->map_transform.m[2][1] -= the_funk->matrix_mod_data.throb_info.y_centre;
MOVE_FUNK_PARAMETER(tmp_v2.v[1], the_funk->matrix_mode, the_funk->matrix_mod_data.throb_info.y_period, the_funk->matrix_mod_data.throb_info.y_magnitude, the_funk->matrix_mod_data.throb_info.y_magnitude);
MOVE_FUNK_PARAMETER(tmp_v2.v[0], the_funk->matrix_mode, the_funk->matrix_mod_data.throb_info.x_period, the_funk->matrix_mod_data.throb_info.x_magnitude, the_funk->matrix_mod_data.throb_info.x_magnitude);
BrMatrix23PostScale(mat_matrix, tmp_v2.v[0] + 1.f, tmp_v2.v[1] + 1.f);
the_material->map_transform.m[2][0] += the_funk->matrix_mod_data.throb_info.x_centre;
the_material->map_transform.m[2][1] += the_funk->matrix_mod_data.throb_info.y_centre;
break;
case eMatrix_mod_slither:
MOVE_FUNK_PARAMETER(the_material->map_transform.m[2][0], the_funk->matrix_mode, the_funk->matrix_mod_data.slither_info.x_period, the_funk->matrix_mod_data.slither_info.x_magnitude, the_funk->matrix_mod_data.slither_info.x_magnitude);
MOVE_FUNK_PARAMETER(the_material->map_transform.m[2][1], the_funk->matrix_mode, the_funk->matrix_mod_data.slither_info.y_period, the_funk->matrix_mod_data.slither_info.y_magnitude, the_funk->matrix_mod_data.slither_info.y_magnitude);
break;
case eMatrix_mod_roll:
MOVE_FUNK_PARAMETER(the_material->map_transform.m[2][0], the_funk->matrix_mode, the_funk->matrix_mod_data.roll_info.x_period, 1.f, 1.875f);
MOVE_FUNK_PARAMETER(the_material->map_transform.m[2][1], the_funk->matrix_mode, the_funk->matrix_mod_data.roll_info.y_period, 1.f, 1.875f);
break;
case eMatrix_mod_none:
break;
}
if (the_funk->matrix_mod_type != eMatrix_mod_none) {
BrMaterialUpdate(the_funk->material, BR_MATU_MAP_TRANSFORM);
}
}
if (the_funk->lighting_animation_type != eMove_none) {
MOVE_FUNK_PARAMETER(the_material->ka, the_funk->lighting_animation_type, the_funk->lighting_animation_period, the_funk->ambient_delta, the_funk->ambient_delta);
the_material->ka += the_funk->ambient_base;
MOVE_FUNK_PARAMETER(the_material->kd, the_funk->lighting_animation_type, the_funk->lighting_animation_period, the_funk->direct_delta, the_funk->direct_delta);
the_material->kd += the_funk->direct_base;
MOVE_FUNK_PARAMETER(the_material->ks, the_funk->lighting_animation_type, the_funk->lighting_animation_period, the_funk->specular_delta, the_funk->specular_delta);
the_material->ks += the_funk->specular_base;
}
if (the_funk->texture_animation_type == eTexture_animation_frames) {
if (!gAction_replay_mode || !ReplayIsPaused() || the_funk->mode == eFunk_mode_all_laps_but_last || the_funk->mode == 4) {
old_colour_map = the_material->colour_map;
if (the_funk->time_mode == eTime_mode_accurate) {
MOVE_FUNK_PARAMETER(rot_amount, the_funk->texture_animation_data.frames_info.mode, the_funk->texture_animation_data.frames_info.period, the_funk->texture_animation_data.frames_info.texture_count, the_funk->texture_animation_data.frames_info.texture_count);
the_material->colour_map = the_funk->texture_animation_data.frames_info.textures[(int)rot_amount];
} else {
if (the_funk->texture_animation_data.frames_info.period <= fabsf(f_the_time - the_funk->last_frame)) {
the_funk->last_frame = f_the_time;
the_funk->texture_animation_data.frames_info.current_frame++;
if (the_funk->texture_animation_data.frames_info.current_frame >= the_funk->texture_animation_data.frames_info.texture_count) {
the_funk->texture_animation_data.frames_info.current_frame = 0;
}
the_material->colour_map = the_funk->texture_animation_data.frames_info.textures[the_funk->texture_animation_data.frames_info.current_frame];
}
}
if (the_material->colour_map != old_colour_map) {
BrMaterialUpdate(the_funk->material, BR_MATU_COLOURMAP);
}
}
} else if (the_funk->texture_animation_type == eTexture_animation_flic && (!gAction_replay_mode || !ReplayIsPaused())) {
f_time_diff = f_the_time;
if (f_time_diff < the_funk->last_frame) {
f_time_diff = 2 * the_funk->last_frame - f_time_diff;
}
if (the_funk->time_mode == eTime_mode_accurate) {
if (the_funk->last_frame) {
iteration_count = (f_time_diff - the_funk->last_frame) / the_funk->texture_animation_data.flic_info.flic_descriptor.frame_period;
} else {
iteration_count = 1;
}
} else {
if (f_time_diff - the_funk->last_frame > the_funk->texture_animation_data.flic_info.flic_descriptor.frame_period) {
iteration_count = 1;
} else {
iteration_count = 0;
}
}
for (j = 0; j < iteration_count; j++) {
finished = PlayNextFlicFrame(&the_funk->texture_animation_data.flic_info.flic_descriptor);
BrMapUpdate(the_funk->material->colour_map, BR_MAPU_ALL);
BrMaterialUpdate(the_funk->material, BR_MATU_COLOURMAP);
if (finished) {
EndFlic(&the_funk->texture_animation_data.flic_info.flic_descriptor);
StartFlic(
the_funk->texture_animation_data.flic_info.flic_descriptor.file_name,
-1,
&the_funk->texture_animation_data.flic_info.flic_descriptor,
the_funk->texture_animation_data.flic_info.flic_data_length,
(tS8*)the_funk->texture_animation_data.flic_info.flic_data,
the_material->colour_map, 0, 0, 0);
}
the_funk->last_frame = f_the_time;
}
}
}
}
LetFlicFuckWithPalettes();
}
// IDA: void __usercall LollipopizeActor(br_actor *pSubject_actor@<EAX>, br_matrix34 *ref_to_world@<EDX>, tLollipop_mode pWhich_axis@<EBX>)
void LollipopizeActor(br_actor* pSubject_actor, br_matrix34* ref_to_world, tLollipop_mode pWhich_axis) {
br_vector3 ref_to_subject;
br_vector3 fixed_axis;
br_vector3 vector_a;
br_vector3 vector_b;
br_matrix34 subject_to_world;
br_matrix34 mat;
LOG_TRACE("(%p, %p, %d)", pSubject_actor, ref_to_world, pWhich_axis);
BrActorToActorMatrix34(&subject_to_world, pSubject_actor, gNon_track_actor);
BrVector3Sub(&ref_to_subject, (br_vector3*)ref_to_world->m[3], (br_vector3*)subject_to_world.m[3]);
switch (pWhich_axis) {
case eLollipop_none:
TELL_ME_IF_WE_PASS_THIS_WAY();
break;
case eLollipop_x_match:
BrVector3SetFloat(&vector_a, 1.f, 0.f, 0.f);
break;
case eLollipop_y_match:
BrVector3SetFloat(&vector_a, 0.f, 1.f, 0.f);
break;
case eLollipop_z_match:
BrVector3SetFloat(&vector_a, 0.f, 0.f, 1.f);
break;
}
BrVector3Cross(&vector_b, &ref_to_subject, &vector_a);
BrVector3Normalise(&vector_b, &vector_b);
BrVector3Cross(&fixed_axis, &vector_a, &vector_b);
switch (pWhich_axis) {
case eLollipop_none:
break;
case eLollipop_x_match:
mat.m[0][0] = vector_a.v[0];
mat.m[1][0] = vector_a.v[1];
mat.m[2][0] = vector_a.v[2];
mat.m[0][1] = vector_b.v[0];
mat.m[1][1] = vector_b.v[1];
mat.m[2][1] = vector_b.v[2];
mat.m[0][2] = fixed_axis.v[0];
mat.m[1][2] = fixed_axis.v[1];
mat.m[2][2] = fixed_axis.v[2];
mat.m[3][0] = 0.f;
mat.m[3][1] = 0.f;
mat.m[3][2] = 0.f;
break;
case eLollipop_y_match:
mat.m[0][0] = vector_b.v[0];
mat.m[1][0] = vector_b.v[1];
mat.m[2][0] = vector_b.v[2];
mat.m[0][1] = vector_a.v[0];
mat.m[1][1] = vector_a.v[1];
mat.m[2][1] = vector_a.v[2];
mat.m[0][2] = fixed_axis.v[0];
mat.m[1][2] = fixed_axis.v[1];
mat.m[2][2] = fixed_axis.v[2];
mat.m[3][0] = 0.f;
mat.m[3][1] = 0.f;
mat.m[3][2] = 0.f;
break;
case eLollipop_z_match:
mat.m[0][0] = vector_b.v[0];
mat.m[1][0] = vector_b.v[1];
mat.m[2][0] = vector_b.v[2];
mat.m[0][1] = vector_a.v[0];
mat.m[1][1] = vector_a.v[1];
mat.m[2][1] = vector_a.v[2];
mat.m[0][2] = fixed_axis.v[0];
mat.m[1][2] = fixed_axis.v[1];
mat.m[2][2] = fixed_axis.v[2];
mat.m[3][0] = 0.f;
mat.m[3][1] = 0.f;
mat.m[3][2] = 0.f;
break;
}
BrMatrix34Pre(&pSubject_actor->t.t.mat, &mat);
}
// IDA: void __usercall CalcActorGlobalPos(br_vector3 *pResult@<EAX>, br_actor *pActor@<EDX>)
void CalcActorGlobalPos(br_vector3* pResult, br_actor* pActor) {
LOG_TRACE("(%p, %p)", pResult, pActor);
pResult->v[0] = 0.0;
pResult->v[1] = 0.0;
pResult->v[2] = 0.0;
while (pActor && pActor != gNon_track_actor) {
pResult->v[0] = pActor->t.t.mat.m[3][0] + pResult->v[0];
pResult->v[1] = pActor->t.t.mat.m[3][1] + pResult->v[1];
pResult->v[2] = pActor->t.t.mat.m[3][2] + pResult->v[2];
pActor = pActor->parent;
}
}
// IDA: int __usercall PointOutOfSight@<EAX>(br_vector3 *pPoint@<EAX>, br_scalar pMax_distance)
int PointOutOfSight(br_vector3* pPoint, br_scalar pMax_distance) {
br_vector3 distance_vector;
LOG_TRACE("(%p, %f)", pPoint, pMax_distance);
if (gMirror_on__graphics) {
distance_vector.v[0] = pPoint->v[0] - gRearview_camera_to_world.m[3][0];
distance_vector.v[1] = pPoint->v[1] - gRearview_camera_to_world.m[3][1];
distance_vector.v[2] = pPoint->v[2] - gRearview_camera_to_world.m[3][2];
if (distance_vector.v[0] * distance_vector.v[0]
+ distance_vector.v[1] * distance_vector.v[1]
+ distance_vector.v[2] * distance_vector.v[2]
< pMax_distance
&& gRearview_camera_to_world.m[2][2] * distance_vector.v[2]
+ gRearview_camera_to_world.m[2][1] * distance_vector.v[1]
+ gRearview_camera_to_world.m[2][0] * distance_vector.v[0]
< 0.0) {
return 0;
}
}
distance_vector.v[0] = pPoint->v[0] - gCamera_to_world.m[3][0];
distance_vector.v[1] = pPoint->v[1] - gCamera_to_world.m[3][1];
distance_vector.v[2] = pPoint->v[2] - gCamera_to_world.m[3][2];
return distance_vector.v[0] * distance_vector.v[0] + distance_vector.v[1] * distance_vector.v[1] + distance_vector.v[2] * distance_vector.v[2] >= pMax_distance
|| gCamera_to_world.m[2][2] * distance_vector.v[2] + gCamera_to_world.m[2][1] * distance_vector.v[1] + gCamera_to_world.m[2][0] * distance_vector.v[0] >= 0.0;
}
// IDA: void __usercall PathGrooveBastard(tGroovidelic_spec *pGroove@<EAX>, tU32 pTime@<EDX>, br_matrix34 *pMat@<EBX>, int pInterrupt_it@<ECX>)
void PathGrooveBastard(tGroovidelic_spec* pGroove, tU32 pTime, br_matrix34* pMat, int pInterrupt_it) {
br_scalar pos;
LOG_TRACE("(%p, %d, %p, %d)", pGroove, pTime, pMat, pInterrupt_it);
if (pGroove->path_type == eGroove_path_straight) {
if (pGroove->path_data.straight_info.x_delta != 0.0f) {
switch (pGroove->path_mode) {
case eMove_continuous:
if (pGroove->path_data.straight_info.period == 0.0f) {
pos = 0.f;
} else {
pos = fmodf(pTime, pGroove->path_data.straight_info.period) / pGroove->path_data.straight_info.period * pGroove->path_data.straight_info.x_delta;
}
break;
case eMove_controlled:
pos = pGroove->path_data.straight_info.period * pGroove->path_data.straight_info.x_delta;
break;
case eMove_absolute:
pos = pGroove->path_data.straight_info.period;
break;
case eMove_flash:
if (fmodf(pTime, pGroove->path_data.straight_info.period) * 2.0f <= pGroove->path_data.straight_info.period) {
pos = pGroove->path_data.straight_info.x_delta;
} else {
pos = -pGroove->path_data.straight_info.x_delta;
}
break;
case eMove_harmonic:
if (pGroove->path_data.straight_info.period == 0.0f) {
pos = 0.f;
} else {
pos = sinf(BrAngleToRadian(BrDegreeToAngle(fmodf(pTime, pGroove->path_data.straight_info.period) / pGroove->path_data.straight_info.period * 360.0f)))
* pGroove->path_data.straight_info.x_delta;
}
break;
case eMove_linear:
if (pGroove->path_data.straight_info.period == 0.0f) {
pos = 0.f;
} else {
pos = MapSawToTriangle(fmodf(pTime, pGroove->path_data.straight_info.period) / pGroove->path_data.straight_info.period)
* pGroove->path_data.straight_info.x_delta;
}
break;
default:
TELL_ME_IF_WE_PASS_THIS_WAY();
}
pos += pGroove->path_data.straight_info.centre.v[0];
if (pInterrupt_it) {
pGroove->path_resumption_value = pos;
if (pMat->m[3][0] <= pos) {
pGroove->path_interrupt_status = eInterrupt_greater_than;
} else {
pGroove->path_interrupt_status = eInterrupt_less_than;
}
} else if (pGroove->path_interrupt_status == eInterrupt_none) {
BrVector3Copy(&pGroove->actor->t.t.translate.t, &pGroove->path_data.straight_info.centre);
pMat->m[3][0] = pos;
} else if (pGroove->path_interrupt_status == eInterrupt_less_than) {
if (pGroove->path_resumption_value > pos) {
pGroove->path_interrupt_status = eInterrupt_none;
BrVector3Copy(&pGroove->actor->t.t.translate.t, &pGroove->path_data.straight_info.centre);
pMat->m[3][0] = pos;
}
} else if (pGroove->path_resumption_value < pos) {
pGroove->path_interrupt_status = eInterrupt_none;
BrVector3Copy(&pGroove->actor->t.t.euler.t, &pGroove->path_data.straight_info.centre);
pMat->m[3][0] = pos;
}
}
if (pGroove->path_data.straight_info.y_delta != 0.0f) {
switch (pGroove->path_mode) {
case eMove_continuous:
if (pGroove->path_data.straight_info.period == 0.0f) {
pos = 0.f;
} else {
pos = fmodf(pTime, pGroove->path_data.straight_info.period) / pGroove->path_data.straight_info.period * pGroove->path_data.straight_info.y_delta;
}
break;
case eMove_controlled:
pos = pGroove->path_data.straight_info.period * pGroove->path_data.straight_info.y_delta;
break;
case eMove_absolute:
pos = pGroove->path_data.straight_info.period;
break;
case eMove_flash:
if (fmodf(pTime, pGroove->path_data.straight_info.period) * 2.0f <= pGroove->path_data.straight_info.period) {
pos = pGroove->path_data.straight_info.y_delta;
} else {
pos = -pGroove->path_data.straight_info.y_delta;
}
break;
case eMove_harmonic:
if (pGroove->path_data.straight_info.period == 0.0) {
pos = 0.f;
} else {
pos = sinf(BrAngleToRadian(BrDegreeToAngle(fmodf(pTime, pGroove->path_data.straight_info.period) / pGroove->path_data.straight_info.period * 360.0f)))
* pGroove->path_data.straight_info.y_delta;
}
break;
case eMove_linear:
if (pGroove->path_data.straight_info.period == 0.0f) {
pos = 0.f;
} else {
pos = MapSawToTriangle(fmodf(pTime, pGroove->path_data.straight_info.period) / pGroove->path_data.straight_info.period)
* pGroove->path_data.straight_info.y_delta;
}
break;
default:
TELL_ME_IF_WE_PASS_THIS_WAY();
}
pos += pGroove->path_data.straight_info.centre.v[1];
if (pInterrupt_it) {
pGroove->path_resumption_value = pos;
if (pMat->m[3][1] <= pos) {
pGroove->path_interrupt_status = eInterrupt_greater_than;
} else {
pGroove->path_interrupt_status = eInterrupt_less_than;
}
} else if (pGroove->path_interrupt_status == eInterrupt_none) {
BrVector3Copy(&pGroove->actor->t.t.translate.t, &pGroove->path_data.straight_info.centre);
pMat->m[3][1] = pos;
} else if (pGroove->path_interrupt_status == eInterrupt_less_than) {
if (pGroove->path_resumption_value > pos) {
pGroove->path_interrupt_status = eInterrupt_none;
BrVector3Copy(&pGroove->actor->t.t.translate.t, &pGroove->path_data.straight_info.centre);
pMat->m[3][1] = pos;
}
} else if (pGroove->path_resumption_value < pos) {
pGroove->path_interrupt_status = eInterrupt_none;
pGroove->actor->t.t.euler.t = pGroove->path_data.straight_info.centre;
pMat->m[3][1] = pos;
}
}
if (pGroove->path_data.straight_info.z_delta != 0.0f) {
switch (pGroove->path_mode) {
case eMove_continuous:
if (pGroove->path_data.straight_info.period == 0.0f) {
pos = 0.f;
} else {
pos = fmodf(pTime, pGroove->path_data.straight_info.period) / pGroove->path_data.straight_info.period * pGroove->path_data.straight_info.z_delta;
}
break;
case eMove_controlled:
pos = pGroove->path_data.straight_info.period * pGroove->path_data.straight_info.z_delta;
break;
case eMove_absolute:
pos = pGroove->path_data.straight_info.period;
break;
case eMove_flash:
if (fmodf(pTime, pGroove->path_data.straight_info.period) * 2.0f <= pGroove->path_data.straight_info.period) {
pos = pGroove->path_data.straight_info.z_delta;
} else {
pos = -pGroove->path_data.straight_info.z_delta;
}
break;
case eMove_harmonic:
if (pGroove->path_data.straight_info.period == 0.0f) {
pos = 0.f;
} else {
pos = sinf(BrAngleToRadian(BrDegreeToAngle(fmodf(pTime, pGroove->path_data.straight_info.period) / pGroove->path_data.straight_info.period * 360.0f)))
* pGroove->path_data.straight_info.z_delta;
}
break;
case eMove_linear:
if (pGroove->path_data.straight_info.period == 0.0f) {
pos = 0.f;
} else {
pos = MapSawToTriangle(fmodf(pTime, pGroove->path_data.straight_info.period) / pGroove->path_data.straight_info.period)
* pGroove->path_data.straight_info.z_delta;
}
break;
default:
TELL_ME_IF_WE_PASS_THIS_WAY();
}
pos += pGroove->path_data.straight_info.centre.v[1];
if (pInterrupt_it) {
pGroove->path_resumption_value = pos;
if (pMat->m[3][2] <= pos) {
pGroove->path_interrupt_status = eInterrupt_greater_than;
} else {
pGroove->path_interrupt_status = eInterrupt_less_than;
}
} else {
if (pGroove->path_interrupt_status == eInterrupt_none) {
BrVector3Copy(&pGroove->actor->t.t.translate.t, &pGroove->path_data.straight_info.centre);
pMat->m[3][2] = pos;
} else if (pGroove->path_interrupt_status == eInterrupt_less_than) {
if (pGroove->path_resumption_value > pos) {
pGroove->path_interrupt_status = eInterrupt_none;
BrVector3Copy(&pGroove->actor->t.t.translate.t, &pGroove->path_data.straight_info.centre);
pMat->m[3][2] = pos;
}
} else if (pGroove->path_resumption_value < pos) {
pGroove->path_interrupt_status = eInterrupt_none;
BrVector3Copy(&pGroove->actor->t.t.translate.t, &pGroove->path_data.straight_info.centre);
pMat->m[3][2] = pos;
}
}
}
BrVector3Copy(&pGroove->object_position, &pGroove->actor->t.t.translate.t);
} else if (pGroove->path_type == eGroove_path_circular) {
BrVector3Copy(&pGroove->actor->t.t.translate.t, &pGroove->path_data.circular_info.centre);
if (pGroove->path_data.circular_info.axis == eGroove_axis_y) {
if (pGroove->path_data.circular_info.period == 0.0f) {
pos = 0.f;
} else {
pos = cosf(BrAngleToRadian(BrDegreeToAngle(fmodf(pTime, pGroove->path_data.circular_info.period) / pGroove->path_data.circular_info.period * 360.0f))) * pGroove->path_data.circular_info.radius;
}
pMat->m[3][0] = pGroove->path_data.circular_info.centre.v[0] + pos;
} else if (pGroove->path_data.circular_info.axis == eGroove_axis_z) {
if (pGroove->path_data.circular_info.period == 0.0f) {
pos = 0.f;
} else {
pos = sinf(BrAngleToRadian(BrDegreeToAngle(fmodf(pTime, pGroove->path_data.circular_info.period) / pGroove->path_data.circular_info.period * 360.0f))) * pGroove->path_data.circular_info.radius;
}
pMat->m[3][0] = pGroove->path_data.circular_info.centre.v[0] + pos;
}
if (pGroove->path_data.circular_info.axis == eGroove_axis_x) {
if (pGroove->path_data.circular_info.period == 0.0f) {
pos = 0.f;
} else {
pos = sinf(BrAngleToRadian(BrDegreeToAngle(fmodf(pTime, pGroove->path_data.circular_info.period) / pGroove->path_data.circular_info.period * 360.0f))) * pGroove->path_data.circular_info.radius;
}
pMat->m[3][1] = pGroove->path_data.circular_info.centre.v[1] + pos;
} else if (pGroove->path_data.circular_info.axis == eGroove_axis_z) {
if (pGroove->path_data.circular_info.period == 0.0f) {
pos = 0.f;
} else {
pos = cosf(BrAngleToRadian(BrDegreeToAngle(fmodf(pTime, pGroove->path_data.circular_info.period) / pGroove->path_data.circular_info.period * 360.0f))) * pGroove->path_data.circular_info.radius;
}
pMat->m[3][1] = pGroove->path_data.circular_info.centre.v[1] + pos;
}
if (pGroove->path_data.circular_info.axis == eGroove_axis_x) {
if (pGroove->path_data.circular_info.period == 0.0f) {
pos = 0.f;
} else {
pos = cosf(BrAngleToRadian(BrDegreeToAngle(fmodf(pTime, pGroove->path_data.circular_info.period) / pGroove->path_data.circular_info.period * 360.0f))) * pGroove->path_data.circular_info.radius;
}
pMat->m[3][2] = pGroove->path_data.circular_info.centre.v[1] + pos;
} else if (pGroove->path_data.circular_info.axis == eGroove_axis_z) {
if (pGroove->path_data.circular_info.period == 0.0f) {
pos = 0.f;
} else {
pos = sinf(BrAngleToRadian(BrDegreeToAngle(fmodf(pTime, pGroove->path_data.circular_info.period) / pGroove->path_data.circular_info.period * 360.0f))) * pGroove->path_data.circular_info.radius;
}
pMat->m[3][2] = pGroove->path_data.circular_info.centre.v[1] + pos;
}
BrVector3Copy(&pGroove->object_position, &pGroove->actor->t.t.translate.t);
}
}
// IDA: void __usercall ObjectGrooveBastard(tGroovidelic_spec *pGroove@<EAX>, tU32 pTime@<EDX>, br_matrix34 *pMat@<EBX>, int pInterrupt_it@<ECX>)
void ObjectGrooveBastard(tGroovidelic_spec* pGroove, tU32 pTime, br_matrix34* pMat, int pInterrupt_it) {
int rock_it;
br_scalar x_size;
br_scalar y_size;
br_scalar z_size;
br_scalar pos;
br_bounds* bounds;
LOG_TRACE("(%p, %d, %p, %d)", pGroove, pTime, pMat, pInterrupt_it);
x_size = 0;
y_size = 0;
z_size = 0;
pos = 0;
switch (pGroove->object_type) {
case eGroove_object_spin:
if (pGroove->object_data.spin_info.axis == eGroove_axis_y) {
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.spin_info.period != 0.0) {
pos = fmod(pTime, pGroove->object_data.spin_info.period) / pGroove->object_data.spin_info.period * 360.0;
}
} else if (pGroove->object_mode == eMove_controlled) {
pos = pGroove->object_data.spin_info.period * 360.0;
} else if (pGroove->object_mode == eMove_absolute) {
pos = pGroove->object_data.spin_info.period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.spin_info.period) * 2.0 <= pGroove->object_data.spin_info.period) {
pos = (br_scalar) 3.85156; // Pierre-Marie Baty -- added type cast
} else {
pos = (br_scalar) -3.85156; // Pierre-Marie Baty -- added type cast
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.spin_info.period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.spin_info.period) / pGroove->object_data.spin_info.period * 360.0)))
* 360.0;
}
} else {
if (pGroove->object_data.spin_info.period != 0.0) {
pos = fmod(pTime, pGroove->object_data.spin_info.period) / pGroove->object_data.spin_info.period;
pos = MapSawToTriangle(pos) * 360.0;
}
}
DRMatrix34PostRotateY(pMat, BrDegreeToAngle(pos));
} else if (pGroove->object_data.spin_info.axis == eGroove_axis_z) {
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.spin_info.period != 0.0) {
pos = fmod(pTime, pGroove->object_data.spin_info.period) / pGroove->object_data.spin_info.period * 360.0;
}
} else if (pGroove->object_mode == eMove_controlled) {
pos = pGroove->object_data.spin_info.period * 360.0;
} else if (pGroove->object_mode == eMove_absolute) {
pos = pGroove->object_data.spin_info.period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.spin_info.period) * 2.0 <= pGroove->object_data.spin_info.period) {
pos = (br_scalar) 3.85156; // Pierre-Marie Baty -- added type cast
} else {
pos = (br_scalar) -3.85156; // Pierre-Marie Baty -- added type cast
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.spin_info.period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.spin_info.period) / pGroove->object_data.spin_info.period * 360.0)))
* 360.0;
}
} else {
if (pGroove->object_data.spin_info.period != 0.0) {
pos = fmod(pTime, pGroove->object_data.spin_info.period) / pGroove->object_data.spin_info.period;
pos = MapSawToTriangle(pos) * 360.0;
}
}
DRMatrix34PostRotateZ(pMat, BrDegreeToAngle(pos));
} else if (pGroove->object_data.spin_info.axis == eGroove_axis_x) {
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.spin_info.period != 0.0) {
pos = fmod(pTime, pGroove->object_data.spin_info.period) / pGroove->object_data.spin_info.period * 360.0;
}
} else if (pGroove->object_mode == eMove_controlled) {
pos = pGroove->object_data.spin_info.period * 360.0;
} else if (pGroove->object_mode == eMove_absolute) {
pos = pGroove->object_data.spin_info.period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.spin_info.period) * 2.0 <= pGroove->object_data.spin_info.period) {
pos = (br_scalar) 3.85156; // Pierre-Marie Baty -- added type cast
} else {
pos = (br_scalar) -3.85156; // Pierre-Marie Baty -- added type cast
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.spin_info.period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.spin_info.period) / pGroove->object_data.spin_info.period * 360.0)))
* 360.0;
}
} else {
if (pGroove->object_data.spin_info.period != 0.0) {
pos = fmod(pTime, pGroove->object_data.spin_info.period) / pGroove->object_data.spin_info.period;
pos = MapSawToTriangle(pos) * 360.0;
}
}
DRMatrix34PostRotateX(pMat, BrDegreeToAngle(pos));
}
break;
case eGroove_object_rock:
rock_it = 1;
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.rock_info.period != 0.0) {
pos = fmod(pTime, pGroove->object_data.rock_info.period) / pGroove->object_data.rock_info.period * pGroove->object_data.rock_info.max_angle;
}
} else if (pGroove->object_mode == eMove_controlled) {
pos = pGroove->object_data.rock_info.period * pGroove->object_data.rock_info.max_angle;
} else if (pGroove->object_mode == eMove_absolute) {
pos = pGroove->object_data.rock_info.period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.rock_info.period) * 2.0 <= pGroove->object_data.rock_info.period) {
pos = pGroove->object_data.rock_info.max_angle;
} else {
pos = -pGroove->object_data.rock_info.max_angle;
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.rock_info.period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.rock_info.period) / pGroove->object_data.rock_info.period * 360.0)))
* pGroove->object_data.rock_info.max_angle;
}
} else {
if (pGroove->object_data.rock_info.period != 0.0) {
pos = fmod(pTime, pGroove->object_data.rock_info.period) / pGroove->object_data.rock_info.period;
pos = MapSawToTriangle(pos) * pGroove->object_data.rock_info.max_angle;
}
}
if (pInterrupt_it) {
pGroove->object_resumption_value = pos;
if (pGroove->object_data.rock_info.current_angle <= pos) {
pGroove->object_interrupt_status = eInterrupt_greater_than;
} else {
pGroove->object_interrupt_status = eInterrupt_less_than;
}
} else if (pGroove->object_interrupt_status == eInterrupt_less_than) {
if (pGroove->object_resumption_value <= pos || gAction_replay_mode) {
rock_it = 0;
} else {
pGroove->object_interrupt_status = eInterrupt_none;
}
} else if (pGroove->object_interrupt_status == eInterrupt_greater_than) {
if (pGroove->object_resumption_value >= pos || gAction_replay_mode) {
rock_it = 0;
} else {
pGroove->object_interrupt_status = eInterrupt_none;
}
}
if (rock_it) {
pGroove->object_data.rock_info.current_angle = pos;
}
if (pGroove->object_data.rock_info.axis == eGroove_axis_y) {
DRMatrix34PostRotateY(pMat, BrDegreeToAngle(pGroove->object_data.rock_info.current_angle));
} else if (pGroove->object_data.rock_info.axis == eGroove_axis_z) {
DRMatrix34PostRotateZ(pMat, BrDegreeToAngle(pGroove->object_data.rock_info.current_angle));
} else if (pGroove->object_data.rock_info.axis == eGroove_axis_x) {
DRMatrix34PostRotateX(pMat, BrDegreeToAngle(pGroove->object_data.rock_info.current_angle));
}
break;
case eGroove_object_throb:
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.throb_info.z_period != 0.0) {
z_size = fmod(pTime, pGroove->object_data.throb_info.z_period) / pGroove->object_data.throb_info.z_period * pGroove->object_data.throb_info.z_magnitude;
}
} else if (pGroove->object_mode == eMove_controlled) {
z_size = pGroove->object_data.throb_info.z_magnitude * pGroove->object_data.throb_info.z_period;
} else if (pGroove->object_mode == eMove_absolute) {
z_size = pGroove->object_data.throb_info.z_period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.throb_info.z_period) * 2.0 <= pGroove->object_data.throb_info.z_period) {
z_size = pGroove->object_data.throb_info.z_magnitude;
} else {
z_size = -pGroove->object_data.throb_info.z_magnitude;
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.throb_info.z_period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.throb_info.z_period) / pGroove->object_data.throb_info.z_period * 360.0)))
* pGroove->object_data.throb_info.z_magnitude;
}
} else {
if (pGroove->object_data.throb_info.z_period != 0.0) {
z_size = fmod(pTime, pGroove->object_data.throb_info.z_period) / pGroove->object_data.throb_info.z_period;
z_size = MapSawToTriangle(z_size) * pGroove->object_data.throb_info.z_magnitude;
}
}
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.throb_info.x_period != 0.0) {
x_size = fmod(pTime, pGroove->object_data.throb_info.x_period) / pGroove->object_data.throb_info.x_period * pGroove->object_data.throb_info.x_magnitude;
}
} else if (pGroove->object_mode == eMove_controlled) {
x_size = pGroove->object_data.throb_info.x_magnitude * pGroove->object_data.throb_info.x_period;
} else if (pGroove->object_mode == eMove_absolute) {
x_size = pGroove->object_data.throb_info.x_period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.throb_info.x_period) * 2.0 <= pGroove->object_data.throb_info.x_period) {
x_size = pGroove->object_data.throb_info.x_magnitude;
} else {
x_size = -pGroove->object_data.throb_info.x_magnitude;
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.throb_info.x_period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.throb_info.x_period) / pGroove->object_data.throb_info.x_period * 360.0)))
* pGroove->object_data.throb_info.x_magnitude;
}
} else {
if (pGroove->object_data.throb_info.x_period != 0.0) {
x_size = fmod(pTime, pGroove->object_data.throb_info.x_period) / pGroove->object_data.throb_info.x_period;
x_size = MapSawToTriangle(x_size) * pGroove->object_data.throb_info.x_magnitude;
}
}
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.throb_info.y_period != 0.0) {
y_size = fmod(pTime, pGroove->object_data.throb_info.y_period) / pGroove->object_data.throb_info.y_period * pGroove->object_data.throb_info.y_magnitude;
}
} else if (pGroove->object_mode == eMove_controlled) {
y_size = pGroove->object_data.throb_info.y_magnitude * pGroove->object_data.throb_info.y_period;
} else if (pGroove->object_mode == eMove_absolute) {
y_size = pGroove->object_data.throb_info.y_period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.throb_info.y_period) * 2.0 <= pGroove->object_data.throb_info.y_period) {
y_size = pGroove->object_data.throb_info.y_magnitude;
} else {
y_size = -pGroove->object_data.throb_info.y_magnitude;
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.throb_info.y_period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.throb_info.y_period) / pGroove->object_data.throb_info.y_period * 360.0)))
* pGroove->object_data.throb_info.y_magnitude;
}
} else {
if (pGroove->object_data.throb_info.y_period != 0.0) {
y_size = fmod(pTime, pGroove->object_data.throb_info.y_period) / pGroove->object_data.throb_info.y_period;
y_size = MapSawToTriangle(y_size) * pGroove->object_data.throb_info.y_magnitude;
}
}
BrMatrix34PostScale(pMat, x_size + 1.0, y_size + 1.0, z_size + 1.0);
/* FALLTHROUGH */
case eGroove_object_shear:
bounds = &pGroove->actor->model->bounds;
if (pGroove->object_data.shear_info.x_magnitude == 0.0) {
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.shear_info.z_period != 0.0) {
z_size = fmod(pTime, pGroove->object_data.shear_info.z_period) / pGroove->object_data.shear_info.z_period * pGroove->object_data.shear_info.z_magnitude;
}
} else if (pGroove->object_mode == eMove_controlled) {
z_size = pGroove->object_data.shear_info.z_magnitude * pGroove->object_data.shear_info.z_period;
} else if (pGroove->object_mode == eMove_absolute) {
z_size = pGroove->object_data.shear_info.z_period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.shear_info.z_period) * 2.0 <= pGroove->object_data.shear_info.z_period) {
z_size = pGroove->object_data.shear_info.z_magnitude;
} else {
z_size = -pGroove->object_data.shear_info.z_magnitude;
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.shear_info.z_period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.shear_info.z_period) / pGroove->object_data.shear_info.z_period * 360.0)))
* pGroove->object_data.shear_info.z_magnitude;
}
} else {
if (pGroove->object_data.shear_info.z_period != 0.0) {
z_size = fmod(pTime, pGroove->object_data.shear_info.z_period) / pGroove->object_data.shear_info.z_period;
z_size = MapSawToTriangle(z_size) * pGroove->object_data.shear_info.z_magnitude;
}
}
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.shear_info.y_period != 0.0) {
y_size = fmod(pTime, pGroove->object_data.shear_info.y_period) / pGroove->object_data.shear_info.y_period * pGroove->object_data.shear_info.y_magnitude;
}
} else if (pGroove->object_mode == eMove_controlled) {
y_size = pGroove->object_data.shear_info.y_magnitude * pGroove->object_data.shear_info.y_period;
} else if (pGroove->object_mode == eMove_absolute) {
y_size = pGroove->object_data.shear_info.y_period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.shear_info.y_period) * 2.0 <= pGroove->object_data.shear_info.y_period) {
y_size = pGroove->object_data.shear_info.y_magnitude;
} else {
y_size = -pGroove->object_data.shear_info.y_magnitude;
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.shear_info.y_period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.shear_info.y_period) / pGroove->object_data.shear_info.y_period * 360.0)))
* pGroove->object_data.shear_info.y_magnitude;
}
} else {
if (pGroove->object_data.shear_info.y_period != 0.0) {
y_size = fmod(pTime, pGroove->object_data.shear_info.y_period) / pGroove->object_data.shear_info.y_period;
y_size = MapSawToTriangle(y_size) * pGroove->object_data.shear_info.y_magnitude;
}
}
BrMatrix34PostShearX(pMat, y_size / (bounds->max.v[1] - bounds->min.v[1]), z_size / bounds->max.v[2] - bounds->min.v[2]);
} else if (pGroove->object_data.shear_info.y_magnitude == 0.0) {
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.shear_info.z_period != 0.0) {
z_size = fmod(pTime, pGroove->object_data.shear_info.z_period) / pGroove->object_data.shear_info.z_period * pGroove->object_data.shear_info.z_magnitude;
}
} else if (pGroove->object_mode == eMove_controlled) {
z_size = pGroove->object_data.shear_info.z_magnitude * pGroove->object_data.shear_info.z_period;
} else if (pGroove->object_mode == eMove_absolute) {
z_size = pGroove->object_data.shear_info.z_period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.shear_info.z_period) * 2.0 <= pGroove->object_data.shear_info.z_period) {
z_size = pGroove->object_data.shear_info.z_magnitude;
} else {
z_size = -pGroove->object_data.shear_info.z_magnitude;
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.shear_info.z_period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.shear_info.z_period) / pGroove->object_data.shear_info.z_period * 360.0)))
* pGroove->object_data.shear_info.z_magnitude;
}
} else {
if (pGroove->object_data.shear_info.z_period != 0.0) {
z_size = fmod(pTime, pGroove->object_data.shear_info.z_period) / pGroove->object_data.shear_info.z_period;
z_size = MapSawToTriangle(z_size) * pGroove->object_data.shear_info.z_magnitude;
}
}
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.shear_info.x_period != 0.0) {
x_size = fmod(pTime, pGroove->object_data.shear_info.x_period) / pGroove->object_data.shear_info.x_period * pGroove->object_data.shear_info.x_magnitude;
}
} else if (pGroove->object_mode == eMove_controlled) {
x_size = pGroove->object_data.shear_info.x_magnitude * pGroove->object_data.shear_info.x_period;
} else if (pGroove->object_mode == eMove_absolute) {
x_size = pGroove->object_data.shear_info.x_period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.shear_info.x_period) * 2.0 <= pGroove->object_data.shear_info.x_period) {
x_size = pGroove->object_data.shear_info.x_magnitude;
} else {
x_size = -pGroove->object_data.shear_info.x_magnitude;
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.shear_info.x_period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.shear_info.x_period) / pGroove->object_data.shear_info.x_period * 360.0)))
* pGroove->object_data.shear_info.x_magnitude;
}
} else { // linear
if (pGroove->object_data.shear_info.x_period != 0.0) {
x_size = fmod(pTime, pGroove->object_data.shear_info.x_period) / pGroove->object_data.shear_info.x_period;
x_size = MapSawToTriangle(x_size) * pGroove->object_data.shear_info.x_magnitude;
}
}
BrMatrix34PostShearY(pMat, x_size / (bounds->max.v[0] - bounds->min.v[0]), z_size / (bounds->max.v[2] - bounds->min.v[2]));
} else { // x_magnitude == 0
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.shear_info.y_period != 0.0) {
y_size = fmod(pTime, pGroove->object_data.shear_info.y_period) / pGroove->object_data.shear_info.y_period * pGroove->object_data.shear_info.y_magnitude;
}
} else if (pGroove->object_mode == eMove_controlled) {
y_size = pGroove->object_data.shear_info.y_magnitude * pGroove->object_data.shear_info.y_period;
} else if (pGroove->object_mode == eMove_absolute) {
y_size = pGroove->object_data.shear_info.y_period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.shear_info.y_period) * 2.0 <= pGroove->object_data.shear_info.y_period) {
y_size = pGroove->object_data.shear_info.y_magnitude;
} else {
y_size = -pGroove->object_data.shear_info.y_magnitude;
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.shear_info.y_period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.shear_info.y_period) / pGroove->object_data.shear_info.y_period * 360.0)))
* pGroove->object_data.shear_info.y_magnitude;
}
} else {
if (pGroove->object_data.shear_info.y_period != 0.0) {
y_size = fmod(pTime, pGroove->object_data.shear_info.y_period) / pGroove->object_data.shear_info.y_period;
y_size = MapSawToTriangle(y_size) * pGroove->object_data.shear_info.y_magnitude;
}
}
if (pGroove->object_mode == eMove_continuous) {
if (pGroove->object_data.shear_info.x_period != 0.0) {
x_size = fmod(pTime, pGroove->object_data.shear_info.x_period) / pGroove->object_data.shear_info.x_period * pGroove->object_data.shear_info.x_magnitude;
}
} else if (pGroove->object_mode == eMove_controlled) {
x_size = pGroove->object_data.shear_info.x_magnitude * pGroove->object_data.shear_info.x_period;
} else if (pGroove->object_mode == eMove_absolute) {
x_size = pGroove->object_data.shear_info.x_period;
} else if (pGroove->object_mode == eMove_flash) {
if (fmod(pTime, pGroove->object_data.shear_info.x_period) * 2.0 <= pGroove->object_data.shear_info.x_period) {
x_size = pGroove->object_data.shear_info.x_magnitude;
} else {
x_size = -pGroove->object_data.shear_info.x_magnitude;
}
} else if (pGroove->object_mode == eMove_harmonic) {
if (pGroove->object_data.shear_info.x_period != 0.0) {
BrAngleToRadian(
BrDegreeToAngle(
fmod(pTime, pGroove->object_data.shear_info.x_period) / pGroove->object_data.shear_info.x_period * 360.0)))
* pGroove->object_data.shear_info.x_magnitude;
}
} else {
if (pGroove->object_data.shear_info.x_period != 0.0) {
x_size = fmod(pTime, pGroove->object_data.shear_info.x_period) / pGroove->object_data.shear_info.x_period;
x_size = MapSawToTriangle(x_size) * pGroove->object_data.shear_info.x_magnitude;
}
}
BrMatrix34PostShearZ(pMat, x_size / (bounds->max.v[0] - bounds->min.v[0]), y_size / (bounds->max.v[1] - bounds->min.v[1]));
}
break;
default:
return;
}
}
// IDA: void __usercall GrooveThisDelic(tGroovidelic_spec *pGroove@<EAX>, tU32 pTime@<EDX>, int pInterrupt_it@<EBX>)
void GrooveThisDelic(tGroovidelic_spec* pGroove, tU32 pTime, int pInterrupt_it) {
br_actor* the_actor;
br_vector3 actor_pos;
br_matrix34* the_mat;
tInterrupt_status old_path_interrupt;
tInterrupt_status old_object_interrupt;
LOG_TRACE8("(%p, %d, %d)", pGroove, pTime, pInterrupt_it);
old_path_interrupt = pGroove->path_interrupt_status;
old_object_interrupt = pGroove->object_interrupt_status;
the_actor = pGroove->actor;
pGroove->done_this_frame = 1;
CalcActorGlobalPos(&actor_pos, the_actor);
if (PointOutOfSight(&actor_pos, pGroove->mode == eGroove_mode_distance ? gYon_squared : 36.f)) {
return;
}
the_mat = &the_actor->t.t.mat;
if (!gAction_replay_mode
|| !ReplayIsPaused()
|| pGroove->path_mode == eMove_controlled
|| pGroove->path_mode == eMove_absolute) {
PathGrooveBastard(pGroove, pTime, the_mat, pInterrupt_it);
}
if ((pGroove->object_type != eGroove_object_none || pGroove->lollipop_mode != eLollipop_none)
&& (!gAction_replay_mode
|| !ReplayIsPaused()
|| pGroove->object_mode == eMove_controlled
|| pGroove->object_mode == eMove_absolute)) {
the_mat->m[0][0] = 1.0f;
the_mat->m[0][1] = 0.0f;
the_mat->m[0][2] = 0.0f;
the_mat->m[1][0] = 0.0f;
the_mat->m[1][1] = 1.0f;
the_mat->m[1][2] = 0.0f;
the_mat->m[2][0] = 0.0f;
the_mat->m[2][1] = 0.0f;
the_mat->m[2][2] = 1.0f;
the_mat->m[3][0] = -pGroove->object_centre.v[0];
the_mat->m[3][1] = -pGroove->object_centre.v[1];
the_mat->m[3][2] = -pGroove->object_centre.v[2];
ObjectGrooveBastard(pGroove, pTime, the_mat, pInterrupt_it);
the_actor->t.t.mat.m[3][0] += pGroove->object_position.v[0] + pGroove->object_centre.v[0];
the_actor->t.t.mat.m[3][1] += pGroove->object_position.v[1] + pGroove->object_centre.v[1];
the_actor->t.t.mat.m[3][2] += pGroove->object_position.v[2] + pGroove->object_centre.v[2];
if (pGroove->lollipop_mode != eLollipop_none) {
LollipopizeActor(pGroove->actor, &gCamera_to_world, pGroove->lollipop_mode);
}
}
if (pGroove->path_interrupt_status != old_path_interrupt || pGroove->object_interrupt_status != old_object_interrupt) {
PipeSingleGrooveStop(
pGroove - gGroovidelics_array,
the_mat,
pGroove->path_interrupt_status,
pGroove->object_interrupt_status,
pGroove->path_resumption_value,
pGroove->object_resumption_value);
}
}
// IDA: void __cdecl GrooveThoseDelics()
void GrooveThoseDelics(void) {
int i;
tGroovidelic_spec* the_groove;
float f_the_time;
LOG_TRACE("()");
if (gGroovidelics_array != NULL) {
f_the_time = (double)GetTotalTime();
gPrevious_groove_times[1] = gPrevious_groove_times[0];
gPrevious_groove_times[0] = f_the_time;
for (i = 0; i < gGroovidelics_array_size; i++) {
the_groove = &gGroovidelics_array[i];
if (the_groove->owner != -999 && !the_groove->done_this_frame) {
GrooveThisDelic(the_groove, f_the_time, 0);
}
}
}
}
// IDA: void __usercall StopGroovidelic(br_actor *pActor@<EAX>)
void StopGroovidelic(br_actor* pActor) {
int i;
tGroovidelic_spec* the_groove;
LOG_TRACE("(%p)", pActor);
for (i = 0; i < gGroovidelics_array_size; i++) {
the_groove = &gGroovidelics_array[i];
if (the_groove->actor == pActor) {
if (the_groove->path_interrupt_status == eInterrupt_none && the_groove->object_interrupt_status == eInterrupt_none) {
GrooveThisDelic(the_groove, gPrevious_groove_times[1], 1);
}
return;
}
}
}
// IDA: void __usercall SetGrooveInterrupt(int pGroove_index@<EAX>, br_matrix34 *pMatrix@<EDX>, int pPath_interrupt@<EBX>, int pObject_interrupt@<ECX>, float pPath_resumption, float pObject_resumption)
void SetGrooveInterrupt(int pGroove_index, br_matrix34* pMatrix, int pPath_interrupt, int pObject_interrupt, float pPath_resumption, float pObject_resumption) {
tGroovidelic_spec* the_groove;
LOG_TRACE("(%d, %p, %d, %d, %f, %f)", pGroove_index, pMatrix, pPath_interrupt, pObject_interrupt, pPath_resumption, pObject_resumption);
the_groove = &gGroovidelics_array[pGroove_index];
the_groove->path_interrupt_status = pPath_interrupt;
the_groove->object_interrupt_status = pObject_interrupt;
the_groove->path_resumption_value = pPath_resumption;
the_groove->object_resumption_value = pObject_resumption;
BrMatrix34Copy(&the_groove->actor->t.t.mat, pMatrix);
}
// IDA: void __cdecl ResetGrooveFlags()
void ResetGrooveFlags(void) {
int i;
tGroovidelic_spec* the_groove;
LOG_TRACE("()");
the_groove = gGroovidelics_array;
for (i = 0; i < gGroovidelics_array_size; i++) {
the_groove->done_this_frame = 0;
the_groove++;
}
}
// IDA: tSpecial_volume* __cdecl GetDefaultSpecialVolumeForWater()
tSpecial_volume* GetDefaultSpecialVolumeForWater(void) {
LOG_TRACE("()");
return gDefault_water_spec_vol;
}
// IDA: tSpecial_volume* __usercall FindSpecialVolume@<EAX>(br_vector3 *pP@<EAX>, tSpecial_volume *pLast_vol@<EDX>)
tSpecial_volume* FindSpecialVolume(br_vector3* pP, tSpecial_volume* pLast_vol) {
int i;
tSpecial_volume* v;
br_vector3 p;
LOG_TRACE("(%p, %p)", pP, pLast_vol);
for (i = 0, v = gProgram_state.special_volumes; i < gProgram_state.special_volume_count; i++, v++) {
if (!v->no_mat && v->bounds.min.v[0] < pP->v[0] && pP->v[0] < v->bounds.max.v[0] && v->bounds.min.v[1] < pP->v[1] && pP->v[1] < v->bounds.max.v[1] && v->bounds.min.v[2] < pP->v[2] && pP->v[2] < v->bounds.max.v[2]) {
BrMatrix34ApplyP(&p, pP, &v->inv_mat);
if (-1.f < p.v[0] && p.v[0] < 1.f && -1.f < p.v[1] && p.v[1] < 1.f && -1.f < p.v[2] && p.v[2] < 1.f) {
return v;
}
}
}
return NULL;
}
// IDA: void __cdecl SaveAdditionalActors()
void SaveAdditionalActors(void) {
LOG_TRACE("()");
if (gAdditional_actors != NULL) {
SaveAdditionalStuff();
}
}
// IDA: br_scalar __usercall DistanceFromFace@<ST0>(br_vector3 *pPos@<EAX>, tFace_ref *pFace@<EDX>)
br_scalar DistanceFromFace(br_vector3* pPos, tFace_ref* pFace) {
//br_vector3 normal; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, %p)", pPos, pFace);
return BrVector3Dot(&pFace->normal, pPos) - pFace->d;
}
// IDA: br_uint_32 __cdecl CalcHighestID(br_actor *pActor, int *pHighest)
br_uint_32 CalcHighestID(br_actor* pActor, int* pHighest) {
char s[256];
int number;
LOG_TRACE("(%p, %p)", pActor, pHighest);
if (pActor->identifier == NULL || pActor->identifier[0] == '@') {
return 0;
}
strcpy(s
, &pActor
->identifier
[4]);
s[4] = '\0';
if (*pHighest < number) {
*pHighest = number;
}
return 0;
}
// IDA: br_uint_32 __cdecl SetID(br_actor *pActor, void *pArg)
br_uint_32 SetID(br_actor* pActor, void* pArg) {
char s[256];
LOG_TRACE("(%p, %p)", pActor, pArg);
if (pActor->identifier == NULL) {
return 0;
}
strcpy(s
, pActor
->identifier
);
sprintf(&s
[4], "%04d", (int)(intptr_t)pArg
);
BrResFree(pActor->identifier);
pActor->identifier = BrResStrDup(pActor, s);
return 0;
}
// IDA: void __usercall UniquificateActorsName(br_actor *pUniverse_actor@<EAX>, br_actor *pActor@<EDX>)
void UniquificateActorsName(br_actor* pUniverse_actor, br_actor* pActor) {
int highest;
LOG_TRACE("(%p, %p)", pUniverse_actor, pActor);
if (pActor->identifier == NULL || pActor->identifier[0] == '@') {
return;
}
highest = 0;
DRActorEnumRecurse(pUniverse_actor, (br_actor_enum_cbfn*)CalcHighestNonAmID, &highest);
DRActorEnumRecurse(pActor, (br_actor_enum_cbfn*)SetID, (void*)(uintptr_t)(highest + 1));
}
// IDA: void __usercall AccessoryHeadup(br_actor *pActor@<EAX>, char *pPrefix@<EDX>)
void AccessoryHeadup(br_actor* pActor, char* pPrefix) {
char s[256];
//int i; // Pierre-Marie Baty -- unused variable
//br_actor* original_actor; // Pierre-Marie Baty -- unused variable
LOG_TRACE("(%p, \"%s\")", pActor, pPrefix);
if (pActor->identifier != NULL) {
strcat(s
, pActor
->identifier
);
}
NewTextHeadupSlot(4, 0, 2000, -2, s);
}
// IDA: br_uint_32 __cdecl CalcHighestNonAmID(br_actor *pActor, int *pHighest)
br_uint_32 CalcHighestNonAmID(br_actor* pActor, int* pHighest) {
char s[256];
int number;
LOG_TRACE("(%p, %p)", pActor, pHighest);
if (pActor->identifier == NULL || pActor->identifier[0] == '&') {
return 0;
}
if (strlen(pActor
->identifier
) == 12) {
strcpy(s
, &pActor
->identifier
[4]);
} else {
number = 0;
}
if (*pHighest < number) {
*pHighest = number;
}
return 0;
}
// IDA: br_uint_32 __cdecl SetIDAndDupModel(br_actor *pActor, void *pArg)
br_uint_32 SetIDAndDupModel(br_actor* pActor, void* pArg) {
char s[256];
char s2[256];
br_model* new_model;
LOG_TRACE("(%p, %p)", pActor, pArg);
if (pActor->identifier == NULL || pActor->identifier[0] == '@') {
return 0;
}
*(int*)(uintptr_t)pArg = *(int*)(uintptr_t)pArg + 1;
strcpy(s
, pActor
->identifier
);
s[0] = '@';
sprintf(&s
[4], "%04d", *(int*)(uintptr_t)pArg
);
BrResFree(pActor->identifier);
pActor->identifier = BrResStrDup(pActor, s);
if (pActor->model != NULL) {
new_model = BrModelAllocate(s2, pActor->model->nvertices, pActor->model->nfaces);
memcpy(new_model
->vertices
, pActor
->model
->vertices
, pActor
->model
->nvertices
* sizeof(br_vertex
));
memcpy(new_model
->faces
, pActor
->model
->faces
, pActor
->model
->nfaces
* sizeof(br_face
));
new_model->flags |= BR_MODF_UPDATEABLE;
BrModelAdd(new_model);
BrModelUpdate(new_model, BR_MODU_ALL);
pActor->model = new_model;
gAdditional_models[gNumber_of_additional_models] = new_model;
gNumber_of_additional_models++;
}
return 0;
}
// IDA: void __usercall DuplicateIfNotAmpersand(br_actor *pActor@<EAX>)
void DuplicateIfNotAmpersand(br_actor* pActor) {
int highest;
LOG_TRACE("(%p)", pActor);
if (pActor->identifier != NULL && pActor->identifier[0] != '&') {
highest = 0;
DRActorEnumRecurse(gUniverse_actor, (br_actor_enum_cbfn*)CalcHighestID, &highest);
DRActorEnumRecurse(pActor, (br_actor_enum_cbfn*)SetIDAndDupModel, &highest);
}
}
// IDA: void __usercall DropActor(int pIndex@<EAX>)
void DropActor(int pIndex) {
FILE* f;
tPath_name the_path;
char s[256];
int i;
int face_bastard;
int face_count;
tBounds kev_bounds;
tFace_ref the_list[50];
br_scalar nearest_bastard;
br_scalar distance_bastard;
//br_scalar ts; // Pierre-Marie Baty -- unused variable
br_vector3 side_vector;
//br_angle phi; // Pierre-Marie Baty -- unused variable
//br_matrix34 mat; // Pierre-Marie Baty -- unused variable
br_transform new_transform;
br_actor* a;
br_actor* last_non_ampersand;
LOG_TRACE("(%d)", pIndex);
if (PDKeyDown(KEY_CTRL_ANY)) {
pIndex += 20;
}
if (PDKeyDown(KEY_ALT_ANY)) {
pIndex += 10;
}
PathCat(the_path, gApplication_path, "ACCESSRY.TXT");
f = DRfopen(the_path, "rt");
for (i = 0; i <= pIndex; i++) {
GetAString(f, s);
} else {
s[0] = '\0';
}
}
if (s[0] != '\0') {
gLast_actor = LoadActor(s);
if (gLast_actor != NULL && gLast_actor->model != NULL) {
BrVector3Set(&kev_bounds.original_bounds.min, -.05f, -.05f, -.05f);
BrVector3Set(&kev_bounds.original_bounds.max, .05f, .05f, .05f);
kev_bounds.mat = &gProgram_state.current_car.car_master_actor->t.t.mat;
do {
face_count = FindFacesInBox(&kev_bounds, the_list, COUNT_OF(the_list));
BrVector3Scale(&kev_bounds.original_bounds.min, &kev_bounds.original_bounds.min, 2.f);
BrVector3Scale(&kev_bounds.original_bounds.max, &kev_bounds.original_bounds.max, 2.f);
} while (face_count == 0);
nearest_bastard = FLT_MAX;
face_bastard = -1;
for (i = 0; i < face_count; i++) {
distance_bastard = DistanceFromFace(gOur_pos, &the_list[i]);
if (distance_bastard < nearest_bastard) {
nearest_bastard = distance_bastard;
face_bastard = i;
}
}
if (face_bastard >= 0) {
BrVector3Scale(&gLast_actor->t.t.translate.t, &the_list[face_bastard].normal, -nearest_bastard);
BrVector3Accumulate(&gLast_actor->t.t.translate.t, gOur_pos);
if (!PDKeyDown(KEY_SHIFT_ANY)) {
if (the_list[face_bastard].normal.v[1] > the_list[face_bastard].normal.v[0] && the_list[face_bastard].normal.v[2] > the_list[face_bastard].normal.v[0]) {
BrVector3Set(&side_vector, -1.f, 0.f, 0.f);
} else if (the_list[face_bastard].normal.v[0] > the_list[face_bastard].normal.v[1] && the_list[face_bastard].normal.v[2] > the_list[face_bastard].normal.v[1]) {
BrVector3Set(&side_vector, 0.f, -1.f, 0.f);
} else {
BrVector3Set(&side_vector, 0.f, 0.f, -1.f);
}
new_transform.type = BR_TRANSFORM_LOOK_UP;
BrVector3Cross(&new_transform.t.look_up.look, &the_list[face_bastard].normal, &side_vector);
BrVector3Copy(&new_transform.t.look_up.up, &the_list[face_bastard].normal);
BrVector3Copy(&new_transform.t.look_up.t, &gLast_actor->t.t.translate.t);
BrTransformToTransform(&gLast_actor->t, &new_transform);
}
gKnown_actor = gLast_actor;
BrVector3Copy(&gActor_centre, &gLast_actor->t.t.translate.t);
DuplicateIfNotAmpersand(gLast_actor);
UniquificateActorsName(gUniverse_actor, gLast_actor);
gLast_actor->model->flags |= BR_MODF_UPDATEABLE;
if (gLast_actor->identifier == NULL || gLast_actor->identifier[0] == '&') {
last_non_ampersand = gAdditional_actors;
for (a = gAdditional_actors->children; a != NULL; a = a->next) {
if (a->identifier != NULL && a->identifier[0] != '&') {
last_non_ampersand = a;
}
}
BrActorAdd(last_non_ampersand, gLast_actor);
} else {
BrActorAdd(gAdditional_actors, gLast_actor);
}
SaveAdditionalStuff();
AccessoryHeadup(gLast_actor, "Shat out ");
}
}
}
}
// IDA: void __cdecl DropActor0()
void DropActor0(void) {
LOG_TRACE("()");
DropActor(0);
}
// IDA: void __cdecl DropActor1()
void DropActor1(void) {
LOG_TRACE("()");
DropActor(1);
}
// IDA: void __cdecl DropActor2()
void DropActor2(void) {
LOG_TRACE("()");
DropActor(2);
}
// IDA: void __cdecl DropActor3()
void DropActor3(void) {
LOG_TRACE("()");
DropActor(3);
}
// IDA: void __cdecl DropActor4()
void DropActor4(void) {
LOG_TRACE("()");
DropActor(4);
}
// IDA: void __cdecl DropActor5()
void DropActor5(void) {
LOG_TRACE("()");
DropActor(5);
}
// IDA: void __cdecl DropActor6()
void DropActor6(void) {
LOG_TRACE("()");
DropActor(6);
}
// IDA: void __cdecl DropActor7()
void DropActor7(void) {
LOG_TRACE("()");
DropActor(7);
}
// IDA: void __cdecl DropActor8()
void DropActor8(void) {
LOG_TRACE("()");
DropActor(8);
}
// IDA: void __cdecl DropActor9()
void DropActor9(void) {
LOG_TRACE("()");
DropActor(9);
}
// IDA: br_uint_32 __cdecl IdentifyAccCB(br_actor *pActor, void *pArg)
br_uint_32 IdentifyAccCB(br_actor* pActor, void* pArg) {
br_scalar distance;
//char s[256]; // Pierre-Marie Baty -- unused variable
br_vector3 v;
LOG_TRACE("(%p, %p)", pActor, pArg);
if (pActor == NULL || pActor->model == NULL) {
return 0;
}
BrVector3Add(&v, &pActor->model->bounds.max, &pActor->model->bounds.min);
BrVector3InvScale(&v, &v, 2.f);
BrVector3Accumulate(&v, &pActor->t.t.translate.t);
BrVector3Sub(&v, &v, gOur_pos);
distance = BrVector3LengthSquared(&v);
if (distance < gNearest_distance) {
gNearest_actor = pActor;
gNearest_distance = distance;
}
return 0;
}
// IDA: void __cdecl IdentifyAcc()
void IdentifyAcc(void) {
LOG_TRACE("()");
gNearest_distance = FLT_MAX;
gNearest_actor = NULL;
DRActorEnumRecurse(gAdditional_actors, (br_actor_enum_cbfn*)IdentifyAccCB, NULL);
if (gNearest_actor != NULL) {
gLast_actor = gNearest_actor;
AccessoryHeadup(gNearest_actor, "Locked onto ");
}
}
// IDA: br_uint_32 __cdecl DelGrooveRef(br_actor *pActor, void *pArg)
br_uint_32 DelGrooveRef(br_actor* pActor, void* pArg) {
tGroovidelic_spec* the_groove;
int i;
LOG_TRACE("(%p, %p)", pActor, pArg);
for (i = 0; i < gGroovidelics_array_size; i++) {
the_groove = &gGroovidelics_array[i];
if (the_groove->actor == pActor) {
the_groove->owner = -999;
}
}
return 0;
}
// IDA: br_uint_32 __cdecl DelReferencedModels(br_actor *pActor, void *pArg)
br_uint_32 DelReferencedModels(br_actor* pActor, void* pArg) {
//tGroovidelic_spec* the_groove; // Pierre-Marie Baty -- unused variable
int i;
LOG_TRACE("(%p, %p)", pActor, pArg);
for (i = 0; i < gNumber_of_additional_models; i++) {
if (pActor->model == gAdditional_models[i]) {
BrModelRemove(pActor->model);
BrModelFree(pActor->model);
memmove(&gAdditional_models
[i
], &gAdditional_models
[i
+ 1], (gNumber_of_additional_models
- i
- 1) * sizeof(br_model
*));
gNumber_of_additional_models--;
}
}
return 0;
}
// IDA: void __cdecl DeleteAcc()
void DeleteAcc(void) {
LOG_TRACE("()");
if (gLast_actor == NULL) {
return;
}
AccessoryHeadup(gLast_actor, "Murdered ");
DRActorEnumRecurse(gLast_actor, (br_actor_enum_cbfn*)DelReferencedModels, NULL);
DRActorEnumRecurse(gLast_actor, (br_actor_enum_cbfn*)DelGrooveRef, NULL);
BrActorRemove(gLast_actor);
BrActorFree(gLast_actor);
gLast_actor = NULL;
SaveAdditionalStuff();
}
// IDA: br_uint_32 __cdecl OffsetModel(br_actor *pActor, void *pArg)
br_uint_32 OffsetModel(br_actor* pActor, void* pArg) {
int i;
LOG_TRACE("(%p, %p)", pActor, pArg);
if (pActor->model == NULL) {
return 0;
}
for (i = 0; i < pActor->model->nvertices; i++) {
BrVector3Accumulate(&pActor->model->vertices[i].p, (br_vector3*)pArg);
}
return 0;
}
// IDA: void __usercall OffsetActor(br_actor *pActor@<EAX>, br_vector3 *pOffset@<EDX>)
void OffsetActor(br_actor* pActor, br_vector3* pOffset) {
LOG_TRACE("(%p, %p)", pActor, pOffset);
DRActorEnumRecurse(pActor, (br_actor_enum_cbfn*)OffsetModel, pOffset);
}
// IDA: void __usercall CentreActor(br_actor *pActor@<EAX>, br_vector3 *pOffset@<EDX>)
void CentreActor(br_actor* pActor, br_vector3* pOffset) {
LOG_TRACE("(%p, %p)", pActor, pOffset);
if (pActor->model == NULL) {
BrVector3Set(pOffset, 0.f, 0.f, 0.f);
} else if (gKnown_actor == gLast_actor) {
BrVector3Scale(pOffset, &gActor_centre, -1.f);
} else {
BrVector3Add(pOffset, &pActor->model->bounds.max, &pActor->model->bounds.min);
BrVector3Scale(pOffset, pOffset, -2.f);
}
DRActorEnumRecurse(pActor, (br_actor_enum_cbfn*)OffsetModel, pOffset);
BrVector3Scale(pOffset, pOffset, -1.f);
}
// IDA: void __cdecl SnapAccToVertical()
void SnapAccToVertical(void) {
LOG_TRACE("()");
if (gLast_actor == NULL) {
return;
}
BrVector3Set((br_vector3*)gLast_actor->t.t.mat.m[0], 1.f, 0.f, 0.f);
BrVector3Set((br_vector3*)gLast_actor->t.t.mat.m[1], 0.f, 1.f, 0.f);
BrVector3Set((br_vector3*)gLast_actor->t.t.mat.m[2], 0.f, 0.f, 1.f);
SaveAdditionalStuff();
}
// IDA: void __usercall RotateAccessory(br_angle pAngle@<EAX>)
void RotateAccessory(br_angle pAngle) {
br_vector3 mr_offset;
LOG_TRACE("(%d)", pAngle);
if (gLast_actor == NULL) {
return;
}
if (!gSpec_vol_mode && gLast_actor->identifier != NULL && gLast_actor->identifier[0] == '@') {
CentreActor(gLast_actor, &mr_offset);
}
switch (gCurrent_rotate_mode) {
case eRotate_mode_x:
BrMatrix34PreRotateX(&gLast_actor->t.t.mat, pAngle);
break;
case eRotate_mode_y:
BrMatrix34PreRotateY(&gLast_actor->t.t.mat, pAngle);
break;
case eRotate_mode_z:
BrMatrix34PreRotateZ(&gLast_actor->t.t.mat, pAngle);
break;
}
if (!gSpec_vol_mode && gLast_actor->identifier != NULL && gLast_actor->identifier[0] == '@') {
DRActorEnumRecurseWithTrans(gLast_actor, NULL, ApplyTransToModels, NULL);
OffsetActor(gLast_actor, &mr_offset);
}
SaveAdditionalStuff();
}
// IDA: void __cdecl ScaleAccessory(float pScaling_factor)
void ScaleAccessory(float pScaling_factor) {
br_vector3 mr_offset;
LOG_TRACE("(%f)", pScaling_factor);
if (gLast_actor == NULL) {
return;
}
if (!gSpec_vol_mode && gLast_actor->identifier != NULL && gLast_actor->identifier[0] == '@') {
CentreActor(gLast_actor, &mr_offset);
}
switch (gCurrent_scale_mode) {
case eScale_mode_all:
BrMatrix34PreScale(&gLast_actor->t.t.mat, pScaling_factor, pScaling_factor, pScaling_factor);
break;
case eScale_mode_x:
BrMatrix34PreScale(&gLast_actor->t.t.mat, pScaling_factor, 1.f, 1.f);
break;
case eScale_mode_y:
BrMatrix34PreScale(&gLast_actor->t.t.mat, 1.f, pScaling_factor, 1.f);
break;
case eScale_mode_z:
BrMatrix34PreScale(&gLast_actor->t.t.mat, 1.f, 1.f, pScaling_factor);
break;
}
if (!gSpec_vol_mode && gLast_actor->identifier != NULL && gLast_actor->identifier[0] == '@') {
DRActorEnumRecurseWithTrans(gLast_actor, NULL, ApplyTransToModels, NULL);
OffsetActor(gLast_actor, &mr_offset);
}
SaveAdditionalStuff();
}
// IDA: void __cdecl MoveAccessory(br_scalar pX_shift, br_scalar pY_shift, br_scalar pZ_shift)
void MoveAccessory(br_scalar pX_shift, br_scalar pY_shift, br_scalar pZ_shift) {
br_vector3 v;
LOG_TRACE("(%f, %f, %f)", pX_shift, pY_shift, pZ_shift);
if (gLast_actor == NULL) {
return;
}
BrVector3SetFloat(&v, pX_shift, pY_shift, pZ_shift);
BrVector3Accumulate(&gLast_actor->t.t.translate.t, &v);
SaveAdditionalStuff();
}
// IDA: void __cdecl RotateAccL()
void RotateAccL(void) {
LOG_TRACE("()");
RotateAccessory(BrDegreeToAngle(90));
}
// IDA: void __cdecl RotateAccL2()
void RotateAccL2(void) {
LOG_TRACE("()");
RotateAccessory(BrDegreeToAngle(15));
}
// IDA: void __cdecl RotateAccL3()
void RotateAccL3(void) {
LOG_TRACE("()");
RotateAccessory(BrDegreeToAngle(5));
}
// IDA: void __cdecl RotateAccL4()
void RotateAccL4(void) {
LOG_TRACE("()");
RotateAccessory(BrDegreeToAngle(1));
}
// IDA: void __cdecl RotateAccR()
void RotateAccR(void) {
LOG_TRACE("()");
RotateAccessory(BrDegreeToAngle(270));
}
// IDA: void __cdecl RotateAccR2()
void RotateAccR2(void) {
LOG_TRACE("()");
RotateAccessory(BrDegreeToAngle(345));
}
// IDA: void __cdecl RotateAccR3()
void RotateAccR3(void) {
LOG_TRACE("()");
RotateAccessory(BrDegreeToAngle(355));
}
// IDA: void __cdecl RotateAccR4()
void RotateAccR4(void) {
LOG_TRACE("()");
RotateAccessory(BrDegreeToAngle(359));
}
// IDA: void __cdecl CycleAccRotate()
void CycleAccRotate(void) {
LOG_TRACE("()");
gCurrent_rotate_mode = (gCurrent_rotate_mode == eRotate_mode_z) ? eRotate_mode_x : (gCurrent_rotate_mode + 1);
switch (gCurrent_rotate_mode) {
case eRotate_mode_x:
NewTextHeadupSlot(4, 0, 2000, -2, "Rotate mode: X");
break;
case eRotate_mode_y:
NewTextHeadupSlot(4, 0, 2000, -2, "Rotate mode: Y");
break;
case eRotate_mode_z:
NewTextHeadupSlot(4, 0, 2000, -2, "Rotate mode: Z");
break;
}
}
// IDA: void __cdecl CycleAccScale()
void CycleAccScale(void) {
LOG_TRACE("()");
gCurrent_scale_mode = (gCurrent_scale_mode == eScale_mode_z) ? eScale_mode_all : (gCurrent_scale_mode + 1);
switch (gCurrent_scale_mode) {
case eScale_mode_all:
NewTextHeadupSlot(4, 0, 2000, -2, "Scale mode: ALL");
break;
case eScale_mode_x:
NewTextHeadupSlot(4, 0, 2000, -2, "Scale mode: X");
break;
case eScale_mode_y:
NewTextHeadupSlot(4, 0, 2000, -2, "Scale mode: Y");
break;
case eScale_mode_z:
NewTextHeadupSlot(4, 0, 2000, -2, "Scale mode: Z");
break;
}
}
// IDA: void __cdecl ScaleAccUp2()
void ScaleAccUp2(void) {
LOG_TRACE("()");
ScaleAccessory(1.2f);
}
// IDA: void __cdecl ScaleAccUp3()
void ScaleAccUp3(void) {
LOG_TRACE("()");
ScaleAccessory(1.05f);
}
// IDA: void __cdecl ScaleAccUp4()
void ScaleAccUp4(void) {
LOG_TRACE("()");
ScaleAccessory(1.002f);
}
// IDA: void __cdecl ScaleAccDown2()
void ScaleAccDown2(void) {
LOG_TRACE("()");
ScaleAccessory(1 / 1.2f);
}
// IDA: void __cdecl ScaleAccDown3()
void ScaleAccDown3(void) {
LOG_TRACE("()");
ScaleAccessory(1 / 1.05f);
}
// IDA: void __cdecl ScaleAccDown4()
void ScaleAccDown4(void) {
LOG_TRACE("()");
ScaleAccessory(1 / 1.002f);
}
// IDA: void __cdecl MoveXAccL()
void MoveXAccL(void) {
LOG_TRACE("()");
MoveAccessory(1.f, 0.f, 0.f);
}
// IDA: void __cdecl MoveXAccL2()
void MoveXAccL2(void) {
LOG_TRACE("()");
MoveAccessory(.2f, 0.f, 0.f);
}
// IDA: void __cdecl MoveXAccL3()
void MoveXAccL3(void) {
LOG_TRACE("()");
MoveAccessory(.02f, 0.f, 0.f);
}
// IDA: void __cdecl MoveXAccL4()
void MoveXAccL4(void) {
LOG_TRACE("()");
MoveAccessory(.002f, 0.f, 0.f);
}
// IDA: void __cdecl MoveXAccR()
void MoveXAccR(void) {
LOG_TRACE("()");
MoveAccessory(-1.f, 0.f, 0.f);
}
// IDA: void __cdecl MoveXAccR2()
void MoveXAccR2(void) {
LOG_TRACE("()");
MoveAccessory(-.2f, 0.f, 0.f);
}
// IDA: void __cdecl MoveXAccR3()
void MoveXAccR3(void) {
LOG_TRACE("()");
MoveAccessory(-.02f, 0.f, 0.f);
}
// IDA: void __cdecl MoveXAccR4()
void MoveXAccR4(void) {
LOG_TRACE("()");
MoveAccessory(-.002f, 0.f, 0.f);
}
// IDA: void __cdecl MoveYAccL()
void MoveYAccL(void) {
LOG_TRACE("()");
MoveAccessory(0.f, 1.f, 0.f);
}
// IDA: void __cdecl MoveYAccL2()
void MoveYAccL2(void) {
LOG_TRACE("()");
MoveAccessory(0.f, .2f, 0.f);
}
// IDA: void __cdecl MoveYAccL3()
void MoveYAccL3(void) {
LOG_TRACE("()");
MoveAccessory(0.f, .02f, 0.f);
}
// IDA: void __cdecl MoveYAccL4()
void MoveYAccL4(void) {
LOG_TRACE("()");
MoveAccessory(0.f, .002f, 0.f);
}
// IDA: void __cdecl MoveYAccR()
void MoveYAccR(void) {
LOG_TRACE("()");
MoveAccessory(0.f, -1.f, 0.f);
}
// IDA: void __cdecl MoveYAccR2()
void MoveYAccR2(void) {
LOG_TRACE("()");
MoveAccessory(0.f, -.2f, 0.f);
}
// IDA: void __cdecl MoveYAccR3()
void MoveYAccR3(void) {
LOG_TRACE("()");
MoveAccessory(0.f, -.02f, 0.f);
}
// IDA: void __cdecl MoveYAccR4()
void MoveYAccR4(void) {
LOG_TRACE("()");
MoveAccessory(0.f, -.002f, 0.f);
}
// IDA: void __cdecl MoveZAccL()
void MoveZAccL(void) {
LOG_TRACE("()");
MoveAccessory(0.f, 0.f, 1.f);
}
// IDA: void __cdecl MoveZAccL2()
void MoveZAccL2(void) {
LOG_TRACE("()");
MoveAccessory(0.f, 0.f, .2f);
}
// IDA: void __cdecl MoveZAccL3()
void MoveZAccL3(void) {
LOG_TRACE("()");
MoveAccessory(0.f, 0.f, .02f);
}
// IDA: void __cdecl MoveZAccL4()
void MoveZAccL4(void) {
LOG_TRACE("()");
MoveAccessory(0.f, 0.f, .002f);
}
// IDA: void __cdecl MoveZAccR()
void MoveZAccR(void) {
LOG_TRACE("()");
MoveAccessory(0.f, 0.f, -1.f);
}
// IDA: void __cdecl MoveZAccR2()
void MoveZAccR2(void) {
LOG_TRACE("()");
MoveAccessory(0.f, 0.f, -.2f);
}
// IDA: void __cdecl MoveZAccR3()
void MoveZAccR3(void) {
LOG_TRACE("()");
MoveAccessory(0.f, 0.f, -.02f);
}
// IDA: void __cdecl MoveZAccR4()
void MoveZAccR4(void) {
LOG_TRACE("()");
MoveAccessory(0.f, 0.f, -.002f);
}
// IDA: br_material* __cdecl GetInternalMat()
br_material* GetInternalMat(void) {
LOG_TRACE("()");
return BrMaterialFind("SPECVOL.MAT");
}
// IDA: br_material* __cdecl GetExternalMat()
br_material* GetExternalMat(void) {
LOG_TRACE("()");
return BrMaterialFind("SPECVOL2.MAT");
}
#define DrVertexSet(V, X, Y, Z) \
do { \
(V)[0] = (X); \
(V)[1] = (Y); \
(V)[2] = (Z); \
} while (0)
// IDA: void __usercall BuildSpecVolModel(tSpecial_volume *pSpec@<EAX>, int pIndex@<EDX>, br_material *pInt_mat@<EBX>, br_material *pExt_mat@<ECX>)
void BuildSpecVolModel(tSpecial_volume* pSpec, int pIndex, br_material* pInt_mat, br_material* pExt_mat) {
int i;
//int j; // Pierre-Marie Baty -- unused variable
int temp;
//tSpecial_volume* v; // Pierre-Marie Baty -- unused variable
br_actor* actor;
br_model* model;
LOG_TRACE("(%p, %d, %p, %p)", pSpec, pIndex, pInt_mat, pExt_mat);
actor = BrActorAllocate(BR_ACTOR_MODEL, NULL);
BrMatrix34Copy(&actor->t.t.mat, &pSpec->mat);
actor->render_style = BR_RSTYLE_FACES;
model = BrModelAllocate("", 12, 24);
model->flags |= BR_MODF_KEEP_ORIGINAL | BR_MODF_DONT_WELD;
BrVector3Set(&model->vertices[0].p, 1.f, -1.f, -1.f);
BrVector2Set(&model->vertices[0].map, 1.f, 0.f);
BrVector3Set(&model->vertices[1].p, -1.f, -1.f, -1.f);
BrVector2Set(&model->vertices[1].map, 0.f, 0.f);
BrVector3Set(&model->vertices[2].p, -1.f, 1.f, -1.f);
BrVector2Set(&model->vertices[2].map, 0.f, 1.f);
BrVector3Set(&model->vertices[3].p, 1.f, 1.f, -1.f);
BrVector2Set(&model->vertices[3].map, 1.f, 1.f);
BrVector3Set(&model->vertices[4].p, 1.f, -1.f, 1.f);
BrVector2Set(&model->vertices[4].map, 1.f, 1.f);
BrVector3Set(&model->vertices[5].p, -1.f, -1.f, 1.f);
BrVector2Set(&model->vertices[5].map, 0.f, 1.f);
BrVector3Set(&model->vertices[6].p, -1.f, 1.f, 1.f);
BrVector2Set(&model->vertices[6].map, 0.f, 0.f);
BrVector3Set(&model->vertices[7].p, 1.f, 1.f, 1.f);
BrVector2Set(&model->vertices[7].map, 1.f, 0.f);
BrVector3Set(&model->vertices[8].p, 1.f, -1.f, -1.f);
BrVector2Set(&model->vertices[8].map, 0.f, 1.f);
BrVector3Set(&model->vertices[9].p, 1.f, -1.f, 1.f);
BrVector2Set(&model->vertices[9].map, 0.f, 0.f);
BrVector3Set(&model->vertices[10].p, -1.f, -1.f, -1.f);
BrVector2Set(&model->vertices[10].map, 1.f, 1.f);
BrVector3Set(&model->vertices[11].p, -1.f, -1.f, 1.f);
BrVector2Set(&model->vertices[11].map, 1.f, 0.f);
DrVertexSet(model->faces[0].vertices, 0, 3, 1);
DrVertexSet(model->faces[1].vertices, 1, 3, 2);
DrVertexSet(model->faces[2].vertices, 3, 7, 2);
DrVertexSet(model->faces[3].vertices, 2, 7, 6);
DrVertexSet(model->faces[4].vertices, 6, 7, 4);
DrVertexSet(model->faces[5].vertices, 6, 4, 5);
DrVertexSet(model->faces[6].vertices, 0, 5, 4);
DrVertexSet(model->faces[7].vertices, 1, 5, 0);
DrVertexSet(model->faces[8].vertices, 9, 7, 8);
DrVertexSet(model->faces[9].vertices, 8, 7, 3);
DrVertexSet(model->faces[10].vertices, 11, 2, 6);
DrVertexSet(model->faces[11].vertices, 11, 10, 2);
memcpy(&model
->faces
[12], model
->faces
, 12 * sizeof(br_face
));
for (i = 12; i < 24; i++) {
temp = model->faces[i].vertices[0];
model->faces[i].vertices[0] = model->faces[i].vertices[1];
model->faces[i].vertices[1] = temp;
}
model->faces[5].material = model->faces[4].material = model->faces[1].material = model->faces[0].material = DRMaterialClone(pExt_mat);
model->faces[11].material = model->faces[10].material = model->faces[9].material = model->faces[8].material = DRMaterialClone(pExt_mat);
model->faces[7].material = model->faces[6].material = model->faces[3].material = model->faces[2].material = DRMaterialClone(pExt_mat);
model->faces[17].material = model->faces[16].material = model->faces[13].material = model->faces[12].material = DRMaterialClone(pInt_mat);
model->faces[23].material = model->faces[22].material = model->faces[21].material = model->faces[20].material = DRMaterialClone(pInt_mat);
model->faces[19].material = model->faces[18].material = model->faces[15].material = model->faces[14].material = DRMaterialClone(pInt_mat);
BrModelUpdate(model, BR_MODU_ALL);
actor->model = model;
BrActorAdd(gNon_track_actor, actor);
gSpec_vol_actors[pIndex] = actor;
SetSpecVolMatSize(actor);
}
#undef DrVertexSet
// IDA: void __usercall DropSpecVol(int pIndex@<EAX>)
void DropSpecVol(int pIndex) {
FILE* f;
tPath_name the_path;
int i;
int spec_count;
tSpecial_volume spec;
tSpecial_volume* new_specs;
char s[256];
LOG_TRACE("(%d)", pIndex);
PathCat(the_path, gApplication_path, "SPECVOL.TXT");
f = DRfopen(the_path, "rt");
if (f == NULL) {
return;
}
spec_count = GetAnInt(f);
// pIndex = 1 means first special volume
if (pIndex > spec_count) {
return;
}
for (i = 0; i < pIndex; i++) {
ParseSpecialVolume(f, &spec, NULL);
}
spec.no_mat = 0;
BrMatrix34Copy(&spec.mat, &gProgram_state.current_car.car_master_actor->t.t.mat);
new_specs = BrMemAllocate((gProgram_state.special_volume_count + 1) * sizeof(tSpecial_volume), kMem_new_special_vol);
memcpy(new_specs
, gProgram_state.
special_volumes, gProgram_state.
special_volume_count * sizeof(tSpecial_volume
));
memcpy(&new_specs
[gProgram_state.
special_volume_count], &spec
, sizeof(tSpecial_volume
));
gProgram_state.special_volume_count++;
BrMemFree(gProgram_state.special_volumes);
gProgram_state.special_volumes = new_specs;
BuildSpecVolModel(&spec, gProgram_state.special_volume_count - 1, GetInternalMat(), GetExternalMat());
gLast_actor = gSpec_vol_actors[gProgram_state.special_volume_count - 1];
UpdateSpecVol();
sprintf(s
, "Shat out special volume #%d (type %d)", gProgram_state.
special_volume_count - 1, pIndex
);
NewTextHeadupSlot(4, 0, 2000, -2, s);
SaveSpecialVolumes();
}
// IDA: void __cdecl DropSpecVol0()
void DropSpecVol0(void) {
LOG_TRACE("()");
}
// IDA: void __cdecl DropSpecVol1()
void DropSpecVol1(void) {
LOG_TRACE("()");
DropSpecVol(1);
}
// IDA: void __cdecl DropSpecVol2()
void DropSpecVol2(void) {
LOG_TRACE("()");
DropSpecVol(2);
}
// IDA: void __cdecl DropSpecVol3()
void DropSpecVol3(void) {
LOG_TRACE("()");
DropSpecVol(3);
}
// IDA: void __cdecl DropSpecVol4()
void DropSpecVol4(void) {
LOG_TRACE("()");
DropSpecVol(4);
}
// IDA: void __cdecl DropSpecVol5()
void DropSpecVol5(void) {
LOG_TRACE("()");
DropSpecVol(5);
}
// IDA: void __cdecl DropSpecVol6()
void DropSpecVol6(void) {
LOG_TRACE("()");
DropSpecVol(6);
}
// IDA: void __cdecl DropSpecVol7()
void DropSpecVol7(void) {
LOG_TRACE("()");
DropSpecVol(7);
}
// IDA: void __cdecl DropSpecVol8()
void DropSpecVol8(void) {
LOG_TRACE("()");
DropSpecVol(8);
}
// IDA: void __cdecl DropSpecVol9()
void DropSpecVol9(void) {
LOG_TRACE("()");
DropSpecVol(9);
}
// IDA: void __cdecl IdentifySpecVol()
void IdentifySpecVol(void) {
int i;
int min_index;
tSpecial_volume* v;
br_scalar min_d;
br_scalar d;
br_vector3* p;
char s[256];
LOG_TRACE("()");
min_d = FLT_MAX;
min_index = -1;
p = &gProgram_state.current_car.pos;
for (i = 0; i < gProgram_state.special_volume_count; i++) {
v = &gProgram_state.special_volumes[i];
d = Vector3DistanceSquared((br_vector3*)v->mat.m[3], p);
if (d < min_d) {
min_index = i;
min_d = d;
}
}
if (min_index < 0) {
gLast_actor = NULL;
NewTextHeadupSlot(4, 0, 2000, -2, "No special volumes to lock onto");
} else {
sprintf(s
, "Locked onto Special Volume #%d", min_index
);
NewTextHeadupSlot(4, 0, 2000, -2, s);
gLast_actor = gSpec_vol_actors[min_index];
}
}
// IDA: void __usercall DelSpecVolumeGraph(int pIndex@<EAX>)
void DelSpecVolumeGraph(int pIndex) {
br_actor* actor;
br_model* model;
LOG_TRACE("(%d)", pIndex);
actor = gSpec_vol_actors[pIndex];
model = actor->model;
BrMaterialRemove(model->faces[5].material);
BrMaterialRemove(model->faces[11].material);
BrMaterialRemove(model->faces[7].material);
BrMaterialRemove(model->faces[17].material);
BrMaterialRemove(model->faces[23].material);
BrMaterialRemove(model->faces[19].material);
BrMaterialFree(model->faces[5].material);
BrMaterialFree(model->faces[11].material);
BrMaterialFree(model->faces[7].material);
BrMaterialFree(model->faces[17].material);
BrMaterialFree(model->faces[23].material);
BrMaterialFree(model->faces[19].material);
BrModelRemove(model);
BrModelFree(model);
BrActorRemove(actor);
BrActorFree(actor);
}
// IDA: void __cdecl DeleteSpecVol()
void DeleteSpecVol(void) {
int index;
LOG_TRACE("()");
index = FindSpecVolIndex(gLast_actor);
if (index < 0) {
return;
}
DelSpecVolumeGraph(index);
memmove(&gProgram_state.
special_volumes[index
], &gProgram_state.
special_volumes[index
+ 1], (gProgram_state.
special_volume_count - index
- 1) * sizeof(tSpecial_volume
));
memmove(&gSpec_vol_actors
[index
], &gSpec_vol_actors
[index
+ 1], (gProgram_state.
special_volume_count - index
- 1) * sizeof(br_actor
*));
gProgram_state.special_volume_count--;
NewTextHeadupSlot(4, 0, 2000, -2, "There's been a special volumes MURDER!!");
gLast_actor = NULL;
if (&gProgram_state.special_volumes[index] < gDefault_water_spec_vol) {
gDefault_water_spec_vol--;
}
SaveSpecialVolumes();
}
// IDA: void __cdecl RotateSpecVolL()
void RotateSpecVolL(void) {
LOG_TRACE("()");
RotateAccL();
}
// IDA: void __cdecl RotateSpecVolL2()
void RotateSpecVolL2(void) {
LOG_TRACE("()");
RotateAccL2();
}
// IDA: void __cdecl RotateSpecVolL3()
void RotateSpecVolL3(void) {
LOG_TRACE("()");
RotateAccL3();
}
// IDA: void __cdecl RotateSpecVolL4()
void RotateSpecVolL4(void) {
LOG_TRACE("()");
RotateAccL4();
}
// IDA: void __cdecl RotateSpecVolR()
void RotateSpecVolR(void) {
LOG_TRACE("()");
RotateAccR();
}
// IDA: void __cdecl RotateSpecVolR2()
void RotateSpecVolR2(void) {
LOG_TRACE("()");
RotateAccR2();
}
// IDA: void __cdecl RotateSpecVolR3()
void RotateSpecVolR3(void) {
LOG_TRACE("()");
RotateAccR3();
}
// IDA: void __cdecl RotateSpecVolR4()
void RotateSpecVolR4(void) {
LOG_TRACE("()");
RotateAccR4();
}
// IDA: void __cdecl CycleSpecVolRotate()
void CycleSpecVolRotate(void) {
LOG_TRACE("()");
CycleAccRotate();
}
// IDA: void __cdecl CycleSpecVolScale()
void CycleSpecVolScale(void) {
LOG_TRACE("()");
CycleAccScale();
}
// IDA: void __cdecl ScaleSpecVolUp2()
void ScaleSpecVolUp2(void) {
LOG_TRACE("()");
ScaleAccUp2();
}
// IDA: void __cdecl ScaleSpecVolUp3()
void ScaleSpecVolUp3(void) {
LOG_TRACE("()");
ScaleAccUp3();
}
// IDA: void __cdecl ScaleSpecVolUp4()
void ScaleSpecVolUp4(void) {
LOG_TRACE("()");
ScaleAccUp4();
}
// IDA: void __cdecl ScaleSpecVolDown2()
void ScaleSpecVolDown2(void) {
LOG_TRACE("()");
ScaleAccDown2();
}
// IDA: void __cdecl ScaleSpecVolDown3()
void ScaleSpecVolDown3(void) {
LOG_TRACE("()");
ScaleAccDown3();
}
// IDA: void __cdecl ScaleSpecVolDown4()
void ScaleSpecVolDown4(void) {
LOG_TRACE("()");
ScaleAccDown4();
}
// IDA: void __cdecl MoveXSpecVolL()
void MoveXSpecVolL(void) {
LOG_TRACE("()");
MoveXAccL();
}
// IDA: void __cdecl MoveXSpecVolL2()
void MoveXSpecVolL2(void) {
LOG_TRACE("()");
MoveXAccL2();
}
// IDA: void __cdecl MoveXSpecVolL3()
void MoveXSpecVolL3(void) {
LOG_TRACE("()");
MoveXAccL3();
}
// IDA: void __cdecl MoveXSpecVolL4()
void MoveXSpecVolL4(void) {
LOG_TRACE("()");
MoveXAccL4();
}
// IDA: void __cdecl MoveXSpecVolR()
void MoveXSpecVolR(void) {
LOG_TRACE("()");
MoveXAccR();
}
// IDA: void __cdecl MoveXSpecVolR2()
void MoveXSpecVolR2(void) {
LOG_TRACE("()");
MoveXAccR2();
}
// IDA: void __cdecl MoveXSpecVolR3()
void MoveXSpecVolR3(void) {
LOG_TRACE("()");
MoveXAccR3();
}
// IDA: void __cdecl MoveXSpecVolR4()
void MoveXSpecVolR4(void) {
LOG_TRACE("()");
MoveXAccR4();
}
// IDA: void __cdecl MoveYSpecVolL()
void MoveYSpecVolL(void) {
LOG_TRACE("()");
MoveYAccL();
}
// IDA: void __cdecl MoveYSpecVolL2()
void MoveYSpecVolL2(void) {
LOG_TRACE("()");
#if defined(DETHRACE_FIX_BUGS)
MoveYAccL2();
#else
MoveYAccL3();
#endif
}
// IDA: void __cdecl MoveYSpecVolL3()
void MoveYSpecVolL3(void) {
LOG_TRACE("()");
MoveYAccL3();
}
// IDA: void __cdecl MoveYSpecVolL4()
void MoveYSpecVolL4(void) {
LOG_TRACE("()");
MoveYAccL4();
}
// IDA: void __cdecl MoveYSpecVolR()
void MoveYSpecVolR(void) {
LOG_TRACE("()");
MoveYAccR();
}
// IDA: void __cdecl MoveYSpecVolR2()
void MoveYSpecVolR2(void) {
LOG_TRACE("()");
MoveYAccR2();
}
// IDA: void __cdecl MoveYSpecVolR3()
void MoveYSpecVolR3(void) {
LOG_TRACE("()");
MoveYAccR3();
}
// IDA: void __cdecl MoveYSpecVolR4()
void MoveYSpecVolR4(void) {
LOG_TRACE("()");
MoveYAccR4();
}
// IDA: void __cdecl MoveZSpecVolL()
void MoveZSpecVolL(void) {
LOG_TRACE("()");
MoveZAccL();
}
// IDA: void __cdecl MoveZSpecVolL2()
void MoveZSpecVolL2(void) {
LOG_TRACE("()");
MoveZAccL2();
}
// IDA: void __cdecl MoveZSpecVolL3()
void MoveZSpecVolL3(void) {
LOG_TRACE("()");
MoveZAccL3();
}
// IDA: void __cdecl MoveZSpecVolL4()
void MoveZSpecVolL4(void) {
LOG_TRACE("()");
MoveZAccL4();
}
// IDA: void __cdecl MoveZSpecVolR()
void MoveZSpecVolR(void) {
LOG_TRACE("()");
MoveZAccR();
}
// IDA: void __cdecl MoveZSpecVolR2()
void MoveZSpecVolR2(void) {
LOG_TRACE("()");
MoveZAccR2();
}
// IDA: void __cdecl MoveZSpecVolR3()
void MoveZSpecVolR3(void) {
LOG_TRACE("()");
MoveZAccR3();
}
// IDA: void __cdecl MoveZSpecVolR4()
void MoveZSpecVolR4(void) {
LOG_TRACE("()");
MoveZAccR4();
}
// IDA: void __cdecl SnapSpecVolToVertical()
void SnapSpecVolToVertical(void) {
LOG_TRACE("()");
SnapAccToVertical();
}
// IDA: void __cdecl ShowSpecialVolumes()
void ShowSpecialVolumes(void) {
int i;
//int j; // Pierre-Marie Baty -- unused variable
//int temp; // Pierre-Marie Baty -- unused variable
tSpecial_volume* v;
//br_actor* actor; // Pierre-Marie Baty -- unused variable
//br_model* model; // Pierre-Marie Baty -- unused variable
br_material* internal_mat;
br_material* external_mat;
LOG_TRACE("()");
gLast_actor = NULL;
gSpec_vol_mode = 1;
internal_mat = GetInternalMat();
external_mat = GetExternalMat();
for (i = 0; i < gProgram_state.special_volume_count; i++) {
v = &gProgram_state.special_volumes[i];
if (!v->no_mat) {
BuildSpecVolModel(v, i, internal_mat, external_mat);
} else {
gSpec_vol_actors[i] = NULL;
}
}
}
// IDA: void __cdecl HideSpecialVolumes()
void HideSpecialVolumes(void) {
int i;
tSpecial_volume* v;
LOG_TRACE("()");
gSpec_vol_mode = 0;
for (i = 0; i < gProgram_state.special_volume_count; i++) {
v = &gProgram_state.special_volumes[i];
if (!v->no_mat) {
DelSpecVolumeGraph(i);
}
}
}