sm64

A Super Mario 64 decompilation
Log | Files | Refs | README | LICENSE

chain_chomp.inc.c (19554B)


      1 
      2 /**
      3  * Behavior for bhvChainChomp, bhvChainChompChainPart, bhvWoodenPost, and bhvChainChompGate.
      4  * bhvChainChomp spawns its bhvWoodenPost in its behavior script. It spawns 5 chain
      5  * parts. Part 0 is the "pivot", which is positioned at the wooden post while
      6  * the chomp is chained up. Parts 1-4 are the other parts, starting from the
      7  * chain chomp and moving toward the pivot.
      8  * Processing order is bhvWoodenPost, bhvChainChompGate, bhvChainChomp, bhvChainChompChainPart.
      9  * The chain parts are processed starting at the post and ending at the chomp.
     10  */
     11 
     12 /**
     13  * Hitbox for chain chomp.
     14  */
     15 static struct ObjectHitbox sChainChompHitbox = {
     16     /* interactType:      */ INTERACT_MR_BLIZZARD,
     17     /* downOffset:        */ 0,
     18     /* damageOrCoinValue: */ 3,
     19     /* health:            */ 1,
     20     /* numLootCoins:      */ 0,
     21     /* radius:            */ 80,
     22     /* height:            */ 160,
     23     /* hurtboxRadius:     */ 80,
     24     /* hurtboxHeight:     */ 160,
     25 };
     26 
     27 /**
     28  * Update function for chain chomp part / pivot.
     29  */
     30 void bhv_chain_chomp_chain_part_update(void) {
     31     if (o->parentObj->oAction == CHAIN_CHOMP_ACT_UNLOAD_CHAIN) {
     32         obj_mark_for_deletion(o);
     33     } else if (o->oBhvParams2ndByte != CHAIN_CHOMP_CHAIN_PART_BP_PIVOT) {
     34         struct ChainSegment *segment = &o->parentObj->oChainChompSegments[o->oBhvParams2ndByte];
     35 
     36         // Set position relative to the pivot
     37         o->oPosX = o->parentObj->parentObj->oPosX + segment->posX;
     38         o->oPosY = o->parentObj->parentObj->oPosY + segment->posY;
     39         o->oPosZ = o->parentObj->parentObj->oPosZ + segment->posZ;
     40     } else if (o->parentObj->oChainChompReleaseStatus != CHAIN_CHOMP_NOT_RELEASED) {
     41         cur_obj_update_floor_and_walls();
     42         cur_obj_move_standard(78);
     43     }
     44 }
     45 
     46 /**
     47  * When mario gets close enough, allocate chain segments and spawn their objects.
     48  */
     49 static void chain_chomp_act_uninitialized(void) {
     50     struct ChainSegment *segments;
     51     s32 i;
     52 
     53     if (o->oDistanceToMario < 3000.0f) {
     54         segments = mem_pool_alloc(gObjectMemoryPool, 5 * sizeof(struct ChainSegment));
     55         if (segments != NULL) {
     56             // Each segment represents the offset of a chain part to the pivot.
     57             // Segment 0 connects the pivot to the chain chomp itself. Segment
     58             // 1 connects the pivot to the chain part next to the chain chomp
     59             // (chain part 1), etc.
     60             o->oChainChompSegments = segments;
     61             for (i = 0; i <= 4; i++) {
     62                 chain_segment_init(&segments[i]);
     63             }
     64 
     65             cur_obj_set_pos_to_home();
     66 
     67             // Spawn the pivot and set to parent
     68             if ((o->parentObj =
     69                      spawn_object(o, CHAIN_CHOMP_CHAIN_PART_BP_PIVOT, bhvChainChompChainPart))
     70                 != NULL) {
     71                 // Spawn the non-pivot chain parts, starting from the chain
     72                 // chomp and moving toward the pivot
     73                 for (i = 1; i <= 4; i++) {
     74                     spawn_object_relative(i, 0, 0, 0, o, MODEL_METALLIC_BALL, bhvChainChompChainPart);
     75                 }
     76 
     77                 o->oAction = CHAIN_CHOMP_ACT_MOVE;
     78                 cur_obj_unhide();
     79             }
     80         }
     81     }
     82 }
     83 
     84 /**
     85  * Apply gravity to each chain part, and cap its distance to the previous
     86  * part as well as from the pivot.
     87  */
     88 static void chain_chomp_update_chain_segments(void) {
     89     struct ChainSegment *prevSegment;
     90     struct ChainSegment *segment;
     91     f32 offsetX;
     92     f32 offsetY;
     93     f32 offsetZ;
     94     f32 offset;
     95     f32 segmentVelY;
     96     f32 maxTotalOffset;
     97     s32 i;
     98 
     99     if (o->oVelY < 0.0f) {
    100         segmentVelY = o->oVelY;
    101     } else {
    102         segmentVelY = -20.0f;
    103     }
    104 
    105     // Segment 0 connects the pivot to the chain chomp itself, and segment i>0
    106     // connects the pivot to chain part i (1 is closest to the chain chomp).
    107 
    108     for (i = 1; i <= 4; i++) {
    109         prevSegment = &o->oChainChompSegments[i - 1];
    110         segment = &o->oChainChompSegments[i];
    111 
    112         // Apply gravity
    113 
    114         if ((segment->posY += segmentVelY) < 0.0f) {
    115             segment->posY = 0.0f;
    116         }
    117 
    118         // Cap distance to previous chain part (so that the tail follows the
    119         // chomp)
    120 
    121         offsetX = segment->posX - prevSegment->posX;
    122         offsetY = segment->posY - prevSegment->posY;
    123         offsetZ = segment->posZ - prevSegment->posZ;
    124         offset = sqrtf(offsetX * offsetX + offsetY * offsetY + offsetZ * offsetZ);
    125 
    126         if (offset > o->oChainChompMaxDistBetweenChainParts) {
    127             offset = o->oChainChompMaxDistBetweenChainParts / offset;
    128             offsetX *= offset;
    129             offsetY *= offset;
    130             offsetZ *= offset;
    131         }
    132 
    133         // Cap distance to pivot (so that it stretches when the chomp moves far
    134         // from the wooden post)
    135 
    136         offsetX += prevSegment->posX;
    137         offsetY += prevSegment->posY;
    138         offsetZ += prevSegment->posZ;
    139         offset = sqrtf(offsetX * offsetX + offsetY * offsetY + offsetZ * offsetZ);
    140 
    141         maxTotalOffset = o->oChainChompMaxDistFromPivotPerChainPart * (5 - i);
    142         if (offset > maxTotalOffset) {
    143             offset = maxTotalOffset / offset;
    144             offsetX *= offset;
    145             offsetY *= offset;
    146             offsetZ *= offset;
    147         }
    148 
    149         segment->posX = offsetX;
    150         segment->posY = offsetY;
    151         segment->posZ = offsetZ;
    152     }
    153 }
    154 
    155 /**
    156  * Lunging increases the maximum distance from the pivot and changes the maximum
    157  * distance between chain parts. Restore these values to normal.
    158  */
    159 static void chain_chomp_restore_normal_chain_lengths(void) {
    160     approach_f32_ptr(&o->oChainChompMaxDistFromPivotPerChainPart, 750.0f / 5, 4.0f);
    161     o->oChainChompMaxDistBetweenChainParts = o->oChainChompMaxDistFromPivotPerChainPart;
    162 }
    163 
    164 /**
    165  * Turn toward mario. Wait a bit and then enter the lunging sub-action.
    166  */
    167 static void chain_chomp_sub_act_turn(void) {
    168     o->oGravity = -4.0f;
    169 
    170     chain_chomp_restore_normal_chain_lengths();
    171     obj_move_pitch_approach(0, 0x100);
    172 
    173     if (o->oMoveFlags & OBJ_MOVE_MASK_ON_GROUND) {
    174         cur_obj_rotate_yaw_toward(o->oAngleToMario, 0x400);
    175         if (abs_angle_diff(o->oAngleToMario, o->oMoveAngleYaw) < 0x800) {
    176             if (o->oTimer > 30) {
    177                 if (cur_obj_check_anim_frame(0)) {
    178                     cur_obj_reverse_animation();
    179                     if (o->oTimer > 40) {
    180                         // Increase the maximum distance from the pivot and enter
    181                         // the lunging sub-action.
    182                         cur_obj_play_sound_2(SOUND_GENERAL_CHAIN_CHOMP2);
    183 
    184                         o->oSubAction = CHAIN_CHOMP_SUB_ACT_LUNGE;
    185                         o->oChainChompMaxDistFromPivotPerChainPart = 900.0f / 5;
    186 
    187                         o->oForwardVel = 140.0f;
    188                         o->oVelY = 20.0f;
    189                         o->oGravity = 0.0f;
    190                         o->oChainChompTargetPitch = obj_get_pitch_from_vel();
    191                     }
    192                 } else {
    193                     o->oTimer--;
    194                 }
    195             } else {
    196                 o->oForwardVel = 0.0f;
    197             }
    198         } else {
    199             cur_obj_play_sound_2(SOUND_GENERAL_CHAIN_CHOMP1);
    200             o->oForwardVel = 10.0f;
    201             o->oVelY = 20.0f;
    202         }
    203     } else {
    204         cur_obj_rotate_yaw_toward(o->oAngleToMario, 400);
    205         o->oTimer = 0;
    206     }
    207 }
    208 
    209 static void chain_chomp_sub_act_lunge(void) {
    210     obj_face_pitch_approach(o->oChainChompTargetPitch, 0x400);
    211 
    212     if (o->oForwardVel != 0.0f) {
    213         f32 val04;
    214 
    215         if (o->oChainChompRestrictedByChain == TRUE) {
    216             o->oForwardVel = o->oVelY = 0.0f;
    217             o->oChainChompUnk104 = 30.0f;
    218         }
    219 
    220         // TODO: What is this
    221         if ((val04 = 900.0f - o->oChainChompDistToPivot) > 220.0f) {
    222             val04 = 220.0f;
    223         }
    224 
    225         o->oChainChompMaxDistBetweenChainParts =
    226             val04 / 220.0f * o->oChainChompMaxDistFromPivotPerChainPart;
    227         o->oTimer = 0;
    228     } else {
    229         // Turn toward pivot
    230         cur_obj_rotate_yaw_toward(atan2s(o->oChainChompSegments[0].posZ, o->oChainChompSegments[0].posX),
    231                                   0x1000);
    232 
    233         if (o->oChainChompUnk104 != 0.0f) {
    234             approach_f32_ptr(&o->oChainChompUnk104, 0.0f, 0.8f);
    235         } else {
    236             o->oSubAction = CHAIN_CHOMP_SUB_ACT_TURN;
    237         }
    238 
    239         o->oChainChompMaxDistBetweenChainParts = o->oChainChompUnk104;
    240         if (gGlobalTimer % 2 != 0) {
    241             o->oChainChompMaxDistBetweenChainParts = -o->oChainChompUnk104;
    242         }
    243     }
    244 
    245     if (o->oTimer < 30) {
    246         cur_obj_reverse_animation();
    247     }
    248 }
    249 
    250 /**
    251  * Fall to the ground and interrupt mario into a cutscene action.
    252  */
    253 static void chain_chomp_released_trigger_cutscene(void) {
    254     o->oForwardVel = 0.0f;
    255     o->oGravity = -4.0f;
    256 
    257     //! Can delay this if we get into a cutscene-unfriendly action after the
    258     //  last post ground pound and before this
    259     if (set_mario_npc_dialog(MARIO_DIALOG_LOOK_UP) == MARIO_DIALOG_STATUS_SPEAK
    260         && (o->oMoveFlags & OBJ_MOVE_MASK_ON_GROUND) && cutscene_object(CUTSCENE_STAR_SPAWN, o) == 1) {
    261         o->oChainChompReleaseStatus = CHAIN_CHOMP_RELEASED_LUNGE_AROUND;
    262         o->oTimer = 0;
    263     }
    264 }
    265 
    266 /**
    267  * Lunge 4 times, each time moving toward mario +/- 0x2000 angular units.
    268  * Finally, begin a lunge toward x=1450, z=562 (near the gate).
    269  */
    270 static void chain_chomp_released_lunge_around(void) {
    271     chain_chomp_restore_normal_chain_lengths();
    272 
    273     // Finish bounce
    274     if (o->oMoveFlags & OBJ_MOVE_MASK_ON_GROUND) {
    275         // Before first bounce, turn toward mario and wait 2 seconds
    276         if (o->oChainChompNumLunges == 0) {
    277             if (cur_obj_rotate_yaw_toward(o->oAngleToMario, 800)) {
    278                 if (o->oTimer > 60) {
    279                     o->oChainChompNumLunges++;
    280                     // enable wall collision
    281                     o->oWallHitboxRadius = 200.0f;
    282                 }
    283             } else {
    284                 o->oTimer = 0;
    285             }
    286         } else {
    287             if (++o->oChainChompNumLunges <= 5) {
    288                 cur_obj_play_sound_2(SOUND_GENERAL_CHAIN_CHOMP1);
    289                 o->oMoveAngleYaw = o->oAngleToMario + random_sign() * 0x2000;
    290                 o->oForwardVel = 30.0f;
    291                 o->oVelY = 50.0f;
    292             } else {
    293                 o->oChainChompReleaseStatus = CHAIN_CHOMP_RELEASED_BREAK_GATE;
    294                 o->oHomeX = 1450.0f;
    295                 o->oHomeZ = 562.0f;
    296                 o->oMoveAngleYaw = cur_obj_angle_to_home();
    297                 o->oForwardVel = cur_obj_lateral_dist_to_home() / 8;
    298                 o->oVelY = 50.0f;
    299             }
    300         }
    301     }
    302 }
    303 
    304 /**
    305  * Continue lunging until a wall collision occurs. Mark the gate as destroyed,
    306  * wait for the chain chomp to land, and then begin a jump toward the final
    307  * target, x=3288, z=-1770.
    308  */
    309 static void chain_chomp_released_break_gate(void) {
    310     if (!o->oChainChompHitGate) {
    311         // If hit wall, assume it's the gate and bounce off of it
    312         //! The wall may not be the gate
    313         //! If the chain chomp gets stuck, it may never hit a wall, resulting
    314         //  in a softlock
    315         if (o->oMoveFlags & OBJ_MOVE_HIT_WALL) {
    316             o->oChainChompHitGate = TRUE;
    317             o->oMoveAngleYaw = cur_obj_reflect_move_angle_off_wall();
    318             o->oForwardVel *= 0.4f;
    319         }
    320     } else if (o->oMoveFlags & OBJ_MOVE_MASK_ON_GROUND) {
    321         o->oChainChompReleaseStatus = CHAIN_CHOMP_RELEASED_JUMP_AWAY;
    322         o->oHomeX = 3288.0f;
    323         o->oHomeZ = -1770.0f;
    324         o->oMoveAngleYaw = cur_obj_angle_to_home();
    325         o->oForwardVel = cur_obj_lateral_dist_to_home() / 50.0f;
    326         o->oVelY = 120.0f;
    327     }
    328 }
    329 
    330 /**
    331  * Wait until the chain chomp lands.
    332  */
    333 static void chain_chomp_released_jump_away(void) {
    334     if (o->oMoveFlags & OBJ_MOVE_MASK_ON_GROUND) {
    335         gObjCutsceneDone = TRUE;
    336         o->oChainChompReleaseStatus = CHAIN_CHOMP_RELEASED_END_CUTSCENE;
    337     }
    338 }
    339 
    340 /**
    341  * Release mario and transition to the unload chain action.
    342  */
    343 static void chain_chomp_released_end_cutscene(void) {
    344     if (cutscene_object(CUTSCENE_STAR_SPAWN, o) == -1) {
    345         set_mario_npc_dialog(MARIO_DIALOG_STOP);
    346         o->oAction = CHAIN_CHOMP_ACT_UNLOAD_CHAIN;
    347     }
    348 }
    349 
    350 /**
    351  * All chain chomp movement behavior, including the cutscene after being
    352  * released.
    353  */
    354 static void chain_chomp_act_move(void) {
    355     f32 maxDistToPivot;
    356 
    357     // Unload chain if mario is far enough
    358     if (o->oChainChompReleaseStatus == CHAIN_CHOMP_NOT_RELEASED && o->oDistanceToMario > 4000.0f) {
    359         o->oAction = CHAIN_CHOMP_ACT_UNLOAD_CHAIN;
    360         o->oForwardVel = o->oVelY = 0.0f;
    361     } else {
    362         cur_obj_update_floor_and_walls();
    363 
    364         switch (o->oChainChompReleaseStatus) {
    365             case CHAIN_CHOMP_NOT_RELEASED:
    366                 switch (o->oSubAction) {
    367                     case CHAIN_CHOMP_SUB_ACT_TURN:
    368                         chain_chomp_sub_act_turn();
    369                         break;
    370                     case CHAIN_CHOMP_SUB_ACT_LUNGE:
    371                         chain_chomp_sub_act_lunge();
    372                         break;
    373                 }
    374                 break;
    375             case CHAIN_CHOMP_RELEASED_TRIGGER_CUTSCENE:
    376                 chain_chomp_released_trigger_cutscene();
    377                 break;
    378             case CHAIN_CHOMP_RELEASED_LUNGE_AROUND:
    379                 chain_chomp_released_lunge_around();
    380                 break;
    381             case CHAIN_CHOMP_RELEASED_BREAK_GATE:
    382                 chain_chomp_released_break_gate();
    383                 break;
    384             case CHAIN_CHOMP_RELEASED_JUMP_AWAY:
    385                 chain_chomp_released_jump_away();
    386                 break;
    387             case CHAIN_CHOMP_RELEASED_END_CUTSCENE:
    388                 chain_chomp_released_end_cutscene();
    389                 break;
    390         }
    391 
    392         cur_obj_move_standard(78);
    393 
    394         // Segment 0 connects the pivot to the chain chomp itself
    395         o->oChainChompSegments[0].posX = o->oPosX - o->parentObj->oPosX;
    396         o->oChainChompSegments[0].posY = o->oPosY - o->parentObj->oPosY;
    397         o->oChainChompSegments[0].posZ = o->oPosZ - o->parentObj->oPosZ;
    398 
    399         o->oChainChompDistToPivot =
    400             sqrtf(o->oChainChompSegments[0].posX * o->oChainChompSegments[0].posX
    401                   + o->oChainChompSegments[0].posY * o->oChainChompSegments[0].posY
    402                   + o->oChainChompSegments[0].posZ * o->oChainChompSegments[0].posZ);
    403 
    404         // If the chain is fully stretched
    405         maxDistToPivot = o->oChainChompMaxDistFromPivotPerChainPart * 5;
    406         if (o->oChainChompDistToPivot > maxDistToPivot) {
    407             f32 ratio = maxDistToPivot / o->oChainChompDistToPivot;
    408             o->oChainChompDistToPivot = maxDistToPivot;
    409 
    410             o->oChainChompSegments[0].posX *= ratio;
    411             o->oChainChompSegments[0].posY *= ratio;
    412             o->oChainChompSegments[0].posZ *= ratio;
    413 
    414             if (o->oChainChompReleaseStatus == CHAIN_CHOMP_NOT_RELEASED) {
    415                 // Restrict chain chomp position
    416                 o->oPosX = o->parentObj->oPosX + o->oChainChompSegments[0].posX;
    417                 o->oPosY = o->parentObj->oPosY + o->oChainChompSegments[0].posY;
    418                 o->oPosZ = o->parentObj->oPosZ + o->oChainChompSegments[0].posZ;
    419 
    420                 o->oChainChompRestrictedByChain = TRUE;
    421             } else {
    422                 // Move pivot like the chain chomp is pulling it along
    423                 f32 oldPivotY = o->parentObj->oPosY;
    424 
    425                 o->parentObj->oPosX = o->oPosX - o->oChainChompSegments[0].posX;
    426                 o->parentObj->oPosY = o->oPosY - o->oChainChompSegments[0].posY;
    427                 o->parentObj->oVelY = o->parentObj->oPosY - oldPivotY;
    428                 o->parentObj->oPosZ = o->oPosZ - o->oChainChompSegments[0].posZ;
    429             }
    430         } else {
    431             o->oChainChompRestrictedByChain = FALSE;
    432         }
    433 
    434         chain_chomp_update_chain_segments();
    435 
    436         // Begin a lunge if mario tries to attack
    437         if (obj_check_attacks(&sChainChompHitbox, o->oAction) != 0) {
    438             o->oSubAction = CHAIN_CHOMP_SUB_ACT_LUNGE;
    439             o->oChainChompMaxDistFromPivotPerChainPart = 900.0f / 5;
    440             o->oForwardVel = 0.0f;
    441             o->oVelY = 300.0f;
    442             o->oGravity = -4.0f;
    443             o->oChainChompTargetPitch = -0x3000;
    444         }
    445     }
    446 }
    447 
    448 /**
    449  * Hide and free the chain chomp segments. The chain objects will unload
    450  * themselves when they see that the chain chomp is in this action.
    451  */
    452 static void chain_chomp_act_unload_chain(void) {
    453     cur_obj_hide();
    454     mem_pool_free(gObjectMemoryPool, o->oChainChompSegments);
    455 
    456     o->oAction = CHAIN_CHOMP_ACT_UNINITIALIZED;
    457 
    458     if (o->oChainChompReleaseStatus != CHAIN_CHOMP_NOT_RELEASED) {
    459         obj_mark_for_deletion(o);
    460     }
    461 }
    462 
    463 /**
    464  * Update function for chain chomp.
    465  */
    466 void bhv_chain_chomp_update(void) {
    467     switch (o->oAction) {
    468         case CHAIN_CHOMP_ACT_UNINITIALIZED:
    469             chain_chomp_act_uninitialized();
    470             break;
    471         case CHAIN_CHOMP_ACT_MOVE:
    472             chain_chomp_act_move();
    473             break;
    474         case CHAIN_CHOMP_ACT_UNLOAD_CHAIN:
    475             chain_chomp_act_unload_chain();
    476             break;
    477     }
    478 }
    479 
    480 /**
    481  * Update function for wooden post.
    482  */
    483 void bhv_wooden_post_update(void) {
    484     // When ground pounded by mario, drop by -45 + -20
    485     if (!o->oWoodenPostMarioPounding) {
    486         if ((o->oWoodenPostMarioPounding = cur_obj_is_mario_ground_pounding_platform())) {
    487             cur_obj_play_sound_2(SOUND_GENERAL_POUND_WOOD_POST);
    488             o->oWoodenPostSpeedY = -70.0f;
    489         }
    490     } else if (approach_f32_ptr(&o->oWoodenPostSpeedY, 0.0f, 25.0f)) {
    491         // Stay still until mario is done ground pounding
    492         o->oWoodenPostMarioPounding = cur_obj_is_mario_ground_pounding_platform();
    493     } else if ((o->oWoodenPostOffsetY += o->oWoodenPostSpeedY) < -190.0f) {
    494         // Once pounded, if this is the chain chomp's post, release the chain
    495         // chomp
    496         o->oWoodenPostOffsetY = -190.0f;
    497         if (o->parentObj != o) {
    498             play_puzzle_jingle();
    499             o->parentObj->oChainChompReleaseStatus = CHAIN_CHOMP_RELEASED_TRIGGER_CUTSCENE;
    500             o->parentObj = o;
    501         }
    502     }
    503 
    504     if (o->oWoodenPostOffsetY != 0.0f) {
    505         o->oPosY = o->oHomeY + o->oWoodenPostOffsetY;
    506     } else if (!(o->oBhvParams & WOODEN_POST_BP_NO_COINS_MASK)) {
    507         // Reset the timer once mario is far enough
    508         if (o->oDistanceToMario > 400.0f) {
    509             o->oTimer = o->oWoodenPostTotalMarioAngle = 0;
    510         } else {
    511             // When mario runs around the post 3 times within 200 frames, spawn
    512             // coins
    513             o->oWoodenPostTotalMarioAngle += (s16)(o->oAngleToMario - o->oWoodenPostPrevAngleToMario);
    514             if (absi(o->oWoodenPostTotalMarioAngle) > 0x30000 && o->oTimer < 200) {
    515                 obj_spawn_loot_yellow_coins(o, 5, 20.0f);
    516                 set_object_respawn_info_bits(o, 1);
    517             }
    518         }
    519 
    520         o->oWoodenPostPrevAngleToMario = o->oAngleToMario;
    521     }
    522 }
    523 
    524 /**
    525  * Init function for chain chomp gate.
    526  */
    527 void bhv_chain_chomp_gate_init(void) {
    528     o->parentObj = cur_obj_nearest_object_with_behavior(bhvChainChomp);
    529 }
    530 
    531 /**
    532  * Update function for chain chomp gate
    533  */
    534 void bhv_chain_chomp_gate_update(void) {
    535     if (o->parentObj->oChainChompHitGate) {
    536         spawn_mist_particles_with_sound(SOUND_GENERAL_WALL_EXPLOSION);
    537         set_camera_shake_from_point(SHAKE_POS_SMALL, o->oPosX, o->oPosY, o->oPosZ);
    538         spawn_mist_particles_variable(0, 127, 200.0f);
    539         spawn_triangle_break_particles(30, MODEL_DIRT_ANIMATION, 3.0f, 4);
    540         obj_mark_for_deletion(o);
    541     }
    542 }