Subversion Repositories Games.Carmageddon

Rev

Rev 18 | Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | Download | RSS feed

  1. #include "crush.h"
  2. #include "brender/brender.h"
  3. #include "car.h"
  4. #include "displays.h"
  5. #include "globvars.h"
  6. #include "globvrkm.h"
  7. #include "globvrpb.h"
  8. #include "graphics.h"
  9. #include "harness/trace.h"
  10. #include "loading.h"
  11. #include "mainloop.h"
  12. #include "netgame.h"
  13. #include "network.h"
  14. #include "oil.h"
  15. #include "opponent.h"
  16. #include "pd/sys.h"
  17. #include "pedestrn.h"
  18. #include "piping.h"
  19. #include "pratcam.h"
  20. #include "raycast.h"
  21. #include "replay.h"
  22. #include "spark.h"
  23. #include "structur.h"
  24. #include "utility.h"
  25. #include "world.h"
  26. #include <stdlib.h>
  27.  
  28. float gWobble_spam_y[8] = { 0.0f, -0.15f, 0.4f, 0.15f, -0.4f, 0.25f, 0.0f, -0.25f };
  29. float gWobble_spam_z[8] = { 0.4f, -0.25f, 0.0f, 0.25f, 0.0f, 0.15f, -0.4f, -0.15f };
  30. br_scalar gWheel_circ_to_width = 0.16f;
  31. tU8 gSmoke_damage_step[12] = { 20u, 20u, 0u, 10u, 10u, 10u, 10u, 10u, 10u, 10u, 10u, 10u };
  32. int gSteal_ranks[5] = { 89, 72, 55, 38, 21 };
  33.  
  34. #define BIGAPC_OPPONENT_INDEX 4
  35.  
  36. // IDA: int __usercall ReadCrushData@<EAX>(FILE *pF@<EAX>, tCrush_data *pCrush_data@<EDX>)
  37. int ReadCrushData(FILE* pF, tCrush_data* pCrush_data) {
  38.     //char s[256]; // Pierre-Marie Baty -- unused variable
  39.     //char* str; // Pierre-Marie Baty -- unused variable
  40.     int i;
  41.     int j;
  42.     //int k; // Pierre-Marie Baty -- unused variable
  43.     tCrush_point_spec* the_spec;
  44.     tCrush_neighbour* the_neighbour;
  45.     LOG_TRACE("(%p, %p)", pF, pCrush_data);
  46.  
  47.     pCrush_data->softness_factor = GetAFloat(pF);
  48.     GetPairOfFloats(pF, &pCrush_data->min_fold_factor, &pCrush_data->max_fold_factor);
  49.     pCrush_data->wibble_factor = GetAFloat(pF);
  50.     pCrush_data->limit_deviant = GetAFloat(pF);
  51.     pCrush_data->split_chance = GetAFloat(pF);
  52.     pCrush_data->min_y_fold_down = GetAFloat(pF);
  53.     pCrush_data->number_of_crush_points = GetAnInt(pF);
  54.     pCrush_data->crush_points = (tCrush_point_spec*)BrMemAllocate(sizeof(tCrush_point_spec) * pCrush_data->number_of_crush_points, kMem_crush_data);
  55.  
  56.     for (i = 0, the_spec = pCrush_data->crush_points; i < pCrush_data->number_of_crush_points; i++, the_spec++) {
  57.         the_spec->vertex_index = GetAnInt(pF);
  58.         GetThreeFloats(pF, &the_spec->limits_neg.v[0], &the_spec->limits_neg.v[1], &the_spec->limits_neg.v[2]);
  59.         GetThreeFloats(pF, &the_spec->limits_pos.v[0], &the_spec->limits_pos.v[1], &the_spec->limits_pos.v[2]);
  60.         GetThreeFloats(pF, &the_spec->softness_neg.v[0], &the_spec->softness_neg.v[1], &the_spec->softness_neg.v[2]);
  61.         GetThreeFloats(pF, &the_spec->softness_pos.v[0], &the_spec->softness_pos.v[1], &the_spec->softness_pos.v[2]);
  62.         the_spec->number_of_neighbours = GetAnInt(pF);
  63.         the_spec->neighbours = BrMemAllocate(sizeof(tCrush_neighbour) * the_spec->number_of_neighbours, kMem_crush_neighbours);
  64.  
  65.         for (j = 0, the_neighbour = the_spec->neighbours; j < the_spec->number_of_neighbours; j++, the_neighbour++) {
  66.             the_neighbour->vertex_index = GetAnInt(pF);
  67.             the_neighbour->factor = GetAnInt(pF);
  68.         }
  69.     }
  70.     return 0;
  71. }
  72.  
  73. // IDA: float __usercall SkipCrushData@<ST0>(FILE *pF@<EAX>)
  74. float SkipCrushData(FILE* pF) {
  75.     int i;
  76.     int j;
  77.     int count_1;
  78.     int count_2;
  79.     char s[256];
  80.     float softness;
  81.     LOG_TRACE("(%p)", pF);
  82.  
  83.     softness = GetAFloat(pF);
  84.     for (i = 0; i < 5; ++i) {
  85.         GetALineAndDontArgue(pF, s);
  86.     }
  87.     count_1 = GetAnInt(pF);
  88.     for (i = 0; i < count_1; i++) {
  89.         for (j = 0; j < 5; j++) {
  90.             GetALineAndDontArgue(pF, s);
  91.         }
  92.         count_2 = GetAnInt(pF);
  93.         for (j = 0; j < 2 * count_2; j++) {
  94.             GetALineAndDontArgue(pF, s);
  95.         }
  96.     }
  97.     return softness;
  98. }
  99.  
  100. // IDA: int __usercall WriteCrushData@<EAX>(FILE *pF@<EAX>, tCrush_data *pCrush_data@<EDX>)
  101. int WriteCrushData(FILE* pF, tCrush_data* pCrush_data) {
  102.     int i;
  103.     int j;
  104.     //int k; // Pierre-Marie Baty -- unused variable
  105.     tCrush_point_spec* the_spec;
  106.     tCrush_neighbour* the_neighbour;
  107.     LOG_TRACE("(%p, %p)", pF, pCrush_data);
  108.  
  109.     fprintf(pF, "%f\n\r", pCrush_data->softness_factor);
  110.     fprintf(pF, "%f,%f\n\r", pCrush_data->min_fold_factor, pCrush_data->max_fold_factor);
  111.     fprintf(pF, "%f\n\r", pCrush_data->wibble_factor);
  112.     fprintf(pF, "%f\n\r", pCrush_data->limit_deviant);
  113.     fprintf(pF, "%f\n\r", pCrush_data->split_chance);
  114.     fprintf(pF, "%f\n\r", pCrush_data->min_y_fold_down);
  115.     fprintf(pF, "%d\n\r", pCrush_data->number_of_crush_points);
  116.     for (i = 0, the_spec = pCrush_data->crush_points; i < pCrush_data->number_of_crush_points; i++, the_spec++) {
  117.         fprintf(pF, "%d\n\r", the_spec->vertex_index);
  118.         fprintf(pF, "%f, %f, %f\n\r", the_spec->limits_neg.v[0], the_spec->limits_neg.v[1], the_spec->limits_neg.v[2]);
  119.         fprintf(pF, "%f, %f, %f\n\r", the_spec->limits_pos.v[0], the_spec->limits_pos.v[1], the_spec->limits_pos.v[2]);
  120.         fprintf(pF, "%f, %f, %f\n\r", the_spec->softness_neg.v[0], the_spec->softness_neg.v[1], the_spec->softness_neg.v[2]);
  121.         fprintf(pF, "%f, %f, %f\n\r", the_spec->softness_pos.v[0], the_spec->softness_pos.v[1], the_spec->softness_pos.v[2]);
  122.         fprintf(pF, "%d\n\r", the_spec->number_of_neighbours);
  123.         for (j = 0, the_neighbour = the_spec->neighbours; j < the_spec->number_of_neighbours; j++, the_neighbour++) {
  124.             fprintf(pF, "%d\n\r", the_neighbour->vertex_index);
  125.             fprintf(pF, "%d\n\r", the_neighbour->factor);
  126.         }
  127.     }
  128.     return 0;
  129. }
  130.  
  131. // IDA: void __usercall DisposeCrushData(tCrush_data *pCrush_data@<EAX>)
  132. void DisposeCrushData(tCrush_data* pCrush_data) {
  133.     int i;
  134.     LOG_TRACE("(%p)", pCrush_data);
  135.  
  136.     for (i = 0; i < pCrush_data->number_of_crush_points; i++) {
  137.         if (pCrush_data->crush_points[i].neighbours != NULL) {
  138.             BrMemFree(pCrush_data->crush_points[i].neighbours);
  139.         }
  140.     }
  141.     if (pCrush_data->crush_points != NULL) {
  142.         BrMemFree(pCrush_data->crush_points);
  143.     }
  144. }
  145.  
  146. // IDA: void __usercall CrushModelPoint(tCar_spec *pCar@<EAX>, int pModel_index@<EDX>, br_model *pModel@<EBX>, int pCrush_point_index@<ECX>, br_vector3 *pEnergy_vector, br_scalar total_energy, tCrush_data *pCrush_data)
  147. void CrushModelPoint(tCar_spec* pCar, int pModel_index, br_model* pModel, int pCrush_point_index, br_vector3* pEnergy_vector, br_scalar total_energy, tCrush_data* pCrush_data) {
  148.     int i;
  149.     int j;
  150.     //int k; // Pierre-Marie Baty -- unused variable
  151.     int pipe_vertex_count;
  152.     int neighbour_index;
  153.     int bend_axis;
  154.     int default_bend_axis[3];
  155.     tCrush_point_spec* the_crush_point;
  156.     tCrush_neighbour* the_neighbour;
  157.     br_vector3* target_point;
  158.     br_vector3 old_vector;
  159.     br_vector3 softnesss;
  160.     br_vector3 movement;
  161.     br_scalar random_range;
  162.     br_scalar bend_amount;
  163.     br_scalar min_y_fold_down;
  164.     float default_bend_factor[3];
  165.     float working_min_fold;
  166.     float working_max_fold;
  167.     float working_wibble;
  168.     float working_limit_deviant;
  169.     float working_split_chance;
  170.     tChanged_vertex pipe_array[600];
  171.     //tCar_spec* car; // Pierre-Marie Baty -- unused variable
  172.     LOG_TRACE("(%p, %d, %p, %d, %p, %f, %p)", pCar, pModel_index, pModel, pCrush_point_index, pEnergy_vector, total_energy, pCrush_data);
  173.  
  174.     pipe_vertex_count = 0;
  175.     if (gNet_mode == eNet_mode_host && pCar->car_model_actors[pModel_index].min_distance_squared == 0.0f) {
  176.         NetSendPointCrush(pCar, pCrush_point_index, pEnergy_vector);
  177.     }
  178.     working_min_fold = pCrush_data->min_fold_factor * gCar_crush_min_fold;
  179.     working_max_fold = pCrush_data->max_fold_factor * gCar_crush_max_fold;
  180.     working_wibble = pCrush_data->wibble_factor * gCar_crush_wibble;
  181.     working_limit_deviant = pCrush_data->limit_deviant * gCar_crush_limit_deviant;
  182.     working_split_chance = pCrush_data->split_chance * gCar_crush_split_chance;
  183.     min_y_fold_down = pCrush_data->min_y_fold_down;
  184.     the_crush_point = &pCrush_data->crush_points[pCrush_point_index];
  185.     if (pModel->nvertices <= the_crush_point->vertex_index) {
  186.         return;
  187.     }
  188.     target_point = &pModel->vertices[the_crush_point->vertex_index].p;
  189.     old_vector = *target_point;
  190.     for (i = 0; i < 3; i++) {
  191.         pEnergy_vector->v[i] = SRandomPosNeg(working_wibble * total_energy) + pEnergy_vector->v[i];
  192.         random_range = (the_crush_point->limits_pos.v[i] - the_crush_point->limits_neg.v[i]) * working_limit_deviant;
  193.         if (pEnergy_vector->v[i] >= 0.0f) {
  194.             softnesss.v[i] = the_crush_point->softness_pos.v[i];
  195.         } else {
  196.             softnesss.v[i] = the_crush_point->softness_neg.v[i];
  197.         }
  198.         movement.v[i] = target_point->v[i];
  199.         target_point->v[i] += pEnergy_vector->v[i] * softnesss.v[i];
  200.         if (the_crush_point->limits_neg.v[i] <= target_point->v[i]) {
  201.             if (target_point->v[i] > the_crush_point->limits_pos.v[i]) {
  202.                 target_point->v[i] = SRandomPosNeg(random_range) + the_crush_point->limits_pos.v[i];
  203.             }
  204.         } else {
  205.             target_point->v[i] = SRandomPosNeg(random_range) + the_crush_point->limits_neg.v[i];
  206.         }
  207.         movement.v[i] = target_point->v[i] - movement.v[i];
  208.         if (pEnergy_vector->v[i] * movement.v[i] < 0.0f) {
  209.             movement.v[i] = 0.0f;
  210.             target_point->v[i] = old_vector.v[i];
  211.         }
  212.     }
  213.  
  214.     if (IsActionReplayAvailable()) {
  215.         pipe_array[pipe_vertex_count].vertex_index = the_crush_point->vertex_index;
  216.         BrVector3Sub(&pipe_array[pipe_vertex_count].delta_coordinates, target_point, &old_vector);
  217.         pipe_vertex_count++;
  218.     }
  219.     neighbour_index = -1;
  220.     for (bend_axis = 0; bend_axis < 3; bend_axis++) {
  221.         default_bend_axis[bend_axis] = (bend_axis + IRandomBetween(1, 2)) % 3;
  222.         default_bend_factor[bend_axis] = FRandomBetween(working_min_fold, working_max_fold);
  223.     }
  224.  
  225.     the_neighbour = the_crush_point->neighbours;
  226.     for (j = 0; j < the_crush_point->number_of_neighbours; j++, the_neighbour++) {
  227.         if (the_neighbour->vertex_index) {
  228.             neighbour_index += the_neighbour->vertex_index;
  229.             if (neighbour_index < 0 || pModel->nvertices <= neighbour_index) {
  230.                 return;
  231.             }
  232.             target_point = &pModel->vertices[neighbour_index].p;
  233.             old_vector = *target_point;
  234.             for (bend_axis = 0; bend_axis < 3; bend_axis++) {
  235.                 target_point->v[bend_axis] += (1.0f - the_neighbour->factor / 256.0f) * movement.v[bend_axis];
  236.                 float v12;
  237.                 if (the_neighbour->factor <= 128) {
  238.                     v12 = the_neighbour->factor / 128.0f;
  239.                 } else {
  240.                     v12 = 2.0 - the_neighbour->factor / 128.0f;
  241.                 }
  242.                 if (((int)((target_point->v[2] + target_point->v[1] + target_point->v[0]) * 63.0f) & 1) * v12 == 0.0) {
  243.                     bend_amount = -default_bend_factor[bend_axis];
  244.                 } else {
  245.                     bend_amount = default_bend_factor[bend_axis];
  246.                 }
  247.                 int axis_tmp = (((int)((target_point->v[2] + target_point->v[1] + target_point->v[0]) * 100.0f) + bend_axis - 1) & 1) % 3;
  248.                 target_point->v[axis_tmp] += fabs(movement.v[bend_axis]) * bend_amount;
  249.             }
  250.             if (IsActionReplayAvailable() && pipe_vertex_count < 600) {
  251.                 pipe_array[pipe_vertex_count].vertex_index = neighbour_index;
  252.                 BrVector3Sub(&pipe_array[pipe_vertex_count].delta_coordinates, target_point, &old_vector);
  253.                 pipe_vertex_count++;
  254.             }
  255.         } else {
  256.             neighbour_index += the_neighbour->factor;
  257.         }
  258.     }
  259.     if (IsActionReplayAvailable() && pipe_vertex_count) {
  260.         PipeSingleModelGeometry(pCar->car_ID, pModel_index, pipe_vertex_count, pipe_array);
  261.     }
  262. }
  263.  
  264. // IDA: void __usercall CrushModel(tCar_spec *pCar@<EAX>, int pModel_index@<EDX>, br_actor *pActor@<EBX>, br_vector3 *pImpact_point@<ECX>, br_vector3 *pEnergy_vector, tCrush_data *pCrush_data)
  265. void CrushModel(tCar_spec* pCar, int pModel_index, br_actor* pActor, br_vector3* pImpact_point, br_vector3* pEnergy_vector, tCrush_data* pCrush_data) {
  266.     br_scalar this_distance;
  267.     br_scalar total_energy;
  268.     br_scalar nearest_so_far;
  269.     br_vector3 impact_point_model;
  270.     br_vector3 energy_vector_scaled;
  271.     br_vector3 energy_vector_model;
  272.     int i;
  273.     int nearest_index;
  274.     br_vertex* vertices;
  275.     br_vertex* the_vertex;
  276.     //br_matrix34 inverse_transform; // Pierre-Marie Baty -- unused variable
  277.     LOG_TRACE("(%p, %d, %p, %p, %p, %p)", pCar, pModel_index, pActor, pImpact_point, pEnergy_vector, pCrush_data);
  278.  
  279.     if (gArrow_mode) {
  280.         return;
  281.     }
  282.     if (pCrush_data->number_of_crush_points == 0) {
  283.         return;
  284.     }
  285.     BrVector3Sub(&impact_point_model, pImpact_point, (br_vector3*)pActor->t.t.mat.m[3]);
  286.     BrVector3Scale(&energy_vector_model, pEnergy_vector, pCrush_data->softness_factor * gCar_crush_softness);
  287.     total_energy = BrVector3Length(&energy_vector_model);
  288.     if (total_energy < 0.06f) {
  289.         return;
  290.     }
  291.     BrVector3Scale(&energy_vector_scaled, &energy_vector_model, (total_energy - 0.06f) / total_energy);
  292.     nearest_so_far = BR_SCALAR_MAX;
  293.     vertices = pActor->model->vertices;
  294.     nearest_index = -1;
  295.     for (i = 0; i < pCrush_data->number_of_crush_points; i++) {
  296.         the_vertex = &vertices[pCrush_data->crush_points[i].vertex_index];
  297.         this_distance = (impact_point_model.v[2] - the_vertex->p.v[2]) * (impact_point_model.v[2] - the_vertex->p.v[2]) + (impact_point_model.v[1] - the_vertex->p.v[1]) * (impact_point_model.v[1] - the_vertex->p.v[1]) + (impact_point_model.v[0] - the_vertex->p.v[0]) * (impact_point_model.v[0] - the_vertex->p.v[0]);
  298.         if (this_distance < nearest_so_far) {
  299.             nearest_so_far = this_distance;
  300.             nearest_index = i;
  301.         }
  302.     }
  303.     if (nearest_index >= 0) {
  304.         CrushModelPoint(pCar, pModel_index, pActor->model, nearest_index, &energy_vector_scaled, total_energy, pCrush_data);
  305.         SetModelForUpdate(pActor->model, pCar, 1);
  306.     }
  307. }
  308.  
  309. // IDA: void __cdecl JitModelUpdate(br_actor *actor, br_model *model, br_material *material, void *render_data, br_uint_8 style, int on_screen)
  310. void JitModelUpdate(br_actor* actor, br_model* model, br_material* material, void* render_data, br_uint_8 style, int on_screen) {
  311.     LOG_TRACE("(%p, %p, %p, %p, %d, %d)", actor, model, material, render_data, style, on_screen);
  312.  
  313.     BrModelUpdate(model, BR_MODU_VERTEX_POSITIONS);
  314.     model->flags &= ~(BR_MODF_CUSTOM);
  315.     BrZbModelRender(actor, model, material, style, BrOnScreenCheck(&model->bounds), 0);
  316. }
  317.  
  318. // IDA: void __usercall SetModelForUpdate(br_model *pModel@<EAX>, tCar_spec *pCar@<EDX>, int crush_only@<EBX>)
  319. void SetModelForUpdate(br_model* pModel, tCar_spec* pCar, int crush_only) {
  320.     LOG_TRACE("(%p, %p, %d)", pModel, pCar, crush_only);
  321.  
  322.     if (crush_only && pCar != NULL && pCar->car_model_actors[pCar->principal_car_actor].actor->model == pModel) {
  323.         CrushBoundingBox(pCar, crush_only);
  324.     }
  325.     if ((pModel->flags & BR_MODF_CUSTOM) != 0) {
  326.         pModel->user = JitModelUpdate;
  327.     } else {
  328.         pModel->custom = JitModelUpdate;
  329.         pModel->flags |= BR_MODF_CUSTOM;
  330.     }
  331. }
  332.  
  333. // IDA: void __usercall TotallySpamTheModel(tCar_spec *pCar@<EAX>, int pModel_index@<EDX>, br_actor *pActor@<EBX>, tCrush_data *pCrush_data@<ECX>, br_scalar pMagnitude)
  334. void TotallySpamTheModel(tCar_spec* pCar, int pModel_index, br_actor* pActor, tCrush_data* pCrush_data, br_scalar pMagnitude) {
  335.     //br_scalar total_energy; // Pierre-Marie Baty -- unused variable
  336.     br_vector3 energy_vector_model;
  337.     int i;
  338.     int the_index;
  339.     br_vertex* the_vertex;
  340.     //br_vertex* vertices; // Pierre-Marie Baty -- unused variable
  341.     LOG_TRACE("(%p, %d, %p, %p, %f)", pCar, pModel_index, pActor, pCrush_data, pMagnitude);
  342.  
  343.     if (gArrow_mode || pCrush_data->number_of_crush_points == 0) {
  344.         return;
  345.     }
  346.     the_vertex = pActor->model->vertices;
  347.     for (i = 0; i < 15; i++) {
  348.         the_index = IRandomBetween(0, pCrush_data->number_of_crush_points - 1);
  349.         energy_vector_model = the_vertex[pCrush_data->crush_points[the_index].vertex_index].p;
  350.         BrVector3Normalise(&energy_vector_model, &energy_vector_model);
  351.         BrVector3Scale(&energy_vector_model, &energy_vector_model, -pMagnitude);
  352.         CrushModelPoint(pCar, pModel_index, pActor->model, the_index, &energy_vector_model, pMagnitude, pCrush_data);
  353.     }
  354.     SetModelForUpdate(pActor->model, pCar, 1);
  355. }
  356.  
  357. // IDA: br_scalar __usercall RepairModel@<ST0>(tCar_spec *pCar@<EAX>, int pModel_index@<EDX>, br_actor *pActor@<EBX>, br_vertex *pUndamaged_vertices@<ECX>, br_scalar pAmount, br_scalar *pTotal_deflection)
  358. br_scalar RepairModel(tCar_spec* pCar, int pModel_index, br_actor* pActor, br_vertex* pUndamaged_vertices, br_scalar pAmount, br_scalar* pTotal_deflection) {
  359.     int i;
  360.     int j;
  361.     int pipe_vertex_count;
  362.     br_vector3 old_point;
  363.     br_vertex* model_vertex;
  364.     br_scalar amount;
  365.     //br_scalar deviation; // Pierre-Marie Baty -- unused variable
  366.     tChanged_vertex pipe_array[600];
  367.     LOG_TRACE("(%p, %d, %p, %p, %f, %p)", pCar, pModel_index, pActor, pUndamaged_vertices, pAmount, pTotal_deflection);
  368.  
  369.     pipe_vertex_count = 0;
  370.     amount = 0.0f;
  371.     *pTotal_deflection = 0.0f;
  372.  
  373.     for (i = 0; i < pActor->model->nvertices; i++) {
  374.         model_vertex = &pActor->model->vertices[i];
  375.         old_point = model_vertex->p;
  376.         for (j = 0; j < 3; ++j) {
  377.             *pTotal_deflection = fabsf(pUndamaged_vertices->p.v[j] - old_point.v[j]) + *pTotal_deflection;
  378.             if (pUndamaged_vertices->p.v[j] >= old_point.v[j]) {
  379.                 if (pUndamaged_vertices->p.v[j] > old_point.v[j]) {
  380.                     model_vertex->p.v[j] = model_vertex->p.v[j] + pAmount;
  381.                     if (pUndamaged_vertices->p.v[j] < model_vertex->p.v[j]) {
  382.                         model_vertex->p.v[j] = pUndamaged_vertices->p.v[j];
  383.                     }
  384.                     amount = model_vertex->p.v[j] - old_point.v[j] + amount;
  385.                 }
  386.             } else {
  387.                 model_vertex->p.v[j] = model_vertex->p.v[j] - pAmount;
  388.                 if (pUndamaged_vertices->p.v[j] > model_vertex->p.v[j]) {
  389.                     model_vertex->p.v[j] = pUndamaged_vertices->p.v[j];
  390.                 }
  391.                 amount = old_point.v[j] - model_vertex->p.v[j] + amount;
  392.             }
  393.         }
  394.         if (amount != 0.0 && IsActionReplayAvailable() && pipe_vertex_count < COUNT_OF(pipe_array)) {
  395.             pipe_array[pipe_vertex_count].vertex_index = i;
  396.             BrVector3Sub(&pipe_array[pipe_vertex_count].delta_coordinates, &model_vertex->p, &old_point);
  397.             pipe_vertex_count++;
  398.         }
  399.         pUndamaged_vertices++;
  400.     }
  401.     SetModelForUpdate(pActor->model, pCar, 0);
  402.     if (IsActionReplayAvailable() && pipe_vertex_count) {
  403.         PipeSingleModelGeometry(pCar->car_ID, pModel_index, pipe_vertex_count, pipe_array);
  404.     }
  405.     return amount;
  406. }
  407.  
  408. // IDA: float __usercall RepairCar2@<ST0>(tCar_spec *pCar@<EAX>, tU32 pFrame_period@<EDX>, br_scalar *pTotal_deflection@<EBX>)
  409. float RepairCar2(tCar_spec* pCar, tU32 pFrame_period, br_scalar* pTotal_deflection) {
  410.     int i;
  411.     tCar_actor* the_car_actor;
  412.     br_scalar amount;
  413.     br_scalar dummy;
  414.     LOG_TRACE("(%p, %d, %p)", pCar, pFrame_period, pTotal_deflection);
  415.  
  416.     if (gArrow_mode) {
  417.         return 0.0f;
  418.     }
  419.     *pTotal_deflection = 0.0;
  420.     amount = 0.0;
  421.  
  422.     for (i = 0; i < gProgram_state.current_car.car_actor_count; i++) {
  423.         the_car_actor = &pCar->car_model_actors[i];
  424.         if (the_car_actor->min_distance_squared == 0.0 || !the_car_actor->undamaged_vertices) {
  425.             if (the_car_actor->undamaged_vertices) {
  426.                 amount = RepairModel(pCar, i, the_car_actor->actor, the_car_actor->undamaged_vertices, pFrame_period * 0.00005f, pTotal_deflection);
  427.             }
  428.         } else {
  429.             RepairModel(pCar, i, the_car_actor->actor, the_car_actor->undamaged_vertices, pFrame_period * 0.00005f, &dummy);
  430.         }
  431.     }
  432.     pCar->repair_time += pFrame_period;
  433.     return amount;
  434. }
  435.  
  436. // IDA: float __usercall RepairCar@<ST0>(tU16 pCar_ID@<EAX>, tU32 pFrame_period@<EDX>, br_scalar *pTotal_deflection@<EBX>)
  437. float RepairCar(tU16 pCar_ID, tU32 pFrame_period, br_scalar* pTotal_deflection) {
  438.     LOG_TRACE("(%d, %d, %p)", pCar_ID, pFrame_period, pTotal_deflection);
  439.  
  440.     if (VEHICLE_TYPE_FROM_ID(pCar_ID) == eVehicle_self) {
  441.         return RepairCar2(&gProgram_state.current_car, pFrame_period, pTotal_deflection);
  442.     }
  443.  
  444.     return RepairCar2(GetCarSpec(VEHICLE_TYPE_FROM_ID(pCar_ID), VEHICLE_INDEX_FROM_ID(pCar_ID)), pFrame_period, pTotal_deflection);
  445. }
  446.  
  447. // IDA: void __usercall TotallyRepairACar(tCar_spec *pCar@<EAX>)
  448. void TotallyRepairACar(tCar_spec* pCar) {
  449.     int i;
  450.     int j;
  451.     int k;
  452.     int pipe_vertex_count;
  453.     tCar_actor* the_car_actor;
  454.     tChanged_vertex pipe_array[600];
  455.     br_bounds storage_bounds;
  456.     LOG_TRACE("(%p)", pCar);
  457.  
  458.     StopCarSmokingInstantly(pCar);
  459.     if (IsActionReplayAvailable()) {
  460.         PipeInstantUnSmudge(pCar);
  461.     }
  462.     pCar->repair_time += 100000;
  463.     for (i = 0; i < COUNT_OF(pCar->damage_units); i++) {
  464.         pCar->damage_units[i].damage_level = 0;
  465.         pCar->damage_units[i].last_level = 0;
  466.         pCar->damage_units[i].smoke_last_level = 0;
  467.     }
  468.     memcpy(&storage_bounds, &pCar->bounds[1], sizeof(br_bounds));
  469.     memcpy(&pCar->bounds[1], &pCar->max_bounds[1], sizeof(br_bounds));
  470.     if (TestForCarInSensiblePlace(pCar)) {
  471.         for (j = 0; j < pCar->car_actor_count; j++) {
  472.             the_car_actor = &pCar->car_model_actors[j];
  473.             if (the_car_actor->undamaged_vertices != NULL) {
  474.                 pipe_vertex_count = 0;
  475.                 for (k = 0; k < the_car_actor->actor->model->nvertices; k++) {
  476.                     if (pipe_vertex_count < COUNT_OF(pipe_array)) {
  477.                         BrVector3Sub(&pipe_array[pipe_vertex_count].delta_coordinates,
  478.                             &the_car_actor->undamaged_vertices[k].p, &the_car_actor->actor->model->vertices[k].p);
  479.                         if (!Vector3IsZero(&pipe_array[pipe_vertex_count].delta_coordinates)) {
  480.                             pipe_array[pipe_vertex_count].vertex_index = k;
  481.                             pipe_vertex_count++;
  482.                         }
  483.                     }
  484.                 }
  485.                 memcpy(the_car_actor->actor->model->vertices,
  486.                     the_car_actor->undamaged_vertices,
  487.                     the_car_actor->actor->model->nvertices * sizeof(br_vertex));
  488.                 // FIXME: BrModelUpdate(..., BR_MODU_VERTEX_COLOURS | BR_MODU_VERTEX_POSITIONS) fails on TELL_ME_IF_WE_PASS_THIS_WAY
  489.                 // BrModelUpdate(the_car_actor->actor->model, BR_MODU_VERTEX_COLOURS | BR_MODU_VERTEX_POSITIONS);
  490.                 BrModelUpdate(the_car_actor->actor->model, BR_MODU_ALL);
  491.                 if (pipe_vertex_count != 0 && IsActionReplayAvailable()) {
  492.                     PipeSingleModelGeometry(pCar->car_ID, j, pipe_vertex_count, pipe_array);
  493.                 }
  494.             }
  495.         }
  496.     } else {
  497.         memcpy(&pCar->bounds[1], &storage_bounds, sizeof(br_bounds));
  498.     }
  499. }
  500.  
  501. // IDA: void __cdecl TotallyRepairCar()
  502. void TotallyRepairCar(void) {
  503.     LOG_TRACE("()");
  504.  
  505.     if (!gArrow_mode) {
  506.         TotallyRepairACar(&gProgram_state.current_car);
  507.     }
  508. }
  509.  
  510. // IDA: void __cdecl CheckLastCar()
  511. void CheckLastCar(void) {
  512.     LOG_TRACE("()");
  513.  
  514.     if (gNet_mode == eNet_mode_none && GetCarCount(eVehicle_opponent) != 0 && NumberOfOpponentsLeft() == 0) {
  515.         NewTextHeadupSlot(4, 0, 5000, -4, GetMiscString(kMiscString_EveryOpponentWasted));
  516.         RaceCompleted(eRace_over_opponents);
  517.     }
  518. }
  519.  
  520. // IDA: void __usercall KnackerThisCar(tCar_spec *pCar@<EAX>)
  521. void KnackerThisCar(tCar_spec* pCar) {
  522.     LOG_TRACE("(%p)", pCar);
  523.  
  524.     pCar->knackered = 1;
  525.     QueueWastedMassage(pCar->index);
  526.     CheckLastCar();
  527.     QueueOilSpill(pCar);
  528.     if (gNet_mode == eNet_mode_none) {
  529.         KillGroovadelic(pCar->index);
  530.         KillFunkotronic(pCar->index);
  531.     }
  532. }
  533.  
  534. // IDA: void __usercall SetKnackeredFlag(tCar_spec *pCar@<EAX>)
  535. void SetKnackeredFlag(tCar_spec* pCar) {
  536.     LOG_TRACE("(%p)", pCar);
  537.  
  538.     if (gNet_mode != eNet_mode_client
  539.         && !pCar->knackered
  540.         && (pCar->damage_units[eDamage_engine].damage_level >= 99
  541.             || pCar->damage_units[eDamage_transmission].damage_level >= 99
  542.             || pCar->damage_units[eDamage_driver].damage_level >= 99
  543.             || (pCar->damage_units[eDamage_lf_wheel].damage_level >= 99
  544.                 && pCar->damage_units[eDamage_rf_wheel].damage_level >= 99
  545.                 && pCar->damage_units[eDamage_lr_wheel].damage_level >= 99
  546.                 && pCar->damage_units[eDamage_rr_wheel].damage_level >= 99))) {
  547.         KnackerThisCar(pCar);
  548.         if (gNet_mode == eNet_mode_none) {
  549.             if (IRandomBetween(0, 1)) {
  550.                 if (gNet_mode == eNet_mode_none) {
  551.                     StopCarSmoking(pCar);
  552.                     CreateSmokeColumn(pCar, 0, IRandomBetween(0, 11), 20000);
  553.                 }
  554.             }
  555.             CreateSmokeColumn(pCar, 0, IRandomBetween(0, 11), 180000);
  556.         }
  557.     }
  558. }
  559.  
  560. // IDA: void __usercall DamageUnit2(tCar_spec *pCar@<EAX>, int pUnit_type@<EDX>, int pDamage_amount@<EBX>)
  561. void DamageUnit2(tCar_spec* pCar, int pUnit_type, int pDamage_amount) {
  562.     tDamage_unit* the_damage;
  563.     int last_level;
  564.     LOG_TRACE("(%p, %d, %d)", pCar, pUnit_type, pDamage_amount);
  565.  
  566.     the_damage = &pCar->damage_units[pUnit_type];
  567.     if ((pCar->driver < eDriver_net_human || pUnit_type != eDamage_driver) && pDamage_amount >= 5 && !pCar->invulnerable) {
  568.         last_level = the_damage->damage_level;
  569.         the_damage->damage_level = pDamage_amount + last_level;
  570.         if (the_damage->damage_level >= 99) {
  571.             if (pDamage_amount >= 10) {
  572.                 the_damage->damage_level = 99;
  573.             } else {
  574.                 the_damage->damage_level = last_level;
  575.             }
  576.         }
  577.         if (pCar->driver == eDriver_oppo || gNet_mode != eNet_mode_none) {
  578.             SetKnackeredFlag(pCar);
  579.         } else if ((pCar->damage_units[eDamage_engine].damage_level >= 99 && pCar->damage_units[eDamage_engine].last_level < 99 && pCar->damage_units[eDamage_transmission].last_level < 99)
  580.             || (pCar->damage_units[eDamage_transmission].damage_level >= 99 && pCar->damage_units[eDamage_engine].last_level < 99 && pCar->damage_units[eDamage_transmission].last_level < 99)) {
  581.             QueueOilSpill(pCar);
  582.         }
  583.     }
  584. }
  585.  
  586. // IDA: void __usercall RecordLastDamage(tCar_spec *pCar@<EAX>)
  587. void RecordLastDamage(tCar_spec* pCar) {
  588.     int i;
  589.     LOG_TRACE("(%p)", pCar);
  590.  
  591.     for (i = 0; i < COUNT_OF(pCar->damage_units); i++) {
  592.         pCar->damage_units[i].last_level = pCar->damage_units[i].damage_level;
  593.     }
  594.     pCar->damage_magnitude_accumulator = 0.0;
  595.     pCar->last_impact_location = eImpact_unknown;
  596.     pCar->pre_car_col_mat = pCar->car_master_actor->t.t.mat;
  597.     pCar->pre_car_col_speed = pCar->speed;
  598.     pCar->pre_car_col_knackered = pCar->knackered;
  599.     pCar->pre_car_col_direction = pCar->direction;
  600.     pCar->pre_car_col_velocity = pCar->v;
  601.     pCar->pre_car_col_velocity_car_space = pCar->velocity_car_space;
  602. }
  603.  
  604. // IDA: void __usercall DoDamage(tCar_spec *pCar@<EAX>, tDamage_type pDamage_type@<EDX>, float pMagnitude, float pNastiness)
  605. void DoDamage(tCar_spec* pCar, tDamage_type pDamage_type, float pMagnitude, float pNastiness) {
  606.     LOG_TRACE("(%p, %d, %f, %f)", pCar, pDamage_type, pMagnitude, pNastiness);
  607.  
  608.     if (pCar->driver < eDriver_net_human) {
  609.         DamageUnit2(pCar, pDamage_type, ((gCurrent_race.suggested_rank < 10 ? 0.5f : gCurrent_race.suggested_rank) / 20.0f + 1.0f) * (pNastiness * pMagnitude * 10.0f));
  610.     } else if (gNet_mode != eNet_mode_none) {
  611.         DamageUnit2(pCar, pDamage_type, pNastiness * pMagnitude * 15.0f);
  612.     } else if (PercentageChance(pNastiness * pMagnitude * 1500.0f)) {
  613.         DamageUnit2(pCar, pDamage_type, pNastiness * pMagnitude * 30.0f);
  614.     }
  615. }
  616.  
  617. // IDA: void __usercall CheckPiledriverBonus(tCar_spec *pCar@<EAX>, br_vector3 *pImpact_point@<EDX>, br_vector3 *pEnergy@<EBX>)
  618. void CheckPiledriverBonus(tCar_spec* pCar, br_vector3* pImpact_point, br_vector3* pEnergy) {
  619.     br_actor* child;
  620.     br_vector3 norm_impact;
  621.     br_vector3 norm_child;
  622.     br_vector3 norm_energy;
  623.     //br_scalar dp; // Pierre-Marie Baty -- unused variable
  624.     LOG_TRACE("(%p, %p, %p)", pCar, pImpact_point, pEnergy);
  625.  
  626.     if (pCar->current_car_actor < 0) {
  627.         return;
  628.     }
  629.  
  630.     BrVector3Normalise(&norm_impact, pImpact_point);
  631.     norm_impact.v[1] = 0.f;
  632.     BrVector3Normalise(&norm_energy, pEnergy);
  633.  
  634.     for (child = pCar->car_master_actor->children; child != NULL; child = child->next) {
  635.         if (ActorIsPedestrian(child) && PedestrianActorIsPerson(child) && pCar->speed > 0.001f) {
  636.             BrVector3Normalise(&norm_child, &child->t.t.translate.t);
  637.             norm_child.v[1] = 0.f;
  638.             if (BrVector3Dot(&norm_child, &norm_impact) > 0.8f && BrVector3Dot(&norm_energy, &norm_child) < -.65) {
  639.                 DoFancyHeadup(kFancyHeadupPileDriverBonus);
  640.                 EarnCredits(((GetPedestrianValue(child) / 2 + 12) / 25) * 25);
  641.                 return;
  642.             }
  643.         }
  644.     }
  645. }
  646.  
  647. // IDA: tImpact_location __usercall CalcModifiedLocation@<EAX>(tCar_spec *pCar@<EAX>)
  648. tImpact_location CalcModifiedLocation(tCar_spec* pCar) {
  649.     LOG_TRACE("(%p)", pCar);
  650.  
  651.     if (pCar->last_impact_location != eImpact_left && pCar->last_impact_location != eImpact_right && pCar->last_impact_location != eImpact_top && pCar->last_impact_location != eImpact_bottom) {
  652.         return pCar->last_impact_location;
  653.     }
  654.     if (pCar->last_col_prop_z < 0.25f) {
  655.         return eImpact_front;
  656.     }
  657.     if (pCar->last_col_prop_z > 0.75f) {
  658.         return eImpact_back;
  659.     } else {
  660.         return pCar->last_impact_location;
  661.     }
  662. }
  663.  
  664. // IDA: void __usercall DoPratcamHit(br_vector3 *pHit_vector@<EAX>)
  665. void DoPratcamHit(br_vector3* pHit_vector) {
  666.     //int strength_modifier; // Pierre-Marie Baty -- unused variable
  667.     //br_scalar strength; // Pierre-Marie Baty -- unused variable
  668.     LOG_TRACE("(%p)", pHit_vector);
  669.  
  670.     STUB_ONCE();
  671. }
  672.  
  673. // IDA: void __usercall DamageSystems(tCar_spec *pCar@<EAX>, br_vector3 *pImpact_point@<EDX>, br_vector3 *pEnergy_vector@<EBX>, int pWas_hitting_a_car@<ECX>)
  674. void DamageSystems(tCar_spec* pCar, br_vector3* pImpact_point, br_vector3* pEnergy_vector, int pWas_hitting_a_car) {
  675.     int i;
  676.     int j;
  677.     int result;
  678.     br_bounds crushed_car_bounds;
  679.     float proportion_x;
  680.     float proportion_y;
  681.     float proportion_z;
  682.     float energy_magnitude;
  683.     float pure_energy_magnitude;
  684.     br_scalar x;
  685.     br_scalar y;
  686.     br_scalar z;
  687.     br_scalar x1;
  688.     br_scalar x2;
  689.     br_scalar y1;
  690.     br_scalar y2;
  691.     br_scalar z1;
  692.     br_scalar z2;
  693.     //br_scalar distance; // Pierre-Marie Baty -- unused variable
  694.     tImpact_location impact_location;
  695.     tDamage_program* the_program;
  696.     tDamage_clause* the_clause;
  697.     tDamage_condition* the_condition;
  698.     tDamage_effect* the_effect;
  699.     //tImpact_location modified_location; // Pierre-Marie Baty -- unused variable
  700.     LOG_TRACE("(%p, %p, %p, %d)", pCar, pImpact_point, pEnergy_vector, pWas_hitting_a_car);
  701.  
  702. #if defined(DETHRACE_FIX_BUGS)
  703.     proportion_x = 0;
  704.     proportion_y = 0;
  705.     proportion_z = 0;
  706. #endif
  707.  
  708.     pure_energy_magnitude = BrVector3Length(pEnergy_vector);
  709.     if (pure_energy_magnitude == 0.0f && !pWas_hitting_a_car) {
  710.         return;
  711.     }
  712.  
  713.     energy_magnitude = pCar->car_model_actors[pCar->principal_car_actor].crush_data.softness_factor * pure_energy_magnitude / 0.7f;
  714.     BrVector3InvScale(&crushed_car_bounds.min, &pCar->bounds[1].min, WORLD_SCALE);
  715.     BrVector3InvScale(&crushed_car_bounds.max, &pCar->bounds[1].max, WORLD_SCALE);
  716.  
  717.     x1 = pImpact_point->v[0] - crushed_car_bounds.min.v[0];
  718.     x2 = crushed_car_bounds.max.v[0] - pImpact_point->v[0];
  719.     if (x1 >= x2) {
  720.         x = x2;
  721.     } else {
  722.         x = x1;
  723.     }
  724.     y1 = pImpact_point->v[1] - crushed_car_bounds.min.v[1];
  725.     y2 = crushed_car_bounds.max.v[1] - pImpact_point->v[1];
  726.     if (y1 >= y2) {
  727.         y = y2;
  728.     } else {
  729.         y = y1;
  730.     }
  731.     z1 = pImpact_point->v[2] - crushed_car_bounds.min.v[2];
  732.     z2 = crushed_car_bounds.max.v[2] - pImpact_point->v[2];
  733.     if (z1 >= z2) {
  734.         z = z2;
  735.     } else {
  736.         z = z1;
  737.     }
  738.     if (z > x || z > y) {
  739.         if (x > y || x > z) {
  740.             impact_location = y1 < y2 ? eImpact_bottom : eImpact_top;
  741.             proportion_z = z1 / (crushed_car_bounds.max.v[2] - crushed_car_bounds.min.v[2]);
  742.             proportion_x = x1 / (crushed_car_bounds.max.v[0] - crushed_car_bounds.min.v[0]);
  743.         } else {
  744.             impact_location = x1 >= x2 ? eImpact_right : eImpact_left;
  745.             proportion_z = z1 / (crushed_car_bounds.max.v[2] - crushed_car_bounds.min.v[2]);
  746.             proportion_y = y1 / (crushed_car_bounds.max.v[1] - crushed_car_bounds.min.v[1]);
  747.         }
  748.     } else {
  749.         impact_location = z1 >= z2 ? eImpact_back : eImpact_front;
  750.         proportion_x = x1 / (crushed_car_bounds.max.v[0] - crushed_car_bounds.min.v[0]);
  751.         proportion_y = y1 / (crushed_car_bounds.max.v[1] - crushed_car_bounds.min.v[1]);
  752.     }
  753.     if (pWas_hitting_a_car && pCar->last_impact_location == eImpact_unknown) {
  754.         pCar->last_impact_location = impact_location;
  755.         pCar->last_col_prop_x = proportion_x;
  756.         pCar->last_col_prop_y = proportion_y;
  757.         pCar->last_col_prop_z = proportion_z;
  758.     }
  759.  
  760.     if (energy_magnitude != 0.0f && !pCar->invulnerable) {
  761.         if (!pWas_hitting_a_car && impact_location == eImpact_bottom) {
  762.             energy_magnitude = energy_magnitude / 2.0f;
  763.         }
  764.  
  765.         the_program = &pCar->damage_programs[impact_location];
  766.         the_clause = the_program->clauses;
  767.         for (i = 0; i < the_program->clause_count; i++) {
  768.             result = 1;
  769.             the_condition = the_clause->conditions;
  770.             for (j = 0; j < the_clause->condition_count; j++) {
  771.                 switch (the_condition->axis_comp) {
  772.                 case eAxis_x:
  773.                     if (the_condition->condition_operator == eCondition_greater_than) {
  774.                         if (the_condition->comparitor >= proportion_x) {
  775.                             result = 0;
  776.                         }
  777.                     } else if (the_condition->comparitor <= proportion_x) {
  778.                         result = 0;
  779.                     }
  780.                     break;
  781.  
  782.                 case eAxis_y:
  783.                     if (the_condition->condition_operator == eCondition_greater_than) {
  784.                         if (the_condition->comparitor >= proportion_y) {
  785.                             result = 0;
  786.                         }
  787.                     } else if (the_condition->comparitor <= proportion_y) {
  788.                         result = 0;
  789.                     }
  790.                     break;
  791.  
  792.                 case eAxis_z:
  793.                     if (the_condition->condition_operator == eCondition_greater_than) {
  794.                         if (the_condition->comparitor >= proportion_z) {
  795.                             result = 0;
  796.                         }
  797.                     } else if (the_condition->comparitor <= proportion_z) {
  798.                         result = 0;
  799.                     }
  800.                     break;
  801.                 }
  802.  
  803.                 if (!result) {
  804.                     break;
  805.                 }
  806.                 the_condition++;
  807.             }
  808.             if (result) {
  809.                 for (j = 0; j < the_clause->effect_count; j++) {
  810.                     the_effect = &the_clause->effects[j];
  811.                     DoDamage(pCar, the_effect->type, energy_magnitude, the_effect->weakness_factor);
  812.                 }
  813.             }
  814.             the_clause++;
  815.         }
  816.         if (pCar->driver == eDriver_local_human) {
  817.             switch (impact_location) {
  818.             case eImpact_top:
  819.             case eImpact_bottom:
  820.                 NewScreenWobble(
  821.                     FRandomBetween(energy_magnitude * 5.0f, energy_magnitude * 20.0f),
  822.                     FRandomBetween(energy_magnitude * 30.0f, energy_magnitude * 60.0f),
  823.                     FRandomBetween(1.0f / energy_magnitude, 5.0f / energy_magnitude));
  824.                 break;
  825.             case eImpact_left:
  826.                 NewScreenWobble(
  827.                     FRandomBetween(energy_magnitude * 50.0f, energy_magnitude * 100.0f),
  828.                     FRandomBetween(energy_magnitude * 5.0f, energy_magnitude * 20.0f),
  829.                     FRandomBetween(4.0f / energy_magnitude, 7.0 / energy_magnitude));
  830.                 break;
  831.             case eImpact_right:
  832.                 NewScreenWobble(
  833.                     -FRandomBetween(energy_magnitude * 50.0f, energy_magnitude * 100.0f),
  834.                     FRandomBetween(energy_magnitude * 5.0f, energy_magnitude * 20.0f),
  835.                     FRandomBetween(4.0f / energy_magnitude, 7.0f / energy_magnitude));
  836.  
  837.                 break;
  838.             case eImpact_front:
  839.                 NewScreenWobble(
  840.                     FRandomBetween(energy_magnitude * 5.0f, energy_magnitude * 20.0f),
  841.                     FRandomBetween(energy_magnitude * 50.0f, energy_magnitude * 150.0f),
  842.                     FRandomBetween(7.0f / energy_magnitude, 25.0f / energy_magnitude));
  843.                 break;
  844.             case eImpact_back:
  845.                 NewScreenWobble(
  846.                     FRandomBetween(energy_magnitude * 5.0f, energy_magnitude * 20.0f),
  847.                     FRandomBetween(-energy_magnitude * 50.0f, energy_magnitude * 150.0f),
  848.                     FRandomBetween(7.0f / energy_magnitude, 25.0f / energy_magnitude));
  849.                 break;
  850.             default:
  851.                 break;
  852.             }
  853.             CheckPiledriverBonus(pCar, pImpact_point, pEnergy_vector);
  854.         }
  855.     }
  856. }
  857.  
  858. // IDA: tImpact_location __usercall GetDirection@<EAX>(br_vector3 *pVelocity@<EAX>)
  859. tImpact_location GetDirection(br_vector3* pVelocity) {
  860.     br_scalar mag_x;
  861.     br_scalar mag_y;
  862.     br_scalar mag_z;
  863.     LOG_TRACE("(%p)", pVelocity);
  864.  
  865.     mag_x = fabsf(pVelocity->v[0]);
  866.     mag_y = fabsf(pVelocity->v[1]);
  867.     mag_z = fabsf(pVelocity->v[2]);
  868.     if (mag_y >= mag_x || mag_z >= mag_x) {
  869.         if (mag_y <= mag_x || mag_z >= mag_y) {
  870.             if (pVelocity->v[2] >= 0.0f) {
  871.                 return eImpact_back;
  872.             } else {
  873.                 return eImpact_front;
  874.             }
  875.         } else {
  876.             return pVelocity->v[1] < 0.0;
  877.         }
  878.     } else if (pVelocity->v[0] >= 0.0) {
  879.         return eImpact_right;
  880.     } else {
  881.         return eImpact_left;
  882.     }
  883. }
  884.  
  885. // IDA: void __usercall SetSmokeLastDamageLevel(tCar_spec *pCar@<EAX>)
  886. void SetSmokeLastDamageLevel(tCar_spec* pCar) {
  887.     int i;
  888.     LOG_TRACE("(%p)", pCar);
  889.  
  890.     for (i = 0; i < COUNT_OF(pCar->damage_units); i++) {
  891.         pCar->damage_units[i].smoke_last_level = pCar->damage_units[i].damage_level;
  892.     }
  893. }
  894.  
  895. // IDA: void __usercall SortOutSmoke(tCar_spec *pCar@<EAX>)
  896. void SortOutSmoke(tCar_spec* pCar) {
  897.     int i;
  898.     int colour;
  899.     int old_colour;
  900.     int step;
  901.     //int pass; // Pierre-Marie Baty -- unused variable
  902.     //int repeat; // Pierre-Marie Baty -- unused variable
  903.     LOG_TRACE("(%p)", pCar);
  904.  
  905.     if (!pCar || pCar->driver <= eDriver_non_car) {
  906.         return;
  907.     }
  908.     for (i = 0; i < COUNT_OF(pCar->damage_units); i++) {
  909.         if (pCar->damage_units[i].damage_level != pCar->damage_units[i].smoke_last_level) {
  910.             step = gSmoke_damage_step[i];
  911.             if (step) {
  912.                 if (pCar->damage_units[i].damage_level > pCar->damage_units[i].smoke_last_level) {
  913.                     old_colour = (99 - pCar->damage_units[i].smoke_last_level) / step;
  914.                     colour = (99 - pCar->damage_units[i].damage_level) / step;
  915.                     if (old_colour != colour && colour <= 2) {
  916.                         ConditionalSmokeColumn(pCar, i, colour);
  917.                     }
  918.                 }
  919.             }
  920.         }
  921.     }
  922.     SetSmokeLastDamageLevel(pCar);
  923. }
  924.  
  925. // IDA: void __usercall StealCar(tCar_spec *pCar@<EAX>)
  926. void StealCar(tCar_spec* pCar) {
  927.     LOG_TRACE("(%p)", pCar);
  928.  
  929.     pCar->has_been_stolen = 1;
  930.     gProgram_state.cars_available[gProgram_state.number_of_cars] = pCar->index;
  931.     gProgram_state.number_of_cars++;
  932.     gOpponents[pCar->index].dead = 1;
  933. }
  934.  
  935. // IDA: int __usercall DoCrashEarnings@<EAX>(tCar_spec *pCar1@<EAX>, tCar_spec *pCar2@<EDX>)
  936. int DoCrashEarnings(tCar_spec* pCar1, tCar_spec* pCar2) {
  937.     tCar_spec* culprit;
  938.     tCar_spec* victim;
  939.     int i;
  940.     int net_loop;
  941.     int head_on;
  942.     int bonus_level;
  943.     int credits;
  944.     int impact_in_moving_direction_1;
  945.     int impact_in_moving_direction_2;
  946.     int car_off_ground_1;
  947.     int car_off_ground_2;
  948.     int total_units_of_damage;
  949.     int inherited_damage;
  950.     int dam_acc_1;
  951.     int dam_acc_2;
  952.     int car_1_culpable;
  953.     int car_2_culpable;
  954.     int mutual_culpability;
  955.     tU32 the_time;
  956.     tU32 time;
  957.     float credits_squared;
  958.     static tU32 last_earn_time;
  959.     //char s[256]; // Pierre-Marie Baty -- unused variable
  960.     tImpact_location modified_location_1;
  961.     tImpact_location modified_location_2;
  962.     tImpact_location car_direction_1;
  963.     tImpact_location car_direction_2;
  964.     br_scalar car_1_height;
  965.     br_scalar car_2_height;
  966.     br_scalar dp;
  967.     br_vector3 car_1_pos;
  968.     br_vector3 car_2_pos;
  969.     br_vector3 car_1_offset;
  970.     br_vector3 car_2_offset;
  971.     tNet_message* message;
  972.     LOG_TRACE("(%p, %p)", pCar1, pCar2);
  973.  
  974.     culprit = 0;
  975.     victim = 0;
  976.     head_on = 0;
  977.     bonus_level = 1;
  978.     car_1_culpable = 0;
  979.     car_2_culpable = 0;
  980.     mutual_culpability = 0;
  981.     the_time = PDGetTotalTime();
  982.     inherited_damage = 0;
  983. #if defined(DETHRACE_FIX_BUGS)
  984.     total_units_of_damage = 0;
  985. #endif
  986.     if (pCar1->driver <= eDriver_non_car) {
  987.         dam_acc_1 = 0;
  988.     } else {
  989.         dam_acc_1 = pCar1->damage_magnitude_accumulator;
  990.     }
  991.  
  992.     dam_acc_2 = 0;
  993.     if (pCar2 != NULL) {
  994.         if (pCar2->driver <= eDriver_non_car) {
  995.             dam_acc_2 = 0;
  996.         } else {
  997.             dam_acc_2 = pCar2->damage_magnitude_accumulator != 0;
  998.         }
  999.     }
  1000.  
  1001.     if (pCar1->driver <= eDriver_non_car) {
  1002.         if (pCar2 == NULL || pCar2->driver <= eDriver_non_car) {
  1003.             return 0;
  1004.         }
  1005.         pCar1 = pCar2;
  1006.         pCar2 = NULL;
  1007.     }
  1008.     if (pCar2 != NULL && pCar2->driver <= eDriver_non_car) {
  1009.         pCar2 = NULL;
  1010.     }
  1011.  
  1012.     if (pCar1->pre_car_col_knackered || (pCar2 && pCar2->pre_car_col_knackered) || (pCar2 && pCar2->damage_magnitude_accumulator <= 0.00005f && pCar1->damage_magnitude_accumulator <= 0.00005f)) {
  1013.         return dam_acc_1 || (pCar2 && dam_acc_2);
  1014.     }
  1015.  
  1016.     modified_location_1 = CalcModifiedLocation(pCar1);
  1017.     car_direction_1 = GetDirection(&pCar1->pre_car_col_velocity_car_space);
  1018.     impact_in_moving_direction_1 = car_direction_1 == modified_location_1;
  1019.     if (pCar2 != NULL) {
  1020.         modified_location_2 = CalcModifiedLocation(pCar2);
  1021.         car_direction_2 = GetDirection(&pCar2->pre_car_col_velocity_car_space);
  1022.         impact_in_moving_direction_2 = car_direction_2 == modified_location_2;
  1023.     }
  1024.     if (pCar1->driver >= eDriver_net_human && pCar2) {
  1025.         if (impact_in_moving_direction_1 && (pCar1->driver < eDriver_net_human || (pCar1->pre_car_col_velocity_car_space.v[2] != 0.0 && (pCar1->pre_car_col_velocity_car_space.v[2] > 0.0) != (pCar1->gear > 0) && (pCar1->keys.acc != 0 || pCar1->joystick.acc > 0x8000)))) {
  1026.             pCar2->time_last_hit = the_time;
  1027.             pCar2->last_hit_by = pCar1;
  1028.         }
  1029.     } else if (pCar2 && pCar2->driver >= eDriver_net_human && impact_in_moving_direction_2 && (pCar2->driver < eDriver_net_human || (pCar2->pre_car_col_velocity_car_space.v[2] != 0.0f && (pCar2->pre_car_col_velocity_car_space.v[2] > 0.0f) != (pCar2->gear > 0) && (pCar2->keys.acc != 0 || pCar2->joystick.acc > 0x8000)))) {
  1030.         pCar1->time_last_hit = the_time;
  1031.         pCar1->last_hit_by = pCar2;
  1032.     }
  1033.     if (pCar2) {
  1034.         if (impact_in_moving_direction_1
  1035.             && pCar1->pre_car_col_speed * 5.0f > pCar2->pre_car_col_speed
  1036.             && pCar1->pre_car_col_speed > 0.0005f
  1037.             && (pCar1->driver < eDriver_net_human
  1038.                 || (pCar1->pre_car_col_velocity_car_space.v[2] != 0.0f
  1039.                     && (pCar1->pre_car_col_velocity_car_space.v[2] > 0.0f) != (pCar1->gear > 0)
  1040.                     && (pCar1->keys.acc != 0 || pCar1->joystick.acc > 0x8000)))) {
  1041.             car_1_culpable = 1;
  1042.         }
  1043.         if (impact_in_moving_direction_2
  1044.             && pCar2->pre_car_col_speed * 5.0f > pCar1->pre_car_col_speed
  1045.             && pCar2->pre_car_col_speed > 0.0005f
  1046.             && (pCar2->driver < eDriver_net_human
  1047.                 || (pCar2->pre_car_col_velocity_car_space.v[2] != 0.0f
  1048.                     && (pCar2->pre_car_col_velocity_car_space.v[2] > 0.0f) != (pCar2->gear > 0)
  1049.                     && (pCar2->keys.acc != 0 || pCar2->joystick.acc > 0x8000)))) {
  1050.             car_2_culpable = 1;
  1051.         }
  1052.         if (gNet_mode && car_1_culpable && car_2_culpable) {
  1053.             mutual_culpability = 1;
  1054.         } else {
  1055.             if (car_2_culpable && pCar2->driver == eDriver_local_human) {
  1056.                 car_1_culpable = 0;
  1057.             }
  1058.             if (car_1_culpable) {
  1059.                 culprit = pCar1;
  1060.                 victim = pCar2;
  1061.                 dp = BrVector3Dot(&pCar1->pre_car_col_direction, &pCar2->pre_car_col_direction);
  1062.                 if (modified_location_1 == eImpact_front && modified_location_2 == eImpact_front && pCar1->pre_car_col_speed > 0.001f && pCar2->pre_car_col_speed > 0.001f && dp < -0.7f) {
  1063.                     head_on = 1;
  1064.                     bonus_level = 2;
  1065.                 } else {
  1066.                     bonus_level = 1;
  1067.                 }
  1068.             } else if (car_2_culpable) {
  1069.                 culprit = pCar2;
  1070.                 victim = pCar1;
  1071.                 dp = BrVector3Dot(&pCar1->pre_car_col_direction, &pCar2->pre_car_col_direction);
  1072.                 if (modified_location_1 == eImpact_front && modified_location_2 == eImpact_front && pCar1->pre_car_col_speed > 0.001f && pCar2->pre_car_col_speed > 0.001f && dp < -0.7f) {
  1073.                     head_on = 1;
  1074.                     bonus_level = 2;
  1075.                 } else {
  1076.                     bonus_level = 1;
  1077.                 }
  1078.             }
  1079.         }
  1080.     } else {
  1081.         if (the_time - pCar1->time_last_hit >= 3000) {
  1082.             return 1;
  1083.         }
  1084.         culprit = pCar1->last_hit_by;
  1085.         victim = pCar1;
  1086.         bonus_level = 1;
  1087.         inherited_damage = 1;
  1088.     }
  1089.     if (!mutual_culpability && (!victim || culprit->driver < eDriver_net_human)) {
  1090.         if (pCar2 && pCar2->last_culprit == pCar1 && the_time - pCar2->time_last_victim < 750) {
  1091.             inherited_damage = 1;
  1092.             culprit = pCar1;
  1093.             victim = pCar2;
  1094.         } else if (pCar2 && pCar1->last_culprit == pCar2 && the_time - pCar1->time_last_victim < 750) {
  1095.             inherited_damage = 1;
  1096.             culprit = pCar2;
  1097.             victim = pCar1;
  1098.         } else if (!pCar2 && the_time - pCar1->time_last_victim < 750) {
  1099.             inherited_damage = 1;
  1100.             culprit = pCar1->last_culprit;
  1101.             victim = pCar1;
  1102.         }
  1103.     }
  1104.     if (culprit && victim) {
  1105.         RecordOpponentTwattageOccurrence(culprit, victim);
  1106.         total_units_of_damage = 0;
  1107.         for (i = 0; i < COUNT_OF(victim->damage_units); i++) {
  1108.             if (victim->damage_units[i].damage_level > victim->damage_units[i].last_level) {
  1109.                 victim->damage_units[i].damage_level = (victim->damage_units[i].damage_level - victim->damage_units[i].last_level) * 2.0f + victim->damage_units[i].last_level;
  1110.                 if (victim->damage_units[i].damage_level >= 99) {
  1111.                     victim->damage_units[i].damage_level = 99;
  1112.                 }
  1113.                 total_units_of_damage += victim->damage_units[i].damage_level - victim->damage_units[i].last_level;
  1114.             }
  1115.             if (culprit->damage_units[i].damage_level > culprit->damage_units[i].last_level) {
  1116.                 culprit->damage_units[i].damage_level = (culprit->damage_units[i].damage_level - culprit->damage_units[i].last_level) * 0.1f + (double)culprit->damage_units[i].last_level;
  1117.                 if (culprit->damage_units[i].damage_level < 0) {
  1118.                     culprit->damage_units[i].damage_level = 0;
  1119.                 }
  1120.             }
  1121.         }
  1122.     }
  1123.     // TODO: tidy this up
  1124.     for (net_loop = 0; 2 - (mutual_culpability == 0) > net_loop; net_loop++) {
  1125.         if (mutual_culpability) {
  1126.             if (net_loop) {
  1127.                 culprit = pCar1;
  1128.                 victim = pCar2;
  1129.             } else {
  1130.                 culprit = pCar2;
  1131.                 victim = pCar1;
  1132.             }
  1133.             total_units_of_damage = 0;
  1134.             for (i = 0; i < COUNT_OF(victim->damage_units); i++) {
  1135.                 if (victim->damage_units[i].damage_level > victim->damage_units[i].last_level) {
  1136.                     total_units_of_damage += victim->damage_units[i].damage_level - victim->damage_units[i].last_level;
  1137.                 }
  1138.             }
  1139.         }
  1140.         if (culprit && (culprit->driver == eDriver_local_human || gNet_mode) && victim) {
  1141.             SetKnackeredFlag(victim);
  1142.             if (victim->knackered && !victim->pre_car_col_knackered) {
  1143.                 victim->pre_car_col_knackered = 1;
  1144.                 credits_squared = sqr(0.7f / victim->car_model_actors[victim->principal_car_actor].crush_data.softness_factor) * gWasted_creds[gProgram_state.skill_level] + 50.0f;
  1145.                 credits = 100 * (int)(credits_squared / 100.0f);
  1146.                 if (gNet_mode) {
  1147.                     message = NetBuildMessage(0x18u, 0);
  1148.                     message->contents.data.wasted.victim = NetPlayerFromCar(victim)->ID;
  1149.                     if (NetPlayerFromCar(culprit)) {
  1150.                         message->contents.data.wasted.culprit = NetPlayerFromCar(culprit)->ID;
  1151.                     } else {
  1152.                         message->contents.data.wasted.culprit = -2;
  1153.                     }
  1154.                     NetGuaranteedSendMessageToEverybody(gCurrent_net_game, message, NULL);
  1155.                     NetEarnCredits(NetPlayerFromCar(culprit), credits);
  1156.                 } else {
  1157.                     PratcamEvent(32);
  1158.                     DoFancyHeadup(11);
  1159.                     credits_squared = sqr(0.7f / victim->car_model_actors[victim->principal_car_actor].crush_data.softness_factor) * gWasted_creds[gProgram_state.skill_level] + 50.0f;
  1160.                     credits = 100 * (int)(credits_squared / 100.0);
  1161.                     AwardTime(gWasted_time[gProgram_state.skill_level]);
  1162.                     EarnCredits(credits);
  1163.                     if (victim->can_be_stolen && !gOpponents[victim->index].dead
  1164.                         // strength_rating is between 1 and 5
  1165.                         && ((PercentageChance(50) && gProgram_state.rank <= gSteal_ranks[gOpponents[victim->index].strength_rating - 1]) || victim->index == BIGAPC_OPPONENT_INDEX)) {
  1166.                         StealCar(victim);
  1167.                     }
  1168.                 }
  1169.             }
  1170.             victim->time_last_hit = the_time;
  1171.             victim->last_hit_by = culprit;
  1172.             if (!inherited_damage) {
  1173.                 victim->time_last_victim = the_time;
  1174.                 victim->last_culprit = culprit;
  1175.             }
  1176.             if (victim && (fabs(victim->omega.v[0]) > 4.0f || fabs(victim->omega.v[1]) > 6.0f || fabs(victim->omega.v[2]) > 4.0f)) {
  1177.                 bonus_level *= 2;
  1178.             }
  1179.             if (pCar1->number_of_wheels_on_ground) {
  1180.                 car_off_ground_1 = 0;
  1181.             } else {
  1182.                 BrVector3InvScale(&car_1_pos, &pCar1->car_master_actor->t.t.translate.t, WORLD_SCALE);
  1183.                 BrMatrix34ApplyV(&car_1_offset, &pCar1->car_model_actors[pCar1->principal_car_actor].actor->t.t.translate.t, &pCar1->car_master_actor->t.t.mat);
  1184.                 BrVector3Accumulate(&car_1_pos, &car_1_offset);
  1185.                 car_1_pos.v[1] += 0.15f;
  1186.                 car_1_height = FindYVerticallyBelow2(&car_1_pos);
  1187.                 car_off_ground_1 = car_1_height > -100.0f
  1188.                     && pCar1->car_model_actors[pCar1->principal_car_actor].actor->t.t.translate.t.v[1] * 4.0f <= car_1_pos.v[1] - car_1_height - 0.15f;
  1189.             }
  1190.             if (!pCar2 || pCar2->number_of_wheels_on_ground) {
  1191.                 car_off_ground_2 = 0;
  1192.             } else {
  1193.                 BrVector3InvScale(&car_2_pos, &pCar2->car_master_actor->t.t.translate.t, WORLD_SCALE);
  1194.                 BrMatrix34ApplyV(&car_2_offset, &pCar2->car_model_actors[pCar2->principal_car_actor].actor->t.t.translate.t, &pCar2->car_master_actor->t.t.mat);
  1195.                 BrVector3Accumulate(&car_2_pos, &car_2_offset);
  1196.                 car_2_pos.v[1] += 0.15f;
  1197.                 car_2_height = FindYVerticallyBelow2(&car_2_pos);
  1198.                 car_off_ground_2 = car_2_height > -100.0f
  1199.                     && pCar2->car_model_actors[pCar2->principal_car_actor].actor->t.t.translate.t.v[1] * 4.0f <= car_2_pos.v[1] - car_2_height - 0.15f;
  1200.             }
  1201.             if (car_off_ground_1) {
  1202.                 bonus_level *= 2;
  1203.             }
  1204.             if (car_off_ground_2) {
  1205.                 bonus_level *= 2;
  1206.             }
  1207.             total_units_of_damage = 0.7f / victim->car_model_actors[victim->principal_car_actor].crush_data.softness_factor * total_units_of_damage;
  1208.             if (!victim->has_been_stolen) {
  1209.                 credits = 100 * (int)((gCar_cred_value[gProgram_state.skill_level] * MIN(bonus_level, 8) * total_units_of_damage + 50.0f) / 100.0f);
  1210.                 if (credits || victim->knackered) {
  1211.                     if (!victim->knackered) {
  1212.                         if (gNet_mode) {
  1213.                             NetEarnCredits(NetPlayerFromCar(culprit), MIN(credits, 2000));
  1214.                         } else {
  1215.                             EarnCredits(MIN(credits, 2000));
  1216.                         }
  1217.                         last_earn_time = the_time;
  1218.                         if (gNet_mode == eNet_mode_none) {
  1219.                             time = 5 * (int)((total_units_of_damage * gCar_time_value[gProgram_state.skill_level] + 2.5f) / 5.0f);
  1220.                             AwardTime(MIN(time, 90));
  1221.                             if (pCar2) {
  1222.                                 if (head_on) {
  1223.                                     DoFancyHeadup(10);
  1224.                                 } else if (bonus_level <= 2) {
  1225.                                     if (bonus_level > 1) {
  1226.                                         DoFancyHeadup(2);
  1227.                                     }
  1228.                                 } else {
  1229.                                     DoFancyHeadup(3);
  1230.                                 }
  1231.                             }
  1232.                         }
  1233.                     }
  1234.                     for (i = 0; i < COUNT_OF(victim->damage_units); i++) {
  1235.                         victim->damage_units[i].last_level = victim->damage_units[i].damage_level;
  1236.                     }
  1237.                 }
  1238.             }
  1239.         } else {
  1240.             pCar1->time_last_hit = 0;
  1241.             if (pCar2) {
  1242.                 pCar2->time_last_hit = 0;
  1243.             }
  1244.         }
  1245.     }
  1246.     pCar1->damage_magnitude_accumulator = 0.0f;
  1247.     if (pCar2) {
  1248.         pCar2->damage_magnitude_accumulator = 0.0f;
  1249.     }
  1250.     return 1;
  1251. }
  1252.  
  1253. // IDA: void __usercall DoWheelDamage(tU32 pFrame_period@<EAX>)
  1254. void DoWheelDamage(tU32 pFrame_period) {
  1255.     int i;
  1256.     int j;
  1257.     int damage;
  1258.     tCar_spec* car;
  1259.     br_scalar y_amount;
  1260.     br_scalar z_amount;
  1261.     br_scalar wheel_circum;
  1262.     br_scalar old_offset;
  1263.     br_vector3 temp_vector;
  1264.     br_vector3 wonky_vector;
  1265.     static int kev_index[4];
  1266.     LOG_TRACE("(%d)", pFrame_period);
  1267.  
  1268.     if (gAction_replay_mode && ReplayIsPaused()) {
  1269.         return;
  1270.     }
  1271.  
  1272.     for (i = 0; i < gNum_active_cars; i++) {
  1273.         car = gActive_car_list[i];
  1274.         for (j = 0; j < COUNT_OF(car->wheel_dam_offset); j++) {
  1275.             if (car->wheel_actors[j] == NULL) {
  1276.                 continue;
  1277.             }
  1278.             old_offset = car->wheel_dam_offset[j];
  1279.             damage = car->damage_units[j + 8].damage_level;
  1280.             if (damage <= 30 || gRace_finished) {
  1281.                 car->wheel_dam_offset[j] = 0.0f;
  1282.                 continue;
  1283.             }
  1284.             if (PointOutOfSight(&car->pos, 32.0f)) {
  1285.                 break;
  1286.             }
  1287.             y_amount = (damage - 30) * gWobble_spam_y[damage & 7];
  1288.             z_amount = (damage - 30) * gWobble_spam_z[damage & 7];
  1289.             BrMatrix34PreRotateY(&car->wheel_actors[j]->t.t.mat, BrDegreeToAngle(y_amount < 0 ? y_amount + 360.f : y_amount));
  1290.             BrMatrix34PreRotateZ(&car->wheel_actors[j]->t.t.mat, BrDegreeToAngle(z_amount < 0 ? z_amount + 360.f : z_amount));
  1291.             if (j < 2 && car->wheel_actors[j + 4] != NULL) {
  1292.                 BrMatrix34PreRotateY(&car->wheel_actors[j + 4]->t.t.mat, BrDegreeToAngle(y_amount < 0 ? y_amount + 360.f : y_amount));
  1293.                 BrMatrix34PreRotateZ(&car->wheel_actors[j + 4]->t.t.mat, BrDegreeToAngle(z_amount < 0 ? z_amount + 360.f : z_amount));
  1294.             }
  1295.             switch (j) {
  1296.             case 0:
  1297.                 if (car->driven_wheels_spin_ref_1 < 0) {
  1298.                     wheel_circum = car->non_driven_wheels_circum;
  1299.                 } else {
  1300.                     wheel_circum = car->driven_wheels_circum;
  1301.                 }
  1302.                 break;
  1303.             case 1:
  1304.                 if (car->driven_wheels_spin_ref_2 < 0) {
  1305.                     wheel_circum = car->non_driven_wheels_circum;
  1306.                 } else {
  1307.                     wheel_circum = car->driven_wheels_circum;
  1308.                 }
  1309.                 break;
  1310.             case 2:
  1311.                 if (car->driven_wheels_spin_ref_3 < 0) {
  1312.                     wheel_circum = car->non_driven_wheels_circum;
  1313.                 } else {
  1314.                     wheel_circum = car->driven_wheels_circum;
  1315.                 }
  1316.                 break;
  1317.             case 3:
  1318.                 if (car->driven_wheels_spin_ref_4 < 0) {
  1319.                     wheel_circum = car->non_driven_wheels_circum;
  1320.                 } else {
  1321.                     wheel_circum = car->driven_wheels_circum;
  1322.                 }
  1323.                 break;
  1324.             default:
  1325.                 TELL_ME_IF_WE_PASS_THIS_WAY();
  1326.                 break;
  1327.             }
  1328.             if (gNet_mode == eNet_mode_none || car->driver == eDriver_local_human) {
  1329.                 BrVector3Set(&temp_vector, wheel_circum * gWheel_circ_to_width, 0.f, 0.f);
  1330.                 BrMatrix34ApplyV(&wonky_vector, &temp_vector, &car->wheel_actors[j]->t.t.mat);
  1331.                 car->wheel_dam_offset[j] = fabsf(wonky_vector.v[1]);
  1332.             }
  1333.         }
  1334.     }
  1335. }
  1336.  
  1337. // IDA: void __usercall CrashEarnings(tCar_spec *pCar1@<EAX>, tCar_spec *pCar2@<EDX>)
  1338. void CrashEarnings(tCar_spec* pCar1, tCar_spec* pCar2) {
  1339.     LOG_TRACE("(%p, %p)", pCar1, pCar2);
  1340.  
  1341.     if (DoCrashEarnings(pCar1, pCar2)) {
  1342.         SortOutSmoke(pCar1);
  1343.         SortOutSmoke(pCar2);
  1344.     }
  1345. }
  1346.