Rev 1 | Rev 20 | Go to most recent revision | Show entire file | Ignore whitespace | Details | Blame | Last modification | View Log | RSS feed
| Rev 1 | Rev 18 | ||
|---|---|---|---|
| Line 1... | Line 1... | ||
| 1 | #include "netgame.h" |
1 | #include "netgame.h" |
| 2 | #include " |
2 | #include "brender.h" |
| - | 3 | #include "brucetrk.h" |
|
| 3 | #include "car.h" |
4 | #include "car.h" |
| - | 5 | #include "controls.h" |
|
| - | 6 | #include "crush.h" |
|
| 4 | #include "displays.h" |
7 | #include "displays.h" |
| 5 | #include "errors.h" |
8 | #include "errors.h" |
| 6 | #include "globvars.h" |
9 | #include "globvars.h" |
| 7 | #include "globvrpb.h" |
10 | #include "globvrpb.h" |
| 8 | #include "grafdata.h" |
11 | #include "grafdata.h" |
| Line 13... | Line 16... | ||
| 13 | #include "newgame.h" |
16 | #include "newgame.h" |
| 14 | #include "opponent.h" |
17 | #include "opponent.h" |
| 15 | #include "pd/sys.h" |
18 | #include "pd/sys.h" |
| 16 | #include "pedestrn.h" |
19 | #include "pedestrn.h" |
| 17 | #include "powerup.h" |
20 | #include "powerup.h" |
| - | 21 | #include "pratcam.h" |
|
| 18 | #include "racestrt.h" |
22 | #include "racestrt.h" |
| - | 23 | #include "spark.h" |
|
| 19 | #include "structur.h" |
24 | #include "structur.h" |
| 20 | #include "utility.h" |
25 | #include "utility.h" |
| - | 26 | #include <limits.h> |
|
| 21 | #include <stdlib.h> |
27 | #include <stdlib.h> |
| 22 | #include <string.h> |
28 | #include <string.h> |
| 23 | 29 | ||
| 24 | int gPowerup_cost[4] = { 1500, 2500, 4000, 6000 }; |
30 | int gPowerup_cost[4] = { 1500, 2500, 4000, 6000 }; |
| 25 | int gGame_scores[6]; |
31 | int gGame_scores[6] = { 1, 2, 3, 4, 6, 10 }; |
| 26 | int gPed_target; |
32 | int gPed_target; |
| 27 | int gNot_shown_race_type_headup; |
33 | int gNot_shown_race_type_headup; |
| 28 | tU32 gLast_it_change; |
34 | tU32 gLast_it_change; |
| 29 | tU32 gTime_for_punishment; |
35 | tU32 gTime_for_punishment; |
| 30 | tNet_game_player_info* gLast_lepper; |
36 | tNet_game_player_info* gLast_lepper; |
| 31 | int gInitialised_grid; |
37 | int gInitialised_grid; |
| 32 | int gIt_or_fox; |
38 | int gIt_or_fox; |
| - | 39 | ||
| - | 40 | #define PACK_POWERUPS(car) (car->power_up_levels[0] & 0xff) + ((car->power_up_levels[2] & 0xff) << 6) + ((car->power_up_levels[1] & 0xff) << 3); |
|
| 33 | 41 | ||
| 34 | // IDA: void __usercall SendCarData(tU32 pNext_frame_time@<EAX>) |
42 | // IDA: void __usercall SendCarData(tU32 pNext_frame_time@<EAX>) |
| 35 | void SendCarData(tU32 pNext_frame_time) { |
43 | void SendCarData(tU32 pNext_frame_time) { |
| 36 |
|
44 | tNet_contents* contents; |
| 37 |
|
45 | tCar_spec* car; |
| 38 |
|
46 | tCollision_info* ncar; |
| 39 |
|
47 | int i; |
| 40 |
|
48 | int j; |
| 41 |
|
49 | static tU32 last_time; |
| 42 |
|
50 | tU32 time; |
| 43 |
|
51 | int damaged_wheels; |
| 44 | LOG_TRACE("(%d)", pNext_frame_time); |
52 | LOG_TRACE("(%d)", pNext_frame_time); |
| 45 | 53 | ||
| - | 54 | time = GetRaceTime(); |
|
| - | 55 | if (gNet_mode == eNet_mode_none || (time > last_time && last_time + 80 > time)) { |
|
| - | 56 | return; |
|
| - | 57 | } |
|
| - | 58 | last_time = time; |
|
| - | 59 | contents = NetGetBroadcastContents(NETMSGID_TIMESYNC, 0); |
|
| - | 60 | contents->data.time_sync.race_start_time = gRace_start; |
|
| - | 61 | ||
| 46 | if (gNet_mode) { |
62 | if (gNet_mode == eNet_mode_host) { |
| - | 63 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| 47 |
|
64 | car = gNet_players[i].car; |
| - | 65 | if (car->disabled) { |
|
| - | 66 | continue; |
|
| - | 67 | } |
|
| - | 68 | damaged_wheels = car->damage_units[eDamage_lf_wheel].damage_level > 30 || car->damage_units[eDamage_rf_wheel].damage_level > 30 || car->damage_units[eDamage_lr_wheel].damage_level > 30 || car->damage_units[eDamage_rr_wheel].damage_level > 30; |
|
| - | 69 | contents = NetGetBroadcastContents(NETMSGID_MECHANICS, damaged_wheels); |
|
| - | 70 | GetReducedMatrix(&contents->data.mech.mat, &car->car_master_actor->t.t.mat); |
|
| - | 71 | contents->data.mech.ID = gNet_players[i].ID; |
|
| - | 72 | contents->data.mech.time = pNext_frame_time; |
|
| - | 73 | BrVector3Copy(&contents->data.mech.omega, &car->omega); |
|
| - | 74 | BrVector3Copy(&contents->data.mech.v, &car->v); |
|
| - | 75 | contents->data.mech.curvature = (car->curvature / car->maxcurve * 32767.0f); |
|
| - | 76 | contents->data.mech.keys = car->keys; |
|
| - | 77 | contents->data.mech.keys.joystick_acc = (tU8)(car->joystick.acc >> 9); |
|
| - | 78 | contents->data.mech.keys.joystick_dec = (tU8)(car->joystick.dec >> 9); |
|
| - | 79 | contents->data.mech.revs = car->revs; |
|
| - | 80 | for (j = 0; j < COUNT_OF(contents->data.mech.d); j++) { |
|
| - | 81 | contents->data.mech.d[j] = (int)(car->oldd[j] / car->susp_height[j >> 1] * 255.0f); |
|
| - | 82 | } |
|
| - | 83 | for (j = 0; j < COUNT_OF(contents->data.mech.damage); j++) { |
|
| - | 84 | contents->data.mech.damage[j] = car->damage_units[j].damage_level; |
|
| - | 85 | } |
|
| - | 86 | contents->data.mech.front = car->bounds[1].min.v[2]; |
|
| - | 87 | contents->data.mech.back = car->bounds[1].max.v[2]; |
|
| - | 88 | contents->data.mech.powerups = PACK_POWERUPS(car); |
|
| - | 89 | contents->data.mech.repair_time = car->repair_time; |
|
| - | 90 | contents->data.mech.cc_coll_time = car->last_car_car_collision; |
|
| - | 91 | if (damaged_wheels) { |
|
| - | 92 | for (j = 0; j < COUNT_OF(contents->data.mech.wheel_dam_offset); j++) { |
|
| - | 93 | contents->data.mech.wheel_dam_offset[j] = car->wheel_dam_offset[j]; |
|
| - | 94 | } |
|
| - | 95 | } |
|
| - | 96 | if (car->time_to_recover != 0) { |
|
| - | 97 | if (car->time_to_recover - 500 < pNext_frame_time) { |
|
| - | 98 | contents = NetGetBroadcastContents(NETMSGID_RECOVER, 0); |
|
| - | 99 | contents->data.recover.ID = gNet_players[i].ID; |
|
| - | 100 | contents->data.recover.time_to_recover = car->time_to_recover; |
|
| - | 101 | } |
|
| - | 102 | } |
|
| - | 103 | } |
|
| - | 104 | for (i = 0; i < gNum_active_non_cars; i++) { |
|
| - | 105 | contents = NetGetBroadcastContents(NETMSGID_NONCAR_INFO, 0); |
|
| - | 106 | ncar = (tCollision_info*)gActive_non_car_list[i]; |
|
| - | 107 | GetReducedMatrix(&contents->data.mech.mat, &ncar->car_master_actor->t.t.mat); |
|
| - | 108 | contents->data.non_car.ID = ncar->car_ID; |
|
| - | 109 | contents->data.non_car.time = pNext_frame_time; |
|
| - | 110 | BrVector3Copy(&contents->data.non_car.omega, &ncar->omega); |
|
| - | 111 | BrVector3Copy(&contents->data.non_car.v, &ncar->v); |
|
| - | 112 | contents->data.non_car.flags = ncar->car_master_actor->identifier[3] == 2 * ncar->doing_nothing_flag + '!'; |
|
| - | 113 | } |
|
| - | 114 | for (i = 0; i < gProgram_state.AI_vehicles.number_of_cops; i++) { |
|
| - | 115 | if (!gProgram_state.AI_vehicles.cops[i].finished_for_this_race) { |
|
| - | 116 | contents = NetGetBroadcastContents(NETMSGID_COPINFO, 0); |
|
| - | 117 | car = gProgram_state.AI_vehicles.cops[i].car_spec; |
|
| - | 118 | GetReducedMatrix(&contents->data.mech.mat, &car->car_master_actor->t.t.mat); |
|
| - | 119 | contents->data.cop_info.ID = car->car_ID; |
|
| - | 120 | contents->data.cop_info.time = pNext_frame_time; |
|
| - | 121 | BrVector3Copy(&contents->data.cop_info.omega, &car->omega); |
|
| - | 122 | BrVector3Copy(&contents->data.cop_info.v, &car->v); |
|
| - | 123 | for (j = 0; j < COUNT_OF(contents->data.cop_info.damage); ++j) { |
|
| - | 124 | contents->data.cop_info.damage[j] = car->damage_units[j].damage_level; |
|
| - | 125 | } |
|
| - | 126 | for (j = 0; j < COUNT_OF(contents->data.cop_info.d); j++) { |
|
| - | 127 | contents->data.cop_info.d[j] = car->oldd[j]; |
|
| - | 128 | } |
|
| - | 129 | } |
|
| - | 130 | } |
|
| - | 131 | } else if (gNet_mode == eNet_mode_client) { |
|
| - | 132 | car = &gProgram_state.current_car; |
|
| - | 133 | if (car->disabled) { |
|
| - | 134 | return; |
|
| - | 135 | } |
|
| - | 136 | damaged_wheels = car->damage_units[eDamage_lf_wheel].damage_level > 30 || car->damage_units[eDamage_rf_wheel].damage_level > 30 || car->damage_units[eDamage_lr_wheel].damage_level > 30 || car->damage_units[eDamage_rr_wheel].damage_level > 30; |
|
| - | 137 | contents = NetGetToHostContents(NETMSGID_MECHANICS, damaged_wheels); |
|
| - | 138 | GetReducedMatrix(&contents->data.mech.mat, &gProgram_state.current_car.car_master_actor->t.t.mat); |
|
| - | 139 | contents->data.mech.ID = gNet_players[gThis_net_player_index].ID; |
|
| - | 140 | contents->data.mech.time = pNext_frame_time; |
|
| - | 141 | BrVector3Copy(&contents->data.mech.omega, &car->omega); |
|
| - | 142 | BrVector3Copy(&contents->data.mech.v, &car->v); |
|
| - | 143 | ||
| - | 144 | contents->data.mech.curvature = (car->curvature / car->maxcurve * 32767.0f); |
|
| - | 145 | contents->data.mech.keys = car->keys; |
|
| - | 146 | contents->data.mech.keys.joystick_acc = (tU8)(car->joystick.acc >> 9); |
|
| - | 147 | contents->data.mech.keys.joystick_dec = (tU8)(car->joystick.dec >> 9); |
|
| - | 148 | contents->data.mech.revs = car->revs; |
|
| - | 149 | contents->data.mech.cc_coll_time = car->last_car_car_collision; |
|
| - | 150 | for (j = 0; j < COUNT_OF(contents->data.mech.d); j++) { |
|
| - | 151 | contents->data.mech.d[j] = (int)(car->oldd[j] / car->susp_height[j >> 1] * 255.f); |
|
| - | 152 | } |
|
| - | 153 | for (j = 0; j < COUNT_OF(contents->data.mech.damage); j++) { |
|
| - | 154 | contents->data.mech.damage[j] = car->damage_units[j].damage_level; |
|
| - | 155 | } |
|
| - | 156 | contents->data.mech.front = car->bounds[1].min.v[2]; |
|
| - | 157 | contents->data.mech.back = car->bounds[1].max.v[2]; |
|
| - | 158 | contents->data.mech.powerups = PACK_POWERUPS(car); |
|
| - | 159 | contents->data.mech.repair_time = car->repair_time; |
|
| - | 160 | if (damaged_wheels) { |
|
| - | 161 | for (j = 0; j < COUNT_OF(contents->data.mech.wheel_dam_offset); j++) { |
|
| - | 162 | contents->data.mech.wheel_dam_offset[j] = car->wheel_dam_offset[j]; |
|
| - | 163 | } |
|
| - | 164 | } |
|
| - | 165 | if (car->time_to_recover > 0 && car->time_to_recover - 500 < pNext_frame_time) { |
|
| - | 166 | contents = NetGetToHostContents(NETMSGID_RECOVER, 0); |
|
| - | 167 | contents->data.recover.ID = gNet_players[gThis_net_player_index].ID; |
|
| - | 168 | contents->data.recover.time_to_recover = gProgram_state.current_car.time_to_recover; |
|
| - | 169 | } |
|
| 48 | } |
170 | } |
| 49 | } |
171 | } |
| 50 | 172 | ||
| 51 | // IDA: void __usercall ReceivedRecover(tNet_contents *pContents@<EAX>) |
173 | // IDA: void __usercall ReceivedRecover(tNet_contents *pContents@<EAX>) |
| 52 | void ReceivedRecover(tNet_contents* pContents) { |
174 | void ReceivedRecover(tNet_contents* pContents) { |
| 53 |
|
175 | int i; |
| 54 | LOG_TRACE("(%p)", pContents); |
176 | LOG_TRACE("(%p)", pContents); |
| - | 177 | ||
| - | 178 | if (gNet_players[gThis_net_player_index].ID != pContents->data.player_list.number_of_players) { |
|
| - | 179 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 180 | if (gNet_players[i].ID == pContents->data.player_list.number_of_players) { |
|
| - | 181 | gNet_players[i].car->time_to_recover = pContents->data.mech.time; |
|
| 55 |
|
182 | } |
| - | 183 | } |
|
| - | 184 | } |
|
| 56 | } |
185 | } |
| 57 | 186 | ||
| 58 | // IDA: void __usercall CopyMechanics(tCar_spec *pCar@<EAX>, tNet_contents *pContents@<EDX>) |
187 | // IDA: void __usercall CopyMechanics(tCar_spec *pCar@<EAX>, tNet_contents *pContents@<EDX>) |
| 59 | void CopyMechanics(tCar_spec* pCar, tNet_contents* pContents) { |
188 | void CopyMechanics(tCar_spec* pCar, tNet_contents* pContents) { |
| 60 |
|
189 | int j; |
| 61 | LOG_TRACE("(%p, %p)", pCar, pContents); |
190 | LOG_TRACE("(%p, %p)", pCar, pContents); |
| - | 191 | ||
| - | 192 | memcpy(&pCar->message, pContents, pContents->header.contents_size); |
|
| - | 193 | // if it is not a full mechanics message... |
|
| - | 194 | if (pContents->header.contents_size != sizeof(tNet_message_mechanics_info)) { |
|
| - | 195 | for (j = 0; j < COUNT_OF(pCar->message.wheel_dam_offset); j++) { |
|
| - | 196 | pCar->message.wheel_dam_offset[j] = 0.0f; |
|
| 62 |
|
197 | } |
| - | 198 | } |
|
| 63 | } |
199 | } |
| 64 | 200 | ||
| 65 | // IDA: void __usercall ReceivedMechanics(tNet_contents *pContents@<EAX>) |
201 | // IDA: void __usercall ReceivedMechanics(tNet_contents *pContents@<EAX>) |
| 66 | void ReceivedMechanics(tNet_contents* pContents) { |
202 | void ReceivedMechanics(tNet_contents* pContents) { |
| 67 |
|
203 | int i; |
| 68 |
|
204 | tCar_spec* car; |
| 69 | LOG_TRACE("(%p)", pContents); |
205 | LOG_TRACE("(%p)", pContents); |
| - | 206 | ||
| - | 207 | car = NULL; |
|
| - | 208 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 209 | if (gNet_players[i].ID == pContents->data.mech.ID) { |
|
| - | 210 | car = gNet_players[i].car; |
|
| - | 211 | break; |
|
| - | 212 | } |
|
| - | 213 | } |
|
| - | 214 | if (car == NULL || car->message.time >= pContents->data.mech.time) { |
|
| - | 215 | return; |
|
| - | 216 | } |
|
| - | 217 | if (car->disabled) { |
|
| 70 |
|
218 | EnableCar(car); |
| - | 219 | GetExpandedMatrix(&car->car_master_actor->t.t.mat, &pContents->data.mech.mat); |
|
| - | 220 | BrMatrix34Copy(&car->oldmat, &car->car_master_actor->t.t.mat); |
|
| - | 221 | BrVector3InvScale(&car->car_master_actor->t.t.translate.t, &car->car_master_actor->t.t.translate.t, WORLD_SCALE); |
|
| - | 222 | // car->car_master_actor->t.t.mat.m[3][0] = car->car_master_actor->t.t.mat.m[3][0] * 0.14492753; |
|
| - | 223 | // car->car_master_actor->t.t.mat.m[3][1] = car->car_master_actor->t.t.mat.m[3][1] * 0.14492753; |
|
| - | 224 | // car->car_master_actor->t.t.mat.m[3][2] = car->car_master_actor->t.t.mat.m[3][2] * 0.14492753; |
|
| - | 225 | car->box_face_ref = gFace_num__car - 2; |
|
| - | 226 | car->message.time = pContents->data.mech.time; |
|
| - | 227 | car->message.type = NETMSGID_NONE; |
|
| - | 228 | BrVector3Copy(&car->v, &pContents->data.mech.v); |
|
| - | 229 | BrVector3Copy(&car->omega, &pContents->data.mech.omega); |
|
| - | 230 | ||
| - | 231 | for (i = 0; i < COUNT_OF(car->message.d); i++) { |
|
| - | 232 | car->message.d[i] = pContents->data.mech.d[i]; |
|
| - | 233 | } |
|
| - | 234 | for (i = 0; i < COUNT_OF(car->message.damage); i++) { |
|
| - | 235 | car->message.damage[i] = pContents->data.mech.damage[i]; |
|
| - | 236 | } |
|
| - | 237 | DisableCar(car); |
|
| - | 238 | return; |
|
| - | 239 | } |
|
| - | 240 | if (gNet_mode == eNet_mode_host) { |
|
| - | 241 | if (gThis_net_player_index == i) { |
|
| - | 242 | return; |
|
| - | 243 | } |
|
| - | 244 | if (car->last_car_car_collision <= pContents->data.mech.cc_coll_time) { |
|
| - | 245 | CopyMechanics(car, pContents); |
|
| - | 246 | } |
|
| - | 247 | car->power_up_levels[0] = pContents->data.mech.powerups & 7; |
|
| - | 248 | car->power_up_levels[1] = (pContents->data.mech.powerups >> 3) & 7; |
|
| - | 249 | car->power_up_levels[2] = (pContents->data.mech.powerups >> 6) & 7; |
|
| - | 250 | car->keys = car->message.keys; |
|
| - | 251 | if (car->message.keys.joystick_acc < 0) { |
|
| - | 252 | car->joystick.acc = -1; |
|
| - | 253 | } else { |
|
| - | 254 | car->joystick.acc = car->message.keys.joystick_acc << 9; |
|
| - | 255 | } |
|
| - | 256 | if (car->message.keys.joystick_dec < 0) { |
|
| - | 257 | car->joystick.dec = -1; |
|
| - | 258 | } else { |
|
| - | 259 | car->joystick.dec = car->message.keys.joystick_dec << 9; |
|
| - | 260 | } |
|
| - | 261 | } else if (gNet_mode == eNet_mode_client) { |
|
| - | 262 | if (gThis_net_player_index == i) { |
|
| - | 263 | if (car->last_car_car_collision < pContents->data.mech.cc_coll_time) { |
|
| - | 264 | CopyMechanics(car, pContents); |
|
| - | 265 | } |
|
| - | 266 | } else { |
|
| - | 267 | CopyMechanics(car, pContents); |
|
| - | 268 | car->power_up_levels[0] = pContents->data.mech.powerups & 7; |
|
| - | 269 | car->power_up_levels[1] = (pContents->data.mech.powerups >> 3) & 7; |
|
| - | 270 | car->power_up_levels[2] = (pContents->data.mech.powerups >> 6) & 7; |
|
| - | 271 | car->keys = car->message.keys; |
|
| - | 272 | if (car->message.keys.joystick_acc < 0) { |
|
| - | 273 | car->joystick.acc = -1; |
|
| - | 274 | } else { |
|
| - | 275 | car->joystick.acc = car->message.keys.joystick_acc << 9; |
|
| - | 276 | } |
|
| - | 277 | if (car->message.keys.joystick_dec < 0) { |
|
| - | 278 | car->joystick.dec = -1; |
|
| - | 279 | } else { |
|
| - | 280 | car->joystick.dec = car->message.keys.joystick_dec << 9; |
|
| - | 281 | } |
|
| - | 282 | if (!car->active) { |
|
| - | 283 | GetExpandedMatrix(&car->car_master_actor->t.t.mat, &pContents->data.mech.mat); |
|
| - | 284 | BrMatrix34Copy(&car->oldmat, &car->car_master_actor->t.t.mat); |
|
| - | 285 | BrMatrix34ApplyP(&car->pos, &car->cmpos, &car->car_master_actor->t.t.mat); |
|
| - | 286 | BrVector3InvScale(&car->car_master_actor->t.t.translate.t, &car->car_master_actor->t.t.translate.t, WORLD_SCALE); |
|
| - | 287 | BrVector3InvScale(&car->pos, &car->pos, WORLD_SCALE); |
|
| - | 288 | car->box_face_ref = gFace_num__car - 2; |
|
| - | 289 | } |
|
| - | 290 | } |
|
| - | 291 | } |
|
| 71 | } |
292 | } |
| 72 | 293 | ||
| 73 | // IDA: void __usercall ReceivedCopInfo(tNet_contents *pContents@<EAX>) |
294 | // IDA: void __usercall ReceivedCopInfo(tNet_contents *pContents@<EAX>) |
| 74 | void ReceivedCopInfo(tNet_contents* pContents) { |
295 | void ReceivedCopInfo(tNet_contents* pContents) { |
| 75 |
|
296 | tCar_spec* c; |
| 76 |
|
297 | int i; |
| 77 | LOG_TRACE("(%p)", pContents); |
298 | LOG_TRACE("(%p)", pContents); |
| - | 299 | ||
| - | 300 | if (gNet_mode != eNet_mode_client) { |
|
| - | 301 | return; |
|
| - | 302 | } |
|
| - | 303 | if (pContents->data.cop_info.ID & 0xffffff00) { |
|
| - | 304 | c = GetCarSpec(pContents->data.cop_info.ID >> 8, pContents->data.cop_info.ID & 0xff); |
|
| - | 305 | } else { |
|
| - | 306 | c = &gProgram_state.current_car; |
|
| - | 307 | } |
|
| - | 308 | if (c == NULL || c->message.time > pContents->data.cop_info.time) { |
|
| - | 309 | return; |
|
| - | 310 | } |
|
| - | 311 | c->message.time = pContents->data.cop_info.time; |
|
| 78 |
|
312 | if (c->active) { |
| - | 313 | c->message.type = NETMSGID_MECHANICS; |
|
| - | 314 | c->message.mat = pContents->data.cop_info.mat; |
|
| - | 315 | BrVector3Copy(&c->message.v, &pContents->data.cop_info.v); |
|
| - | 316 | BrVector3Copy(&c->message.omega, &pContents->data.cop_info.omega); |
|
| - | 317 | c->message.curvature = (tS16)pContents->data.cop_info.curvature; |
|
| - | 318 | for (i = 0; i < COUNT_OF(c->message.d); i++) { |
|
| - | 319 | c->message.d[i] = pContents->data.cop_info.d[i]; |
|
| - | 320 | } |
|
| - | 321 | for (i = 0; i < COUNT_OF(c->message.damage); i++) { |
|
| - | 322 | c->message.damage[i] = pContents->data.cop_info.damage[i]; |
|
| - | 323 | } |
|
| - | 324 | } else { |
|
| - | 325 | GetExpandedMatrix(&c->car_master_actor->t.t.mat, &pContents->data.cop_info.mat); |
|
| - | 326 | BrVector3InvScale(&c->car_master_actor->t.t.translate.t, &c->car_master_actor->t.t.translate.t, WORLD_SCALE); |
|
| - | 327 | for (i = 0; i < COUNT_OF(c->damage_units); i++) { |
|
| - | 328 | c->damage_units[i].damage_level = pContents->data.cop_info.damage[i]; |
|
| - | 329 | } |
|
| - | 330 | } |
|
| 79 | } |
331 | } |
| 80 | 332 | ||
| 81 | // IDA: void __cdecl SendAllNonCarPositions() |
333 | // IDA: void __cdecl SendAllNonCarPositions() |
| 82 | void SendAllNonCarPositions(void) { |
334 | void SendAllNonCarPositions(void) { |
| 83 | int i; |
335 | int i; |
| Line 101... | Line 353... | ||
| 101 | NetSendMessageStacks(); |
353 | NetSendMessageStacks(); |
| 102 | } |
354 | } |
| 103 | 355 | ||
| 104 | // IDA: void __usercall ReceivedNonCarPosition(tNet_contents *pContents@<EAX>) |
356 | // IDA: void __usercall ReceivedNonCarPosition(tNet_contents *pContents@<EAX>) |
| 105 | void ReceivedNonCarPosition(tNet_contents* pContents) { |
357 | void ReceivedNonCarPosition(tNet_contents* pContents) { |
| 106 |
|
358 | br_actor* actor; |
| 107 | LOG_TRACE("(%p)", pContents); |
359 | LOG_TRACE("(%p)", pContents); |
| - | 360 | ||
| - | 361 | actor = gProgram_state.track_spec.non_car_list[pContents->data.non_car_position.ID]; |
|
| - | 362 | if (actor != NULL && gNet_mode != eNet_mode_none) { |
|
| - | 363 | BrMatrix34Copy(&actor->t.t.mat, &pContents->data.non_car_position.mat); |
|
| - | 364 | if (pContents->data.non_car_position.flags) { |
|
| - | 365 | actor->identifier[3] = '!'; |
|
| 108 |
|
366 | } |
| - | 367 | } |
|
| 109 | } |
368 | } |
| 110 | 369 | ||
| 111 | // IDA: void __usercall ReceivedNonCar(tNet_contents *pContents@<EAX>) |
370 | // IDA: void __usercall ReceivedNonCar(tNet_contents *pContents@<EAX>) |
| 112 | void ReceivedNonCar(tNet_contents* pContents) { |
371 | void ReceivedNonCar(tNet_contents* pContents) { |
| 113 |
|
372 | br_actor* actor; |
| 114 |
|
373 | br_vector3 tv; |
| 115 |
|
374 | tU8 cx; |
| 116 |
|
375 | tU8 cz; |
| 117 |
|
376 | tTrack_spec* track_spec; |
| 118 |
|
377 | tNon_car_spec* ncar; |
| 119 |
|
378 | tCollision_info* c; |
| 120 | LOG_TRACE("(%p)", pContents); |
379 | LOG_TRACE("(%p)", pContents); |
| - | 380 | ||
| - | 381 | track_spec = &gProgram_state.track_spec; |
|
| - | 382 | if (pContents->data.non_car.ID >= track_spec->ampersand_digits) { |
|
| - | 383 | return; |
|
| - | 384 | } |
|
| - | 385 | actor = track_spec->non_car_list[pContents->data.non_car.ID]; |
|
| - | 386 | if (actor == NULL) { |
|
| - | 387 | return; |
|
| - | 388 | } |
|
| - | 389 | ncar = (tNon_car_spec*)actor->type_data; |
|
| - | 390 | if (ncar != NULL && (ncar->collision_info.driver != eDriver_non_car || ncar->collision_info.car_ID != pContents->data.non_car.ID)) { |
|
| 121 |
|
391 | ncar = NULL; |
| - | 392 | } |
|
| - | 393 | if ((pContents->data.non_car.flags & 1) != 0) { |
|
| - | 394 | actor->identifier[3] = '!'; |
|
| - | 395 | } else { |
|
| - | 396 | actor->identifier[3] = 'x'; |
|
| - | 397 | } |
|
| - | 398 | if (ncar == NULL && actor->identifier[1] >= '0' && actor->identifier[1] <= '9') { |
|
| - | 399 | BrVector3Sub(&tv, &gProgram_state.current_car.car_master_actor->t.t.translate.t, &actor->t.t.translate.t); |
|
| - | 400 | if (BrVector3LengthSquared(&tv) < 900.0f) { |
|
| - | 401 | DoPullActorFromWorld(actor); |
|
| - | 402 | ncar = (tNon_car_spec*)actor->type_data; |
|
| - | 403 | } |
|
| - | 404 | } |
|
| - | 405 | if (ncar != NULL) { |
|
| - | 406 | c = &ncar->collision_info; |
|
| - | 407 | if ((pContents->data.non_car.flags & 2) != 0) { |
|
| - | 408 | GetExpandedMatrix(&c->car_master_actor->t.t.mat, &pContents->data.non_car.mat); |
|
| - | 409 | BrVector3Copy(&c->v, &pContents->data.non_car.v); |
|
| - | 410 | BrVector3Copy(&c->omega, &pContents->data.non_car.omega); |
|
| - | 411 | c->doing_nothing_flag = 1; |
|
| - | 412 | } else { |
|
| - | 413 | BrVector3Copy(&c->message.v, &pContents->data.non_car.v); |
|
| - | 414 | BrVector3Copy(&c->message.omega, &pContents->data.non_car.omega); |
|
| - | 415 | memcpy(&c->message.mat, &pContents->data.non_car.mat, sizeof(c->message.mat)); |
|
| - | 416 | c->message.time = pContents->data.non_car.time; |
|
| - | 417 | c->message.type = NETMSGID_NONCAR_INFO; |
|
| - | 418 | c->doing_nothing_flag = 0; |
|
| - | 419 | } |
|
| - | 420 | } else { |
|
| - | 421 | GetExpandedMatrix(&actor->t.t.mat, &pContents->data.mech.mat); |
|
| - | 422 | BrVector3InvScale(&actor->t.t.translate.t, &actor->t.t.translate.t, WORLD_SCALE); |
|
| - | 423 | XZToColumnXZ(&cx, &cz, actor->t.t.translate.t.v[0], actor->t.t.translate.t.v[2], track_spec); |
|
| - | 424 | if (track_spec->columns[cz][cx] != actor->parent) { |
|
| - | 425 | if (track_spec->columns[cz][cx] != NULL) { |
|
| - | 426 | BrActorRemove(actor); |
|
| - | 427 | BrActorAdd(track_spec->columns[cz][cx], actor); |
|
| - | 428 | } |
|
| - | 429 | } |
|
| - | 430 | } |
|
| 122 | } |
431 | } |
| 123 | 432 | ||
| 124 | // IDA: void __usercall SignalToStartRace2(int pIndex@<EAX>) |
433 | // IDA: void __usercall SignalToStartRace2(int pIndex@<EAX>) |
| 125 | void SignalToStartRace2(int pIndex) { |
434 | void SignalToStartRace2(int pIndex) { |
| 126 | tNet_message* the_message; |
435 | tNet_message* the_message; |
| Line 249... | Line 558... | ||
| 249 | ReenableNetService(); |
558 | ReenableNetService(); |
| 250 | } |
559 | } |
| 251 | 560 | ||
| 252 | // IDA: void __usercall ReinitialiseCar(tCar_spec *pCar@<EAX>) |
561 | // IDA: void __usercall ReinitialiseCar(tCar_spec *pCar@<EAX>) |
| 253 | void ReinitialiseCar(tCar_spec* pCar) { |
562 | void ReinitialiseCar(tCar_spec* pCar) { |
| 254 |
|
563 | int i; |
| 255 | LOG_TRACE("(%p)", pCar); |
564 | LOG_TRACE("(%p)", pCar); |
| - | 565 | ||
| - | 566 | StopCarSmokingInstantly(pCar); |
|
| - | 567 | LoseAllLocalPowerups(pCar); |
|
| 256 |
|
568 | InitialiseCar(pCar); |
| - | 569 | TotallyRepairACar(pCar); |
|
| - | 570 | if (pCar->driver == eDriver_local_human) { |
|
| - | 571 | gLast_it_change = PDGetTotalTime() + 2000; |
|
| - | 572 | } |
|
| 257 | } |
573 | } |
| 258 | 574 | ||
| 259 | // IDA: void __usercall RepositionPlayer(int pIndex@<EAX>) |
575 | // IDA: void __usercall RepositionPlayer(int pIndex@<EAX>) |
| 260 | void RepositionPlayer(int pIndex) { |
576 | void RepositionPlayer(int pIndex) { |
| 261 |
|
577 | tNet_message* the_message; |
| 262 |
|
578 | tCar_spec* car; |
| 263 | LOG_TRACE("(%d)", pIndex); |
579 | LOG_TRACE("(%d)", pIndex); |
| - | 580 | ||
| - | 581 | car = gNet_players[pIndex].car; |
|
| - | 582 | gNet_players[pIndex].wasted = 0; |
|
| - | 583 | gNet_players[pIndex].reposition_time = 0; |
|
| - | 584 | if (gCurrent_net_game->type == eNet_game_type_carnage |
|
| - | 585 | || gCurrent_net_game->type == eNet_game_type_checkpoint |
|
| - | 586 | || gCurrent_net_game->type == eNet_game_type_sudden_death) { |
|
| - | 587 | BrMatrix34Copy(&car->car_master_actor->t.t.mat, &gNet_players[pIndex].initial_position); |
|
| - | 588 | } else { |
|
| - | 589 | SetInitialPosition(&gCurrent_race, gNet_players[pIndex].opponent_list_index, gNet_players[pIndex].grid_index); |
|
| - | 590 | } |
|
| - | 591 | ReinitialiseCar(car); |
|
| 264 |
|
592 | SetFlipUpCar(car); |
| - | 593 | car->last_car_car_collision = GetRaceTime() + 100; |
|
| - | 594 | SignalToStartRace2(pIndex); |
|
| - | 595 | if (car->driver == eDriver_local_human) { |
|
| - | 596 | CancelPendingCunningStunt(); |
|
| - | 597 | gProgram_state.credits_earned = gInitial_net_credits[gCurrent_net_game->options.starting_money_index]; |
|
| - | 598 | gProgram_state.credits_lost = 0; |
|
| - | 599 | } |
|
| 265 | } |
600 | } |
| 266 | 601 | ||
| 267 | // IDA: void __usercall DisableCar(tCar_spec *pCar@<EAX>) |
602 | // IDA: void __usercall DisableCar(tCar_spec *pCar@<EAX>) |
| 268 | void DisableCar(tCar_spec* pCar) { |
603 | void DisableCar(tCar_spec* pCar) { |
| 269 | LOG_TRACE("(%p)", pCar); |
604 | LOG_TRACE("(%p)", pCar); |
| Line 360... | Line 695... | ||
| 360 | s[0] = '\0'; |
695 | s[0] = '\0'; |
| 361 | break; |
696 | break; |
| 362 | } |
697 | } |
| 363 | ChangeHeadupText(gNet_ped_headup, s); |
698 | ChangeHeadupText(gNet_ped_headup, s); |
| 364 | } |
699 | } |
| - | 700 | ||
| - | 701 | #define HEADUP1 ((tHeadup_pair*)pFirst_one) |
|
| - | 702 | #define HEADUP2 ((tHeadup_pair*)pSecond_one) |
|
| 365 | 703 | ||
| 366 | // IDA: int __usercall SortNetHeadAscending@<EAX>(void *pFirst_one@<EAX>, void *pSecond_one@<EDX>) |
704 | // IDA: int __usercall SortNetHeadAscending@<EAX>(void *pFirst_one@<EAX>, void *pSecond_one@<EDX>) |
| 367 | int SortNetHeadAscending(void* pFirst_one, void* pSecond_one) { |
705 | int SortNetHeadAscending(void* pFirst_one, void* pSecond_one) { |
| 368 | LOG_TRACE("(%p, %p)", pFirst_one, pSecond_one); |
706 | LOG_TRACE("(%p, %p)", pFirst_one, pSecond_one); |
| - | 707 | ||
| - | 708 | if (HEADUP1->out_of_game) { |
|
| - | 709 | if (HEADUP2->out_of_game) { |
|
| - | 710 | return HEADUP1->out_of_game - HEADUP2->out_of_game; |
|
| - | 711 | } else { |
|
| - | 712 | return INT_MAX; |
|
| - | 713 | } |
|
| - | 714 | } else if (HEADUP2->out_of_game) { |
|
| 369 |
|
715 | return -INT_MAX; |
| - | 716 | } else if (HEADUP2->score == HEADUP1->score) { |
|
| - | 717 | return gNet_players[HEADUP1->player_index].last_score_index |
|
| - | 718 | - gNet_players[HEADUP2->player_index].last_score_index; |
|
| - | 719 | } else { |
|
| - | 720 | return HEADUP1->score - HEADUP2->score; |
|
| - | 721 | } |
|
| 370 | } |
722 | } |
| 371 | 723 | ||
| 372 | // IDA: int __usercall SortNetHeadDescending@<EAX>(void *pFirst_one@<EAX>, void *pSecond_one@<EDX>) |
724 | // IDA: int __usercall SortNetHeadDescending@<EAX>(void *pFirst_one@<EAX>, void *pSecond_one@<EDX>) |
| 373 | int SortNetHeadDescending(void* pFirst_one, void* pSecond_one) { |
725 | int SortNetHeadDescending(void* pFirst_one, void* pSecond_one) { |
| 374 | LOG_TRACE("(%p, %p)", pFirst_one, pSecond_one); |
726 | LOG_TRACE("(%p, %p)", pFirst_one, pSecond_one); |
| - | 727 | ||
| - | 728 | if (HEADUP1->out_of_game) { |
|
| - | 729 | if (HEADUP2->out_of_game) { |
|
| - | 730 | return HEADUP1->out_of_game - HEADUP2->out_of_game; |
|
| - | 731 | } else { |
|
| - | 732 | return INT_MAX; |
|
| - | 733 | } |
|
| - | 734 | } else if (HEADUP2->out_of_game) { |
|
| 375 |
|
735 | return -INT_MAX; |
| - | 736 | } else if (HEADUP2->score == HEADUP1->score) { |
|
| - | 737 | return gNet_players[HEADUP1->player_index].last_score_index |
|
| - | 738 | - gNet_players[HEADUP2->player_index].last_score_index; |
|
| - | 739 | } else { |
|
| - | 740 | return HEADUP2->score - HEADUP1->score; |
|
| - | 741 | } |
|
| 376 | } |
742 | } |
| - | 743 | ||
| - | 744 | #undef HEADUP2 |
|
| - | 745 | #undef HEADUP1 |
|
| 377 | 746 | ||
| 378 | // IDA: void __usercall ClipName(char *pName@<EAX>, tDR_font *pFont@<EDX>, int pMax_width@<EBX>) |
747 | // IDA: void __usercall ClipName(char *pName@<EAX>, tDR_font *pFont@<EDX>, int pMax_width@<EBX>) |
| 379 | void ClipName(char* pName, tDR_font* pFont, int pMax_width) { |
748 | void ClipName(char* pName, tDR_font* pFont, int pMax_width) { |
| 380 | LOG_TRACE("(\"%s\", %p, %d)", pName, pFont, pMax_width); |
749 | LOG_TRACE("(\"%s\", %p, %d)", pName, pFont, pMax_width); |
| - | 750 | ||
| - | 751 | while (DRTextWidth(pFont, pName) > pMax_width) { |
|
| 381 |
|
752 | pName[strlen(pName) - 1] = 0; |
| - | 753 | } |
|
| 382 | } |
754 | } |
| 383 | 755 | ||
| 384 | // IDA: void __usercall DoNetScores2(int pOnly_sort_scores@<EAX>) |
756 | // IDA: void __usercall DoNetScores2(int pOnly_sort_scores@<EAX>) |
| 385 | void DoNetScores2(int pOnly_sort_scores) { |
757 | void DoNetScores2(int pOnly_sort_scores) { |
| 386 |
|
758 | int i; |
| 387 |
|
759 | int j; |
| 388 |
|
760 | int score; |
| 389 |
|
761 | int flags; |
| 390 |
|
762 | int index; |
| 391 |
|
763 | int right_edge; |
| 392 |
|
764 | int x; |
| 393 |
|
765 | int len; |
| 394 |
|
766 | int ascending_order; |
| 395 |
|
767 | char s[256]; |
| 396 |
|
768 | static tU32 last_flash; |
| 397 |
|
769 | static int flash_state; |
| 398 |
|
770 | tHeadup_pair headup_pairs[6]; |
| 399 | LOG_TRACE("(%d)", pOnly_sort_scores); |
771 | LOG_TRACE("(%d)", pOnly_sort_scores); |
| - | 772 | ||
| - | 773 | ascending_order = gCurrent_net_game->type == eNet_game_type_checkpoint || gCurrent_net_game->type == eNet_game_type_tag; |
|
| - | 774 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 775 | if (gNet_players[i].player_status < ePlayer_status_racing) { |
|
| - | 776 | headup_pairs[i].player_index = -1; |
|
| - | 777 | if (ascending_order) { |
|
| - | 778 | headup_pairs[i].score = 1000001; |
|
| - | 779 | } else { |
|
| - | 780 | headup_pairs[i].score = -1000001; |
|
| - | 781 | } |
|
| - | 782 | headup_pairs[i].out_of_game = 1000; |
|
| - | 783 | } else { |
|
| - | 784 | headup_pairs[i].player_index = i; |
|
| - | 785 | headup_pairs[i].score = gNet_players[i].score; |
|
| - | 786 | if (abs(gNet_players[i].score) != 1000000 && (gNet_players[i].score >= 0 || gCurrent_net_game->type == eNet_game_type_car_crusher)) { |
|
| - | 787 | headup_pairs[i].out_of_game = 0; |
|
| - | 788 | } else { |
|
| - | 789 | headup_pairs[i].out_of_game = gNet_players[i].last_score_index + 1; |
|
| - | 790 | } |
|
| - | 791 | } |
|
| - | 792 | } |
|
| - | 793 | for (i = gNumber_of_net_players; i < COUNT_OF(headup_pairs); i++) { |
|
| - | 794 | headup_pairs[i].player_index = -1; |
|
| - | 795 | if (ascending_order) { |
|
| - | 796 | headup_pairs[i].score = 1000001; |
|
| - | 797 | } else { |
|
| - | 798 | headup_pairs[i].score = -1000001; |
|
| - | 799 | } |
|
| - | 800 | headup_pairs[i].out_of_game = 1000; |
|
| - | 801 | } |
|
| - | 802 | qsort(headup_pairs, COUNT_OF(headup_pairs), sizeof(tHeadup_pair), (int (*)(const void*, const void*))(ascending_order ? SortNetHeadAscending : SortNetHeadDescending)); |
|
| - | 803 | ||
| - | 804 | right_edge = gCurrent_graf_data->net_head_box_x + gCurrent_graf_data->net_head_box_width + 5 * gCurrent_graf_data->net_head_box_pitch; |
|
| - | 805 | for (i = 0; i < COUNT_OF(headup_pairs); i++) { |
|
| - | 806 | index = headup_pairs[i].player_index; |
|
| - | 807 | if (index >= 0) { |
|
| - | 808 | gNet_players[index].last_score_index = i; |
|
| - | 809 | } |
|
| - | 810 | if (pOnly_sort_scores) { |
|
| - | 811 | continue; |
|
| - | 812 | } |
|
| - | 813 | x = gCurrent_graf_data->net_head_box_x + i * gCurrent_graf_data->net_head_box_pitch; |
|
| - | 814 | if (gCurrent_graf_data->net_head_box_bot > gProgram_state.current_render_top) { |
|
| - | 815 | DimRectangle(gBack_screen, x, gCurrent_graf_data->net_head_box_top, x + gCurrent_graf_data->net_head_box_width, gCurrent_graf_data->net_head_box_bot, 1); |
|
| - | 816 | } |
|
| - | 817 | if (index >= 0) { |
|
| - | 818 | Flash(200, &last_flash, &flash_state); |
|
| - | 819 | if (flash_state |
|
| - | 820 | || (gCurrent_net_game->type != eNet_game_type_tag && gCurrent_net_game->type != eNet_game_type_foxy) |
|
| - | 821 | || index != gIt_or_fox) { |
|
| - | 822 | if (gNet_players[index].name_not_clipped) { |
|
| - | 823 | ClipName(gNet_players[index].player_name, &gFonts[6], gCurrent_graf_data->net_head_box_width - gCurrent_graf_data->net_head_name_x_marg - 2); |
|
| - | 824 | gNet_players[index].name_not_clipped = 0; |
|
| - | 825 | } |
|
| - | 826 | TransDRPixelmapText(gBack_screen, x + gCurrent_graf_data->net_head_name_x_marg, gCurrent_graf_data->net_head_name_y, &gFonts[6], gNet_players[index].player_name, right_edge); |
|
| - | 827 | } |
|
| - | 828 | if (abs(gNet_players[index].score) == 1000000) { |
|
| - | 829 | if (flash_state) { |
|
| - | 830 | strcpy(s, GetMiscString(kMiscString_WASTED_173)); |
|
| - | 831 | } else { |
|
| - | 832 | s[0] = '\0'; |
|
| - | 833 | } |
|
| - | 834 | } else { |
|
| - | 835 | switch (gCurrent_net_game->type) { |
|
| - | 836 | case eNet_game_type_fight_to_death: |
|
| - | 837 | sprintf(s, "%d%%", gNet_players[index].score); |
|
| - | 838 | break; |
|
| - | 839 | case eNet_game_type_car_crusher: |
|
| - | 840 | case eNet_game_type_carnage: |
|
| - | 841 | sprintf(s, "%d", gNet_players[index].score); |
|
| - | 842 | break; |
|
| - | 843 | case eNet_game_type_checkpoint: |
|
| - | 844 | sprintf(s, "%d left", gNet_players[index].score >> 16); |
|
| - | 845 | break; |
|
| - | 846 | case eNet_game_type_sudden_death: |
|
| - | 847 | if (gNet_players[index].score < 0) { |
|
| - | 848 | if (flash_state) { |
|
| - | 849 | sprintf(s, "%s", GetMiscString(kMiscString_OUT)); |
|
| - | 850 | } else { |
|
| - | 851 | s[0] = '\0'; |
|
| - | 852 | } |
|
| - | 853 | } else { |
|
| - | 854 | score = gNet_players[index].score; |
|
| - | 855 | sprintf(s, "%s -%d-", GetMiscString(kMiscString_IN), score); |
|
| - | 856 | } |
|
| - | 857 | break; |
|
| - | 858 | case eNet_game_type_tag: |
|
| - | 859 | case eNet_game_type_foxy: |
|
| - | 860 | if (gNet_players[index].score >= 0) { |
|
| - | 861 | if (index == gIt_or_fox && !flash_state) { |
|
| - | 862 | s[0] = '\0'; |
|
| - | 863 | } else { |
|
| - | 864 | TimerString(gNet_players[index].score, s, 0, 1); |
|
| - | 865 | } |
|
| - | 866 | } else { |
|
| - | 867 | sprintf(s, "%s", GetMiscString(kMiscString_OUT)); |
|
| - | 868 | } |
|
| - | 869 | break; |
|
| - | 870 | default: |
|
| - | 871 | break; |
|
| - | 872 | } |
|
| - | 873 | } |
|
| - | 874 | len = DRTextWidth(&gFonts[6], s); |
|
| - | 875 | TransDRPixelmapText(gBack_screen, x + gCurrent_graf_data->net_head_score_x - len, gCurrent_graf_data->net_head_score_y, &gFonts[6], s, right_edge); |
|
| - | 876 | DRPixelmapRectangleMaskedCopy(gBack_screen, x + gCurrent_graf_data->net_head_num_x, gCurrent_graf_data->net_head_num_y, gDigits_pix, 0, i * gCurrent_graf_data->net_head_num_height, gDigits_pix->width, gCurrent_graf_data->net_head_num_height); |
|
| 400 |
|
877 | DRPixelmapRectangleMaskedCopy(gBack_screen, x + gCurrent_graf_data->net_head_icon_x, gCurrent_graf_data->net_head_icon_y, gIcons_pix, 0, gCurrent_graf_data->net_head_icon_height * gNet_players[index].car_index, gIcons_pix->width, gCurrent_graf_data->net_head_icon_height); |
| - | 878 | if (gNet_players[index].ID == gLocal_net_ID) { |
|
| - | 879 | BrPixelmapLine(gBack_screen, x, gCurrent_graf_data->net_head_box_top, x, gCurrent_graf_data->net_head_box_bot - 1, 0); |
|
| - | 880 | BrPixelmapLine(gBack_screen, x + gCurrent_graf_data->net_head_box_width - 1, gCurrent_graf_data->net_head_box_top, x + gCurrent_graf_data->net_head_box_width - 1, gCurrent_graf_data->net_head_box_bot - 1, 0); |
|
| - | 881 | BrPixelmapLine(gBack_screen, x + 1, gCurrent_graf_data->net_head_box_bot, x + gCurrent_graf_data->net_head_box_width - 2, gCurrent_graf_data->net_head_box_bot, 0); |
|
| - | 882 | BrPixelmapLine(gBack_screen, x - 1, gCurrent_graf_data->net_head_box_top, x - 1, gCurrent_graf_data->net_head_box_bot - 1, 255); |
|
| - | 883 | BrPixelmapLine(gBack_screen, x + gCurrent_graf_data->net_head_box_width, gCurrent_graf_data->net_head_box_top, x + gCurrent_graf_data->net_head_box_width, gCurrent_graf_data->net_head_box_bot - 1, 255); |
|
| - | 884 | BrPixelmapLine(gBack_screen, x + 1, gCurrent_graf_data->net_head_box_bot + 1, x + gCurrent_graf_data->net_head_box_width - 2, gCurrent_graf_data->net_head_box_bot + 1, 255); |
|
| - | 885 | BrPixelmapPixelSet(gBack_screen, x, gCurrent_graf_data->net_head_box_bot, 255); |
|
| - | 886 | BrPixelmapPixelSet(gBack_screen, x + gCurrent_graf_data->net_head_box_width - 1, gCurrent_graf_data->net_head_box_bot, 255); |
|
| - | 887 | } |
|
| - | 888 | } |
|
| - | 889 | } |
|
| 401 | } |
890 | } |
| 402 | 891 | ||
| 403 | // IDA: void __cdecl DoNetScores() |
892 | // IDA: void __cdecl DoNetScores() |
| 404 | void DoNetScores(void) { |
893 | void DoNetScores(void) { |
| 405 | LOG_TRACE("()"); |
894 | LOG_TRACE("()"); |
| - | 895 | ||
| 406 |
|
896 | DoNetScores2(0); |
| 407 | } |
897 | } |
| 408 | 898 | ||
| 409 | // IDA: void __cdecl InitNetHeadups() |
899 | // IDA: void __cdecl InitNetHeadups() |
| 410 | void InitNetHeadups(void) { |
900 | void InitNetHeadups(void) { |
| 411 | LOG_TRACE("()"); |
901 | LOG_TRACE("()"); |
| Line 447... | Line 937... | ||
| 447 | } |
937 | } |
| 448 | } |
938 | } |
| 449 | 939 | ||
| 450 | // IDA: void __cdecl EverybodysLost() |
940 | // IDA: void __cdecl EverybodysLost() |
| 451 | void EverybodysLost(void) { |
941 | void EverybodysLost(void) { |
| 452 |
|
942 | tNet_message* the_message; |
| 453 |
|
943 | int i; |
| 454 | LOG_TRACE("()"); |
944 | LOG_TRACE("()"); |
| - | 945 | ||
| - | 946 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| 455 |
|
947 | gNet_players[i].played += 1; |
| - | 948 | the_message = NetBuildMessage(NETMSGID_RACEOVER, 0); |
|
| - | 949 | the_message->contents.data.race_over.reason = eRace_over_network_loss; |
|
| - | 950 | NetGuaranteedSendMessageToPlayer(gCurrent_net_game, the_message, gNet_players[i].ID, NULL); |
|
| - | 951 | } |
|
| 456 | } |
952 | } |
| 457 | 953 | ||
| 458 | // IDA: void __usercall DeclareWinner(int pWinner_index@<EAX>) |
954 | // IDA: void __usercall DeclareWinner(int pWinner_index@<EAX>) |
| 459 | void DeclareWinner(int pWinner_index) { |
955 | void DeclareWinner(int pWinner_index) { |
| 460 |
|
956 | tNet_message* the_message; |
| 461 |
|
957 | int i; |
| 462 |
|
958 | int j; |
| 463 |
|
959 | int best_score_index; |
| 464 |
|
960 | char s[256]; |
| 465 | LOG_TRACE("(%d)", pWinner_index); |
961 | LOG_TRACE("(%d)", pWinner_index); |
| - | 962 | ||
| 466 |
|
963 | DoNetScores2(1); |
| - | 964 | the_message = NetBuildMessage(NETMSGID_RACEOVER, 0); |
|
| - | 965 | the_message->contents.data.race_over.reason = eRace_over_network_victory; |
|
| - | 966 | NetGuaranteedSendMessageToPlayer(gCurrent_net_game, the_message, gNet_players[pWinner_index].ID, NULL); |
|
| - | 967 | gNet_players[pWinner_index].won++; |
|
| - | 968 | if (gCurrent_net_game->type == eNet_game_type_sudden_death |
|
| - | 969 | || gCurrent_net_game->type == eNet_game_type_tag |
|
| - | 970 | || gCurrent_net_game->type == eNet_game_type_fight_to_death) { |
|
| - | 971 | gNet_players[pWinner_index].games_score += gGame_scores[5]; |
|
| - | 972 | } |
|
| - | 973 | sprintf(s, "%s %s", gNet_players[pWinner_index].player_name, GetMiscString(kMiscString_IS_THE_WINNER)); |
|
| - | 974 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 975 | if (gCurrent_net_game->type != eNet_game_type_sudden_death && gCurrent_net_game->type != eNet_game_type_tag && gCurrent_net_game->type != eNet_game_type_fight_to_death) { |
|
| - | 976 | best_score_index = gNet_players[i].last_score_index; |
|
| - | 977 | for (j = 0; j < gNumber_of_net_players; j++) { |
|
| - | 978 | if (gNet_players[j].score == gNet_players[i].score && gNet_players[j].last_score_index < best_score_index) { |
|
| - | 979 | best_score_index = gNet_players[j].last_score_index; |
|
| - | 980 | } |
|
| - | 981 | } |
|
| - | 982 | gNet_players[i].games_score += gGame_scores[5 - best_score_index]; |
|
| - | 983 | } |
|
| - | 984 | gNet_players[i].played++; |
|
| - | 985 | if (i != pWinner_index) { |
|
| - | 986 | the_message = NetBuildMessage(NETMSGID_RACEOVER, 0); |
|
| - | 987 | the_message->contents.data.race_over.reason = eRace_over_network_loss; |
|
| - | 988 | NetGuaranteedSendMessageToPlayer(gCurrent_net_game, the_message, gNet_players[i].ID, NULL); |
|
| - | 989 | NetSendHeadupToPlayer(s, gNet_players[i].ID); |
|
| - | 990 | } |
|
| - | 991 | } |
|
| - | 992 | gReceived_game_scores = 1; |
|
| - | 993 | the_message = NetBuildMessage(NETMSGID_GAMESCORES, 0); |
|
| - | 994 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 995 | the_message->contents.data.game_scores.scores[i].played = gNet_players[i].played; |
|
| - | 996 | the_message->contents.data.game_scores.scores[i].won = gNet_players[i].won; |
|
| - | 997 | the_message->contents.data.game_scores.scores[i].score = gNet_players[i].games_score; |
|
| - | 998 | } |
|
| - | 999 | NetGuaranteedSendMessageToAllPlayers(gCurrent_net_game, the_message, NULL); |
|
| 467 | } |
1000 | } |
| 468 | 1001 | ||
| 469 | // IDA: void __usercall PlayerIsIt(tNet_game_player_info *pPlayer@<EAX>) |
1002 | // IDA: void __usercall PlayerIsIt(tNet_game_player_info *pPlayer@<EAX>) |
| 470 | void PlayerIsIt(tNet_game_player_info* pPlayer) { |
1003 | void PlayerIsIt(tNet_game_player_info* pPlayer) { |
| 471 |
|
1004 | int i; |
| 472 |
|
1005 | char s[256]; |
| 473 | LOG_TRACE("(%p)", pPlayer); |
1006 | LOG_TRACE("(%p)", pPlayer); |
| - | 1007 | ||
| - | 1008 | if (pPlayer - gNet_players == gIt_or_fox) { |
|
| - | 1009 | return; |
|
| - | 1010 | } |
|
| - | 1011 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 1012 | StopCarBeingIt(gNet_players[i].car); |
|
| - | 1013 | } |
|
| - | 1014 | if (gCurrent_net_game->type == eNet_game_type_foxy) { |
|
| - | 1015 | pPlayer->car->power_up_levels[1] = 0; |
|
| - | 1016 | } else if (gCurrent_net_game->type == eNet_game_type_tag && gIt_or_fox >= 0) { |
|
| - | 1017 | gNet_players[gIt_or_fox].car->power_up_levels[1] = 0; |
|
| - | 1018 | } |
|
| 474 |
|
1019 | MakeCarIt(pPlayer->car); |
| - | 1020 | gIt_or_fox = pPlayer - gNet_players; |
|
| - | 1021 | sprintf(s, "%s", GetMiscString(gCurrent_net_game->type == eNet_game_type_tag ? kMiscString_HA_HA_YOU_ARE_IT : kMiscString_YOU_ARE_THE_FOX)); |
|
| - | 1022 | NetSendHeadupToPlayer(s, pPlayer->ID); |
|
| - | 1023 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 1024 | if (&gNet_players[i] != pPlayer) { |
|
| - | 1025 | sprintf(s, "%s %s", pPlayer->player_name, GetMiscString(gCurrent_net_game->type == eNet_game_type_tag ? 185 : 187)); |
|
| - | 1026 | NetSendHeadupToPlayer(s, gNet_players[i].ID); |
|
| - | 1027 | } |
|
| - | 1028 | } |
|
| 475 | } |
1029 | } |
| 476 | 1030 | ||
| 477 | // IDA: int __usercall FarEnoughAway@<EAX>(tNet_game_player_info *pPlayer_1@<EAX>, tNet_game_player_info *pPlayer_2@<EDX>) |
1031 | // IDA: int __usercall FarEnoughAway@<EAX>(tNet_game_player_info *pPlayer_1@<EAX>, tNet_game_player_info *pPlayer_2@<EDX>) |
| 478 | int FarEnoughAway(tNet_game_player_info* pPlayer_1, tNet_game_player_info* pPlayer_2) { |
1032 | int FarEnoughAway(tNet_game_player_info* pPlayer_1, tNet_game_player_info* pPlayer_2) { |
| 479 |
|
1033 | br_vector3 difference; |
| 480 | LOG_TRACE("(%p, %p)", pPlayer_1, pPlayer_2); |
1034 | LOG_TRACE("(%p, %p)", pPlayer_1, pPlayer_2); |
| - | 1035 | ||
| - | 1036 | BrVector3Sub(&difference, &pPlayer_1->car->pos, &pPlayer_2->car->pos); |
|
| 481 |
|
1037 | return BrVector3LengthSquared(&difference) >= 4.0f; |
| 482 | } |
1038 | } |
| 483 | 1039 | ||
| 484 | // IDA: void __usercall CarInContactWithItOrFox(tNet_game_player_info *pPlayer@<EAX>) |
1040 | // IDA: void __usercall CarInContactWithItOrFox(tNet_game_player_info *pPlayer@<EAX>) |
| 485 | void CarInContactWithItOrFox(tNet_game_player_info* pPlayer) { |
1041 | void CarInContactWithItOrFox(tNet_game_player_info* pPlayer) { |
| 486 | LOG_TRACE("(%p)", pPlayer); |
1042 | LOG_TRACE("(%p)", pPlayer); |
| - | 1043 | ||
| - | 1044 | if (gCurrent_net_game->type == eNet_game_type_tag || gCurrent_net_game->type == eNet_game_type_foxy) { |
|
| - | 1045 | if (PDGetTotalTime() - gLast_it_change > 500) { |
|
| - | 1046 | gLast_it_change = PDGetTotalTime(); |
|
| - | 1047 | if (gIt_or_fox >= 0) { |
|
| - | 1048 | gLast_lepper = &gNet_players[gIt_or_fox]; |
|
| - | 1049 | } |
|
| 487 |
|
1050 | PlayerIsIt(pPlayer); |
| - | 1051 | } |
|
| - | 1052 | } |
|
| 488 | } |
1053 | } |
| 489 | 1054 | ||
| 490 | // IDA: void __usercall SelectRandomItOrFox(int pNot_this_one@<EAX>) |
1055 | // IDA: void __usercall SelectRandomItOrFox(int pNot_this_one@<EAX>) |
| 491 | void SelectRandomItOrFox(int pNot_this_one) { |
1056 | void SelectRandomItOrFox(int pNot_this_one) { |
| 492 |
|
1057 | int i; |
| 493 |
|
1058 | int new_choice; |
| 494 | LOG_TRACE("(%d)", pNot_this_one); |
1059 | LOG_TRACE("(%d)", pNot_this_one); |
| - | 1060 | ||
| - | 1061 | new_choice = 0; |
|
| - | 1062 | gLast_lepper = NULL; |
|
| - | 1063 | if (gCurrent_net_game->type == eNet_game_type_tag) { |
|
| - | 1064 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 1065 | if (gNet_players[i].last_score_index == 0) { |
|
| - | 1066 | new_choice = i; |
|
| 495 |
|
1067 | break; |
| - | 1068 | } |
|
| - | 1069 | } |
|
| - | 1070 | } else { |
|
| - | 1071 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 1072 | if (i != pNot_this_one && gNet_players[i].last_score_index == gNumber_of_net_players - 1) { |
|
| - | 1073 | PlayerIsIt(&gNet_players[i]); |
|
| - | 1074 | return; |
|
| - | 1075 | } |
|
| - | 1076 | } |
|
| - | 1077 | do { |
|
| - | 1078 | new_choice = IRandomBetween(0, gNumber_of_net_players - 1); |
|
| - | 1079 | } while (new_choice == pNot_this_one && !gNet_players[new_choice].car->knackered); |
|
| - | 1080 | } |
|
| - | 1081 | PlayerIsIt(&gNet_players[new_choice]); |
|
| 496 | } |
1082 | } |
| 497 | 1083 | ||
| 498 | // IDA: void __cdecl CalcPlayerScores() |
1084 | // IDA: void __cdecl CalcPlayerScores() |
| 499 | void CalcPlayerScores(void) { |
1085 | void CalcPlayerScores(void) { |
| 500 |
|
1086 | int i; |
| 501 |
|
1087 | int j; |
| 502 |
|
1088 | int knock_out_bit; |
| 503 |
|
1089 | int e_dam; |
| 504 |
|
1090 | int t_dam; |
| 505 |
|
1091 | int d_dam; |
| 506 |
|
1092 | int w_dam; |
| 507 |
|
1093 | int cars_left; |
| 508 |
|
1094 | int car_left; |
| 509 |
|
1095 | int flags; |
| 510 |
|
1096 | int score; |
| 511 |
|
1097 | int highest; |
| 512 |
|
1098 | int next_highest; |
| 513 |
|
1099 | int lowest_score; |
| 514 |
|
1100 | int player_left; |
| 515 |
|
1101 | int new_choice; |
| 516 |
|
1102 | tCar_spec* car; |
| 517 |
|
1103 | tNet_message* message; |
| 518 |
|
1104 | tS32 time; |
| 519 |
|
1105 | char s[256]; |
| 520 |
|
1106 | tNet_game_player_info* lowest_score_player; |
| 521 | LOG_TRACE("()"); |
1107 | LOG_TRACE("()"); |
| - | 1108 | ||
| 522 |
|
1109 | time = GetTotalTime(); |
| - | 1110 | ||
| - | 1111 | if (gCurrent_net_game->type == eNet_game_type_carnage) { |
|
| - | 1112 | highest = 0; |
|
| - | 1113 | next_highest = 0; |
|
| - | 1114 | for (i = 0; i < gNumber_of_net_players; ++i) { |
|
| - | 1115 | if (gNet_players[i].score > highest) { |
|
| - | 1116 | next_highest = highest; |
|
| - | 1117 | highest = gNet_players[i].score; |
|
| - | 1118 | } else if (gNet_players[i].score > next_highest) { |
|
| - | 1119 | next_highest = gNet_players[i].score; |
|
| - | 1120 | } |
|
| - | 1121 | } |
|
| - | 1122 | gPed_target = (gTotal_peds - (gProgram_state.peds_killed - highest - next_highest)) / 2 + 1; |
|
| - | 1123 | if (gCurrent_net_game->options.race_end_target < gPed_target) { |
|
| - | 1124 | gPed_target = gCurrent_net_game->options.race_end_target; |
|
| - | 1125 | } |
|
| - | 1126 | } else if (gCurrent_net_game->type == eNet_game_type_tag || gCurrent_net_game->type == eNet_game_type_foxy) { |
|
| - | 1127 | if (gIt_or_fox < 0) { |
|
| - | 1128 | SelectRandomItOrFox(-1); |
|
| - | 1129 | } |
|
| - | 1130 | if (gLast_lepper != NULL && gIt_or_fox >= 0 && FarEnoughAway(gLast_lepper, &gNet_players[gIt_or_fox])) { |
|
| - | 1131 | gLast_lepper = NULL; |
|
| - | 1132 | } |
|
| - | 1133 | } |
|
| - | 1134 | lowest_score = 9999; |
|
| - | 1135 | lowest_score_player = NULL; |
|
| - | 1136 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 1137 | car = gNet_players[i].car; |
|
| - | 1138 | if (gNet_players[i].reposition_time != 0 && gNet_players[i].reposition_time <= time && (!gRace_finished || gRace_over_reason == -1)) { |
|
| - | 1139 | RepositionPlayer(i); |
|
| - | 1140 | } |
|
| - | 1141 | if (gNet_players[i].last_waste_message != 0 |
|
| - | 1142 | && !gNet_players[i].wasteage_attributed |
|
| - | 1143 | && time - gNet_players[i].last_waste_message > 500) { |
|
| - | 1144 | sprintf(s, "%s %s", gNet_players[i].player_name, GetMiscString(kMiscString_COMMITTED_SUICIDE)); |
|
| - | 1145 | gNet_players[i].last_waste_message = 0; |
|
| - | 1146 | gNet_players[i].wasteage_attributed = 0; |
|
| - | 1147 | if (gCurrent_net_game->type == eNet_game_type_car_crusher) { |
|
| - | 1148 | gNet_players[i].score--; |
|
| - | 1149 | } |
|
| - | 1150 | NetSendHeadupToEverybody(s); |
|
| - | 1151 | } |
|
| - | 1152 | SetKnackeredFlag(car); |
|
| - | 1153 | if (car->knackered && !gNet_players[i].wasted) { |
|
| - | 1154 | gNet_players[i].wasted = 1; |
|
| - | 1155 | message = NetBuildMessage(NETMSGID_WASTED, 0); |
|
| - | 1156 | message->contents.data.wasted.victim = gNet_players[i].ID; |
|
| - | 1157 | message->contents.data.wasted.culprit = -1; |
|
| - | 1158 | NetGuaranteedSendMessageToEverybody(gCurrent_net_game, message, 0); |
|
| - | 1159 | switch (gCurrent_net_game->type) { |
|
| - | 1160 | case eNet_game_type_fight_to_death: |
|
| - | 1161 | cars_left = 0; |
|
| - | 1162 | for (j = 0; j < gNumber_of_net_players; j++) { |
|
| - | 1163 | if (!gNet_players[j].wasted) { |
|
| - | 1164 | cars_left++; |
|
| - | 1165 | car_left = j; |
|
| - | 1166 | } |
|
| - | 1167 | } |
|
| - | 1168 | gNet_players[i].games_score += gGame_scores[5 - cars_left]; |
|
| - | 1169 | if (cars_left == 1) { |
|
| - | 1170 | DeclareWinner(car_left); |
|
| - | 1171 | } else if (cars_left <= 0) { |
|
| - | 1172 | EverybodysLost(); |
|
| - | 1173 | } |
|
| - | 1174 | break; |
|
| - | 1175 | case eNet_game_type_car_crusher: |
|
| - | 1176 | case eNet_game_type_sudden_death: |
|
| - | 1177 | gNet_players[i].reposition_time = GetTotalTime() + 5000; |
|
| - | 1178 | break; |
|
| - | 1179 | case eNet_game_type_carnage: |
|
| - | 1180 | gNet_players[i].reposition_time = GetTotalTime() + 5000; |
|
| - | 1181 | gNet_players[i].score /= 2; |
|
| - | 1182 | break; |
|
| - | 1183 | case eNet_game_type_checkpoint: |
|
| - | 1184 | if (gNet_players[i].score >> 16 != gCurrent_race.check_point_count) { |
|
| - | 1185 | knock_out_bit = IRandomBetween(0, gCurrent_race.check_point_count - 1); |
|
| - | 1186 | while (((1 << knock_out_bit) & gNet_players[i].score) != 0) { |
|
| - | 1187 | knock_out_bit++; |
|
| - | 1188 | if (gCurrent_race.check_point_count <= knock_out_bit) { |
|
| - | 1189 | knock_out_bit = 0; |
|
| - | 1190 | } |
|
| - | 1191 | } |
|
| - | 1192 | gNet_players[i].score |= 1 << knock_out_bit; |
|
| - | 1193 | } |
|
| - | 1194 | gNet_players[i].reposition_time = GetTotalTime() + 5000; |
|
| - | 1195 | break; |
|
| - | 1196 | case eNet_game_type_tag: |
|
| - | 1197 | gNet_players[i].reposition_time = GetTotalTime() + 5000; |
|
| - | 1198 | PlayerIsIt(&gNet_players[i]); |
|
| - | 1199 | break; |
|
| - | 1200 | case eNet_game_type_foxy: |
|
| - | 1201 | gNet_players[i].reposition_time = GetTotalTime() + 5000; |
|
| - | 1202 | gNet_players[i].score /= 2; |
|
| - | 1203 | if (gNumber_of_net_players > 1 && i == gIt_or_fox) { |
|
| - | 1204 | SelectRandomItOrFox(i); |
|
| - | 1205 | } |
|
| - | 1206 | break; |
|
| - | 1207 | default: |
|
| - | 1208 | break; |
|
| - | 1209 | } |
|
| - | 1210 | } |
|
| - | 1211 | switch (gCurrent_net_game->type) { |
|
| - | 1212 | case eNet_game_type_fight_to_death: |
|
| - | 1213 | if (car->knackered) { |
|
| - | 1214 | if (gCurrent_net_game->type == eNet_game_type_checkpoint |
|
| - | 1215 | || gCurrent_net_game->type == eNet_game_type_tag) { |
|
| - | 1216 | gNet_players[i].score = 1000000; |
|
| - | 1217 | } else { |
|
| - | 1218 | gNet_players[i].score = -1000000; |
|
| - | 1219 | } |
|
| - | 1220 | } else { |
|
| - | 1221 | e_dam = car->damage_units[eDamage_engine].damage_level; |
|
| - | 1222 | t_dam = car->damage_units[eDamage_transmission].damage_level; |
|
| - | 1223 | d_dam = car->damage_units[eDamage_driver].damage_level; |
|
| - | 1224 | w_dam = (car->damage_units[eDamage_lr_wheel].damage_level |
|
| - | 1225 | + car->damage_units[eDamage_lf_wheel].damage_level |
|
| - | 1226 | + car->damage_units[eDamage_rr_wheel].damage_level |
|
| - | 1227 | + car->damage_units[eDamage_rf_wheel].damage_level) |
|
| - | 1228 | / 4; |
|
| - | 1229 | ||
| - | 1230 | if (e_dam >= t_dam && e_dam >= d_dam && e_dam >= w_dam) { |
|
| - | 1231 | gNet_players[i].score = 100 - e_dam; |
|
| - | 1232 | } else if (t_dam >= d_dam && t_dam >= w_dam) { |
|
| - | 1233 | gNet_players[i].score = 100 - t_dam; |
|
| - | 1234 | } else if (w_dam >= d_dam) { |
|
| - | 1235 | gNet_players[i].score = 100 - w_dam; |
|
| - | 1236 | } else { |
|
| - | 1237 | gNet_players[i].score = 100 - d_dam; |
|
| - | 1238 | } |
|
| - | 1239 | } |
|
| - | 1240 | break; |
|
| - | 1241 | case eNet_game_type_carnage: |
|
| - | 1242 | if (gNet_players[i].score >= gPed_target && !gRace_finished) { |
|
| - | 1243 | DeclareWinner(i); |
|
| - | 1244 | } |
|
| - | 1245 | break; |
|
| - | 1246 | case eNet_game_type_checkpoint: |
|
| - | 1247 | score = 0; |
|
| - | 1248 | gNet_players[i].score = gNet_players[i].score & 0xffff; |
|
| - | 1249 | flags = gNet_players[i].score; |
|
| - | 1250 | for (j = 0; j < gCurrent_race.check_point_count; j++) { |
|
| - | 1251 | if ((flags & 1) != 0) { |
|
| - | 1252 | score++; |
|
| - | 1253 | } |
|
| - | 1254 | flags >>= 1; |
|
| - | 1255 | } |
|
| - | 1256 | gNet_players[i].score |= score << 16; |
|
| - | 1257 | if (!score && !gRace_finished) { |
|
| - | 1258 | DeclareWinner(i); |
|
| - | 1259 | } |
|
| - | 1260 | break; |
|
| - | 1261 | case eNet_game_type_sudden_death: |
|
| - | 1262 | if (gNet_players[i].score >= 0) { |
|
| - | 1263 | if (gNet_players[i].score >= lowest_score) { |
|
| - | 1264 | if (gNet_players[i].score == lowest_score) { |
|
| - | 1265 | lowest_score_player = 0; |
|
| - | 1266 | } |
|
| - | 1267 | } else { |
|
| - | 1268 | lowest_score = gNet_players[i].score; |
|
| - | 1269 | lowest_score_player = &gNet_players[i]; |
|
| - | 1270 | } |
|
| - | 1271 | } |
|
| - | 1272 | break; |
|
| - | 1273 | case eNet_game_type_tag: |
|
| - | 1274 | if (i == gIt_or_fox && !gCountdown && gNet_players[i].score >= 0) { |
|
| - | 1275 | gNet_players[i].score += gFrame_period; |
|
| - | 1276 | if (gNet_players[i].score >= gCurrent_net_game->options.race_end_target) { |
|
| - | 1277 | lowest_score_player = &gNet_players[i]; |
|
| - | 1278 | } |
|
| - | 1279 | } |
|
| - | 1280 | break; |
|
| - | 1281 | case eNet_game_type_foxy: |
|
| - | 1282 | if (i == gIt_or_fox && !gCountdown && !gRace_finished) { |
|
| - | 1283 | gNet_players[i].score += gFrame_period; |
|
| - | 1284 | if (gNet_players[i].score >= gCurrent_net_game->options.race_end_target) { |
|
| - | 1285 | DeclareWinner(i); |
|
| - | 1286 | } |
|
| - | 1287 | } |
|
| - | 1288 | break; |
|
| - | 1289 | default: |
|
| - | 1290 | continue; |
|
| - | 1291 | } |
|
| - | 1292 | } |
|
| - | 1293 | if ((gCurrent_net_game->type == eNet_game_type_sudden_death || gCurrent_net_game->type == eNet_game_type_tag) |
|
| - | 1294 | && lowest_score_player != NULL |
|
| - | 1295 | && lowest_score_player->score >= 0) { |
|
| - | 1296 | player_left = -1; |
|
| - | 1297 | cars_left = 0; |
|
| - | 1298 | for (i = 0; i < gNumber_of_net_players; ++i) { |
|
| - | 1299 | if (gNet_players[i].score >= 0 && &gNet_players[i] != lowest_score_player) { |
|
| - | 1300 | ++cars_left; |
|
| - | 1301 | if (player_left == -1) { |
|
| - | 1302 | player_left = i; |
|
| - | 1303 | } else { |
|
| - | 1304 | player_left = -2; |
|
| - | 1305 | } |
|
| - | 1306 | } |
|
| - | 1307 | } |
|
| - | 1308 | if (cars_left) { |
|
| - | 1309 | lowest_score_player->car->knackered = 1; |
|
| - | 1310 | lowest_score_player->wasted = 1; |
|
| - | 1311 | lowest_score_player->games_score += gGame_scores[5 - cars_left]; |
|
| - | 1312 | lowest_score_player->score = -1; |
|
| - | 1313 | if (player_left == -1) { |
|
| - | 1314 | EverybodysLost(); |
|
| - | 1315 | } else if (player_left < 0) { |
|
| - | 1316 | if (gCurrent_net_game->type == eNet_game_type_tag) { |
|
| - | 1317 | SelectRandomItOrFox(i); |
|
| - | 1318 | } |
|
| - | 1319 | SendGameplay(lowest_score_player->ID, eNet_gameplay_suddenly_death, 0, 0, 0, 0); |
|
| - | 1320 | sprintf(s, "%s %s", lowest_score_player->player_name, GetMiscString(kMiscString_IS_OUT)); |
|
| - | 1321 | NetSendHeadupToEverybody(s); |
|
| - | 1322 | } else { |
|
| - | 1323 | DeclareWinner(player_left); |
|
| - | 1324 | } |
|
| - | 1325 | } |
|
| - | 1326 | } |
|
| 523 | } |
1327 | } |
| 524 | 1328 | ||
| 525 | // IDA: void __cdecl SendPlayerScores() |
1329 | // IDA: void __cdecl SendPlayerScores() |
| 526 | void SendPlayerScores(void) { |
1330 | void SendPlayerScores(void) { |
| 527 | tNet_contents* the_contents; |
1331 | tNet_contents* the_contents; |
| Line 649... | Line 1453... | ||
| 649 | BuyPSPowerup(2); |
1453 | BuyPSPowerup(2); |
| 650 | } |
1454 | } |
| 651 | 1455 | ||
| 652 | // IDA: void __usercall UseGeneralScore(int pScore@<EAX>) |
1456 | // IDA: void __usercall UseGeneralScore(int pScore@<EAX>) |
| 653 | void UseGeneralScore(int pScore) { |
1457 | void UseGeneralScore(int pScore) { |
| 654 |
|
1458 | int i; |
| 655 | LOG_TRACE("(%d)", pScore); |
1459 | LOG_TRACE("(%d)", pScore); |
| - | 1460 | ||
| - | 1461 | if (gCurrent_net_game->type == eNet_game_type_carnage) { |
|
| 656 |
|
1462 | gPed_target = pScore; |
| - | 1463 | } else if ((gCurrent_net_game->type == eNet_game_type_tag || gCurrent_net_game->type == eNet_game_type_foxy) && gNet_players[gIt_or_fox].ID != pScore) { |
|
| - | 1464 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 1465 | StopCarBeingIt(gNet_players[i].car); |
|
| - | 1466 | } |
|
| - | 1467 | for (i = 0; i < gNumber_of_net_players; i++) { |
|
| - | 1468 | if (gNet_players[i].ID == pScore) { |
|
| - | 1469 | MakeCarIt(gNet_players[i].car); |
|
| - | 1470 | gIt_or_fox = i; |
|
| - | 1471 | } |
|
| - | 1472 | } |
|
| - | 1473 | } |
|
| 657 | } |
1474 | } |
| 658 | 1475 | ||
| 659 | // IDA: void __usercall NetSendEnvironmentChanges(tNet_game_player_info *pPlayer@<EAX>) |
1476 | // IDA: void __usercall NetSendEnvironmentChanges(tNet_game_player_info *pPlayer@<EAX>) |
| 660 | void NetSendEnvironmentChanges(tNet_game_player_info* pPlayer) { |
1477 | void NetSendEnvironmentChanges(tNet_game_player_info* pPlayer) { |
| 661 | LOG_TRACE("(%p)", pPlayer); |
1478 | LOG_TRACE("(%p)", pPlayer); |
| Line 679... | Line 1496... | ||
| 679 | } |
1496 | } |
| 680 | } |
1497 | } |
| 681 | 1498 | ||
| 682 | // IDA: void __usercall ReceivedGameplay(tNet_contents *pContents@<EAX>, tNet_message *pMessage@<EDX>, tU32 pReceive_time@<EBX>) |
1499 | // IDA: void __usercall ReceivedGameplay(tNet_contents *pContents@<EAX>, tNet_message *pMessage@<EDX>, tU32 pReceive_time@<EBX>) |
| 683 | void ReceivedGameplay(tNet_contents* pContents, tNet_message* pMessage, tU32 pReceive_time) { |
1500 | void ReceivedGameplay(tNet_contents* pContents, tNet_message* pMessage, tU32 pReceive_time) { |
| 684 |
|
1501 | int must_revert_reentrancy; |
| 685 |
|
1502 | int gPixel_buffer_size; |
| 686 |
|
1503 | char* gPixels_copy; |
| 687 |
|
1504 | char* gPalette_copy; |
| 688 |
|
1505 | static int pause_semaphore; |
| 689 | LOG_TRACE("(%p, %p, %d)", pContents, pMessage, pReceive_time); |
1506 | LOG_TRACE("(%p, %p, %d)", pContents, pMessage, pReceive_time); |
| - | 1507 | ||
| - | 1508 | switch (pContents->data.gameplay.mess) { |
|
| - | 1509 | case eNet_gameplay_host_paused: |
|
| - | 1510 | if (!pause_semaphore) { |
|
| - | 1511 | gPixel_buffer_size = gBack_screen->row_bytes * gBack_screen->height; |
|
| - | 1512 | gPixels_copy = BrMemAllocate(gPixel_buffer_size, kMem_quit_vfy_pixels); |
|
| - | 1513 | gPalette_copy = BrMemAllocate(1024, kMem_quit_vfy_pal); |
|
| - | 1514 | memcpy(gPixels_copy, gBack_screen->pixels, gPixel_buffer_size); |
|
| - | 1515 | memcpy(gPalette_copy, gCurrent_palette_pixels, 1024); |
|
| - | 1516 | pause_semaphore = 1; |
|
| - | 1517 | NetFullScreenMessage(kMiscString_PLEASE_WAIT_HOST_HAS_PAUSED, 1); |
|
| - | 1518 | must_revert_reentrancy = PermitNetServiceReentrancy(); |
|
| - | 1519 | do { |
|
| - | 1520 | NetService(0); |
|
| - | 1521 | if (CheckQuit()) { |
|
| - | 1522 | NetFullScreenMessage(kMiscString_PLEASE_WAIT_HOST_HAS_PAUSED, 1); |
|
| - | 1523 | } |
|
| - | 1524 | } while (gWaiting_for_unpause |
|
| - | 1525 | && gProgram_state.prog_status != eProg_idling |
|
| - | 1526 | && (!gRace_finished || gRace_over_reason != eRace_over_abandoned)); |
|
| - | 1527 | if (must_revert_reentrancy) { |
|
| - | 1528 | HaltNetServiceReentrancy(); |
|
| - | 1529 | } |
|
| - | 1530 | gWaiting_for_unpause = 1; |
|
| - | 1531 | FadePaletteDown(); |
|
| - | 1532 | memcpy(gBack_screen->pixels, gPixels_copy, gPixel_buffer_size); |
|
| - | 1533 | memcpy(gCurrent_palette_pixels, gPalette_copy, 1024); |
|
| - | 1534 | BrMemFree(gPixels_copy); |
|
| - | 1535 | BrMemFree(gPalette_copy); |
|
| - | 1536 | PDScreenBufferSwap(0); |
|
| - | 1537 | FadePaletteUp(); |
|
| - | 1538 | pause_semaphore = 0; |
|
| - | 1539 | } |
|
| - | 1540 | break; |
|
| - | 1541 | case eNet_gameplay_earn_credits: |
|
| - | 1542 | EarnCredits(pContents->data.gameplay.param_1); |
|
| - | 1543 | break; |
|
| - | 1544 | case eNet_gameplay_host_unpaused: |
|
| - | 1545 | gWaiting_for_unpause = 0; |
|
| - | 1546 | break; |
|
| - | 1547 | case eNet_gameplay_suicide: |
|
| - | 1548 | KnackerThisCar(NetCarFromPlayerID(pMessage->sender)); |
|
| - | 1549 | break; |
|
| - | 1550 | case eNet_gameplay_go_for_it: |
|
| - | 1551 | gWait_for_it = 0; |
|
| - | 1552 | break; |
|
| - | 1553 | default: |
|
| - | 1554 | if (gCurrent_net_game->type == eNet_game_type_checkpoint || gCurrent_net_game->type == eNet_game_type_sudden_death || gCurrent_net_game->type == eNet_game_type_tag) { |
|
| - | 1555 | switch (pContents->data.gameplay.mess) { |
|
| - | 1556 | case eNet_gameplay_checkpoint: |
|
| - | 1557 | Checkpoint(pContents->data.gameplay.param_1, 1); |
|
| - | 1558 | break; |
|
| - | 1559 | case eNet_gameplay_wrong_checkpoint: |
|
| - | 1560 | WrongCheckpoint(pContents->data.gameplay.param_1); |
|
| - | 1561 | break; |
|
| - | 1562 | case eNet_gameplay_suddenly_death: |
|
| - | 1563 | DoFancyHeadup(kFancyHeadupNetworkRaceOverNetworkLoss); |
|
| - | 1564 | ChangeAmbientPratcam(36); |
|
| - | 1565 | gRace_finished = 1; |
|
| - | 1566 | break; |
|
| - | 1567 | default: |
|
| 690 |
|
1568 | break; |
| - | 1569 | } |
|
| - | 1570 | } |
|
| - | 1571 | break; |
|
| - | 1572 | } |
|
| 691 | } |
1573 | } |
| 692 | 1574 | ||
| 693 | // IDA: void __usercall SendGameplay(tPlayer_ID pPlayer@<EAX>, tNet_gameplay_mess pMess@<EDX>, int pParam_1@<EBX>, int pParam_2@<ECX>, int pParam_3, int pParam_4) |
1575 | // IDA: void __usercall SendGameplay(tPlayer_ID pPlayer@<EAX>, tNet_gameplay_mess pMess@<EDX>, int pParam_1@<EBX>, int pParam_2@<ECX>, int pParam_3, int pParam_4) |
| 694 | void SendGameplay(tPlayer_ID pPlayer, tNet_gameplay_mess pMess, int pParam_1, int pParam_2, int pParam_3, int pParam_4) { |
1576 | void SendGameplay(tPlayer_ID pPlayer, tNet_gameplay_mess pMess, int pParam_1, int pParam_2, int pParam_3, int pParam_4) { |
| 695 | tNet_message* the_message; |
1577 | tNet_message* the_message; |
| Line 718... | Line 1600... | ||
| 718 | NetGuaranteedSendMessageToAllPlayers(gCurrent_net_game, the_message, NULL); |
1600 | NetGuaranteedSendMessageToAllPlayers(gCurrent_net_game, the_message, NULL); |
| 719 | } |
1601 | } |
| 720 | 1602 | ||
| 721 | // IDA: void __usercall SendGameplayToHost(tNet_gameplay_mess pMess@<EAX>, int pParam_1@<EDX>, int pParam_2@<EBX>, int pParam_3@<ECX>, int pParam_4) |
1603 | // IDA: void __usercall SendGameplayToHost(tNet_gameplay_mess pMess@<EAX>, int pParam_1@<EDX>, int pParam_2@<EBX>, int pParam_3@<ECX>, int pParam_4) |
| 722 | void SendGameplayToHost(tNet_gameplay_mess pMess, int pParam_1, int pParam_2, int pParam_3, int pParam_4) { |
1604 | void SendGameplayToHost(tNet_gameplay_mess pMess, int pParam_1, int pParam_2, int pParam_3, int pParam_4) { |
| 723 |
|
1605 | tNet_message* the_message; |
| 724 | LOG_TRACE("(%d, %d, %d, %d, %d)", pMess, pParam_1, pParam_2, pParam_3, pParam_4); |
1606 | LOG_TRACE("(%d, %d, %d, %d, %d)", pMess, pParam_1, pParam_2, pParam_3, pParam_4); |
| - | 1607 | ||
| 725 |
|
1608 | if (gNet_mode == eNet_mode_client) { |
| - | 1609 | the_message = NetBuildMessage(NETMSGID_GAMEPLAY, 0); |
|
| - | 1610 | the_message->contents.data.gameplay.mess = pMess; |
|
| - | 1611 | the_message->contents.data.gameplay.param_1 = pParam_1; |
|
| - | 1612 | the_message->contents.data.gameplay.param_2 = pParam_2; |
|
| - | 1613 | the_message->contents.data.gameplay.param_3 = pParam_3; |
|
| - | 1614 | the_message->contents.data.gameplay.param_4 = pParam_4; |
|
| - | 1615 | NetGuaranteedSendMessageToHost(gCurrent_net_game, the_message, NULL); |
|
| - | 1616 | } |
|
| 726 | } |
1617 | } |
| 727 | 1618 | ||
| 728 | // IDA: void __cdecl InitNetGameplayStuff() |
1619 | // IDA: void __cdecl InitNetGameplayStuff() |
| 729 | void InitNetGameplayStuff(void) { |
1620 | void InitNetGameplayStuff(void) { |
| 730 | LOG_TRACE("()"); |
1621 | LOG_TRACE("()"); |
| Line 735... | Line 1626... | ||
| 735 | NetObtainSystemUserName(gNet_player_name, 32); |
1626 | NetObtainSystemUserName(gNet_player_name, 32); |
| 736 | } |
1627 | } |
| 737 | 1628 | ||
| 738 | // IDA: void __usercall NetSendPointCrush(tCar_spec *pCar@<EAX>, tU16 pCrush_point_index@<EDX>, br_vector3 *pEnergy_vector@<EBX>) |
1629 | // IDA: void __usercall NetSendPointCrush(tCar_spec *pCar@<EAX>, tU16 pCrush_point_index@<EDX>, br_vector3 *pEnergy_vector@<EBX>) |
| 739 | void NetSendPointCrush(tCar_spec* pCar, tU16 pCrush_point_index, br_vector3* pEnergy_vector) { |
1630 | void NetSendPointCrush(tCar_spec* pCar, tU16 pCrush_point_index, br_vector3* pEnergy_vector) { |
| 740 |
|
1631 | tNet_contents* contents; |
| 741 | LOG_TRACE("(%p, %d, %p)", pCar, pCrush_point_index, pEnergy_vector); |
1632 | LOG_TRACE("(%p, %d, %p)", pCar, pCrush_point_index, pEnergy_vector); |
| 742 | 1633 | ||
| - | 1634 | contents = NetGetBroadcastContents(NETMSGID_CRUSHPOINT, 0); |
|
| 743 |
|
1635 | contents->data.crush.id = NetPlayerFromCar(pCar)->ID; |
| - | 1636 | contents->data.crush.vertex = pCrush_point_index; |
|
| - | 1637 | BrVector3Copy(&contents->data.crush.energy_vector, pEnergy_vector); |
|
| 744 | } |
1638 | } |
| 745 | 1639 | ||
| 746 | // IDA: void __usercall RecievedCrushPoint(tNet_contents *pContents@<EAX>) |
1640 | // IDA: void __usercall RecievedCrushPoint(tNet_contents *pContents@<EAX>) |
| 747 | void RecievedCrushPoint(tNet_contents* pContents) { |
1641 | void RecievedCrushPoint(tNet_contents* pContents) { |
| 748 |
|
1642 | tCar_spec* car; |
| 749 | LOG_TRACE("(%p)", pContents); |
1643 | LOG_TRACE("(%p)", pContents); |
| - | 1644 | ||
| - | 1645 | car = NetCarFromPlayerID(pContents->data.crush.id); |
|
| - | 1646 | if (car == NULL || gNet_mode == eNet_mode_host || car->active || gArrow_mode) { |
|
| - | 1647 | return; |
|
| - | 1648 | } |
|
| - | 1649 | if (car->car_model_actors[car->principal_car_actor].crush_data.number_of_crush_points == 0) { |
|
| - | 1650 | return; |
|
| - | 1651 | } |
|
| - | 1652 | ||
| 750 |
|
1653 | CrushModelPoint( |
| - | 1654 | car, |
|
| - | 1655 | car->principal_car_actor, |
|
| - | 1656 | car->car_model_actors[car->principal_car_actor].actor->model, |
|
| - | 1657 | pContents->data.crush.vertex, |
|
| - | 1658 | &pContents->data.crush.energy_vector, |
|
| - | 1659 | BrVector3Length(&pContents->data.crush.energy_vector) + 0.06f, |
|
| - | 1660 | &car->car_model_actors[car->principal_car_actor].crush_data); |
|
| - | 1661 | SetModelForUpdate(car->car_model_actors[car->principal_car_actor].actor->model, car, 0); |
|
| 751 | } |
1662 | } |
| 752 | 1663 | ||
| 753 | // IDA: void __usercall GetReducedMatrix(tReduced_matrix *m1@<EAX>, br_matrix34 *m2@<EDX>) |
1664 | // IDA: void __usercall GetReducedMatrix(tReduced_matrix *m1@<EAX>, br_matrix34 *m2@<EDX>) |
| 754 | void GetReducedMatrix(tReduced_matrix* m1, br_matrix34* m2) { |
1665 | void GetReducedMatrix(tReduced_matrix* m1, br_matrix34* m2) { |
| 755 | LOG_TRACE("(%p, %p)", m1, m2); |
1666 | LOG_TRACE("(%p, %p)", m1, m2); |
| - | 1667 | ||
| - | 1668 | m1->row1.v[0] = m2->m[0][0]; |
|
| - | 1669 | m1->row1.v[1] = m2->m[0][1]; |
|
| - | 1670 | m1->row1.v[2] = m2->m[0][2]; |
|
| - | 1671 | m1->row2.v[0] = m2->m[1][0]; |
|
| - | 1672 | m1->row2.v[1] = m2->m[1][1]; |
|
| 756 |
|
1673 | m1->row2.v[2] = m2->m[1][2]; |
| - | 1674 | m1->translation.v[0] = m2->m[3][0]; |
|
| - | 1675 | m1->translation.v[1] = m2->m[3][1]; |
|
| - | 1676 | m1->translation.v[2] = m2->m[3][2]; |
|
| 757 | } |
1677 | } |
| 758 | 1678 | ||
| 759 | // IDA: void __usercall GetExpandedMatrix(br_matrix34 *m1@<EAX>, tReduced_matrix *m2@<EDX>) |
1679 | // IDA: void __usercall GetExpandedMatrix(br_matrix34 *m1@<EAX>, tReduced_matrix *m2@<EDX>) |
| 760 | void GetExpandedMatrix(br_matrix34* m1, tReduced_matrix* m2) { |
1680 | void GetExpandedMatrix(br_matrix34* m1, tReduced_matrix* m2) { |
| 761 | LOG_TRACE("(%p, %p)", m1, m2); |
1681 | LOG_TRACE("(%p, %p)", m1, m2); |
| - | 1682 | ||
| - | 1683 | m1->m[0][0] = m2->row1.v[0]; |
|
| - | 1684 | m1->m[0][1] = m2->row1.v[1]; |
|
| - | 1685 | m1->m[0][2] = m2->row1.v[2]; |
|
| - | 1686 | ||
| - | 1687 | m1->m[1][0] = m2->row2.v[0]; |
|
| 762 |
|
1688 | m1->m[1][1] = m2->row2.v[1]; |
| - | 1689 | m1->m[1][2] = m2->row2.v[2]; |
|
| - | 1690 | ||
| - | 1691 | m1->m[3][0] = m2->translation.v[0]; |
|
| - | 1692 | m1->m[3][1] = m2->translation.v[1]; |
|
| - | 1693 | m1->m[3][2] = m2->translation.v[2]; |
|
| - | 1694 | ||
| - | 1695 | m1->m[2][0] = m2->row2.v[2] * m2->row1.v[1] - m2->row2.v[1] * m2->row1.v[2]; |
|
| - | 1696 | m1->m[2][1] = m2->row1.v[2] * m2->row2.v[0] - m2->row2.v[2] * m2->row1.v[0]; |
|
| - | 1697 | m1->m[2][2] = m2->row2.v[1] * m2->row1.v[0] - m2->row1.v[1] * m2->row2.v[0]; |
|
| 763 | } |
1698 | } |
| 764 | 1699 | ||
| 765 | // IDA: void __usercall NetEarnCredits(tNet_game_player_info *pPlayer@<EAX>, tS32 pCredits@<EDX>) |
1700 | // IDA: void __usercall NetEarnCredits(tNet_game_player_info *pPlayer@<EAX>, tS32 pCredits@<EDX>) |
| 766 | void NetEarnCredits(tNet_game_player_info* pPlayer, tS32 pCredits) { |
1701 | void NetEarnCredits(tNet_game_player_info* pPlayer, tS32 pCredits) { |
| 767 | LOG_TRACE("(%p, %d)", pPlayer, pCredits); |
1702 | LOG_TRACE("(%p, %d)", pPlayer, pCredits); |
| - | 1703 | ||
| 768 |
|
1704 | // empty function |
| 769 | } |
1705 | } |