sm64

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

particles.c (16395B)


      1 #include <PR/ultratypes.h>
      2 
      3 #include "debug_utils.h"
      4 #include "draw_objects.h"
      5 #include "dynlist_proc.h"
      6 #include "gd_math.h"
      7 #include "gd_types.h"
      8 #include "macros.h"
      9 #include "objects.h"
     10 #include "particles.h"
     11 #include "renderer.h"
     12 #include "skin.h"
     13 
     14 // static types
     15 typedef union {
     16     struct ObjVertex *vtx;
     17     struct ObjParticle *ptc;
     18 } VtxPtc;
     19 
     20 struct Connection {
     21     struct GdObj header;  // this header is never used
     22     u8 filler[8];
     23     VtxPtc node1;  // first connected vertex or particle
     24     VtxPtc node2;  // second connected vertex or particle
     25     f32 unk24;
     26     u32 unk28; // union tag? 0 = vertex; 1 = particle?
     27 };
     28 
     29 // data
     30 UNUSED static void *sUnused801A81D0 = NULL;
     31 static s32 D_801A81D4[25] = {
     32     /*  ID?    X    Y    Z */
     33     9,  3,  12,  -14, 25, 5,  16,  -25, 42, 4,  15, -39, 55,
     34     -6, 20, -23, 70,  -2, 20, -23, 135, 0,  16, 0,  0 /* Terminator */
     35 };
     36 static s32 D_801A8238[5] = {
     37     /*  ID? X   Y  Z */
     38     15, 0, 22, 0, 0 /* Terminator */
     39 };
     40 
     41 // static bss
     42 static struct ObjFace *D_801B9EF0;
     43 
     44 // fn declarations
     45 struct Connection *make_connection(struct ObjVertex *, struct ObjVertex *);
     46 void Unknown80181D14(struct ObjFace *);
     47 void func_80181EB0(struct Connection *);
     48 void func_80182088(struct Connection *);
     49 void move_particle(struct ObjParticle *);
     50 struct Connection *make_connection(struct ObjVertex *, struct ObjVertex *);
     51 int func_80182778(struct ObjParticle *);
     52 void func_80182A08(struct ObjParticle *, struct GdVec3f *b);
     53 void func_801838D0(struct ObjParticle *);
     54 void Unknown801835C8(struct ObjParticle *ptc);
     55 
     56 static void connect_vertices(struct ObjVertex *vtx1, struct ObjVertex *vtx2) {
     57     struct Connection *newConn;
     58     register struct ListNode *link;
     59 
     60     if (vtx1 == vtx2) {
     61         return;
     62     }
     63     link = gGdSkinNet->unk1C0->firstMember;
     64     while (link != NULL) {
     65         // FIXME: types
     66         struct Connection *conn = (struct Connection *) link->obj;
     67 
     68         if ((conn->node1.vtx == vtx1 || conn->node1.vtx == vtx2)
     69          && (conn->node2.vtx == vtx1 || conn->node2.vtx == vtx2)) {
     70             break;
     71         }
     72         link = link->next;
     73     }
     74     if (link == NULL) {
     75         newConn = make_connection(vtx1, vtx2);
     76         //! make_connection never sets the header, so not sure what happens here
     77         addto_group(gGdSkinNet->unk1C0, &newConn->header);
     78     }
     79 }
     80 
     81 /* 2304E4 -> 230680 */
     82 void Unknown80181D14(struct ObjFace *face) {
     83     s32 i;
     84     s32 j;
     85     struct ObjVertex *vtx1;
     86     struct ObjVertex *vtx2;
     87 
     88     for (i = 0; i < face->vtxCount - 1; i++) {
     89         vtx1 = face->vertices[i];
     90         vtx2 = face->vertices[i + 1];
     91         connect_vertices(vtx1, vtx2);
     92     }
     93     if (D_801B9EF0 != NULL) {
     94         for (i = 0; i < face->vtxCount; i++) {
     95             vtx1 = face->vertices[i];
     96             for (j = 0; j < D_801B9EF0->vtxCount; j++) {
     97                 vtx2 = D_801B9EF0->vertices[j];
     98                 connect_vertices(vtx1, vtx2);
     99             }
    100         }
    101     }
    102     D_801B9EF0 = face;
    103 }
    104 
    105 /* 230680 -> 230858 */
    106 void func_80181EB0(struct Connection *cxn) {
    107     struct GdVec3f sp34;
    108     UNUSED u8 filler[44];
    109     struct ObjParticle *sp4 = cxn->node1.ptc;
    110     struct ObjParticle *sp0 = cxn->node2.ptc;
    111 
    112     sp34.x = 0.0f;
    113     sp34.y = sp4->pos.y - sp0->pos.y;
    114     sp34.z = 0.0f;
    115     sp34.y *= 0.01;
    116     sp4->unk38.x -= sp34.x;
    117     sp4->unk38.y -= sp34.y;
    118     sp4->unk38.z -= sp34.z;
    119     sp0->unk38.x += sp34.x;
    120     sp0->unk38.y += sp34.y;
    121     sp0->unk38.z += sp34.z;
    122     if (!(sp4->flags & 2)) {
    123         sp4->pos.x -= sp34.x;
    124         sp4->pos.y -= sp34.y;
    125         sp4->pos.z -= sp34.z;
    126     }
    127     if (!(sp0->flags & 2)) {
    128         sp0->pos.x += sp34.x;
    129         sp0->pos.y += sp34.y;
    130         sp0->pos.z += sp34.z;
    131     }
    132 }
    133 
    134 /* @ 230858 -> 230B70 */
    135 void func_80182088(struct Connection *cxn) {
    136     struct GdVec3f sp4C;
    137     UNUSED u8 filler[36];
    138     f32 sp24;
    139     f32 sp20;
    140     struct ObjParticle *sp1C;
    141     struct ObjParticle *sp18;
    142 
    143     if (cxn->unk28 & 1) {
    144         func_80181EB0(cxn);
    145         return;
    146     }
    147     sp1C = cxn->node1.ptc;
    148     sp18 = cxn->node2.ptc;
    149     sp4C.x = sp1C->pos.x - sp18->pos.x;
    150     sp4C.y = sp1C->pos.y - sp18->pos.y;
    151     sp4C.z = sp1C->pos.z - sp18->pos.z;
    152     sp20 = gd_vec3f_magnitude(&sp4C);
    153     sp24 = sp20 - cxn->unk24;
    154     sp4C.x /= sp20;
    155     sp4C.y /= sp20;
    156     sp4C.z /= sp20;
    157     sp4C.x *= sp24 * 0.1;
    158     sp4C.y *= sp24 * 0.1;
    159     sp4C.z *= sp24 * 0.1;
    160     sp1C->unk38.x -= sp4C.x;
    161     sp1C->unk38.y -= sp4C.y;
    162     sp1C->unk38.z -= sp4C.z;
    163     sp18->unk38.x += sp4C.x;
    164     sp18->unk38.y += sp4C.y;
    165     sp18->unk38.z += sp4C.z;
    166     if (!(sp1C->flags & 2)) {
    167         sp1C->pos.x -= sp4C.x;
    168         sp1C->pos.y -= sp4C.y;
    169         sp1C->pos.z -= sp4C.z;
    170     }
    171     if (!(sp18->flags & 2)) {
    172         sp18->pos.x += sp4C.x;
    173         sp18->pos.y += sp4C.y;
    174         sp18->pos.z += sp4C.z;
    175     }
    176 }
    177 
    178 /* 230B70 -> 230CC0 */
    179 void func_801823A0(struct ObjNet *net) {
    180     register struct ListNode *link;
    181     struct Connection *cxn;
    182 
    183     gGdSkinNet = net;
    184     switch (net->unk3C) {
    185         case 1: // Shape; Are these flags the same as net->netType (+0x1EC)?
    186             net->unk1C8 = net->shapePtr->vtxGroup;
    187             net->unk1C0 = make_group(0);
    188             D_801B9EF0 = NULL;
    189 
    190             apply_to_obj_types_in_group(OBJ_TYPE_FACES, (applyproc_t) Unknown80181D14,
    191                                         net->shapePtr->faceGroup);
    192             net->unk3C = 2;
    193             break;
    194         case 2:
    195             link = net->unk1C0->firstMember;
    196             while (link != NULL) {
    197                 // FIXME: types
    198                 cxn = (struct Connection *) link->obj;
    199                 func_80182088(cxn);
    200                 link = link->next;
    201             }
    202             apply_to_obj_types_in_group(OBJ_TYPE_PARTICLES, (applyproc_t) move_particle, net->unk1C8);
    203             apply_to_obj_types_in_group(OBJ_TYPE_PLANES, (applyproc_t) reset_plane, net->unk1CC);
    204             break;
    205     }
    206 }
    207 
    208 /* 230CC0 -> 230DCC */
    209 struct ObjParticle *make_particle(u32 flags, s32 colourNum, f32 x, f32 y, f32 z) {
    210     struct ObjParticle *particle = (struct ObjParticle *) make_object(OBJ_TYPE_PARTICLES);
    211     UNUSED u8 filler[8];
    212 
    213     particle->pos.x = x;
    214     particle->pos.y = y;
    215     particle->pos.z = z;
    216     particle->unk38.x = particle->unk38.y = particle->unk38.z = 0.0f;
    217     particle->colourNum = colourNum;
    218     particle->flags = flags | 8;
    219     particle->timeout = -1;
    220     particle->id = D_801B9E40; /* should this be D_801B9E40++? */
    221     particle->shapePtr = NULL;
    222     particle->unkB0 = 1;
    223     return particle;
    224 }
    225 
    226 /* 230DCC -> 230F48 */
    227 struct Connection *make_connection(struct ObjVertex *vtx1, struct ObjVertex *vtx2) {
    228     struct Connection *conn = gd_malloc_perm(sizeof(struct Connection));
    229     struct GdVec3f sp28;
    230     struct GdVec3f sp1C;
    231 
    232     if (conn == NULL) {
    233         fatal_print("Cant allocate connection memory!");
    234     }
    235     conn->node1.vtx = vtx1;
    236     conn->node2.vtx = vtx2;
    237     d_stash_dynobj();
    238     set_cur_dynobj((struct GdObj *)vtx1);
    239     d_get_world_pos(&sp28);
    240     set_cur_dynobj((struct GdObj *)vtx2);
    241     d_get_world_pos(&sp1C);
    242     sp28.x -= sp1C.x;
    243     sp28.y -= sp1C.y;
    244     sp28.z -= sp1C.z;
    245     conn->unk24 = gd_vec3f_magnitude(&sp28);
    246     // Duplicate conditional. Possibly should've checked `vtx2`;
    247     // Also, this shouldn't be called with particle types...
    248     if (vtx1->header.type == OBJ_TYPE_PARTICLES && vtx1->header.type == OBJ_TYPE_PARTICLES) {
    249         if ((((struct ObjParticle *) vtx1)->flags & 4) && (((struct ObjParticle *) vtx2)->flags & 4)) {
    250             conn->unk28 |= 1;
    251         }
    252     }
    253     d_unstash_dynobj();
    254     return conn;
    255 }
    256 
    257 /* 230F48 -> 2311D8 */
    258 int func_80182778(struct ObjParticle *ptc) {
    259     s32 sp4 = 0;
    260 
    261     if (ptc->unk7C->animSeqNum == 2 && ptc->unk74 == 1) {
    262         while (D_801A81D4[sp4] != 0) {
    263             if (D_801A81D4[sp4] == ptc->unk7C->frame) {
    264                 ptc->pos.x = D_801A81D4[sp4 + 1] * 10.0f;
    265                 ptc->pos.y = D_801A81D4[sp4 + 2] * 10.0f;
    266                 ptc->pos.z = D_801A81D4[sp4 + 3] * 10.0f;
    267                 return TRUE;
    268             }
    269             sp4 += 4;
    270         }
    271     }
    272     if (ptc->unk7C->animSeqNum == 1 && ptc->unk74 == 1) {
    273         while (D_801A8238[sp4] != 0) {
    274             if (D_801A8238[sp4] == ptc->unk7C->frame) {
    275                 ptc->pos.x = D_801A8238[sp4 + 1] * 10.0f;
    276                 ptc->pos.y = D_801A8238[sp4 + 2] * 10.0f;
    277                 ptc->pos.z = D_801A8238[sp4 + 3] * 10.0f;
    278                 return TRUE;
    279             }
    280             sp4 += 4;
    281         }
    282     }
    283     return FALSE;
    284 }
    285 
    286 /* 2311D8 -> 231454 */
    287 void func_80182A08(struct ObjParticle *ptc, struct GdVec3f *b) {
    288     register struct ListNode *link;
    289     struct ObjParticle *sp20;
    290 
    291     if (ptc->subParticlesGrp != NULL) {
    292         link = ptc->subParticlesGrp->firstMember;
    293         while (link != NULL) {
    294             // FIXME: types
    295             sp20 = (struct ObjParticle *) link->obj;
    296             if (sp20->timeout <= 0) {
    297                 sp20->pos.x = ptc->pos.x;
    298                 sp20->pos.y = ptc->pos.y;
    299                 sp20->pos.z = ptc->pos.z;
    300                 sp20->timeout = 12.0f - gd_rand_float() * 5.0f;
    301                 do {
    302                     sp20->unk38.x = gd_rand_float() * 50.0 - 25.0;
    303                     sp20->unk38.y = gd_rand_float() * 50.0 - 25.0;
    304                     sp20->unk38.z = gd_rand_float() * 50.0 - 25.0;
    305                 } while (gd_vec3f_magnitude(&sp20->unk38) > 30.0);
    306                 sp20->unk38.x += b->x;
    307                 sp20->unk38.y += b->y;
    308                 sp20->unk38.z += b->z;
    309                 sp20->header.drawFlags &= ~OBJ_INVISIBLE;
    310                 sp20->flags |= 8;
    311             }
    312             link = link->next;
    313         }
    314     }
    315 }
    316 
    317 /* 231454 -> 231D40; orig name: Unknown80182C84 */
    318 void move_particle(struct ObjParticle *ptc) {
    319     f32 sp7C;
    320     UNUSED u8 filler1[12];
    321     struct GdVec3f sp64;
    322     struct ObjParticle *sp60;
    323     UNUSED u8 filler2[4];
    324     s32 i;
    325     UNUSED u8 filler3[8];
    326     struct ObjCamera *sp4C;
    327     struct GdVec3f sp40;
    328     struct GdVec3f sp34;
    329 
    330     if (ptc->flags & 2) {
    331         return;
    332     }
    333     if (!(ptc->flags & 8)) {
    334         return;
    335     }
    336     if (ptc->unk60 == 3) {
    337         sp40.x = -gViewUpdateCamera->unkE8[2][0] * 50.0f;
    338         sp40.y = -gViewUpdateCamera->unkE8[2][1] * 50.0f;
    339         sp40.z = gViewUpdateCamera->unkE8[2][2] * 50.0f;
    340         sp34.x = gViewUpdateCamera->unkE8[2][0] * -20.0f;
    341         sp34.y = gViewUpdateCamera->unkE8[2][1] * -20.0f;
    342         sp34.z = gViewUpdateCamera->unkE8[2][2] * -20.0f;
    343     }
    344     if (ptc->attachedToObj != NULL) {
    345         set_cur_dynobj(ptc->attachedToObj);
    346         if (ptc->unk60 == 3) {
    347             if (ptc->unk64 == 3) {
    348                 sp4C = (struct ObjCamera *) ptc->attachedToObj;
    349                 // Camera->unk18C = ObjView here
    350                 if (sp4C->unk18C->pickedObj != NULL) {
    351                     set_cur_dynobj(sp4C->unk18C->pickedObj);
    352                     ptc->flags |= 0x20;
    353                     ; // needed to match
    354                 } else {
    355                     ptc->flags &= ~0x10;
    356                     ptc->flags &= ~0x20;
    357                 }
    358             }
    359         }
    360         d_get_world_pos(&sp64);
    361         ptc->pos.x = sp64.x;
    362         ptc->pos.y = sp64.y;
    363         ptc->pos.z = sp64.z;
    364     }
    365     sp7C = -0.4f;
    366     ptc->pos.x += ptc->unk38.x;
    367     ptc->pos.y += ptc->unk38.y;
    368     ptc->pos.z += ptc->unk38.z;
    369     if (ptc->flags & 1) {
    370         ptc->unk38.y += sp7C;
    371     }
    372     func_801838D0(ptc);
    373     switch (ptc->unkB0) {
    374         case 1:
    375             ptc->unkB0 = 2;
    376             if (ptc->unk60 == 3) {
    377                 switch (ptc->unk64) {
    378                     case 1:
    379                         ptc->subParticlesGrp = make_group(0);
    380                         for (i = 0; i < 50; i++) {
    381                             sp60 = make_particle(1, -1, ptc->pos.x, ptc->pos.y, ptc->pos.z);
    382                             sp60->shapePtr = ptc->shapePtr;
    383                             addto_group(ptc->subParticlesGrp, &sp60->header);
    384                             sp60->flags &= ~8;
    385                         }
    386                         break;
    387                     case 2:
    388                     case 3:
    389                         ptc->subParticlesGrp = make_group(0);
    390                         for (i = 0; i < 30; i++) {
    391                             sp60 = make_particle(1, -1, ptc->pos.x, ptc->pos.y, ptc->pos.z);
    392                             sp60->shapePtr = ptc->shapePtr;
    393                             addto_group(ptc->subParticlesGrp, &sp60->header);
    394                             sp60->flags &= ~8;
    395                         }
    396                         break;
    397                 }
    398             }
    399             break;
    400         default:
    401             break;
    402     }
    403     ptc->unk38.x *= 0.9;
    404     ptc->unk38.y *= 0.9;
    405     ptc->unk38.z *= 0.9;
    406     if (ptc->unk60 == 3) {
    407         switch (ptc->unk64) {
    408             case 1:
    409                 if (func_80182778(ptc) && ptc->subParticlesGrp != NULL) {
    410                     register struct ListNode *link;
    411 
    412                     if (ptc->unk80 != NULL) {
    413                         ptc->unk80->unk3C |= 1;
    414                         ptc->unk80->position.x = ptc->pos.x;
    415                         ptc->unk80->position.y = ptc->pos.y;
    416                         ptc->unk80->position.z = ptc->pos.z;
    417                     }
    418                     link = ptc->subParticlesGrp->firstMember;
    419                     while (link != NULL) {
    420                         struct ObjParticle *sp2C = (struct ObjParticle *) link->obj;
    421 
    422                         sp2C->pos.x = ptc->pos.x;
    423                         sp2C->pos.y = ptc->pos.y;
    424                         sp2C->pos.z = ptc->pos.z;
    425                         sp2C->timeout = 20;
    426                         do {
    427                             sp2C->unk38.x = gd_rand_float() * 64.0 - 32.0;
    428                             sp2C->unk38.y = gd_rand_float() * 64.0 - 32.0;
    429                             sp2C->unk38.z = gd_rand_float() * 64.0 - 32.0;
    430                         } while (gd_vec3f_magnitude(&sp2C->unk38) > 32.0);
    431                         sp2C->unk30 = gd_rand_float() * 180.0f;
    432                         sp2C->header.drawFlags &= ~OBJ_INVISIBLE;
    433                         sp2C->flags |= 8;
    434                         link = link->next;
    435                     }
    436                 }
    437                 break;
    438             case 3:
    439                 if ((ptc->flags & 0x20) && !(ptc->flags & 0x10)) {
    440                     func_80182A08(ptc, &sp40);
    441                     ptc->flags |= 0x10;
    442                 }
    443                 break;
    444             case 2:
    445                 func_80182A08(ptc, &sp34);
    446                 break;
    447         }
    448         apply_to_obj_types_in_group(OBJ_TYPE_PARTICLES, (applyproc_t) move_particle, ptc->subParticlesGrp);
    449     }
    450     if (ptc->timeout >= 0) {
    451         if (ptc->timeout-- <= 0) {
    452             ptc->header.drawFlags |= OBJ_INVISIBLE;
    453             ptc->flags &= ~8;
    454         }
    455     }
    456 }
    457 
    458 /* 231D40 -> 231D98; orig name: func_80183570 */
    459 void move_particles_in_grp(struct ObjGroup *group) {
    460     start_timer("particles");
    461     gGdSkinNet = NULL;
    462     apply_to_obj_types_in_group(OBJ_TYPE_PARTICLES, (applyproc_t) move_particle, group);
    463     stop_timer("particles");
    464 }
    465 
    466 #define ABS(x) ((x) < 0.0f ? -(x) : (x))
    467 /* 231D98 -> 232040 */
    468 void Unknown801835C8(struct ObjParticle *ptc) {
    469     struct GdVec3f sp54;
    470     f32 sp50;
    471     register struct ListNode *link;
    472 
    473     gd_printf("p(%d)=", ptc->attachedObjsGrp->memberCount);
    474     link = ptc->attachedObjsGrp->firstMember;
    475     while (link != NULL) {
    476         // FIXME: types
    477         struct ObjParticle *sp48 = (struct ObjParticle *) link->obj;
    478 
    479         sp54.x = sp48->pos.x - ptc->pos.x;
    480         sp54.y = sp48->pos.y - ptc->pos.y;
    481         sp54.z = sp48->pos.z - ptc->pos.z;
    482         sp50 = 150.0f - (ABS(sp54.x) + ABS(sp54.y) + ABS(sp54.z));
    483         gd_printf(",%f ", sp50);
    484         sp50 *= 0.00000005;
    485         ptc->pos.x += sp50 * sp54.x;
    486         ptc->pos.y += sp50 * sp54.y;
    487         ptc->pos.z += sp50 * sp54.z;
    488         sp48->pos.x -= sp50 * sp54.x;
    489         sp48->pos.y -= sp50 * sp54.y;
    490         sp48->pos.z -= sp50 * sp54.z;
    491         link = link->next;
    492     }
    493     gd_printf("\n");
    494 }
    495 
    496 /**
    497  * Unused
    498  */
    499 void stub_particles_1(UNUSED s32 a) {
    500 }
    501 
    502 /**
    503  * Unused
    504  */
    505 void stub_particles_2(UNUSED s32 a) {
    506 }
    507 
    508 /**
    509  * Unused
    510  */
    511 void stub_particles_3(UNUSED s32 a, UNUSED s32 b, UNUSED s32 c) {
    512 }
    513 
    514 /**
    515  * Unused
    516  */
    517 void stub_particles_4(UNUSED s32 a, UNUSED s32 b, UNUSED s32 c) {
    518 }
    519 
    520 /* 2320A0 -> 2320D4; pad to 2320E0 */
    521 void func_801838D0(struct ObjParticle *ptc) {
    522     D_801B9E3C = ptc;
    523     if (ptc->pos.y < -15.0f) {
    524     }
    525 }