Subversion Repositories Games.Carmageddon

Rev

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

  1. #include "displays.h"
  2. #include "brender.h"
  3. #include "constants.h"
  4. #include "controls.h"
  5. #include "depth.h"
  6. #include "flicplay.h"
  7. #include "globvars.h"
  8. #include "globvrkm.h"
  9. #include "globvrpb.h"
  10. #include "grafdata.h"
  11. #include "graphics.h"
  12. #include "harness/trace.h"
  13. #include "netgame.h"
  14. #include "pd/sys.h"
  15. #include "utility.h"
  16. #include <stdlib.h>
  17.  
  18. int gLast_fancy_index;
  19. int gLast_credit_headup__displays; // suffix added to avoid duplicate symbol
  20. int gLast_time_credit_headup;
  21. tDR_font* gCached_font;
  22. br_font* gBR_fonts[4];
  23. tQueued_headup gQueued_headups[4];
  24. int gOld_times[10];
  25. int gLast_fancy_headup;
  26. tU32 gLast_time_earn_time;
  27. tU32 gLast_centre_headup;
  28. tU32 gLast_fancy_time;
  29. int gQueued_headup_count;
  30. tU32 gLast_earn_time;
  31. tU32 gLast_time_credit_amount;
  32. int gLast_credit_amount;
  33. tHeadup gHeadups[15];
  34. int gLaps_headup;
  35. int gCar_kill_count_headup;
  36. int gTimer_headup;
  37. int gTime_awarded_headup;
  38. int gPed_kill_count_headup;
  39. int gDim_amount;
  40. br_pixelmap* gHeadup_images[32]; // Modified by DethRace for the demo
  41. int gNet_cash_headup;
  42. int gNet_ped_headup;
  43. int gCredits_lost_headup;
  44. int gCredits_won_headup;
  45.  
  46. // IDA: void __usercall GetTimerString(char *pStr@<EAX>, int pFudge_colon@<EDX>)
  47. void GetTimerString(char* pStr, int pFudge_colon) {
  48.     LOG_TRACE("(\"%s\", %d)", pStr, pFudge_colon);
  49.  
  50.     TimerString(gTimer, pStr, pFudge_colon, 0);
  51. }
  52.  
  53. // IDA: void __cdecl InitHeadups()
  54. void InitHeadups(void) {
  55.     int i;
  56.     LOG_TRACE("()");
  57.  
  58.     for (i = 0; i < 15; i++) {
  59.         gHeadups[i].type = eHeadup_unused;
  60.     }
  61.     gBR_fonts[0] = BrFontProp7x9;
  62.     gBR_fonts[1] = BrFontFixed3x5;
  63.     gBR_fonts[2] = gFont_7;
  64.     gBR_fonts[3] = gHeadup_font;
  65. }
  66.  
  67. // IDA: void __usercall ClearHeadup(int pIndex@<EAX>)
  68. void ClearHeadup(int pIndex) {
  69.     LOG_TRACE("(%d)", pIndex);
  70.  
  71.     gHeadups[pIndex].type = eHeadup_unused;
  72. }
  73.  
  74. // IDA: void __usercall ClearHeadupSlot(int pSlot_index@<EAX>)
  75. void ClearHeadupSlot(int pSlot_index) {
  76.     int i;
  77.     tHeadup* the_headup;
  78.     LOG_TRACE("(%d)", pSlot_index);
  79.  
  80.     the_headup = gHeadups;
  81.     for (i = 0; i < COUNT_OF(gHeadups); i++) {
  82.         if (the_headup->type != eHeadup_unused && the_headup->slot_index == pSlot_index) {
  83.             ClearHeadup(i);
  84.             return;
  85.         }
  86.         the_headup++;
  87.     }
  88. }
  89.  
  90. // IDA: void __cdecl ClearHeadups()
  91. void ClearHeadups(void) {
  92.     int i;
  93.     LOG_TRACE("()");
  94.  
  95.     for (i = 0; i < COUNT_OF(gHeadups); i++) {
  96.         if (gHeadups[i].type) {
  97.             ClearHeadup(i);
  98.         }
  99.     }
  100.     gLast_fancy_index = -1;
  101.     gLast_credit_headup__displays = -1;
  102.     gLast_time_credit_headup = -1;
  103.     gLast_earn_time = 0;
  104.     gLast_fancy_time = 0;
  105.     gLast_time_earn_time = 0;
  106.     for (i = 0; i < COUNT_OF(gOld_times); i++) {
  107.         gOld_times[i] = 0;
  108.     }
  109.     gQueued_headup_count = 0;
  110.     gLast_centre_headup = 0;
  111. }
  112.  
  113. // IDA: int __usercall HeadupActive@<EAX>(int pIndex@<EAX>)
  114. int HeadupActive(int pIndex) {
  115.     LOG_TRACE("(%d)", pIndex);
  116.  
  117.     return gHeadups[pIndex].type != eHeadup_unused;
  118. }
  119.  
  120. // IDA: void __usercall DRPixelmapText(br_pixelmap *pPixelmap@<EAX>, int pX@<EDX>, int pY@<EBX>, tDR_font *pFont@<ECX>, char *pText, int pRight_edge)
  121. void DRPixelmapText(br_pixelmap* pPixelmap, int pX, int pY, tDR_font* pFont, char* pText, int pRight_edge) {
  122.     int i;
  123.     int x;
  124.     int len;
  125.     int chr;
  126.     int ch_width;
  127.     unsigned char* ch;
  128.     LOG_TRACE9("(%p, %d, %d, %p, \"%s\", %d)", pPixelmap, pX, pY, pFont, pText, pRight_edge);
  129.  
  130.     len = strlen(pText);
  131.     ch = (unsigned char*)pText;
  132.     if (pX >= 0 && pPixelmap->width >= pRight_edge && pY >= 0 && (pY + pFont->height) <= pPixelmap->height) {
  133.         x = pX;
  134.         for (i = 0; i < len; i++) {
  135.             chr = ch[i] - pFont->offset;
  136.             ch_width = pFont->width_table[chr];
  137.             DRPixelmapRectangleOnscreenCopy(
  138.                 gBack_screen,
  139.                 x,
  140.                 pY,
  141.                 pFont->images,
  142.                 0,
  143.                 pFont->height * chr,
  144.                 ch_width,
  145.                 pFont->height);
  146.  
  147.             x += ch_width + pFont->spacing;
  148.         }
  149.     } else {
  150.         x = pX;
  151.         for (i = 0; i < len; i++) {
  152.             chr = ch[i] - pFont->offset;
  153.             ch_width = pFont->width_table[chr];
  154.             DRPixelmapRectangleMaskedCopy(
  155.                 gBack_screen,
  156.                 x,
  157.                 pY,
  158.                 pFont->images,
  159.                 0,
  160.                 pFont->height * chr,
  161.                 ch_width,
  162.                 pFont->height);
  163.  
  164.             x += ch_width + pFont->spacing;
  165.         }
  166.     }
  167. }
  168.  
  169. // IDA: void __usercall DRPixelmapCleverText2(br_pixelmap *pPixelmap@<EAX>, int pX@<EDX>, int pY@<EBX>, tDR_font *pFont@<ECX>, signed char *pText, int pRight_edge)
  170. void DRPixelmapCleverText2(br_pixelmap* pPixelmap, int pX, int pY, tDR_font* pFont, char* pText, int pRight_edge) {
  171.     int i;
  172.     int x;
  173.     int len;
  174.     int chr;
  175.     int ch_width;
  176.     unsigned char* ch;
  177.     tDR_font* new_font;
  178.     LOG_TRACE("(%p, %d, %d, %p, %p, %d)", pPixelmap, pX, pY, pFont, pText, pRight_edge);
  179.  
  180.     x = pX;
  181.     len = strlen(pText);
  182.     ch = (unsigned char*)pText;
  183.     if (pX >= 0 && pPixelmap->width >= pRight_edge && pY >= 0 && pY + pFont->height <= pPixelmap->height) {
  184.         for (i = 0; i < len; i++) {
  185.             if (*ch < 224) {
  186.                 chr = *ch - pFont->offset;
  187.                 ch_width = pFont->width_table[chr];
  188.                 DRPixelmapRectangleOnscreenCopy(
  189.                     gBack_screen,
  190.                     x,
  191.                     pY,
  192.                     pFont->images,
  193.                     0,
  194.                     chr * pFont->height,
  195.                     ch_width,
  196.                     pFont->height);
  197.                 x += ch_width + pFont->spacing;
  198.             } else {
  199.                 new_font = &gFonts[-*ch + 256];
  200.                 pY -= (new_font->height - pFont->height) / 2;
  201.                 pFont = new_font;
  202.             }
  203.             ch++;
  204.         }
  205.     } else {
  206.         for (i = 0; i < len; i++) {
  207.             if (*ch < 224) {
  208.                 chr = *ch - pFont->offset;
  209.                 ch_width = pFont->width_table[chr];
  210.                 DRPixelmapRectangleMaskedCopy(
  211.                     gBack_screen,
  212.                     x,
  213.                     pY,
  214.                     pFont->images,
  215.                     0,
  216.                     chr * pFont->height,
  217.                     ch_width,
  218.                     pFont->height);
  219.                 x += ch_width + pFont->spacing;
  220.             } else {
  221.                 new_font = &gFonts[-*ch + 256];
  222.                 pY -= (new_font->height - pFont->height) / 2;
  223.                 pFont = new_font;
  224.             }
  225.             ch++;
  226.         }
  227.     }
  228. }
  229.  
  230. // IDA: void __usercall DeviouslyDimRectangle(br_pixelmap *pPixelmap@<EAX>, int pLeft@<EDX>, int pTop@<EBX>, int pRight@<ECX>, int pBottom, int pKnock_out_corners)
  231. void DeviouslyDimRectangle(br_pixelmap* pPixelmap, int pLeft, int pTop, int pRight, int pBottom, int pKnock_out_corners) {
  232.     LOG_TRACE("(%p, %d, %d, %d, %d, %d)", pPixelmap, pLeft, pTop, pRight, pBottom, pKnock_out_corners);
  233.     NOT_IMPLEMENTED();
  234. }
  235.  
  236. // IDA: void __cdecl DimRectangle(br_pixelmap *pPixelmap, int pLeft, int pTop, int pRight, int pBottom, int pKnock_out_corners)
  237. void DimRectangle(br_pixelmap* pPixelmap, int pLeft, int pTop, int pRight, int pBottom, int pKnock_out_corners) {
  238.     tU8* ptr;
  239.     tU8* depth_table_ptr;
  240.     tU8* right_ptr;
  241.     int x;
  242.     int y;
  243.     int line_skip;
  244.     int width;
  245.     LOG_TRACE9("(%p, %d, %d, %d, %d, %d)", pPixelmap, pLeft, pTop, pRight, pBottom, pKnock_out_corners);
  246.  
  247.     ptr = (tU8*)pPixelmap->pixels + pLeft + pPixelmap->row_bytes * pTop;
  248.     line_skip = pPixelmap->row_bytes - pRight + pLeft;
  249.     depth_table_ptr = gDepth_shade_table->pixels;
  250.     x = gDepth_shade_table->row_bytes * gDim_amount;
  251.     width = pRight - pLeft;
  252.  
  253.     if (pKnock_out_corners) {
  254.         ptr++;
  255.         for (right_ptr = ptr + width - 2; ptr < right_ptr; ptr++) {
  256.             *ptr = depth_table_ptr[*ptr + x];
  257.         }
  258.         ptr += line_skip + 1;
  259.         for (y = pTop + 1; y < (pBottom - 1); y++, ptr += line_skip) {
  260.             for (right_ptr = ptr + width; ptr < right_ptr; ptr++) {
  261.                 *ptr = depth_table_ptr[*ptr + x];
  262.             }
  263.         }
  264.         ptr++;
  265.         for (right_ptr = ptr + width - 2; ptr < right_ptr; ptr++) {
  266.             *ptr = depth_table_ptr[*ptr + x];
  267.         }
  268.     } else {
  269.         for (y = pTop; y < pBottom; y++) {
  270.             for (right_ptr = ptr + width; ptr < right_ptr; ptr++) {
  271.                 *ptr = depth_table_ptr[*ptr + x];
  272.             }
  273.             ptr += line_skip;
  274.         }
  275.     }
  276. }
  277.  
  278. // IDA: void __cdecl DimAFewBits()
  279. void DimAFewBits(void) {
  280.     int i;
  281.     LOG_TRACE("()");
  282.  
  283.     int dim_index; // Added
  284.     dim_index = gProgram_state.cockpit_on && gProgram_state.cockpit_image_index >= 0;
  285.     for (i = 0; i < gProgram_state.current_car.dim_count[dim_index]; i++) {
  286.         DimRectangle(
  287.             gBack_screen,
  288.             gProgram_state.current_car.dim_left[dim_index][i],
  289.             gProgram_state.current_car.dim_top[dim_index][i],
  290.             gProgram_state.current_car.dim_right[dim_index][i],
  291.             gProgram_state.current_car.dim_bottom[dim_index][i],
  292.             1);
  293.     }
  294. }
  295.  
  296. // IDA: void __cdecl KillOldestQueuedHeadup()
  297. void KillOldestQueuedHeadup(void) {
  298.     LOG_TRACE("()");
  299.  
  300.     gQueued_headup_count--;
  301.     memmove(&gQueued_headups[0], &gQueued_headups[1], gQueued_headup_count * sizeof(tQueued_headup));
  302. }
  303.  
  304. // IDA: void __usercall DubreyBar(int pX_index@<EAX>, int pY@<EDX>, int pColour@<EBX>)
  305. void DubreyBar(int pX_index, int pY, int pColour) {
  306.     int x;
  307.     LOG_TRACE("(%d, %d, %d)", pX_index, pY, pColour);
  308.  
  309.     x = gCurrent_graf_data->ps_bar_left - gCurrent_graf_data->ps_x_pitch * pX_index;
  310.     BrPixelmapLine(gBack_screen, x, pY, x, gCurrent_graf_data->ps_bar_height + pY, pColour);
  311. }
  312.  
  313. // IDA: void __usercall DoPSPowerHeadup(int pY@<EAX>, int pLevel@<EDX>, char *pName@<EBX>, int pBar_colour@<ECX>)
  314. void DoPSPowerHeadup(int pY, int pLevel, char* pName, int pBar_colour) {
  315.     //char s[16]; // Pierre-Marie Baty -- unused variable
  316.     int i;
  317.     LOG_TRACE("(%d, %d, \"%s\", %d)", pY, pLevel, pName, pBar_colour);
  318.  
  319.     DimRectangle(gBack_screen, gCurrent_graf_data->ps_dim_left, pY, gCurrent_graf_data->ps_dim_right, gCurrent_graf_data->ps_dim_height + pY, 1);
  320.     TransDRPixelmapText(gBack_screen, gCurrent_graf_data->ps_name_left, gCurrent_graf_data->ps_name_top_border + pY, gFonts + 6, pName, gBack_screen->width);
  321.  
  322.     for (i = (6 - gCurrent_graf_data->ps_bars_per_level) * gCurrent_graf_data->ps_bars_per_level + 1; i > (gCurrent_graf_data->ps_bars_per_level * pLevel + 1); i--) {
  323.         DubreyBar(i, pY + gCurrent_graf_data->ps_bar_top_border, 0);
  324.     }
  325.     for (i = gCurrent_graf_data->ps_bars_per_level * pLevel + 1; i >= 0; i--) {
  326.         DubreyBar(i, pY + gCurrent_graf_data->ps_bar_top_border, pBar_colour);
  327.     }
  328. }
  329.  
  330. // IDA: void __cdecl DoPSPowerupHeadups()
  331. void DoPSPowerupHeadups(void) {
  332.     LOG_TRACE("()");
  333.  
  334.     DoPSPowerHeadup(gCurrent_graf_data->armour_headup_y[gProgram_state.cockpit_on], gProgram_state.current_car.power_up_levels[0], "A", 45);
  335.     DoPSPowerHeadup(gCurrent_graf_data->power_headup_y[gProgram_state.cockpit_on], gProgram_state.current_car.power_up_levels[1], "P", 99);
  336.     DoPSPowerHeadup(gCurrent_graf_data->offense_headup_y[gProgram_state.cockpit_on], gProgram_state.current_car.power_up_levels[2], "O", 4);
  337. }
  338.  
  339. // IDA: void __usercall DoHeadups(tU32 pThe_time@<EAX>)
  340. void DoHeadups(tU32 pThe_time) {
  341.     int i;
  342.     int x_offset;
  343.     int y_offset;
  344.     tHeadup* the_headup;
  345.     int time_factor;
  346.     LOG_TRACE("(%d)", pThe_time);
  347.  
  348.     if (gNet_mode) {
  349.         DoNetScores();
  350.     }
  351.     if (gQueued_headup_count && PDGetTotalTime() - gLast_centre_headup >= 1000) {
  352.         NewTextHeadupSlot(4, gQueued_headups[0].flash_rate,
  353.             gQueued_headups[0].lifetime,
  354.             gQueued_headups[0].font_index,
  355.             gQueued_headups[0].text);
  356.         KillOldestQueuedHeadup();
  357.     }
  358.  
  359.     for (i = 0; i < COUNT_OF(gHeadups); i++) {
  360.         the_headup = &gHeadups[i];
  361.         if (the_headup->type != eHeadup_unused
  362.             && (gProgram_state.which_view == eView_forward || !the_headup->cockpit_anchored)
  363.             && (the_headup->type == eHeadup_image
  364.                 || the_headup->type == eHeadup_fancy
  365.                 || (the_headup->type == eHeadup_text && the_headup->data.text_info.text[0] != '\0')
  366.                 || ((the_headup->type == eHeadup_coloured_text || the_headup->type == eHeadup_box_text)
  367.                     && the_headup->data.text_info.text[0] != '\0'))) {
  368.             if (the_headup->type == eHeadup_fancy || the_headup->end_time == 0 || pThe_time < the_headup->end_time) {
  369.                 if (the_headup->dimmed_background) {
  370.                     DimRectangle(
  371.                         gBack_screen,
  372.                         the_headup->dim_left,
  373.                         the_headup->dim_top,
  374.                         the_headup->dim_right,
  375.                         the_headup->dim_bottom,
  376.                         1);
  377.                 }
  378.                 if (the_headup->flash_period == 0
  379.                     || Flash(the_headup->flash_period, &the_headup->last_flash, &the_headup->flash_state)) {
  380.                     switch (the_headup->type) {
  381.                     case eHeadup_text:
  382.                         if (the_headup->cockpit_anchored) {
  383.                             y_offset = gScreen_wobble_y;
  384.                         } else {
  385.                             y_offset = 0;
  386.                         }
  387.                         if (the_headup->cockpit_anchored) {
  388.                             x_offset = gScreen_wobble_x;
  389.                         } else {
  390.                             x_offset = 0;
  391.                         }
  392.                         TransBrPixelmapText(
  393.                             gBack_screen,
  394.                             x_offset + the_headup->x,
  395.                             y_offset + the_headup->y,
  396.                             the_headup->data.text_info.colour,
  397.                             the_headup->data.text_info.font,
  398.                             the_headup->data.text_info.text);
  399.                         break;
  400.                     case eHeadup_coloured_text:
  401.                         if (the_headup->clever) {
  402.                             if (the_headup->cockpit_anchored) {
  403.                                 y_offset = gScreen_wobble_y;
  404.                             } else {
  405.                                 y_offset = 0;
  406.                             }
  407.                             if (the_headup->cockpit_anchored) {
  408.                                 x_offset = gScreen_wobble_x;
  409.                             } else {
  410.                                 x_offset = 0;
  411.                             }
  412.                             TransDRPixelmapCleverText(
  413.                                 gBack_screen,
  414.                                 x_offset + the_headup->x,
  415.                                 y_offset + the_headup->y,
  416.                                 the_headup->data.coloured_text_info.coloured_font,
  417.                                 the_headup->data.coloured_text_info.text,
  418.                                 the_headup->right_edge);
  419.                         } else {
  420.                             if (the_headup->cockpit_anchored) {
  421.                                 y_offset = gScreen_wobble_y;
  422.                             } else {
  423.                                 y_offset = 0;
  424.                             }
  425.                             if (the_headup->cockpit_anchored) {
  426.                                 x_offset = gScreen_wobble_x;
  427.                             } else {
  428.                                 x_offset = 0;
  429.                             }
  430.                             TransDRPixelmapText(
  431.                                 gBack_screen,
  432.                                 x_offset + the_headup->x,
  433.                                 y_offset + the_headup->y,
  434.                                 the_headup->data.coloured_text_info.coloured_font,
  435.                                 the_headup->data.coloured_text_info.text,
  436.                                 the_headup->right_edge);
  437.                         }
  438.                         break;
  439.                     case eHeadup_image:
  440.                         if (the_headup->cockpit_anchored) {
  441.                             y_offset = gScreen_wobble_y;
  442.                         } else {
  443.                             y_offset = 0;
  444.                         }
  445.                         if (the_headup->cockpit_anchored) {
  446.                             x_offset = gScreen_wobble_x;
  447.                         } else {
  448.                             x_offset = 0;
  449.                         }
  450.                         DRPixelmapRectangleMaskedCopy(
  451.                             gBack_screen,
  452.                             x_offset + the_headup->x,
  453.                             y_offset + the_headup->y,
  454.                             the_headup->data.image_info.image,
  455.                             0,
  456.                             0,
  457.                             the_headup->data.image_info.image->width,
  458.                             the_headup->data.image_info.image->height);
  459.                         break;
  460.  
  461.                     case eHeadup_fancy:
  462.                         switch (the_headup->data.fancy_info.fancy_stage) {
  463.                         case eFancy_stage_incoming:
  464.                             the_headup->data.fancy_info.offset -= 500 * gFrame_period / 1000;
  465.                             if (the_headup->data.fancy_info.offset <= the_headup->data.fancy_info.shear_amount) {
  466.                                 the_headup->data.fancy_info.offset = the_headup->data.fancy_info.shear_amount;
  467.                                 the_headup->data.fancy_info.fancy_stage = eFancy_stage_halting;
  468.                                 the_headup->data.fancy_info.start_time = GetTotalTime();
  469.                             }
  470.                             DRPixelmapRectangleShearedCopy(
  471.                                 gBack_screen,
  472.                                 the_headup->x + the_headup->data.fancy_info.offset,
  473.                                 the_headup->y,
  474.                                 the_headup->data.fancy_info.image,
  475.                                 0,
  476.                                 0,
  477.                                 the_headup->data.fancy_info.image->width,
  478.                                 the_headup->data.fancy_info.image->height,
  479.                                 -65536);
  480.                             break;
  481.                         case eFancy_stage_halting:
  482.                             time_factor = 1000 * (pThe_time - the_headup->data.fancy_info.start_time) / 100;
  483.                             if (time_factor > 1000) {
  484.                                 if (time_factor > 1500) {
  485.                                     time_factor = 1500;
  486.                                     the_headup->data.fancy_info.fancy_stage = eFancy_stage_waiting;
  487.                                     the_headup->data.fancy_info.start_time = GetTotalTime();
  488.                                 }
  489.                                 DRPixelmapRectangleShearedCopy(
  490.                                     gBack_screen,
  491.                                     the_headup->x - (1500 - time_factor) * the_headup->data.fancy_info.shear_amount / 500,
  492.                                     the_headup->y,
  493.                                     the_headup->data.image_info.image,
  494.                                     0,
  495.                                     0,
  496.                                     the_headup->data.image_info.image->width,
  497.                                     the_headup->data.image_info.image->height,
  498.                                     (((1500 - time_factor) * the_headup->data.fancy_info.shear_amount / 500) << 16)
  499.                                         / the_headup->data.image_info.image->height);
  500.                             } else {
  501. #if DETHRACE_FIX_BUGS
  502.                                 time_factor = MAX(time_factor, 0);
  503. #endif
  504.                                 DRPixelmapRectangleShearedCopy(
  505.                                     gBack_screen,
  506.                                     the_headup->x - the_headup->data.fancy_info.shear_amount * (time_factor - 500) / 500,
  507.                                     the_headup->y,
  508.                                     the_headup->data.image_info.image,
  509.                                     0,
  510.                                     0,
  511.                                     the_headup->data.image_info.image->width,
  512.                                     the_headup->data.image_info.image->height,
  513.                                     ((the_headup->data.fancy_info.shear_amount * (time_factor - 500) / 500) << 16)
  514.                                         / the_headup->data.image_info.image->height);
  515.                             }
  516.                             break;
  517.                         case eFancy_stage_waiting:
  518.                             if (pThe_time - the_headup->data.fancy_info.start_time > 1000) {
  519.                                 the_headup->data.fancy_info.fancy_stage = eFancy_stage_readying;
  520.                                 the_headup->data.fancy_info.start_time = GetTotalTime();
  521.                             }
  522.                             DRPixelmapRectangleMaskedCopy(
  523.                                 gBack_screen,
  524.                                 the_headup->x,
  525.                                 the_headup->y,
  526.                                 the_headup->data.image_info.image,
  527.                                 0,
  528.                                 0,
  529.                                 the_headup->data.image_info.image->width,
  530.                                 the_headup->data.image_info.image->height);
  531.                             break;
  532.                         case eFancy_stage_readying:
  533.                             time_factor = 1000 * (pThe_time - the_headup->data.fancy_info.start_time) / 100;
  534.                             if (time_factor > 1000) {
  535.                                 time_factor = 1000;
  536.                                 the_headup->data.fancy_info.fancy_stage = eFancy_stage_leaving;
  537.                                 the_headup->data.fancy_info.start_time = GetTotalTime();
  538.                                 the_headup->data.fancy_info.offset = 0;
  539.                             }
  540.                             DRPixelmapRectangleShearedCopy(
  541.                                 gBack_screen,
  542.                                 the_headup->x,
  543.                                 the_headup->y,
  544.                                 the_headup->data.image_info.image,
  545.                                 0,
  546.                                 0,
  547.                                 the_headup->data.image_info.image->width,
  548.                                 the_headup->data.image_info.image->height,
  549.                                 -(((time_factor * the_headup->data.fancy_info.shear_amount / 1000) << 16)
  550.                                     / the_headup->data.image_info.image->height));
  551.                             break;
  552.                         case eFancy_stage_leaving:
  553.                             the_headup->data.fancy_info.offset -= 500 * gFrame_period / 1000;
  554.                             if (the_headup->data.fancy_info.offset <= the_headup->data.fancy_info.end_offset) {
  555.                                 ClearHeadup(i);
  556.                             } else {
  557.                                 DRPixelmapRectangleShearedCopy(
  558.                                     gBack_screen,
  559.                                     the_headup->data.fancy_info.offset + the_headup->x,
  560.                                     the_headup->y,
  561.                                     the_headup->data.image_info.image,
  562.                                     0,
  563.                                     0,
  564.                                     the_headup->data.image_info.image->width,
  565.                                     the_headup->data.image_info.image->height,
  566.                                     -65536);
  567.                             }
  568.                             break;
  569.                         default:
  570.                             break;
  571.                         }
  572.                         break;
  573.  
  574.                     case eHeadup_box_text:
  575.                         if (the_headup->cockpit_anchored) {
  576.                             y_offset = gScreen_wobble_y;
  577.                         } else {
  578.                             y_offset = 0;
  579.                         }
  580.                         if (the_headup->cockpit_anchored) {
  581.                             x_offset = gScreen_wobble_x;
  582.                         } else {
  583.                             x_offset = 0;
  584.                         }
  585.                         OoerrIveGotTextInMeBoxMissus(
  586.                             the_headup->data.coloured_text_info.coloured_font - gFonts,
  587.                             the_headup->data.coloured_text_info.text,
  588.                             gBack_screen,
  589.                             gBack_screen->width / 10,
  590.                             x_offset + the_headup->y,
  591.                             9 * gBack_screen->width / 10,
  592.                             y_offset + the_headup->y + 60,
  593.                             1);
  594.                         break;
  595.                     default:
  596.                         break;
  597.                     }
  598.                 }
  599.             } else {
  600.                 ClearHeadup(i);
  601.             }
  602.         }
  603.     }
  604.     DoPSPowerupHeadups();
  605. }
  606.  
  607. // IDA: int __usercall FindAHeadupHoleWoofBarkSoundsABitRude@<EAX>(int pSlot_index@<EAX>)
  608. int FindAHeadupHoleWoofBarkSoundsABitRude(int pSlot_index) {
  609.     int i;
  610.     int empty_one;
  611.     tHeadup* the_headup;
  612.     LOG_TRACE("(%d)", pSlot_index);
  613.  
  614.     empty_one = -1;
  615.     for (i = 0, the_headup = gHeadups; i < COUNT_OF(gHeadups); i++, the_headup++) {
  616.         if (pSlot_index >= 0 && the_headup->slot_index == pSlot_index) {
  617.             return i;
  618.         }
  619.         if (the_headup->type == eHeadup_unused) {
  620.             empty_one = i;
  621.         }
  622.     }
  623.     return empty_one;
  624. }
  625.  
  626. // IDA: int __usercall DRTextWidth@<EAX>(tDR_font *pFont@<EAX>, char *pText@<EDX>)
  627. int DRTextWidth(tDR_font* pFont, char* pText) {
  628.     int i;
  629.     int len;
  630.     int result;
  631.     char* c;
  632.     LOG_TRACE("(%p, \"%s\")", pFont, pText);
  633.  
  634.     c = pText;
  635.     result = 0;
  636.     len = strlen(pText);
  637.  
  638.     for (i = 0; i < len; i++) {
  639.         result += pFont->width_table[*c - pFont->offset];
  640.         c++;
  641.     }
  642.     return result + pFont->spacing * (len - 1);
  643. }
  644.  
  645. // IDA: int __usercall DRTextCleverWidth@<EAX>(tDR_font *pFont@<EAX>, signed char *pText@<EDX>)
  646. int DRTextCleverWidth(tDR_font* pFont, signed char* pText) {
  647.     int i;
  648.     int len;
  649.     int result;
  650.     unsigned char* c;
  651.     LOG_TRACE("(%p, %p)", pFont, pText);
  652.  
  653.     result = 0;
  654.     len = strlen((char*)pText) + 1;
  655.  
  656.     for (i = 0, c = (unsigned char*)pText; i < len; i++, c++) {
  657.         if (*c < 224) {
  658.             if (i < (len - 1)) {
  659.                 result += pFont->spacing;
  660.             }
  661.             result += pFont->width_table[*c - pFont->offset];
  662.         } else {
  663.             pFont = &gFonts[256 - *c];
  664.         }
  665.     }
  666.     return result;
  667. }
  668.  
  669. // IDA: void __usercall DRPixelmapCentredText(br_pixelmap *pPixelmap@<EAX>, int pX@<EDX>, int pY@<EBX>, tDR_font *pFont@<ECX>, char *pText)
  670. void DRPixelmapCentredText(br_pixelmap* pPixelmap, int pX, int pY, tDR_font* pFont, char* pText) {
  671.     int width_over_2;
  672.     LOG_TRACE("(%p, %d, %d, %p, \"%s\")", pPixelmap, pX, pY, pFont, pText);
  673.  
  674.     width_over_2 = DRTextWidth(pFont, pText) / 2;
  675.     TransDRPixelmapText(pPixelmap, pX - width_over_2, pY, pFont, pText, width_over_2 + pX);
  676. }
  677.  
  678. // IDA: int __usercall IsHeadupTextClever@<EAX>(signed char *pText@<EAX>)
  679. int IsHeadupTextClever(signed char* pText) {
  680.     LOG_TRACE("(%p)", pText);
  681.  
  682.     while (*pText) {
  683.         if (*pText < 0) {
  684.             return 1;
  685.         }
  686.         pText++;
  687.     }
  688.     return 0;
  689. }
  690.  
  691. // IDA: int __usercall MungeHeadupWidth@<EAX>(tHeadup *pHeadup@<EAX>)
  692. int MungeHeadupWidth(tHeadup* pHeadup) {
  693.     int width;
  694.     LOG_TRACE("(%p)", pHeadup);
  695.  
  696.     width = 0;
  697.     if (pHeadup->type == eHeadup_box_text) {
  698.         return 0;
  699.     }
  700.     if (pHeadup->type == eHeadup_coloured_text) {
  701.         pHeadup->clever = IsHeadupTextClever((signed char*)(&pHeadup->data.text_info.text));
  702.         if (pHeadup->justification) {
  703.             if (pHeadup->justification == eJust_right) {
  704.                 if (pHeadup->clever) {
  705.                     width = DRTextCleverWidth(
  706.                         pHeadup->data.coloured_text_info.coloured_font,
  707.                         (signed char*)(&pHeadup->data.text_info.text));
  708.                 } else {
  709.                     width = DRTextWidth(pHeadup->data.coloured_text_info.coloured_font, pHeadup->data.text_info.text);
  710.                 }
  711.                 pHeadup->x = pHeadup->original_x - width;
  712.             } else if (pHeadup->justification == eJust_centre) {
  713.                 if (pHeadup->clever) {
  714.                     width = DRTextCleverWidth(
  715.                         pHeadup->data.coloured_text_info.coloured_font,
  716.                         (signed char*)(&pHeadup->data.text_info.text));
  717.                 } else {
  718.                     width = DRTextWidth(pHeadup->data.coloured_text_info.coloured_font, pHeadup->data.text_info.text);
  719.                 }
  720.                 pHeadup->x = pHeadup->original_x - width / 2;
  721.             }
  722.         } else {
  723.             pHeadup->x = pHeadup->original_x;
  724.         }
  725.     } else {
  726.         pHeadup->clever = 0;
  727.         if (pHeadup->justification) {
  728.             if (pHeadup->justification == eJust_right) {
  729.                 width = BrPixelmapTextWidth(gBack_screen, pHeadup->data.text_info.font, pHeadup->data.text_info.text);
  730.                 pHeadup->x = pHeadup->original_x - width;
  731.             } else if (pHeadup->justification == eJust_centre) {
  732.                 width = BrPixelmapTextWidth(gBack_screen, pHeadup->data.text_info.font, pHeadup->data.text_info.text);
  733.                 pHeadup->x = pHeadup->original_x - width / 2;
  734.             }
  735.         } else {
  736.             pHeadup->x = pHeadup->original_x;
  737.         }
  738.     }
  739.     return width;
  740. }
  741.  
  742. // IDA: int __usercall NewTextHeadupSlot2@<EAX>(int pSlot_index@<EAX>, int pFlash_rate@<EDX>, int pLifetime@<EBX>, int pFont_index@<ECX>, char *pText, int pQueue_it)
  743. int NewTextHeadupSlot2(int pSlot_index, int pFlash_rate, int pLifetime, int pFont_index, char* pText, int pQueue_it) {
  744.     int index;
  745.     tHeadup* the_headup;
  746.     tHeadup_slot* headup_slot;
  747.     tU32 time;
  748.     LOG_TRACE("(%d, %d, %d, %d, \"%s\", %d)", pSlot_index, pFlash_rate, pLifetime, pFont_index, pText, pQueue_it);
  749.  
  750.     time = PDGetTotalTime();
  751.     if (pQueue_it && pSlot_index == 4 && time - gLast_centre_headup < 1000) {
  752.         if (gQueued_headup_count == 4) {
  753.             KillOldestQueuedHeadup();
  754.         }
  755.         gQueued_headups[gQueued_headup_count].flash_rate = pFlash_rate;
  756.         gQueued_headups[gQueued_headup_count].lifetime = pLifetime;
  757.         gQueued_headups[gQueued_headup_count].font_index = pFont_index;
  758.         strcpy(gQueued_headups[gQueued_headup_count].text, pText);
  759.         gQueued_headup_count++;
  760.         index = -1;
  761.     } else {
  762.         index = FindAHeadupHoleWoofBarkSoundsABitRude(pSlot_index);
  763.         if (index >= 0) {
  764.             if (pSlot_index == 4) {
  765.                 gLast_centre_headup = time;
  766.             }
  767.             headup_slot = &gProgram_state.current_car.headup_slots[gProgram_state.cockpit_on][pSlot_index];
  768.             the_headup = &gHeadups[index];
  769.             the_headup->data.coloured_text_info.coloured_font = &gFonts[-pFont_index];
  770.             if (pSlot_index == 4) {
  771.                 the_headup->type = eHeadup_box_text;
  772.             } else {
  773.                 the_headup->type = eHeadup_coloured_text;
  774.             }
  775.             if (!pText) {
  776.                 LOG_PANIC("panic");
  777.             }
  778.             strcpy(the_headup->data.text_info.text, pText);
  779.  
  780.             the_headup->slot_index = pSlot_index;
  781.             the_headup->justification = headup_slot->justification;
  782.             if (pSlot_index < 0) {
  783.                 the_headup->cockpit_anchored = 0;
  784.             } else {
  785.                 the_headup->cockpit_anchored = headup_slot->cockpit_anchored;
  786.             }
  787.             the_headup->dimmed_background = headup_slot->dimmed_background;
  788.             the_headup->dim_left = headup_slot->dim_left;
  789.             the_headup->dim_top = headup_slot->dim_top;
  790.             the_headup->dim_right = headup_slot->dim_right;
  791.             the_headup->dim_bottom = headup_slot->dim_bottom;
  792.             the_headup->original_x = headup_slot->x;
  793.             the_headup->right_edge = MungeHeadupWidth(the_headup) + the_headup->x;
  794.             the_headup->y = headup_slot->y;
  795.             if (pFlash_rate) {
  796.                 the_headup->flash_period = 1000 / pFlash_rate;
  797.             } else {
  798.                 the_headup->flash_period = 0;
  799.             }
  800.             the_headup->last_flash = 0;
  801.             the_headup->flash_state = 0;
  802.             if (pLifetime) {
  803.                 the_headup->end_time = GetTotalTime() + pLifetime;
  804.             } else {
  805.                 the_headup->end_time = 0;
  806.             }
  807.         }
  808.     }
  809.     return index;
  810. }
  811.  
  812. // IDA: int __usercall NewTextHeadupSlot@<EAX>(int pSlot_index@<EAX>, int pFlash_rate@<EDX>, int pLifetime@<EBX>, int pFont_index@<ECX>, char *pText)
  813. int NewTextHeadupSlot(int pSlot_index, int pFlash_rate, int pLifetime, int pFont_index, char* pText) {
  814.     LOG_TRACE("(%d, %d, %d, %d, \"%s\")", pSlot_index, pFlash_rate, pLifetime, pFont_index, pText);
  815.  
  816.     return NewTextHeadupSlot2(pSlot_index, pFlash_rate, pLifetime, pFont_index, pText, 1);
  817. }
  818.  
  819. // IDA: int __usercall NewImageHeadupSlot@<EAX>(int pSlot_index@<EAX>, int pFlash_rate@<EDX>, int pLifetime@<EBX>, int pImage_index@<ECX>)
  820. int NewImageHeadupSlot(int pSlot_index, int pFlash_rate, int pLifetime, int pImage_index) {
  821.     int index;
  822.     tHeadup* the_headup;
  823.     tHeadup_slot* headup_slot;
  824.     LOG_TRACE("(%d, %d, %d, %d)", pSlot_index, pFlash_rate, pLifetime, pImage_index);
  825.  
  826.     index = FindAHeadupHoleWoofBarkSoundsABitRude(pSlot_index);
  827.     if (index >= 0) {
  828.         headup_slot = &gProgram_state.current_car.headup_slots[gProgram_state.cockpit_on][pSlot_index];
  829.         the_headup = &gHeadups[index];
  830.         the_headup->type = eHeadup_image;
  831.         the_headup->slot_index = pSlot_index;
  832.         the_headup->justification = headup_slot->justification;
  833.         if (pSlot_index < 0) {
  834.             the_headup->cockpit_anchored = 0;
  835.         } else {
  836.             the_headup->cockpit_anchored = headup_slot->cockpit_anchored;
  837.         }
  838.         the_headup->dimmed_background = headup_slot->dimmed_background;
  839.         the_headup->dim_left = headup_slot->dim_left;
  840.         the_headup->dim_top = headup_slot->dim_top;
  841.         the_headup->dim_right = headup_slot->dim_right;
  842.         the_headup->dim_bottom = headup_slot->dim_bottom;
  843.         the_headup->original_x = headup_slot->x;
  844.  
  845.         if (headup_slot->justification) {
  846.             if (headup_slot->justification == eJust_right) {
  847.                 the_headup->x = the_headup->original_x - gHeadup_images[pImage_index]->width;
  848.             } else if (headup_slot->justification == eJust_centre) {
  849.                 the_headup->x = the_headup->original_x - gHeadup_images[pImage_index]->width / 2;
  850.             }
  851.         } else {
  852.             the_headup->x = the_headup->original_x;
  853.         }
  854.         the_headup->y = headup_slot->y;
  855.         if (pFlash_rate) {
  856.             the_headup->flash_period = 1000 / pFlash_rate;
  857.         } else {
  858.             the_headup->flash_period = 0;
  859.         }
  860.         the_headup->last_flash = 0;
  861.         the_headup->flash_state = 0;
  862.         if (pLifetime) {
  863.             the_headup->end_time = GetTotalTime() + pLifetime;
  864.         } else {
  865.             the_headup->end_time = 0;
  866.         }
  867.         the_headup->data.image_info.image = gHeadup_images[pImage_index];
  868.     }
  869.     return index;
  870. }
  871.  
  872. // IDA: void __usercall DoFancyHeadup(int pIndex@<EAX>)
  873. void DoFancyHeadup(int pIndex) {
  874.     tU32 the_time;
  875.     tHeadup* the_headup;
  876.     int temp_ref;
  877.     LOG_TRACE("(%d)", pIndex);
  878.  
  879.     the_time = GetTotalTime();
  880.     if (!gMap_mode && (gLast_fancy_index < 0 || the_time - gLast_fancy_time > 2000 || gLast_fancy_index <= pIndex)) {
  881.         temp_ref = NewImageHeadupSlot(6, 0, 2000, pIndex + 10);
  882.         if (temp_ref >= 0) {
  883.             gLast_fancy_headup = temp_ref;
  884.             gLast_fancy_index = pIndex;
  885.             gLast_fancy_time = the_time;
  886.             the_headup = &gHeadups[temp_ref];
  887.             the_headup->type = eHeadup_fancy;
  888.             the_headup->data.fancy_info.offset = (the_headup->data.image_info.image->width + gBack_screen->width) / 2;
  889.             the_headup->data.fancy_info.end_offset = -the_headup->data.fancy_info.offset;
  890.             the_headup->data.fancy_info.fancy_stage = eFancy_stage_incoming;
  891.             the_headup->data.fancy_info.shear_amount = the_headup->data.image_info.image->height;
  892.         }
  893.     }
  894. }
  895.  
  896. // IDA: void __cdecl AdjustHeadups()
  897. void AdjustHeadups(void) {
  898.     int i;
  899.     int delta_x;
  900.     int delta_y;
  901.     tHeadup* the_headup;
  902.     LOG_TRACE("()");
  903.  
  904.     the_headup = gHeadups;
  905.     for (i = 0; i < COUNT_OF(gHeadups); i++) {
  906.         the_headup = &gHeadups[i];
  907.         if (the_headup->type == eHeadup_unused) {
  908.             continue;
  909.         }
  910.         delta_x = gProgram_state.current_car.headup_slots[gProgram_state.cockpit_on && gProgram_state.cockpit_image_index >= 0][the_headup->slot_index].x
  911.             - gProgram_state.current_car.headup_slots[!gProgram_state.cockpit_on || gProgram_state.cockpit_image_index < 0][the_headup->slot_index].x;
  912.         delta_y = gProgram_state.current_car.headup_slots[gProgram_state.cockpit_on && gProgram_state.cockpit_image_index >= 0][the_headup->slot_index].y
  913.             - gProgram_state.current_car.headup_slots[!gProgram_state.cockpit_on || gProgram_state.cockpit_image_index < 0][the_headup->slot_index].y;
  914.         the_headup->x += delta_x;
  915.         the_headup->original_x += delta_x;
  916.         the_headup->y += delta_y;
  917.         the_headup->dim_left += delta_x;
  918.         the_headup->dim_top += delta_y;
  919.         the_headup->dim_right += delta_x;
  920.         the_headup->dim_bottom += delta_y;
  921.     }
  922. }
  923.  
  924. // IDA: void __usercall MoveHeadupTo(int pHeadup_index@<EAX>, int pNew_x@<EDX>, int pNew_y@<EBX>)
  925. void MoveHeadupTo(int pHeadup_index, int pNew_x, int pNew_y) {
  926.     int delta_x;
  927.     //tHeadup* the_headup; // Pierre-Marie Baty -- unused variable
  928.     LOG_TRACE("(%d, %d, %d)", pHeadup_index, pNew_x, pNew_y);
  929.  
  930.     if (pHeadup_index >= 0) {
  931.         delta_x = gHeadups[pHeadup_index].x - gHeadups[pHeadup_index].original_x;
  932.         gHeadups[pHeadup_index].original_x = pNew_x;
  933.         gHeadups[pHeadup_index].x = pNew_x + delta_x;
  934.         gHeadups[pHeadup_index].y = pNew_y;
  935.     }
  936. }
  937.  
  938. // IDA: void __usercall ChangeHeadupText(int pHeadup_index@<EAX>, char *pNew_text@<EDX>)
  939. void ChangeHeadupText(int pHeadup_index, char* pNew_text) {
  940.     tHeadup* the_headup;
  941.     LOG_TRACE("(%d, \"%s\")", pHeadup_index, pNew_text);
  942.  
  943.     if (pHeadup_index >= 0) {
  944.         the_headup = &gHeadups[pHeadup_index];
  945.         strcpy(the_headup->data.text_info.text, pNew_text);
  946.         MungeHeadupWidth(the_headup);
  947.     }
  948. }
  949.  
  950. // IDA: void __usercall ChangeHeadupImage(int pHeadup_index@<EAX>, int pNew_image@<EDX>)
  951. void ChangeHeadupImage(int pHeadup_index, int pNew_image) {
  952.     tHeadup* the_headup;
  953.     LOG_TRACE("(%d, %d)", pHeadup_index, pNew_image);
  954.  
  955.     if (pHeadup_index >= 0) {
  956.         the_headup = &gHeadups[pHeadup_index];
  957.         the_headup->data.image_info.image = gHeadup_images[pNew_image];
  958.         switch (the_headup->justification) {
  959.         case eJust_left:
  960.             the_headup->x = the_headup->original_x;
  961.             break;
  962.         case eJust_right:
  963.             the_headup->x = the_headup->original_x - the_headup->data.image_info.image->width;
  964.             break;
  965.         case eJust_centre:
  966.             the_headup->x = the_headup->original_x - the_headup->data.image_info.image->width / 2;
  967.             break;
  968.         }
  969.     }
  970. }
  971.  
  972. // IDA: void __usercall ChangeHeadupColour(int pHeadup_index@<EAX>, int pNew_colour@<EDX>)
  973. void ChangeHeadupColour(int pHeadup_index, int pNew_colour) {
  974.     LOG_TRACE("(%d, %d)", pHeadup_index, pNew_colour);
  975.  
  976.     if (pHeadup_index >= 0) {
  977.         gHeadups[pHeadup_index].data.text_info.colour = gColours[pNew_colour];
  978.     }
  979. }
  980.  
  981. // IDA: void __usercall DoDamageScreen(tU32 pThe_time@<EAX>)
  982. void DoDamageScreen(tU32 pThe_time) {
  983.     int i;
  984.     int y_pitch;
  985.     int the_step;
  986.     int the_wobble_x;
  987.     int the_wobble_y;
  988.     br_pixelmap* the_image;
  989.     tDamage_unit* the_damage;
  990.     LOG_TRACE("(%d)", pThe_time);
  991.  
  992.     if (&gProgram_state.current_car != gCar_to_view || gProgram_state.current_car_index != gProgram_state.current_car.index) {
  993.         return;
  994.     }
  995.     if (gProgram_state.cockpit_on && gProgram_state.cockpit_image_index >= 0) {
  996.         if (gProgram_state.which_view != eView_forward) {
  997.             return;
  998.         }
  999.         the_wobble_x = gScreen_wobble_x;
  1000.         the_wobble_y = gScreen_wobble_y;
  1001.     } else {
  1002.         the_wobble_x = gProgram_state.current_car.damage_x_offset;
  1003.         the_wobble_y = gProgram_state.current_car.damage_y_offset;
  1004.         if (gProgram_state.current_car.damage_background != NULL) {
  1005.             DRPixelmapRectangleMaskedCopy(
  1006.                 gBack_screen,
  1007.                 gProgram_state.current_car.damage_background_x,
  1008.                 gProgram_state.current_car.damage_background_y,
  1009.                 gProgram_state.current_car.damage_background,
  1010.                 0,
  1011.                 0,
  1012.                 gProgram_state.current_car.damage_background->width,
  1013.                 gProgram_state.current_car.damage_background->height);
  1014.         }
  1015.     }
  1016.     for (i = 0; i < COUNT_OF(gProgram_state.current_car.damage_units); i++) {
  1017.         the_damage = &gProgram_state.current_car.damage_units[i];
  1018.         if (i != eDamage_driver) {
  1019.             the_image = the_damage->images;
  1020.             the_step = 5 * the_damage->damage_level / 100;
  1021.             y_pitch = (the_image->height / 2) / 5;
  1022.             DRPixelmapRectangleMaskedCopy(
  1023.                 gBack_screen,
  1024.                 the_wobble_x + gProgram_state.current_car.damage_units[i].x_coord,
  1025.                 the_wobble_y + gProgram_state.current_car.damage_units[i].y_coord,
  1026.                 the_image,
  1027.                 0,
  1028.                 y_pitch * (2 * the_step + ((pThe_time / the_damage->periods[the_step]) & 1)),
  1029.                 the_image->width,
  1030.                 y_pitch);
  1031.         }
  1032.     }
  1033. }
  1034.  
  1035. // IDA: void __cdecl PoshDrawLine(float pAngle, br_pixelmap *pDestn, int pX1, int pY1, int pX2, int pY2, int pColour)
  1036. void PoshDrawLine(float pAngle, br_pixelmap* pDestn, int pX1, int pY1, int pX2, int pY2, int pColour) {
  1037.     LOG_TRACE("(%f, %p, %d, %d, %d, %d, %d)", pAngle, pDestn, pX1, pY1, pX2, pY2, pColour);
  1038.  
  1039.     if (pColour < 0) {
  1040.         if (pAngle >= 0.785 && pAngle <= 5.498 && (pAngle <= 2.356 || pAngle >= 3.926)) {
  1041.             if ((pAngle <= 0.785 || pAngle >= 1.57) && (pAngle <= 3.926 || pAngle >= 4.712)) {
  1042.                 DRDrawLine(pDestn, pX1 - 1, pY1, pX2 - 1, pY2, -pColour - 1);
  1043.                 DRDrawLine(pDestn, pX1 + 1, pY1, pX2 + 1, pY2, 1 - pColour);
  1044.             } else {
  1045.                 DRDrawLine(pDestn, pX1 - 1, pY1, pX2 - 1, pY2, 1 - pColour);
  1046.                 DRDrawLine(pDestn, pX1 + 1, pY1, pX2 + 1, pY2, -pColour - 1);
  1047.             }
  1048.         } else {
  1049.             DRDrawLine(pDestn, pX1, pY1 + 1, pX2, pY2 + 1, -pColour - 1);
  1050.             DRDrawLine(pDestn, pX1, pY1 - 1, pX2, pY2 - 1, 1 - pColour);
  1051.         }
  1052.         DRDrawLine(pDestn, pX1, pY1, pX2, pY2, -pColour);
  1053.     } else {
  1054.         DRDrawLine(pDestn, pX1, pY1, pX2, pY2, pColour);
  1055.     }
  1056. }
  1057.  
  1058. // IDA: void __usercall DoInstruments(tU32 pThe_time@<EAX>)
  1059. void DoInstruments(tU32 pThe_time) {
  1060.     br_pixelmap* speedo_image;
  1061.     br_pixelmap* tacho_image;
  1062.     int the_wobble_x;
  1063.     int the_wobble_y;
  1064.     int gear;
  1065.     int gear_height; /* Added by dethrace. */
  1066.     double the_angle;
  1067.     double the_angle2;
  1068.     double sin_angle;
  1069.     double cos_angle;
  1070.     double speed_mph;
  1071.     LOG_TRACE("(%d)", pThe_time);
  1072.  
  1073.     if (gProgram_state.current_car_index == gProgram_state.current_car.index) {
  1074.         speed_mph = gCar_to_view->speedo_speed * WORLD_SCALE / 1600.0f * 3600000.0f;
  1075.         if (speed_mph < 0.0f) {
  1076.             speed_mph = 0.0f;
  1077.         }
  1078.         if (gProgram_state.cockpit_on && gProgram_state.cockpit_image_index >= 0) {
  1079.             if (gProgram_state.which_view != eView_forward) {
  1080.                 return;
  1081.             }
  1082.             the_wobble_x = gScreen_wobble_x;
  1083.             the_wobble_y = gScreen_wobble_y;
  1084.         } else {
  1085.             the_wobble_x = 0;
  1086.             the_wobble_y = 0;
  1087.         }
  1088.         tacho_image = gProgram_state.current_car.tacho_image[gProgram_state.cockpit_on];
  1089.         if (gProgram_state.current_car.tacho_radius_2[gProgram_state.cockpit_on] >= 0) {
  1090.             if (gCar_to_view->red_line >= gCar_to_view->revs) {
  1091.                 the_angle = DEG_TO_RAD((double)(gProgram_state.current_car.tacho_end_angle[gProgram_state.cockpit_on] - gProgram_state.current_car.tacho_start_angle[gProgram_state.cockpit_on]) * gCar_to_view->revs / (double)gCar_to_view->red_line + (double)gProgram_state.current_car.tacho_start_angle[gProgram_state.cockpit_on]);
  1092.             } else {
  1093.                 the_angle = DEG_TO_RAD((double)gProgram_state.current_car.tacho_end_angle[gProgram_state.cockpit_on]);
  1094.             }
  1095.             if (the_angle >= 0.0) {
  1096.                 if (the_angle >= TAU) {
  1097.                     the_angle -= TAU;
  1098.                 }
  1099.             } else {
  1100.                 the_angle += TAU;
  1101.             }
  1102.             the_angle2 = DR_PI_OVER_2 - the_angle;
  1103.             if (the_angle2 < 0) {
  1104.                 the_angle2 += TAU;
  1105.             }
  1106.             if (the_angle2 > DR_3PI_OVER_2) {
  1107.                 cos_angle = gCosine_array[(unsigned int)((TAU - the_angle2) / DR_PI * 128.0)];
  1108.             } else if (the_angle2 > DR_PI) {
  1109.                 cos_angle = -gCosine_array[(unsigned int)((the_angle2 - DR_PI) / DR_PI * 128.0)];
  1110. #if defined(DETHRACE_FIX_BUGS)
  1111.             } else if (the_angle2 >= DR_PI_OVER_2) {
  1112. #else
  1113.             } else if (the_angle2 > DR_PI_OVER_2) {
  1114. #endif
  1115.                 cos_angle = -gCosine_array[(unsigned int)((DR_PI - the_angle2) / DR_PI * 128.0)];
  1116.             } else {
  1117.                 cos_angle = gCosine_array[(unsigned int)(the_angle2 / DR_PI * 128.0)];
  1118.             }
  1119.             if (the_angle > DR_3PI_OVER_2) {
  1120.                 sin_angle = gCosine_array[(unsigned int)((TAU - the_angle) / DR_PI * 128.0)];
  1121.             } else if (the_angle > DR_PI) {
  1122.                 sin_angle = -gCosine_array[(unsigned int)((the_angle - DR_PI) / DR_PI * 128.0)];
  1123. #if defined(DETHRACE_FIX_BUGS)
  1124.             } else if (the_angle >= DR_PI_OVER_2) {
  1125. #else
  1126.             } else if (the_angle > DR_PI_OVER_2) {
  1127. #endif
  1128.                 sin_angle = -gCosine_array[(unsigned int)((DR_PI - the_angle) / DR_PI * 128.0)];
  1129.             } else {
  1130.                 sin_angle = gCosine_array[(unsigned int)(the_angle / DR_PI * 128.0)];
  1131.             }
  1132.             if (tacho_image != NULL) {
  1133.                 DRPixelmapRectangleMaskedCopy(
  1134.                     gBack_screen,
  1135.                     the_wobble_x + gProgram_state.current_car.tacho_x[gProgram_state.cockpit_on],
  1136.                     the_wobble_y + gProgram_state.current_car.tacho_y[gProgram_state.cockpit_on],
  1137.                     tacho_image,
  1138.                     0,
  1139.                     0,
  1140.                     tacho_image->width,
  1141.                     tacho_image->height);
  1142.             }
  1143.  
  1144.             PoshDrawLine(
  1145.                 the_angle,
  1146.                 gBack_screen,
  1147.                 ((double)gProgram_state.current_car.tacho_radius_1[gProgram_state.cockpit_on] * sin_angle
  1148.                     + (double)gProgram_state.current_car.tacho_centre_x[gProgram_state.cockpit_on]
  1149.                     + (double)the_wobble_x),
  1150.                 ((double)gProgram_state.current_car.tacho_centre_y[gProgram_state.cockpit_on]
  1151.                     - (double)gProgram_state.current_car.tacho_radius_1[gProgram_state.cockpit_on] * cos_angle
  1152.                     + (double)the_wobble_y),
  1153.                 ((double)gProgram_state.current_car.tacho_radius_2[gProgram_state.cockpit_on] * sin_angle
  1154.                     + (double)gProgram_state.current_car.tacho_centre_x[gProgram_state.cockpit_on]
  1155.                     + (double)the_wobble_x),
  1156.                 ((double)gProgram_state.current_car.tacho_centre_y[gProgram_state.cockpit_on]
  1157.                     - (double)gProgram_state.current_car.tacho_radius_2[gProgram_state.cockpit_on] * cos_angle
  1158.                     + (double)the_wobble_y),
  1159.                 gProgram_state.current_car.tacho_needle_colour[gProgram_state.cockpit_on]);
  1160.         } else if (tacho_image != NULL) {
  1161.             BrPixelmapRectangleCopy(
  1162.                 gBack_screen,
  1163.                 the_wobble_x + gProgram_state.current_car.tacho_x[gProgram_state.cockpit_on],
  1164.                 the_wobble_y + gProgram_state.current_car.tacho_y[gProgram_state.cockpit_on],
  1165.                 gProgram_state.current_car.tacho_image[gProgram_state.cockpit_on],
  1166.                 0,
  1167.                 0,
  1168.                 ((gCar_to_view->revs - 1.0) / (double)gCar_to_view->red_line * (double)gProgram_state.current_car.tacho_image[gProgram_state.cockpit_on]->width + 1.0),
  1169.                 gProgram_state.current_car.tacho_image[gProgram_state.cockpit_on]->height);
  1170.         }
  1171.         if (!gProgram_state.cockpit_on || gProgram_state.cockpit_image_index < 0 || gProgram_state.which_view == eView_forward) {
  1172.             if (gCar_to_view->gear < 0) {
  1173.                 gear = -1;
  1174.             } else {
  1175.                 gear = gCar_to_view->gear;
  1176.             }
  1177. #if defined(DETHRACE_FIX_BUGS)
  1178. /*
  1179.  * The OG derives gear mask height of 16 or 28 by `gears_image->height / 8`, but
  1180.  * this is only valid for HGEARS.PIX, which contains 8 gear images. Hardcoding
  1181.  * this number fixes gear rendering for cars using HGEARS4.PIX, which consists
  1182.  * of 11 gear images.
  1183.  */
  1184. #define GEAR_HEIGHT 16
  1185. #define GEAR_HEIGHT_HIRES 28
  1186. #else
  1187. #define GEAR_HEIGHT ((int)gProgram_state.current_car.gears_image->height / 8)
  1188. #define GEAR_HEIGHT_HIRES GEAR_HEIGHT
  1189. #endif
  1190.             gear_height = gGraf_spec_index ? GEAR_HEIGHT_HIRES : GEAR_HEIGHT;
  1191.             DRPixelmapRectangleMaskedCopy(
  1192.                 gBack_screen,
  1193.                 the_wobble_x + gProgram_state.current_car.gear_x[gProgram_state.cockpit_on],
  1194.                 the_wobble_y + gProgram_state.current_car.gear_y[gProgram_state.cockpit_on],
  1195.                 gProgram_state.current_car.gears_image,
  1196.                 0,
  1197.                 (gear + 1) * gear_height,
  1198.                 gProgram_state.current_car.gears_image->width,
  1199.                 gear_height);
  1200.         }
  1201.         speedo_image = gProgram_state.current_car.speedo_image[gProgram_state.cockpit_on];
  1202.         if (gProgram_state.current_car.speedo_radius_2[gProgram_state.cockpit_on] >= 0) {
  1203.             if (speedo_image && (!gProgram_state.cockpit_on || gProgram_state.cockpit_image_index < 0)) {
  1204.                 DRPixelmapRectangleMaskedCopy(
  1205.                     gBack_screen,
  1206.                     the_wobble_x + gProgram_state.current_car.speedo_x[gProgram_state.cockpit_on],
  1207.                     the_wobble_y + gProgram_state.current_car.speedo_y[gProgram_state.cockpit_on],
  1208.                     speedo_image,
  1209.                     0,
  1210.                     0,
  1211.                     speedo_image->width,
  1212.                     speedo_image->height);
  1213.             }
  1214.             if ((double)gProgram_state.current_car.max_speed >= speed_mph) {
  1215.                 the_angle = DEG_TO_RAD((double)(gProgram_state.current_car.speedo_end_angle[gProgram_state.cockpit_on] - gProgram_state.current_car.speedo_start_angle[gProgram_state.cockpit_on]) * speed_mph / (double)gProgram_state.current_car.max_speed + (double)gProgram_state.current_car.speedo_start_angle[gProgram_state.cockpit_on]);
  1216.             } else {
  1217.                 the_angle = DEG_TO_RAD((double)gProgram_state.current_car.speedo_end_angle[gProgram_state.cockpit_on]);
  1218.             }
  1219.  
  1220.             if (the_angle < 0.0) {
  1221.                 the_angle = the_angle + TAU;
  1222.             } else if (the_angle >= TAU) {
  1223.                 the_angle -= TAU;
  1224.             }
  1225.             the_angle2 = DR_PI_OVER_2 - the_angle;
  1226.             if (the_angle2 < 0.0) {
  1227.                 the_angle2 += TAU;
  1228.             }
  1229.             if (the_angle2 > DR_3PI_OVER_2) {
  1230.                 cos_angle = gCosine_array[(unsigned int)((TAU - the_angle2) / DR_PI * 128.0)];
  1231.             } else if (the_angle2 > DR_PI) {
  1232.                 cos_angle = -gCosine_array[(unsigned int)((the_angle2 - DR_PI) / DR_PI * 128.0)];
  1233.             } else if (the_angle2 > DR_PI_OVER_2) {
  1234.                 cos_angle = -gCosine_array[(unsigned int)((DR_PI - the_angle2) / DR_PI * 128.0)];
  1235.             } else {
  1236.                 cos_angle = gCosine_array[(unsigned int)(the_angle2 / DR_PI * 128.0)];
  1237.             }
  1238.  
  1239.             if (the_angle > DR_3PI_OVER_2) {
  1240.                 sin_angle = gCosine_array[(unsigned int)((TAU - the_angle) / DR_PI * 128.0)];
  1241.             } else if (the_angle > DR_PI) {
  1242.                 sin_angle = -gCosine_array[(unsigned int)((the_angle - DR_PI) / DR_PI * 128.0)];
  1243.             } else if (the_angle > DR_PI_OVER_2) {
  1244.                 sin_angle = -gCosine_array[(unsigned int)((DR_PI - the_angle) / DR_PI * 128.0)];
  1245.             } else {
  1246.                 sin_angle = gCosine_array[(unsigned int)(the_angle / DR_PI * 128.0)];
  1247.             }
  1248.  
  1249.             PoshDrawLine(
  1250.                 the_angle,
  1251.                 gBack_screen,
  1252.                 ((double)gProgram_state.current_car.speedo_radius_1[gProgram_state.cockpit_on] * sin_angle
  1253.                     + (double)gProgram_state.current_car.speedo_centre_x[gProgram_state.cockpit_on]
  1254.                     + (double)the_wobble_x),
  1255.                 ((double)gProgram_state.current_car.speedo_centre_y[gProgram_state.cockpit_on]
  1256.                     - (double)gProgram_state.current_car.speedo_radius_1[gProgram_state.cockpit_on] * cos_angle
  1257.                     + (double)the_wobble_y),
  1258.                 ((double)gProgram_state.current_car.speedo_radius_2[gProgram_state.cockpit_on] * sin_angle
  1259.                     + (double)gProgram_state.current_car.speedo_centre_x[gProgram_state.cockpit_on]
  1260.                     + (double)the_wobble_x),
  1261.                 ((double)gProgram_state.current_car.speedo_centre_y[gProgram_state.cockpit_on]
  1262.                     - (double)gProgram_state.current_car.speedo_radius_2[gProgram_state.cockpit_on] * cos_angle
  1263.                     + (double)the_wobble_y),
  1264.                 gProgram_state.current_car.speedo_needle_colour[gProgram_state.cockpit_on]);
  1265.             if (speedo_image != NULL && gProgram_state.cockpit_on && gProgram_state.cockpit_image_index >= 0) {
  1266.                 DRPixelmapRectangleMaskedCopy(
  1267.                     gBack_screen,
  1268.                     the_wobble_x + gProgram_state.current_car.speedo_x[gProgram_state.cockpit_on],
  1269.                     the_wobble_y + gProgram_state.current_car.speedo_y[gProgram_state.cockpit_on],
  1270.                     speedo_image,
  1271.                     0,
  1272.                     0,
  1273.                     speedo_image->width,
  1274.                     speedo_image->height);
  1275.             }
  1276.         } else if (speedo_image != NULL) {
  1277.             DrawNumberAt(
  1278.                 speedo_image,
  1279.                 the_wobble_x + gProgram_state.current_car.speedo_x[gProgram_state.cockpit_on],
  1280.                 the_wobble_y + gProgram_state.current_car.speedo_y[gProgram_state.cockpit_on],
  1281.                 gProgram_state.current_car.speedo_x_pitch[gProgram_state.cockpit_on],
  1282.                 gProgram_state.current_car.speedo_y_pitch[gProgram_state.cockpit_on],
  1283.                 speed_mph,
  1284.                 3,
  1285.                 1);
  1286.         }
  1287.     }
  1288. }
  1289.  
  1290. // IDA: void __usercall DoSteeringWheel(tU32 pThe_time@<EAX>)
  1291. void DoSteeringWheel(tU32 pThe_time) {
  1292.     br_pixelmap* hands_image;
  1293.     int hands_index;
  1294.     LOG_TRACE("(%d)", pThe_time);
  1295.  
  1296.     if (gProgram_state.current_car_index == gProgram_state.current_car.index && gProgram_state.cockpit_on && gProgram_state.cockpit_image_index >= 0 && gProgram_state.which_view == eView_forward) {
  1297.         hands_index = (int)floor(gProgram_state.current_car.number_of_hands_images * ((1.f - gProgram_state.current_car.steering_angle / 10.f) / 2.f));
  1298.         if (hands_index < 0) {
  1299.             hands_index = 0;
  1300.         } else if (hands_index >= gProgram_state.current_car.number_of_hands_images) {
  1301.             hands_index = gProgram_state.current_car.number_of_hands_images - 1;
  1302.         }
  1303.         hands_image = gProgram_state.current_car.lhands_images[hands_index];
  1304.         if (hands_image != NULL) {
  1305.             DRPixelmapRectangleMaskedCopy(gBack_screen,
  1306.                 gProgram_state.current_car.lhands_x[hands_index] + gScreen_wobble_x,
  1307.                 gProgram_state.current_car.lhands_y[hands_index] + gScreen_wobble_y,
  1308.                 hands_image, 0, 0, hands_image->width, hands_image->height);
  1309.         }
  1310.         hands_image = gProgram_state.current_car.rhands_images[hands_index];
  1311.         if (hands_image != NULL) {
  1312.             DRPixelmapRectangleMaskedCopy(gBack_screen,
  1313.                 gProgram_state.current_car.rhands_x[hands_index] + gScreen_wobble_x,
  1314.                 gProgram_state.current_car.rhands_y[hands_index] + gScreen_wobble_y,
  1315.                 hands_image, 0, 0, hands_image->width, hands_image->height);
  1316.         }
  1317.     }
  1318. }
  1319.  
  1320. // IDA: void __cdecl ChangingView()
  1321. void ChangingView(void) {
  1322.     tU32 the_time;
  1323.     LOG_TRACE("()");
  1324.  
  1325.     if (gProgram_state.new_view == eView_undefined) {
  1326.         return;
  1327.     }
  1328.     the_time = PDGetTotalTime() - gProgram_state.view_change_start;
  1329.     gScreen_wobble_x = 0;
  1330.     gScreen_wobble_y = 0;
  1331.     if (the_time > 175 && gProgram_state.which_view == gProgram_state.new_view) {
  1332.         if (gProgram_state.pending_view != eView_undefined) {
  1333.             if (gProgram_state.pending_view == eView_left) {
  1334.                 LookLeft();
  1335.                 return;
  1336.             }
  1337.             if (gProgram_state.pending_view == eView_forward) {
  1338.                 LookForward();
  1339.                 return;
  1340.             }
  1341.             if (gProgram_state.pending_view == eView_right) {
  1342.                 LookRight();
  1343.                 return;
  1344.             }
  1345.             gScreen_wobble_x = 0;
  1346.             gScreen_wobble_y = 0;
  1347.             return;
  1348.         }
  1349.         if (gProgram_state.which_view == gProgram_state.new_view) {
  1350.             gProgram_state.new_view = eView_undefined;
  1351.             gScreen_wobble_x = 0;
  1352.             gScreen_wobble_y = 0;
  1353.             return;
  1354.         }
  1355.     }
  1356.     if (the_time < 88) {
  1357.         if (gProgram_state.old_view < gProgram_state.new_view) {
  1358.             gScreen_wobble_x = -gCurrent_graf_data->cock_margin_x * the_time * 2.f / 175.f;
  1359.         } else {
  1360.             gScreen_wobble_x = gCurrent_graf_data->cock_margin_x * the_time * 2.f / 175.f;
  1361.         }
  1362.     } else {
  1363.         gProgram_state.which_view = gProgram_state.new_view;
  1364.         switch (gProgram_state.new_view) {
  1365.         case eView_left:
  1366.             gProgram_state.cockpit_image_index = 1;
  1367.             break;
  1368.         case eView_forward:
  1369.             gProgram_state.cockpit_image_index = 0;
  1370.             break;
  1371.         case eView_right:
  1372.             gProgram_state.cockpit_image_index = 2;
  1373.             break;
  1374.         default:
  1375.             break;
  1376.         }
  1377.         AdjustRenderScreenSize();
  1378.         if (gProgram_state.old_view < gProgram_state.new_view) {
  1379.             gScreen_wobble_x = gCurrent_graf_data->cock_margin_x * (175 - the_time) * 2.f / 175.f;
  1380.         } else {
  1381.             gScreen_wobble_x = -gCurrent_graf_data->cock_margin_x * (175 - the_time) * 2.f / 175.f;
  1382.         }
  1383.     }
  1384. }
  1385.  
  1386. // IDA: void __usercall EarnCredits2(int pAmount@<EAX>, char *pPrefix_text@<EDX>)
  1387. void EarnCredits2(int pAmount, char* pPrefix_text) {
  1388.     char s[256];
  1389.     int original_amount;
  1390.     tU32 the_time;
  1391.     LOG_TRACE("(%d, \"%s\")", pAmount, pPrefix_text);
  1392.  
  1393.     if (gRace_finished) {
  1394.         return;
  1395.     }
  1396.     the_time = GetTotalTime();
  1397.     if (pAmount == 0) {
  1398.         return;
  1399.     }
  1400.     if (gNet_mode != eNet_mode_none && gProgram_state.credits_earned - gProgram_state.credits_lost + pAmount < 0) {
  1401.         pAmount = gProgram_state.credits_lost - gProgram_state.credits_lost;
  1402.     }
  1403.     original_amount = pAmount;
  1404.     if (gLast_credit_headup__displays >= 0 && the_time - gLast_earn_time < 2000) {
  1405.         pAmount += gLast_credit_amount;
  1406.     }
  1407.     gLast_credit_amount = pAmount;
  1408.     if (pAmount >= 2) {
  1409.         sprintf(s, "%s%d %s", pPrefix_text, pAmount, GetMiscString(kMiscString_Credits));
  1410.         gProgram_state.credits_earned += original_amount;
  1411.     } else if (pAmount >= 1) {
  1412.         sprintf(s, "%s1 %s", pPrefix_text, GetMiscString(kMiscString_Credit));
  1413.         gProgram_state.credits_earned += original_amount;
  1414.     } else if (pAmount >= -1) {
  1415.         sprintf(s, "%s%s 1 %s", pPrefix_text, GetMiscString(kMiscString_Lost), GetMiscString(kMiscString_Credit));
  1416.         gProgram_state.credits_lost -= original_amount;
  1417.     } else {
  1418.         sprintf(s, "%s%s %d %s", GetMiscString(kMiscString_Lost), pPrefix_text, -pAmount, GetMiscString(kMiscString_Credits));
  1419.         gProgram_state.credits_lost -= original_amount;
  1420.     }
  1421.     gLast_credit_headup__displays = NewTextHeadupSlot(4, 0, 2000, -4, s);
  1422.     gLast_earn_time = the_time;
  1423. }
  1424.  
  1425. // IDA: void __usercall EarnCredits(int pAmount@<EAX>)
  1426. void EarnCredits(int pAmount) {
  1427.     LOG_TRACE("(%d)", pAmount);
  1428.  
  1429.     EarnCredits2(pAmount, "");
  1430. }
  1431.  
  1432. // IDA: int __usercall SpendCredits@<EAX>(int pAmount@<EAX>)
  1433. int SpendCredits(int pAmount) {
  1434.     int amount;
  1435.     LOG_TRACE("(%d)", pAmount);
  1436.  
  1437.     gProgram_state.credits_lost += pAmount;
  1438.     if (gNet_mode == eNet_mode_none) {
  1439.         return 0;
  1440.     }
  1441.     amount = gProgram_state.credits_earned - gProgram_state.credits_lost;
  1442.     if (gProgram_state.credits_earned - gProgram_state.credits_lost >= 0) {
  1443.         return 0;
  1444.     }
  1445.     gProgram_state.credits_lost = gProgram_state.credits_earned;
  1446.     return amount;
  1447. }
  1448.  
  1449. // IDA: void __usercall AwardTime(tU32 pTime@<EAX>)
  1450. void AwardTime(tU32 pTime) {
  1451.     char s[256];
  1452.     tU32 original_amount;
  1453.     tU32 the_time;
  1454.     int i;
  1455.     LOG_TRACE("(%d)", pTime);
  1456.  
  1457.     if (gRace_finished || gFreeze_timer || gNet_mode != eNet_mode_none || pTime == 0) {
  1458.         return;
  1459.     }
  1460.  
  1461.     original_amount = pTime;
  1462.     the_time = GetTotalTime();
  1463.     for (i = COUNT_OF(gOld_times) - 1; i > 0; i--) {
  1464.         gOld_times[i] = gOld_times[i - 1];
  1465.     }
  1466.     gOld_times[0] = pTime;
  1467.     if (gLast_time_credit_headup >= 0 && (the_time - gLast_time_earn_time) < 2000) {
  1468.         pTime += gLast_time_credit_amount;
  1469.     }
  1470.     gLast_time_credit_amount = pTime;
  1471.     gTimer += original_amount * 1000;
  1472.     s[0] = '+';
  1473.     TimerString(1000 * pTime, &s[1], 0, 0);
  1474.     gLast_time_credit_headup = NewTextHeadupSlot(11, 0, 2000, -2, s);
  1475.     gLast_time_earn_time = the_time;
  1476. }
  1477.  
  1478. // IDA: void __usercall DrawRectangle(br_pixelmap *pPixelmap@<EAX>, int pLeft@<EDX>, int pTop@<EBX>, int pRight@<ECX>, int pBottom, int pColour)
  1479. void DrawRectangle(br_pixelmap* pPixelmap, int pLeft, int pTop, int pRight, int pBottom, int pColour) {
  1480.     LOG_TRACE("(%p, %d, %d, %d, %d, %d)", pPixelmap, pLeft, pTop, pRight, pBottom, pColour);
  1481.  
  1482.     BrPixelmapLine(pPixelmap, pLeft, pTop, pRight, pTop, pColour);
  1483.     BrPixelmapLine(pPixelmap, pLeft, pBottom, pRight, pBottom, pColour);
  1484.     BrPixelmapLine(pPixelmap, pLeft, pTop, pLeft, pBottom, pColour);
  1485.     BrPixelmapLine(pPixelmap, pRight, pTop, pRight, pBottom, pColour);
  1486. }
  1487.  
  1488. // IDA: void __usercall DrawRRectangle(br_pixelmap *pPixelmap@<EAX>, int pLeft@<EDX>, int pTop@<EBX>, int pRight@<ECX>, int pBottom, int pColour)
  1489. void DrawRRectangle(br_pixelmap* pPixelmap, int pLeft, int pTop, int pRight, int pBottom, int pColour) {
  1490.     LOG_TRACE("(%p, %d, %d, %d, %d, %d)", pPixelmap, pLeft, pTop, pRight, pBottom, pColour);
  1491.  
  1492.     BrPixelmapLine(pPixelmap, pLeft + 1, pTop, pRight - 1, pTop, pColour);
  1493.     BrPixelmapLine(pPixelmap, pLeft + 1, pBottom, pRight - 1, pBottom, pColour);
  1494.     BrPixelmapLine(pPixelmap, pLeft, pTop + 1, pLeft, pBottom - 1, pColour);
  1495.     BrPixelmapLine(pPixelmap, pRight, pTop + 1, pRight, pBottom - 1, pColour);
  1496. }
  1497.  
  1498. // IDA: void __usercall OoerrIveGotTextInMeBoxMissus(int pFont_index@<EAX>, char *pText@<EDX>, br_pixelmap *pPixelmap@<EBX>, int pLeft@<ECX>, int pTop, int pRight, int pBottom, int pCentred)
  1499. void OoerrIveGotTextInMeBoxMissus(int pFont_index, char* pText, br_pixelmap* pPixelmap, int pLeft, int pTop, int pRight, int pBottom, int pCentred) {
  1500.     tDR_font* font;
  1501.     int width;
  1502.     int current_width;
  1503.     int i;
  1504.     int centre;
  1505.     int line_char_index;
  1506.     int input_str_index;
  1507.     int start_line;
  1508.     int current_y;
  1509.     int font_needed_loading;
  1510.     char line[256];
  1511.     LOG_TRACE("(%d, \"%s\", %p, %d, %d, %d, %d, %d)", pFont_index, pText, pPixelmap, pLeft, pTop, pRight, pBottom, pCentred);
  1512.  
  1513.     font = &gFonts[pFont_index];
  1514.     current_width = 0;
  1515.     font_needed_loading = font->images == NULL;
  1516.     if (font_needed_loading) {
  1517.         LoadFont(pFont_index);
  1518.     }
  1519.     centre = (pRight + pLeft) / 2;
  1520.     current_y = pTop;
  1521.     width = pRight - pLeft;
  1522.     line_char_index = 0;
  1523.     input_str_index = 0;
  1524.     start_line = 0;
  1525.  
  1526.     while (pText[input_str_index]) {
  1527.         line[line_char_index] = pText[input_str_index];
  1528.         line[line_char_index + 1] = 0;
  1529.         current_width += font->spacing + font->width_table[pText[input_str_index] - font->offset];
  1530.         if (current_width > width) {
  1531.             for (i = input_str_index; i >= start_line; i--) {
  1532.                 if (pText[i] == ' ') {
  1533.                     break;
  1534.                 }
  1535.             }
  1536.             if (i == start_line) {
  1537.                 i = input_str_index;
  1538.             }
  1539.             line_char_index += i - input_str_index;
  1540.             input_str_index = i;
  1541.             if (pText[input_str_index] == ' ') {
  1542.                 input_str_index++;
  1543.             }
  1544.             line[line_char_index] = 0;
  1545.             if (pCentred) {
  1546.                 DRPixelmapCentredText(gBack_screen, centre, current_y, font, line);
  1547.             } else {
  1548.                 TransDRPixelmapText(gBack_screen, pLeft, current_y, font, line, pRight);
  1549.             }
  1550.             current_width = 0;
  1551.             current_y += 3 * (font->height - (TranslationMode() ? 2 : 0)) / 2;
  1552.             line_char_index = 0;
  1553.             start_line = input_str_index;
  1554.         } else {
  1555.             line_char_index++;
  1556.             input_str_index++;
  1557.         }
  1558.     }
  1559.     if (line_char_index != 0) {
  1560.         if (pCentred) {
  1561.             TransDRPixelmapText(gBack_screen, centre - (DRTextWidth(font, line) / 2), current_y, font, line, (DRTextWidth(font, line) / 2) + centre);
  1562.         } else {
  1563.             TransDRPixelmapText(gBack_screen, pLeft, current_y, font, line, pRight);
  1564.         }
  1565.     }
  1566.     if (font_needed_loading) {
  1567.         DisposeFont(pFont_index);
  1568.     }
  1569. }
  1570.  
  1571. // IDA: void __usercall TransBrPixelmapText(br_pixelmap *pPixelmap@<EAX>, int pX@<EDX>, int pY@<EBX>, br_uint_32 pColour@<ECX>, br_font *pFont, signed char *pText)
  1572. void TransBrPixelmapText(br_pixelmap* pPixelmap, int pX, int pY, br_uint_32 pColour, br_font* pFont, char* pText) {
  1573.     int len;
  1574.     LOG_TRACE("(%p, %d, %d, %d, %p, %p)", pPixelmap, pX, pY, pColour, pFont, pText);
  1575.  
  1576.     len = TranslationMode() ? 2 : 0;
  1577.     BrPixelmapText(pPixelmap, pX, pY - len, pColour, pFont, (char*)pText);
  1578. }
  1579.  
  1580. // IDA: void __usercall TransDRPixelmapText(br_pixelmap *pPixelmap@<EAX>, int pX@<EDX>, int pY@<EBX>, tDR_font *pFont@<ECX>, char *pText, int pRight_edge)
  1581. void TransDRPixelmapText(br_pixelmap* pPixelmap, int pX, int pY, tDR_font* pFont, char* pText, int pRight_edge) {
  1582.     LOG_TRACE("(%p, %d, %d, %p, \"%s\", %d)", pPixelmap, pX, pY, pFont, pText, pRight_edge);
  1583.  
  1584.     if (gAusterity_mode && FlicsPlayedFromDisk() && pFont != gCached_font) {
  1585.         if (gCached_font != NULL && gCached_font - gFonts > 13) {
  1586.             DisposeFont(gCached_font - gFonts);
  1587.         }
  1588.         gCached_font = pFont;
  1589.     }
  1590.     LoadFont(pFont - gFonts);
  1591.     DRPixelmapText(pPixelmap, pX, pY - (TranslationMode() ? 2 : 0), pFont, pText, pRight_edge);
  1592. }
  1593.  
  1594. // IDA: void __usercall TransDRPixelmapCleverText(br_pixelmap *pPixelmap@<EAX>, int pX@<EDX>, int pY@<EBX>, tDR_font *pFont@<ECX>, char *pText, int pRight_edge)
  1595. void TransDRPixelmapCleverText(br_pixelmap* pPixelmap, int pX, int pY, tDR_font* pFont, char* pText, int pRight_edge) {
  1596.     LOG_TRACE("(%p, %d, %d, %p, \"%s\", %d)", pPixelmap, pX, pY, pFont, pText, pRight_edge);
  1597.  
  1598.     if (gAusterity_mode && FlicsPlayedFromDisk() && gCached_font != pFont) {
  1599.         if (gCached_font && gCached_font - gFonts > 13) {
  1600.             DisposeFont(gCached_font - gFonts);
  1601.         }
  1602.         gCached_font = pFont;
  1603.     }
  1604.     LoadFont(pFont - gFonts);
  1605.     DRPixelmapCleverText2(pPixelmap, pX, pY - (TranslationMode() == 0 ? 0 : 2), pFont, pText, pRight_edge);
  1606. }
  1607.