sm64

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

mario_misc.c (25419B)


      1 #include <PR/ultratypes.h>
      2 
      3 #include "sm64.h"
      4 #include "area.h"
      5 #include "audio/external.h"
      6 #include "behavior_actions.h"
      7 #include "behavior_data.h"
      8 #include "camera.h"
      9 #include "dialog_ids.h"
     10 #include "engine/behavior_script.h"
     11 #include "engine/graph_node.h"
     12 #include "engine/math_util.h"
     13 #include "envfx_snow.h"
     14 #include "game_init.h"
     15 #include "goddard/renderer.h"
     16 #include "interaction.h"
     17 #include "level_update.h"
     18 #include "mario_actions_cutscene.h"
     19 #include "mario_misc.h"
     20 #include "memory.h"
     21 #include "object_helpers.h"
     22 #include "object_list_processor.h"
     23 #include "rendering_graph_node.h"
     24 #include "save_file.h"
     25 #include "skybox.h"
     26 #include "sound_init.h"
     27 
     28 #define TOAD_STAR_1_REQUIREMENT 12
     29 #define TOAD_STAR_2_REQUIREMENT 25
     30 #define TOAD_STAR_3_REQUIREMENT 35
     31 
     32 #define TOAD_STAR_1_DIALOG DIALOG_082
     33 #define TOAD_STAR_2_DIALOG DIALOG_076
     34 #define TOAD_STAR_3_DIALOG DIALOG_083
     35 
     36 #define TOAD_STAR_1_DIALOG_AFTER DIALOG_154
     37 #define TOAD_STAR_2_DIALOG_AFTER DIALOG_155
     38 #define TOAD_STAR_3_DIALOG_AFTER DIALOG_156
     39 
     40 enum ToadMessageStates {
     41     TOAD_MESSAGE_FADED,
     42     TOAD_MESSAGE_OPAQUE,
     43     TOAD_MESSAGE_OPACIFYING,
     44     TOAD_MESSAGE_FADING,
     45     TOAD_MESSAGE_TALKING
     46 };
     47 
     48 enum UnlockDoorStarStates {
     49     UNLOCK_DOOR_STAR_RISING,
     50     UNLOCK_DOOR_STAR_WAITING,
     51     UNLOCK_DOOR_STAR_SPAWNING_PARTICLES,
     52     UNLOCK_DOOR_STAR_DONE
     53 };
     54 
     55 /**
     56  * The eye texture on succesive frames of Mario's blink animation.
     57  * He intentionally blinks twice each time.
     58  */
     59 static s8 gMarioBlinkAnimation[7] = { 1, 2, 1, 0, 1, 2, 1 };
     60 
     61 /**
     62  * The scale values per frame for Mario's foot/hand for his attack animation
     63  * There are 3 scale animations in groups of 6 frames.
     64  * The first animation starts at frame index 3 and goes down, the others start at frame index 5.
     65  * The values get divided by 10 before assigning, so e.g. 12 gives a scale factor 1.2.
     66  * All combined, this means e.g. the first animation scales Mario's fist by {2.4, 1.6, 1.2, 1.0} on
     67  * successive frames.
     68  */
     69 static s8 gMarioAttackScaleAnimation[3 * 6] = {
     70     10, 12, 16, 24, 10, 10, 10, 14, 20, 30, 10, 10, 10, 16, 20, 26, 26, 20,
     71 };
     72 
     73 struct MarioBodyState gBodyStates[2]; // 2nd is never accessed in practice, most likely Luigi related
     74 struct GraphNodeObject gMirrorMario;  // copy of Mario's geo node for drawing mirror Mario
     75 
     76 // This whole file is weirdly organized. It has to be the same file due
     77 // to rodata boundaries and function aligns, which means the programmer
     78 // treated this like a "misc" file for vaguely Mario related things
     79 // (message NPC related things, the Mario head geo, and Mario geo
     80 // functions)
     81 
     82 /**
     83  * Geo node script that draws Mario's head on the title screen.
     84  */
     85 Gfx *geo_draw_mario_head_goddard(s32 callContext, struct GraphNode *node, Mat4 *c) {
     86     Gfx *gfx = NULL;
     87     s16 sfx = 0;
     88     struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node;
     89     UNUSED Mat4 *transform = c;
     90 
     91     if (callContext == GEO_CONTEXT_RENDER) {
     92         if (gPlayer1Controller->controllerData != NULL && !gWarpTransition.isActive) {
     93             gd_copy_p1_contpad(gPlayer1Controller->controllerData);
     94         }
     95         gfx = (Gfx *) PHYSICAL_TO_VIRTUAL(gdm_gettestdl(asGenerated->parameter));
     96         gGoddardVblankCallback = gd_vblank;
     97         sfx = gd_sfx_to_play();
     98         play_menu_sounds(sfx);
     99     }
    100     return gfx;
    101 }
    102 
    103 static void toad_message_faded(void) {
    104     if (gCurrentObject->oDistanceToMario > 700.0f) {
    105         gCurrentObject->oToadMessageRecentlyTalked = FALSE;
    106     }
    107     if (!gCurrentObject->oToadMessageRecentlyTalked && gCurrentObject->oDistanceToMario < 600.0f) {
    108         gCurrentObject->oToadMessageState = TOAD_MESSAGE_OPACIFYING;
    109     }
    110 }
    111 
    112 static void toad_message_opaque(void) {
    113     if (gCurrentObject->oDistanceToMario > 700.0f) {
    114         gCurrentObject->oToadMessageState = TOAD_MESSAGE_FADING;
    115     } else if (!gCurrentObject->oToadMessageRecentlyTalked) {
    116         gCurrentObject->oInteractionSubtype = INT_SUBTYPE_NPC;
    117         if (gCurrentObject->oInteractStatus & INT_STATUS_INTERACTED) {
    118             gCurrentObject->oInteractStatus = 0;
    119             gCurrentObject->oToadMessageState = TOAD_MESSAGE_TALKING;
    120             play_toads_jingle();
    121         }
    122     }
    123 }
    124 
    125 static void toad_message_talking(void) {
    126     if (cur_obj_update_dialog_with_cutscene(MARIO_DIALOG_LOOK_DOWN,
    127         DIALOG_FLAG_TURN_TO_MARIO, CUTSCENE_DIALOG, gCurrentObject->oToadMessageDialogID)) {
    128         gCurrentObject->oToadMessageRecentlyTalked = TRUE;
    129         gCurrentObject->oToadMessageState = TOAD_MESSAGE_FADING;
    130         switch (gCurrentObject->oToadMessageDialogID) {
    131             case TOAD_STAR_1_DIALOG:
    132                 gCurrentObject->oToadMessageDialogID = TOAD_STAR_1_DIALOG_AFTER;
    133                 bhv_spawn_star_no_level_exit(STAR_INDEX_ACT_1);
    134                 break;
    135             case TOAD_STAR_2_DIALOG:
    136                 gCurrentObject->oToadMessageDialogID = TOAD_STAR_2_DIALOG_AFTER;
    137                 bhv_spawn_star_no_level_exit(STAR_INDEX_ACT_2);
    138                 break;
    139             case TOAD_STAR_3_DIALOG:
    140                 gCurrentObject->oToadMessageDialogID = TOAD_STAR_3_DIALOG_AFTER;
    141                 bhv_spawn_star_no_level_exit(STAR_INDEX_ACT_3);
    142                 break;
    143         }
    144     }
    145 }
    146 
    147 static void toad_message_opacifying(void) {
    148     if ((gCurrentObject->oOpacity += 6) == 255) {
    149         gCurrentObject->oToadMessageState = TOAD_MESSAGE_OPAQUE;
    150     }
    151 }
    152 
    153 static void toad_message_fading(void) {
    154     if ((gCurrentObject->oOpacity -= 6) == 81) {
    155         gCurrentObject->oToadMessageState = TOAD_MESSAGE_FADED;
    156     }
    157 }
    158 
    159 void bhv_toad_message_loop(void) {
    160     if (gCurrentObject->header.gfx.node.flags & GRAPH_RENDER_ACTIVE) {
    161         gCurrentObject->oInteractionSubtype = 0;
    162         switch (gCurrentObject->oToadMessageState) {
    163             case TOAD_MESSAGE_FADED:
    164                 toad_message_faded();
    165                 break;
    166             case TOAD_MESSAGE_OPAQUE:
    167                 toad_message_opaque();
    168                 break;
    169             case TOAD_MESSAGE_OPACIFYING:
    170                 toad_message_opacifying();
    171                 break;
    172             case TOAD_MESSAGE_FADING:
    173                 toad_message_fading();
    174                 break;
    175             case TOAD_MESSAGE_TALKING:
    176                 toad_message_talking();
    177                 break;
    178         }
    179     }
    180 }
    181 
    182 void bhv_toad_message_init(void) {
    183     s32 saveFlags = save_file_get_flags();
    184     s32 starCount = save_file_get_total_star_count(gCurrSaveFileNum - 1, COURSE_MIN - 1, COURSE_MAX - 1);
    185     s32 dialogID = (gCurrentObject->oBhvParams >> 24) & 0xFF;
    186     s32 enoughStars = TRUE;
    187 
    188     switch (dialogID) {
    189         case TOAD_STAR_1_DIALOG:
    190             enoughStars = (starCount >= TOAD_STAR_1_REQUIREMENT);
    191             if (saveFlags & SAVE_FLAG_COLLECTED_TOAD_STAR_1) {
    192                 dialogID = TOAD_STAR_1_DIALOG_AFTER;
    193             }
    194             break;
    195         case TOAD_STAR_2_DIALOG:
    196             enoughStars = (starCount >= TOAD_STAR_2_REQUIREMENT);
    197             if (saveFlags & SAVE_FLAG_COLLECTED_TOAD_STAR_2) {
    198                 dialogID = TOAD_STAR_2_DIALOG_AFTER;
    199             }
    200             break;
    201         case TOAD_STAR_3_DIALOG:
    202             enoughStars = (starCount >= TOAD_STAR_3_REQUIREMENT);
    203             if (saveFlags & SAVE_FLAG_COLLECTED_TOAD_STAR_3) {
    204                 dialogID = TOAD_STAR_3_DIALOG_AFTER;
    205             }
    206             break;
    207     }
    208 
    209     if (enoughStars) {
    210         gCurrentObject->oToadMessageDialogID = dialogID;
    211         gCurrentObject->oToadMessageRecentlyTalked = FALSE;
    212         gCurrentObject->oToadMessageState = TOAD_MESSAGE_FADED;
    213         gCurrentObject->oOpacity = 81;
    214     } else {
    215         obj_mark_for_deletion(gCurrentObject);
    216     }
    217 }
    218 
    219 static void star_door_unlock_spawn_particles(s16 angleOffset) {
    220     struct Object *sparkleParticle = spawn_object(gCurrentObject, 0, bhvSparkleSpawn);
    221 
    222     sparkleParticle->oPosX +=
    223         100.0f * sins((gCurrentObject->oUnlockDoorStarTimer * 0x2800) + angleOffset);
    224     sparkleParticle->oPosZ +=
    225         100.0f * coss((gCurrentObject->oUnlockDoorStarTimer * 0x2800) + angleOffset);
    226     // Particles are spawned lower each frame
    227     sparkleParticle->oPosY -= gCurrentObject->oUnlockDoorStarTimer * 10.0f;
    228 }
    229 
    230 void bhv_unlock_door_star_init(void) {
    231     gCurrentObject->oUnlockDoorStarState = UNLOCK_DOOR_STAR_RISING;
    232     gCurrentObject->oUnlockDoorStarTimer = 0;
    233     gCurrentObject->oUnlockDoorStarYawVel = 0x1000;
    234     gCurrentObject->oPosX += 30.0f * sins(gMarioState->faceAngle[1] - 0x4000);
    235     gCurrentObject->oPosY += 160.0f;
    236     gCurrentObject->oPosZ += 30.0f * coss(gMarioState->faceAngle[1] - 0x4000);
    237     gCurrentObject->oMoveAngleYaw = 0x7800;
    238     obj_scale(gCurrentObject, 0.5f);
    239 }
    240 
    241 void bhv_unlock_door_star_loop(void) {
    242     UNUSED u8 filler1[4];
    243     s16 prevYaw = gCurrentObject->oMoveAngleYaw;
    244     UNUSED u8 filler2[4];
    245 
    246     // Speed up the star every frame
    247     if (gCurrentObject->oUnlockDoorStarYawVel < 0x2400) {
    248         gCurrentObject->oUnlockDoorStarYawVel += 0x60;
    249     }
    250     switch (gCurrentObject->oUnlockDoorStarState) {
    251         case UNLOCK_DOOR_STAR_RISING:
    252             gCurrentObject->oPosY += 3.4f; // Raise the star up in the air
    253             gCurrentObject->oMoveAngleYaw +=
    254                 gCurrentObject->oUnlockDoorStarYawVel; // Apply yaw velocity
    255             obj_scale(gCurrentObject, gCurrentObject->oUnlockDoorStarTimer / 50.0f
    256                                              + 0.5f); // Scale the star to be bigger
    257             if (++gCurrentObject->oUnlockDoorStarTimer == 30) {
    258                 gCurrentObject->oUnlockDoorStarTimer = 0;
    259                 gCurrentObject->oUnlockDoorStarState++; // Sets state to UNLOCK_DOOR_STAR_WAITING
    260             }
    261             break;
    262         case UNLOCK_DOOR_STAR_WAITING:
    263             gCurrentObject->oMoveAngleYaw +=
    264                 gCurrentObject->oUnlockDoorStarYawVel; // Apply yaw velocity
    265             if (++gCurrentObject->oUnlockDoorStarTimer == 30) {
    266                 play_sound(SOUND_MENU_STAR_SOUND,
    267                            gCurrentObject->header.gfx.cameraToObject); // Play final sound
    268                 cur_obj_hide();                                            // Hide the object
    269                 gCurrentObject->oUnlockDoorStarTimer = 0;
    270                 gCurrentObject
    271                     ->oUnlockDoorStarState++; // Sets state to UNLOCK_DOOR_STAR_SPAWNING_PARTICLES
    272             }
    273             break;
    274         case UNLOCK_DOOR_STAR_SPAWNING_PARTICLES:
    275             // Spawn two particles, opposite sides of the star.
    276             star_door_unlock_spawn_particles(0);
    277             star_door_unlock_spawn_particles(0x8000);
    278             if (gCurrentObject->oUnlockDoorStarTimer++ == 20) {
    279                 gCurrentObject->oUnlockDoorStarTimer = 0;
    280                 gCurrentObject->oUnlockDoorStarState++; // Sets state to UNLOCK_DOOR_STAR_DONE
    281             }
    282             break;
    283         case UNLOCK_DOOR_STAR_DONE: // The object stays loaded for an additional 50 frames so that the
    284                                     // sound doesn't immediately stop.
    285             if (gCurrentObject->oUnlockDoorStarTimer++ == 50) {
    286                 obj_mark_for_deletion(gCurrentObject);
    287             }
    288             break;
    289     }
    290     // Checks if the angle has cycled back to 0.
    291     // This means that the code will execute when the star completes a full revolution.
    292     if (prevYaw > (s16) gCurrentObject->oMoveAngleYaw) {
    293         play_sound(
    294             SOUND_GENERAL_SHORT_STAR,
    295             gCurrentObject->header.gfx.cameraToObject); // Play a sound every time the star spins once
    296     }
    297 }
    298 
    299 /**
    300  * Generate a display list that sets the correct blend mode and color for mirror Mario.
    301  */
    302 static Gfx *make_gfx_mario_alpha(struct GraphNodeGenerated *node, s16 alpha) {
    303     Gfx *gfx;
    304     Gfx *gfxHead = NULL;
    305 
    306     if (alpha == 255) {
    307         node->fnNode.node.flags = (node->fnNode.node.flags & 0xFF) | (LAYER_OPAQUE << 8);
    308         gfxHead = alloc_display_list(2 * sizeof(*gfxHead));
    309         gfx = gfxHead;
    310     } else {
    311         node->fnNode.node.flags = (node->fnNode.node.flags & 0xFF) | (LAYER_TRANSPARENT << 8);
    312         gfxHead = alloc_display_list(3 * sizeof(*gfxHead));
    313         gfx = gfxHead;
    314         gDPSetAlphaCompare(gfx++, G_AC_DITHER);
    315     }
    316     gDPSetEnvColor(gfx++, 255, 255, 255, alpha);
    317     gSPEndDisplayList(gfx);
    318     return gfxHead;
    319 }
    320 
    321 /**
    322  * Sets the correct blend mode and color for mirror Mario.
    323  */
    324 Gfx *geo_mirror_mario_set_alpha(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) {
    325     UNUSED u8 filler1[4];
    326     Gfx *gfx = NULL;
    327     struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node;
    328     struct MarioBodyState *bodyState = &gBodyStates[asGenerated->parameter];
    329     s16 alpha;
    330     UNUSED u8 filler2[4];
    331 
    332     if (callContext == GEO_CONTEXT_RENDER) {
    333         alpha = (bodyState->modelState & 0x100) ? (bodyState->modelState & 0xFF) : 255;
    334         gfx = make_gfx_mario_alpha(asGenerated, alpha);
    335     }
    336     return gfx;
    337 }
    338 
    339 /**
    340  * Determines if Mario is standing or running for the level of detail of his model.
    341  * If Mario is standing still, he is always high poly. If he is running,
    342  * his level of detail depends on the distance to the camera.
    343  */
    344 Gfx *geo_switch_mario_stand_run(s32 callContext, struct GraphNode *node, UNUSED Mat4 *mtx) {
    345     struct GraphNodeSwitchCase *switchCase = (struct GraphNodeSwitchCase *) node;
    346     struct MarioBodyState *bodyState = &gBodyStates[switchCase->numCases];
    347 
    348     if (callContext == GEO_CONTEXT_RENDER) {
    349         // assign result. 0 if moving, 1 if stationary.
    350         switchCase->selectedCase = ((bodyState->action & ACT_FLAG_STATIONARY) == 0);
    351     }
    352     return NULL;
    353 }
    354 
    355 /**
    356  * Geo node script that makes Mario blink
    357  */
    358 Gfx *geo_switch_mario_eyes(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) {
    359     struct GraphNodeSwitchCase *switchCase = (struct GraphNodeSwitchCase *) node;
    360     struct MarioBodyState *bodyState = &gBodyStates[switchCase->numCases];
    361     s16 blinkFrame;
    362 
    363     if (callContext == GEO_CONTEXT_RENDER) {
    364         if (bodyState->eyeState == 0) {
    365             blinkFrame = ((switchCase->numCases * 32 + gAreaUpdateCounter) >> 1) & 0x1F;
    366             if (blinkFrame < 7) {
    367                 switchCase->selectedCase = gMarioBlinkAnimation[blinkFrame];
    368             } else {
    369                 switchCase->selectedCase = 0;
    370             }
    371         } else {
    372             switchCase->selectedCase = bodyState->eyeState - 1;
    373         }
    374     }
    375     return NULL;
    376 }
    377 
    378 /**
    379  * Makes Mario's upper body tilt depending on the rotation stored in his bodyState
    380  */
    381 Gfx *geo_mario_tilt_torso(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) {
    382     struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node;
    383     struct MarioBodyState *bodyState = &gBodyStates[asGenerated->parameter];
    384     s32 action = bodyState->action;
    385 
    386     if (callContext == GEO_CONTEXT_RENDER) {
    387         struct GraphNodeRotation *rotNode = (struct GraphNodeRotation *) node->next;
    388 
    389         if (action != ACT_BUTT_SLIDE && action != ACT_HOLD_BUTT_SLIDE && action != ACT_WALKING
    390             && action != ACT_RIDING_SHELL_GROUND) {
    391             vec3s_copy(bodyState->torsoAngle, gVec3sZero);
    392         }
    393         rotNode->rotation[0] = bodyState->torsoAngle[1];
    394         rotNode->rotation[1] = bodyState->torsoAngle[2];
    395         rotNode->rotation[2] = bodyState->torsoAngle[0];
    396     }
    397     return NULL;
    398 }
    399 
    400 /**
    401  * Makes Mario's head rotate with the camera angle when in C-up mode
    402  */
    403 Gfx *geo_mario_head_rotation(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) {
    404     struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node;
    405     struct MarioBodyState *bodyState = &gBodyStates[asGenerated->parameter];
    406     s32 action = bodyState->action;
    407 
    408     if (callContext == GEO_CONTEXT_RENDER) {
    409         struct GraphNodeRotation *rotNode = (struct GraphNodeRotation *) node->next;
    410         struct Camera *camera = gCurGraphNodeCamera->config.camera;
    411 
    412         if (camera->mode == CAMERA_MODE_C_UP) {
    413             rotNode->rotation[0] = gPlayerCameraState->headRotation[1];
    414             rotNode->rotation[2] = gPlayerCameraState->headRotation[0];
    415         } else if (action & ACT_FLAG_WATER_OR_TEXT) {
    416             rotNode->rotation[0] = bodyState->headAngle[1];
    417             rotNode->rotation[1] = bodyState->headAngle[2];
    418             rotNode->rotation[2] = bodyState->headAngle[0];
    419         } else {
    420             vec3s_set(bodyState->headAngle, 0, 0, 0);
    421             vec3s_set(rotNode->rotation, 0, 0, 0);
    422         }
    423     }
    424     return NULL;
    425 }
    426 
    427 /**
    428  * Switch between hand models.
    429  * Possible options are described in the MarioHandGSCId enum.
    430  */
    431 Gfx *geo_switch_mario_hand(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) {
    432     struct GraphNodeSwitchCase *switchCase = (struct GraphNodeSwitchCase *) node;
    433     struct MarioBodyState *bodyState = &gBodyStates[0];
    434 
    435     if (callContext == GEO_CONTEXT_RENDER) {
    436         if (bodyState->handState == MARIO_HAND_FISTS) {
    437             // switch between fists (0) and open (1)
    438             switchCase->selectedCase = ((bodyState->action & ACT_FLAG_SWIMMING_OR_FLYING) != 0);
    439         } else {
    440             if (switchCase->numCases == 0) {
    441                 switchCase->selectedCase =
    442                     (bodyState->handState < 5) ? bodyState->handState : MARIO_HAND_OPEN;
    443             } else {
    444                 switchCase->selectedCase =
    445                     (bodyState->handState < 2) ? bodyState->handState : MARIO_HAND_FISTS;
    446             }
    447         }
    448     }
    449     return NULL;
    450 }
    451 
    452 /**
    453  * Increase Mario's hand / foot size when he punches / kicks.
    454  * Since animation geo nodes only support rotation, this scaling animation
    455  * was scripted separately. The node with this script should be placed before
    456  * a scaling node containing the hand / foot geo layout.
    457  * ! Since the animation gets updated in GEO_CONTEXT_RENDER, drawing Mario multiple times
    458  * (such as in the mirror room) results in a faster and desynced punch / kick animation.
    459  */
    460 Gfx *geo_mario_hand_foot_scaler(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) {
    461     static s16 sMarioAttackAnimCounter = 0;
    462     struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node;
    463     struct GraphNodeScale *scaleNode = (struct GraphNodeScale *) node->next;
    464     struct MarioBodyState *bodyState = &gBodyStates[0];
    465 
    466     if (callContext == GEO_CONTEXT_RENDER) {
    467         scaleNode->scale = 1.0f;
    468         if (asGenerated->parameter == bodyState->punchState >> 6) {
    469             if (sMarioAttackAnimCounter != gAreaUpdateCounter && (bodyState->punchState & 0x3F) > 0) {
    470                 bodyState->punchState -= 1;
    471                 sMarioAttackAnimCounter = gAreaUpdateCounter;
    472             }
    473             scaleNode->scale =
    474                 gMarioAttackScaleAnimation[asGenerated->parameter * 6 + (bodyState->punchState & 0x3F)]
    475                 / 10.0f;
    476         }
    477     }
    478     return NULL;
    479 }
    480 
    481 /**
    482  * Switch between normal cap, wing cap, vanish cap and metal cap.
    483  */
    484 Gfx *geo_switch_mario_cap_effect(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) {
    485     struct GraphNodeSwitchCase *switchCase = (struct GraphNodeSwitchCase *) node;
    486     struct MarioBodyState *bodyState = &gBodyStates[switchCase->numCases];
    487 
    488     if (callContext == GEO_CONTEXT_RENDER) {
    489         switchCase->selectedCase = bodyState->modelState >> 8;
    490     }
    491     return NULL;
    492 }
    493 
    494 /**
    495  * Determine whether Mario's head is drawn with or without a cap on.
    496  * Also sets the visibility of the wing cap wings on or off.
    497  */
    498 Gfx *geo_switch_mario_cap_on_off(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) {
    499     struct GraphNode *next = node->next;
    500     struct GraphNodeSwitchCase *switchCase = (struct GraphNodeSwitchCase *) node;
    501     struct MarioBodyState *bodyState = &gBodyStates[switchCase->numCases];
    502 
    503     if (callContext == GEO_CONTEXT_RENDER) {
    504         switchCase->selectedCase = bodyState->capState & 1;
    505         while (next != node) {
    506             if (next->type == GRAPH_NODE_TYPE_TRANSLATION_ROTATION) {
    507                 if (bodyState->capState & 2) {
    508                     next->flags |= GRAPH_RENDER_ACTIVE;
    509                 } else {
    510                     next->flags &= ~GRAPH_RENDER_ACTIVE;
    511                 }
    512             }
    513             next = next->next;
    514         }
    515     }
    516     return NULL;
    517 }
    518 
    519 /**
    520  * Geo node script that makes the wings on Mario's wing cap flap.
    521  * Should be placed before a rotation node.
    522  */
    523 Gfx *geo_mario_rotate_wing_cap_wings(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) {
    524     s16 rotX;
    525     struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node;
    526 
    527     if (callContext == GEO_CONTEXT_RENDER) {
    528         struct GraphNodeRotation *rotNode = (struct GraphNodeRotation *) node->next;
    529 
    530         if (!gBodyStates[asGenerated->parameter >> 1].wingFlutter) {
    531             rotX = (coss((gAreaUpdateCounter & 0xF) << 12) + 1.0f) * 4096.0f;
    532         } else {
    533             rotX = (coss((gAreaUpdateCounter & 7) << 13) + 1.0f) * 6144.0f;
    534         }
    535         if (!(asGenerated->parameter & 1)) {
    536             rotNode->rotation[0] = -rotX;
    537         } else {
    538             rotNode->rotation[0] = rotX;
    539         }
    540     }
    541     return NULL;
    542 }
    543 
    544 /**
    545  * Geo node that updates the held object node and the HOLP.
    546  */
    547 Gfx *geo_switch_mario_hand_grab_pos(s32 callContext, struct GraphNode *b, Mat4 *mtx) {
    548     struct GraphNodeHeldObject *asHeldObj = (struct GraphNodeHeldObject *) b;
    549     Mat4 *curTransform = mtx;
    550     struct MarioState *marioState = &gMarioStates[asHeldObj->playerIndex];
    551 
    552     if (callContext == GEO_CONTEXT_RENDER) {
    553         asHeldObj->objNode = NULL;
    554         if (marioState->heldObj != NULL) {
    555             asHeldObj->objNode = marioState->heldObj;
    556             switch (marioState->marioBodyState->grabPos) {
    557                 case GRAB_POS_LIGHT_OBJ:
    558                     if (marioState->action & ACT_FLAG_THROWING) {
    559                         vec3s_set(asHeldObj->translation, 50, 0, 0);
    560                     } else {
    561                         vec3s_set(asHeldObj->translation, 50, 0, 110);
    562                     }
    563                     break;
    564                 case GRAB_POS_HEAVY_OBJ:
    565                     vec3s_set(asHeldObj->translation, 145, -173, 180);
    566                     break;
    567                 case GRAB_POS_BOWSER:
    568                     vec3s_set(asHeldObj->translation, 80, -270, 1260);
    569                     break;
    570             }
    571         }
    572     } else if (callContext == GEO_CONTEXT_HELD_OBJ) {
    573         // ! The HOLP is set here, which is why it only updates when the held object is drawn.
    574         // This is why it won't update during a pause buffered hitstun or when the camera is very far
    575         // away.
    576         get_pos_from_transform_mtx(marioState->marioBodyState->heldObjLastPosition, *curTransform,
    577                                    *gCurGraphNodeCamera->matrixPtr);
    578     }
    579     return NULL;
    580 }
    581 
    582 // X position of the mirror
    583 #define MIRROR_X 4331.53
    584 
    585 /**
    586  * Geo node that creates a clone of Mario's geo node and updates it to becomes
    587  * a mirror image of the player.
    588  */
    589 Gfx *geo_render_mirror_mario(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) {
    590     f32 mirroredX;
    591     struct Object *mario = gMarioStates[0].marioObj;
    592 
    593     switch (callContext) {
    594         case GEO_CONTEXT_CREATE:
    595             init_graph_node_object(NULL, &gMirrorMario, NULL, gVec3fZero, gVec3sZero, gVec3fOne);
    596             break;
    597         case GEO_CONTEXT_AREA_LOAD:
    598             geo_add_child(node, &gMirrorMario.node);
    599             break;
    600         case GEO_CONTEXT_AREA_UNLOAD:
    601             geo_remove_child(&gMirrorMario.node);
    602             break;
    603         case GEO_CONTEXT_RENDER:
    604             if (mario->header.gfx.pos[0] > 1700.0f) {
    605                 // TODO: Is this a geo layout copy or a graph node copy?
    606                 gMirrorMario.sharedChild = mario->header.gfx.sharedChild;
    607                 gMirrorMario.areaIndex = mario->header.gfx.areaIndex;
    608                 vec3s_copy(gMirrorMario.angle, mario->header.gfx.angle);
    609                 vec3f_copy(gMirrorMario.pos, mario->header.gfx.pos);
    610                 vec3f_copy(gMirrorMario.scale, mario->header.gfx.scale);
    611 
    612                 gMirrorMario.animInfo = mario->header.gfx.animInfo;
    613                 mirroredX = MIRROR_X - gMirrorMario.pos[0];
    614                 gMirrorMario.pos[0] = mirroredX + MIRROR_X;
    615                 gMirrorMario.angle[1] = -gMirrorMario.angle[1];
    616                 gMirrorMario.scale[0] *= -1.0f;
    617                 ((struct GraphNode *) &gMirrorMario)->flags |= GRAPH_RENDER_ACTIVE;
    618             } else {
    619                 ((struct GraphNode *) &gMirrorMario)->flags &= ~GRAPH_RENDER_ACTIVE;
    620             }
    621             break;
    622     }
    623     return NULL;
    624 }
    625 
    626 /**
    627  * Since Mirror Mario has an x scale of -1, the mesh becomes inside out.
    628  * This node corrects that by changing the culling mode accordingly.
    629  */
    630 Gfx *geo_mirror_mario_backface_culling(s32 callContext, struct GraphNode *node, UNUSED Mat4 *c) {
    631     struct GraphNodeGenerated *asGenerated = (struct GraphNodeGenerated *) node;
    632     Gfx *gfx = NULL;
    633 
    634     if (callContext == GEO_CONTEXT_RENDER && gCurGraphNodeObject == &gMirrorMario) {
    635         gfx = alloc_display_list(3 * sizeof(*gfx));
    636 
    637         if (asGenerated->parameter == 0) {
    638             gSPClearGeometryMode(&gfx[0], G_CULL_BACK);
    639             gSPSetGeometryMode(&gfx[1], G_CULL_FRONT);
    640             gSPEndDisplayList(&gfx[2]);
    641         } else {
    642             gSPClearGeometryMode(&gfx[0], G_CULL_FRONT);
    643             gSPSetGeometryMode(&gfx[1], G_CULL_BACK);
    644             gSPEndDisplayList(&gfx[2]);
    645         }
    646         asGenerated->fnNode.node.flags = (asGenerated->fnNode.node.flags & 0xFF) | (LAYER_OPAQUE << 8);
    647     }
    648 
    649     return gfx;
    650 }