sm64coopdx/src/goddard/skin.c
Prince Frizzy 75bb9b7912
Update to Refresh 13 (#19)
* Refresh 13

Co-authored-by: n64 <n64>
2022-03-13 00:17:10 -08:00

507 lines
16 KiB
C

#include <PR/ultratypes.h>
#if defined(VERSION_EU) || defined(VERSION_SH)
#include "prevent_bss_reordering.h"
#endif
#include "debug_utils.h"
#include "gd_main.h"
#include "gd_math.h"
#include "gd_types.h"
#include "joints.h"
#include "macros.h"
#include "objects.h"
#include "particles.h"
#include "renderer.h"
#include "skin.h"
#include "skin_movement.h"
// bss
struct ObjNet *gGdSkinNet; // @ 801BAAF0
static s32 D_801BAAF4;
static s32 sNetCount; // @ 801BAAF8
/* 2406E0 -> 240894 */
void compute_net_bounding_box(struct ObjNet *net) {
reset_bounding_box();
if (net->unk1D0 != NULL) {
apply_to_obj_types_in_group(OBJ_TYPE_ALL, (applyproc_t) add_obj_pos_to_bounding_box, net->unk1D0);
}
if (net->unk1C8 != NULL) {
apply_to_obj_types_in_group(OBJ_TYPE_ALL, (applyproc_t) add_obj_pos_to_bounding_box, net->unk1C8);
}
gSomeBoundingBox.minX *= net->scale.x;
gSomeBoundingBox.maxX *= net->scale.x;
gSomeBoundingBox.minY *= net->scale.y;
gSomeBoundingBox.maxY *= net->scale.y;
gSomeBoundingBox.minZ *= net->scale.z;
gSomeBoundingBox.maxZ *= net->scale.z;
net->boundingBox.minX = gSomeBoundingBox.minX;
net->boundingBox.minY = gSomeBoundingBox.minY;
net->boundingBox.minZ = gSomeBoundingBox.minZ;
net->boundingBox.maxX = gSomeBoundingBox.maxX;
net->boundingBox.maxY = gSomeBoundingBox.maxY;
net->boundingBox.maxZ = gSomeBoundingBox.maxZ;
}
/* 240894 -> 240A64; orig name: func_801920C4 */
void reset_net(struct ObjNet *net) {
struct ObjGroup *grp;
printf("reset_net %d\n", net->id);
net->worldPos.x = net->initPos.x;
net->worldPos.y = net->initPos.y;
net->worldPos.z = net->initPos.z;
net->velocity.x = net->velocity.y = net->velocity.z = 0.0f;
net->torque.x = net->torque.y = net->torque.z = 0.0f;
compute_net_bounding_box(net);
gd_print_vec("net scale: ", &net->scale);
gd_print_bounding_box("net box: ", &net->boundingBox);
gGdSkinNet = net;
D_801BAAF4 = 0;
gd_set_identity_mat4(&net->mat168);
gd_set_identity_mat4(&net->matE8);
gd_rot_mat_about_vec(&net->matE8, &net->unk68); // set rot mtx to initial rotation?
gd_add_vec3f_to_mat4f_offset(&net->matE8, &net->worldPos); // set to initial position?
gd_copy_mat4f(&net->matE8, &net->mat128);
if ((grp = net->unk1C8) != NULL) {
apply_to_obj_types_in_group(OBJ_TYPE_JOINTS, (applyproc_t) reset_joint, grp);
apply_to_obj_types_in_group(OBJ_TYPE_JOINTS, (applyproc_t) func_80191220, grp);
apply_to_obj_types_in_group(OBJ_TYPE_BONES, (applyproc_t) func_8018FB58, grp);
apply_to_obj_types_in_group(OBJ_TYPE_BONES, (applyproc_t) func_8018FA68, grp);
}
}
/* 240A64 -> 240ACC */
void func_80192294(struct ObjNet *net) {
UNUSED s32 sp1C = 0;
if (net->attachedToObj == NULL) {
restart_timer("childpos");
sp1C = transform_child_objects_recursive(&net->header, NULL);
split_timer("childpos");
}
}
/* 240ACC -> 240B84 */
void func_801922FC(struct ObjNet *net) {
struct ObjGroup *group; // 24
UNUSED u32 pad18[2];
gGdSkinNet = net;
// TODO: netype constants?
if (net->netType == 4) {
if (net->shapePtr != NULL) {
D_801B9E38 = &net->mat128;
scale_verts(net->shapePtr->vtxGroup);
}
if ((group = net->unk1C8) != NULL) {
apply_to_obj_types_in_group(OBJ_TYPE_JOINTS, (applyproc_t) reset_joint_weights, group);
}
}
}
/* 240B84 -> 240CF8 */
struct ObjNet *make_net(UNUSED s32 a0, struct ObjShape *shapedata, struct ObjGroup *a2,
struct ObjGroup *a3, struct ObjGroup *a4) {
struct ObjNet *net;
net = (struct ObjNet *) make_object(OBJ_TYPE_NETS);
gd_set_identity_mat4(&net->mat128);
net->initPos.x = net->initPos.y = net->initPos.z = 0.0f;
net->id = ++sNetCount;
net->scale.x = net->scale.y = net->scale.z = 1.0f;
net->shapePtr = shapedata;
net->unk1C8 = a2;
net->unk1CC = a3;
net->unk1D0 = a4;
net->netType = 0;
net->ctrlType = 0;
net->unk21C = NULL;
net->unk3C = 1;
net->colourNum = 0;
net->skinGrp = NULL;
reset_net(net);
return net;
}
/* 240CF8 -> 240E74 */
void func_80192528(struct ObjNet *net) {
net->unusedForce.x = net->unusedForce.y = net->unusedForce.z = 0.0f;
net->collDisp.x = net->collDisp.y = net->collDisp.z = 0.0f;
net->collTorque.x = net->collTorque.y = net->collTorque.z = 0.0f;
net->unusedCollDispOff.x = net->unusedCollDispOff.y = net->unusedCollDispOff.z = 0.0f;
net->unusedCollMaxD = 0.0f;
gGdCounter.ctr0 = 0;
gGdCounter.ctr1 = 0;
D_801B9E18.x = 0.0f;
D_801B9E18.y = 0.0f;
D_801B9E18.z = 0.0f;
D_801B9E28.x = 0.0f;
D_801B9E28.y = 0.0f;
D_801B9E28.z = 0.0f;
D_801B9E34 = 0.0f;
if (net->flags & 0x1) {
net->velocity.y += -4.0; //? 4.0f
}
net->worldPos.x += net->velocity.x / 1.0f;
net->worldPos.y += net->velocity.y / 1.0f;
net->worldPos.z += net->velocity.z / 1.0f;
}
/* 240E74 -> 2412A0 */
void collision_something_801926A4(struct ObjNet *net) {
if (gGdCounter.ctr1 != 0) {
if (D_801B9E34 != 0.0f) {
D_801B9E28.x /= D_801B9E34;
D_801B9E28.y /= D_801B9E34;
D_801B9E28.z /= D_801B9E34;
}
D_801B9E28.x *= 1.0 / gGdCounter.ctr1; // !1.0f
D_801B9E28.y *= 1.0 / gGdCounter.ctr1; // !1.0f
D_801B9E28.z *= 1.0 / gGdCounter.ctr1; // !1.0f
D_801B9E18.x *= 1.0 / gGdCounter.ctr1; // !1.0f
D_801B9E18.y *= 1.0 / gGdCounter.ctr1; // !1.0f
D_801B9E18.z *= 1.0 / gGdCounter.ctr1; // !1.0f
func_8017E584(gGdSkinNet, &D_801B9E28, &D_801B9E18);
func_8017E838(gGdSkinNet, &D_801B9E28, &D_801B9E18);
}
net->torque.x += net->collTorque.x;
net->torque.y += net->collTorque.y;
net->torque.z += net->collTorque.z;
net->collDisp.x *= 1.0; // 1.0f;
net->collDisp.y *= 1.0; // 1.0f;
net->collDisp.z *= 1.0; // 1.0f;
net->velocity.x += net->collDisp.x;
net->velocity.y += net->collDisp.y;
net->velocity.z += net->collDisp.z;
net->worldPos.x += net->collDisp.x;
net->worldPos.y += net->collDisp.y;
net->worldPos.z += net->collDisp.z;
func_8017E9EC(net);
net->torque.x *= 0.98; //? 0.98f
net->torque.z *= 0.98; //? 0.98f
net->torque.y *= 0.9; //? 0.9f
}
/* 2412A0 -> 24142C; not called */
void func_80192AD0(struct ObjNet *net) {
UNUSED u32 pad64;
struct ObjGroup *sp60;
UNUSED u32 pad20[0x10];
UNUSED u32 sp1C;
struct ObjNet *sp18;
if ((sp60 = net->unk1C8) == NULL) {
return;
}
sp18 = net->unk1F0;
net->worldPos.x = net->unk1F4.x;
net->worldPos.y = net->unk1F4.y;
net->worldPos.z = net->unk1F4.z;
gd_rotate_and_translate_vec3f(&net->worldPos, &sp18->mat128);
net->worldPos.x += net->unk1F0->worldPos.x;
net->worldPos.y += net->unk1F0->worldPos.y;
net->worldPos.z += net->unk1F0->worldPos.z;
net->unk200.x = 0.0f;
net->unk200.y = 10.0f;
net->unk200.z = -4.0f;
gd_rotate_and_translate_vec3f(&net->unk200, &sp18->mat128);
apply_to_obj_types_in_group(OBJ_TYPE_JOINTS, (applyproc_t) func_80191824, sp60);
func_80191E88(sp60);
apply_to_obj_types_in_group(OBJ_TYPE_BONES, (applyproc_t) func_8018F328, net->unk20C);
}
/* 24142C -> 24149C; orig name: func_80192C5C */
void move_bonesnet(struct ObjNet *net) {
struct ObjGroup *sp24;
UNUSED u32 pad18[3];
imin("move_bonesnet");
gd_set_identity_mat4(&D_801B9DC8);
if ((sp24 = net->unk1C8) != NULL) {
apply_to_obj_types_in_group(OBJ_TYPE_JOINTS, (applyproc_t) func_801913C0, sp24);
}
imout();
}
/* 24149C -> 241768 */
void func_80192CCC(struct ObjNet *net) {
Mat4f sp38;
UNUSED struct GdControl *ctrl; // 34
struct ObjGroup *group; // 30
struct GdVec3f sp24;
ctrl = &gGdCtrl;
if (gGdCtrl.unk2C != NULL) {
menu_cb_reset_positions();
}
gd_set_identity_mat4(&D_801B9DC8);
if (gGdCtrl.unk30 != NULL) {
sp24.x = net->mat128[0][0];
sp24.y = net->mat128[0][1];
sp24.z = net->mat128[0][2];
gd_create_rot_mat_angular(&sp38, &sp24, 4.0f);
gd_mult_mat4f(&sp38, &D_801B9DC8, &D_801B9DC8);
net->torque.x = net->torque.y = net->torque.z = 0.0f;
}
if (gGdCtrl.unk28 != NULL) {
sp24.x = net->mat128[0][0];
sp24.y = net->mat128[0][1];
sp24.z = net->mat128[0][2];
gd_create_rot_mat_angular(&sp38, &sp24, -4.0f);
gd_mult_mat4f(&sp38, &D_801B9DC8, &D_801B9DC8);
net->torque.x = net->torque.y = net->torque.z = 0.0f;
}
if (gGdCtrl.newStartPress) {
return;
} // start was pressed
switch (net->ctrlType) {
case 2:
break;
}
func_80192528(net);
if ((group = net->unk1C8) != NULL) {
apply_to_obj_types_in_group(OBJ_TYPE_JOINTS, (applyproc_t) func_80191220, group);
apply_to_obj_types_in_group(OBJ_TYPE_JOINTS, (applyproc_t) func_801913F0, group);
apply_to_obj_types_in_group(OBJ_TYPE_JOINTS, (applyproc_t) stub_joints_2, group);
apply_to_obj_types_in_group(OBJ_TYPE_JOINTS, (applyproc_t) func_801911A8, group);
}
collision_something_801926A4(net);
gd_mult_mat4f(&net->mat128, &D_801B9DC8, &net->mat128);
if (group != NULL) {
apply_to_obj_types_in_group(OBJ_TYPE_JOINTS, (applyproc_t) func_801913C0, group);
apply_to_obj_types_in_group(OBJ_TYPE_BONES, (applyproc_t) func_8018FA68, group);
}
}
/* 241768 -> 241AB4; orig name: func_80192F98 */
void convert_gd_verts_to_Vn(struct ObjGroup *grp) {
UNUSED u8 pad[0x40 - 0x2c];
Vtx *vn; // 28
u8 nx, ny, nz; // 24, 25, 26
UNUSED u32 pad20;
register struct VtxLink *vtxlink; // a1
register s16 x; // a3
register s16 y; // t0
register s16 z; // t1
register struct ObjVertex *vtx; // t2
register struct ListNode *link; // t3
struct GdObj *obj; // sp4
for (link = grp->firstMember; link != NULL; link = link->next) {
obj = link->obj;
vtx = (struct ObjVertex *) obj;
x = (s16) vtx->pos.x;
y = (s16) vtx->pos.y;
z = (s16) vtx->pos.z;
nx = (u8)(vtx->normal.x * 255.0f);
ny = (u8)(vtx->normal.y * 255.0f);
nz = (u8)(vtx->normal.z * 255.0f);
for (vtxlink = vtx->gbiVerts; vtxlink != NULL; vtxlink = vtxlink->prev) {
vn = vtxlink->data;
vn->n.ob[0] = x;
vn->n.ob[1] = y;
vn->n.ob[2] = z;
vn->n.n[0] = nx;
vn->n.n[1] = ny;
vn->n.n[2] = nz;
}
}
}
/* 241AB4 -> 241BCC; orig name: func_801932E4 */
void convert_gd_verts_to_Vtx(struct ObjGroup *grp) {
UNUSED u32 pad24[6];
register struct VtxLink *vtxlink; // a1
register s16 x; // a3
register s16 y; // t0
register s16 z; // t1
register struct ObjVertex *vtx; // t2
register struct ListNode *link; // t3
struct GdObj *obj; // sp4
for (link = grp->firstMember; link != NULL; link = link->next) {
obj = link->obj;
vtx = (struct ObjVertex *) obj;
x = (s16) vtx->pos.x;
y = (s16) vtx->pos.y;
z = (s16) vtx->pos.z;
for (vtxlink = vtx->gbiVerts; vtxlink != NULL; vtxlink = vtxlink->prev) {
vtxlink->data->v.ob[0] = x;
vtxlink->data->v.ob[1] = y;
vtxlink->data->v.ob[2] = z;
}
}
}
/* 241BCC -> 241CA0; orig name: Proc801933FC */
void convert_net_verts(struct ObjNet *net) {
if (net->shapePtr != NULL) {
if (net->shapePtr->unk30) {
convert_gd_verts_to_Vn(net->shapePtr->vtxGroup);
}
}
switch (net->netType) {
case 2:
if (net->shapePtr != NULL) {
convert_gd_verts_to_Vtx(net->shapePtr->scaledVtxGroup);
}
break;
}
}
/* 241CA0 -> 241D6C */
static void move_joints_in_net(struct ObjNet *net) {
struct ObjGroup *grp; // 2c
register struct ListNode *link; // s0
struct GdObj *obj; // 24
if ((grp = net->unk1C8) != NULL) {
for (link = grp->firstMember; link != NULL; link = link->next) {
obj = link->obj;
switch (obj->type) {
case OBJ_TYPE_JOINTS:
if (((struct ObjJoint *) obj)->updateFunc != NULL) {
(*((struct ObjJoint *) obj)->updateFunc)((struct ObjJoint *) obj);
}
break;
default:;
}
}
}
}
/* 241D6C -> 241E94; orig name: func_8019359C */
void move_net(struct ObjNet *net) {
gGdSkinNet = net;
switch (net->netType) {
case 1:
break;
case 7:
func_80192CCC(net);
break;
case 4:
restart_timer("move_bones");
move_bonesnet(net);
split_timer("move_bones");
break;
case 2:
restart_timer("move_skin");
move_skin(net);
split_timer("move_skin");
break;
case 3:
move_joints_in_net(net);
break;
case 5:
func_801823A0(net);
break;
case 6:
break;
default:
fatal_printf("move_net(%d(%d)): Undefined net type", net->id, net->netType);
}
}
/* 241E94 -> 241F0C; orig name: func_801936C4 */
void move_nets(struct ObjGroup *group) {
imin("move_nets");
restart_timer("move_nets");
apply_to_obj_types_in_group(OBJ_TYPE_NETS, (applyproc_t) func_80192294, group);
apply_to_obj_types_in_group(OBJ_TYPE_NETS, (applyproc_t) move_net, group);
split_timer("move_nets");
imout();
}
/* 241F0C -> 242018 */
void func_8019373C(struct ObjNet *net) {
register struct ListNode *link;
struct ObjVertex *vtx;
switch (net->netType) {
case 2:
if (net->shapePtr != NULL) {
net->shapePtr->scaledVtxGroup = make_group(0);
for (link = net->shapePtr->vtxGroup->firstMember; link != NULL; link = link->next) {
vtx = (struct ObjVertex *) link->obj;
if (vtx->scaleFactor != 1.0) {
addto_group(net->shapePtr->scaledVtxGroup, &vtx->header);
}
}
}
break;
}
}
/* 242018 -> 24208C */
void func_80193848(struct ObjGroup *group) {
apply_to_obj_types_in_group(OBJ_TYPE_NETS, (applyproc_t) reset_net, group);
apply_to_obj_types_in_group(OBJ_TYPE_NETS, (applyproc_t) func_80192294, group);
apply_to_obj_types_in_group(OBJ_TYPE_NETS, (applyproc_t) func_801922FC, group);
apply_to_obj_types_in_group(OBJ_TYPE_NETS, (applyproc_t) func_8019373C, group);
}
/* 24208C -> 2422E0; not called; orig name: func_801938BC */
void gd_print_net(struct ObjNet *net) {
gd_printf("Flags:%x\n", net->flags);
gd_print_vec("World:", &net->worldPos);
gd_print_vec("Force:", &net->unusedForce);
gd_print_vec("Vel:", &net->velocity);
gd_print_vec("Rot:", &net->rotation);
gd_print_vec("CollDisp:", &net->collDisp);
gd_print_vec("CollTorque:", &net->collTorque);
gd_print_vec("CollTorqueL:", &net->unusedCollTorqueL);
gd_print_vec("CollTorqueD:", &net->unusedCollTorqueD);
gd_print_vec("Torque:", &net->torque);
gd_print_vec("CofG:", &net->centerOfGravity);
gd_print_bounding_box("BoundBox:", &net->boundingBox);
gd_print_vec("CollDispOff:", &net->unusedCollDispOff);
gd_printf("CollMaxD: %f\n", net->unusedCollMaxD);
gd_printf("MaxRadius: %f\n", net->maxRadius);
gd_print_mtx("Matrix:", &net->mat128);
if (net->shapePtr != NULL) {
gd_printf("ShapePtr: %x (%s)\n", (u32) (uintptr_t) net->shapePtr, net->shapePtr->name);
} else {
gd_printf("ShapePtr: NULL\n");
}
gd_print_vec("Scale:", &net->scale);
gd_printf("Mass: %f\n", net->unusedMass);
gd_printf("NumModes: %d\n", net->numModes);
gd_printf("NodeGroup: %x\n", (u32) (uintptr_t) net->unk1C8);
gd_printf("PlaneGroup: %x\n", (u32) (uintptr_t) net->unk1CC);
gd_printf("VertexGroup: %x\n", (u32) (uintptr_t) net->unk1D0);
}
/* 2422E0 -> 2422F8; orig name: func_80193B10 */
void reset_net_count(void) {
sNetCount = 0;
}