mirror of
https://github.com/zeldaret/oot.git
synced 2025-01-15 12:47:04 +00:00
37f0774778
* T() in idle.c, irqmgr.c, padmgr.c, z_locale.c * T() in z_sram.c * T() in z_bgcheck.c, z_scene.c * format
4588 lines
167 KiB
C
4588 lines
167 KiB
C
#include "global.h"
|
||
#include "terminal.h"
|
||
|
||
u16 DynaSSNodeList_GetNextNodeIdx(DynaSSNodeList* nodeList);
|
||
void BgCheck_GetStaticLookupIndicesFromPos(CollisionContext* colCtx, Vec3f* pos, Vec3i* sector);
|
||
s32 BgCheck_PosInStaticBoundingBox(CollisionContext* colCtx, Vec3f* pos);
|
||
s32 BgCheck_CheckLineImpl(CollisionContext* colCtx, u16 xpFlags1, u16 xpFlags2, Vec3f* posA, Vec3f* posB,
|
||
Vec3f* posResult, CollisionPoly** outPoly, s32* outBgId, Actor* actor, f32 chkDist,
|
||
u32 bccFlags);
|
||
void SSNodeList_Initialize(SSNodeList* this);
|
||
void SSNodeList_Alloc(PlayState* play, SSNodeList* this, s32 tblMax, s32 numPolys);
|
||
u16 SSNodeList_GetNextNodeIdx(SSNodeList* this);
|
||
void DynaPoly_Init(PlayState* play, DynaCollisionContext* dyna);
|
||
void DynaPoly_Alloc(PlayState* play, DynaCollisionContext* dyna);
|
||
f32 BgCheck_RaycastDownDyna(DynaRaycastDown* dynaRaycastDown);
|
||
s32 BgCheck_SphVsDynaWall(CollisionContext* colCtx, u16 xpFlags, f32* outX, f32* outZ, Vec3f* pos, f32 radius,
|
||
CollisionPoly** outPoly, s32* outBgId, Actor* actor);
|
||
s32 BgCheck_CheckDynaCeiling(CollisionContext* colCtx, u16 xpFlags, f32* outY, Vec3f* pos, f32 chkDist,
|
||
CollisionPoly** outPoly, s32* outBgId, Actor* actor);
|
||
s32 BgCheck_CheckLineAgainstDyna(CollisionContext* colCtx, u16 xpFlags, Vec3f* posA, Vec3f* posB, Vec3f* posResult,
|
||
CollisionPoly** outPoly, f32* distSq, s32* outBgId, Actor* actor, f32 chkDist,
|
||
s32 bccFlags);
|
||
s32 BgCheck_SphVsFirstDynaPoly(CollisionContext* colCtx, u16 xpFlags, CollisionPoly** outPoly, s32* outBgId,
|
||
Vec3f* center, f32 radius, Actor* actor, u16 bciFlags);
|
||
void BgCheck_ResetPolyCheckTbl(SSNodeList* nodeList, s32 numPolys);
|
||
|
||
#define SS_NULL 0xFFFF
|
||
|
||
// bccFlags
|
||
#define BGCHECK_CHECK_WALL (1 << 0)
|
||
#define BGCHECK_CHECK_FLOOR (1 << 1)
|
||
#define BGCHECK_CHECK_CEILING (1 << 2)
|
||
#define BGCHECK_CHECK_ONE_FACE (1 << 3)
|
||
#define BGCHECK_CHECK_DYNA (1 << 4)
|
||
#define BGCHECK_CHECK_ALL \
|
||
(BGCHECK_CHECK_WALL | BGCHECK_CHECK_FLOOR | BGCHECK_CHECK_CEILING | BGCHECK_CHECK_ONE_FACE | BGCHECK_CHECK_DYNA)
|
||
|
||
// bciFlags
|
||
#define BGCHECK_IGNORE_NONE 0
|
||
#define BGCHECK_IGNORE_CEILING (1 << 0)
|
||
#define BGCHECK_IGNORE_WALL (1 << 1)
|
||
#define BGCHECK_IGNORE_FLOOR (1 << 2)
|
||
|
||
// raycast down flags (downChkFlags)
|
||
#define BGCHECK_RAYCAST_DOWN_CHECK_CEILINGS (1 << 0)
|
||
#define BGCHECK_RAYCAST_DOWN_CHECK_WALLS (1 << 1)
|
||
#define BGCHECK_RAYCAST_DOWN_CHECK_FLOORS (1 << 2)
|
||
#define BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE (1 << 3) // stops checking dyna walls on finding first candidate result
|
||
#define BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY (1 << 4) // skips walls and ceilings with normal.y < 0
|
||
|
||
// raycast down groundChk flag. When enabled, search range is limited to floors and walls with a normal.y >= 0
|
||
#define BGCHECK_GROUND_CHECK_ON (1 << 0)
|
||
|
||
s32 D_80119D90[WALL_TYPE_MAX] = {
|
||
0, // WALL_TYPE_0
|
||
WALL_FLAG_0, // WALL_TYPE_1
|
||
WALL_FLAG_0 | WALL_FLAG_1, // WALL_TYPE_2
|
||
WALL_FLAG_0 | WALL_FLAG_2, // WALL_TYPE_3
|
||
WALL_FLAG_3, // WALL_TYPE_4
|
||
WALL_FLAG_CRAWLSPACE_1, // WALL_TYPE_5
|
||
WALL_FLAG_CRAWLSPACE_2, // WALL_TYPE_6
|
||
WALL_FLAG_6, // WALL_TYPE_7
|
||
};
|
||
|
||
u16 sSurfaceMaterialToSfxOffset[SURFACE_MATERIAL_MAX] = {
|
||
SURFACE_SFX_OFFSET_DIRT, // SURFACE_MATERIAL_DIRT
|
||
SURFACE_SFX_OFFSET_SAND, // SURFACE_MATERIAL_SAND
|
||
SURFACE_SFX_OFFSET_STONE, // SURFACE_MATERIAL_STONE
|
||
SURFACE_SFX_OFFSET_JABU, // SURFACE_MATERIAL_JABU
|
||
SURFACE_SFX_OFFSET_WATER_SHALLOW, // SURFACE_MATERIAL_WATER_SHALLOW
|
||
SURFACE_SFX_OFFSET_WATER_DEEP, // SURFACE_MATERIAL_WATER_DEEP
|
||
SURFACE_SFX_OFFSET_TALL_GRASS, // SURFACE_MATERIAL_TALL_GRASS
|
||
SURFACE_SFX_OFFSET_LAVA, // SURFACE_MATERIAL_LAVA
|
||
SURFACE_SFX_OFFSET_GRASS, // SURFACE_MATERIAL_GRASS
|
||
SURFACE_SFX_OFFSET_BRIDGE, // SURFACE_MATERIAL_BRIDGE
|
||
SURFACE_SFX_OFFSET_WOOD, // SURFACE_MATERIAL_WOOD
|
||
SURFACE_SFX_OFFSET_DIRT, // SURFACE_MATERIAL_DIRT_SOFT
|
||
SURFACE_SFX_OFFSET_ICE, // SURFACE_MATERIAL_ICE
|
||
SURFACE_SFX_OFFSET_CARPET, // SURFACE_MATERIAL_CARPET
|
||
};
|
||
|
||
#if OOT_DEBUG
|
||
/**
|
||
* original name: T_BGCheck_PosErrorCheck
|
||
*/
|
||
s32 BgCheck_PosErrorCheck(Vec3f* pos, const char* file, int line) {
|
||
if (pos->x >= BGCHECK_XYZ_ABSMAX || pos->x <= -BGCHECK_XYZ_ABSMAX || pos->y >= BGCHECK_XYZ_ABSMAX ||
|
||
pos->y <= -BGCHECK_XYZ_ABSMAX || pos->z >= BGCHECK_XYZ_ABSMAX || pos->z <= -BGCHECK_XYZ_ABSMAX) {
|
||
PRINTF(VT_FGCOL(RED));
|
||
PRINTF(T("T_BGCheck_PosErrorCheck():位置が妥当ではありません。pos (%f,%f,%f) file:%s line:%d\n",
|
||
"T_BGCheck_PosErrorCheck(): Position is invalid. pos (%f,%f,%f) file:%s line:%d\n"),
|
||
pos->x, pos->y, pos->z, file, line);
|
||
PRINTF(VT_RST);
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
#endif
|
||
|
||
/**
|
||
* Set SSNode
|
||
*/
|
||
void SSNode_SetValue(SSNode* node, s16* polyId, u16 next) {
|
||
node->polyId = *polyId;
|
||
node->next = next;
|
||
}
|
||
|
||
/**
|
||
* Set SSList to SS_NULL
|
||
*/
|
||
void SSList_SetNull(SSList* ssList) {
|
||
ssList->head = SS_NULL;
|
||
}
|
||
|
||
/**
|
||
* Insert `polyId` at the start of the static `ssList` list
|
||
*/
|
||
void SSNodeList_SetSSListHead(SSNodeList* nodeList, SSList* ssList, s16* polyId) {
|
||
u16 newNodeId = SSNodeList_GetNextNodeIdx(nodeList);
|
||
|
||
SSNode_SetValue(&nodeList->tbl[newNodeId], polyId, ssList->head);
|
||
ssList->head = newNodeId;
|
||
}
|
||
|
||
/**
|
||
* Insert `polyId` at the start of the dyna `ssList` list
|
||
*/
|
||
void DynaSSNodeList_SetSSListHead(DynaSSNodeList* nodeList, SSList* ssList, s16* polyId) {
|
||
u16 newNodeId = DynaSSNodeList_GetNextNodeIdx(nodeList);
|
||
|
||
ASSERT(newNodeId != SS_NULL, "new_node != SS_NULL", "../z_bgcheck.c", 1776);
|
||
SSNode_SetValue(&nodeList->tbl[newNodeId], polyId, ssList->head);
|
||
ssList->head = newNodeId;
|
||
}
|
||
|
||
/**
|
||
* Initialize DynaSSNodeList
|
||
*/
|
||
void DynaSSNodeList_Initialize(PlayState* play, DynaSSNodeList* nodeList) {
|
||
nodeList->tbl = NULL;
|
||
nodeList->count = 0;
|
||
}
|
||
|
||
/**
|
||
* Initialize DynaSSNodeList tbl
|
||
*/
|
||
void DynaSSNodeList_Alloc(PlayState* play, DynaSSNodeList* nodeList, s32 max) {
|
||
nodeList->tbl = THA_AllocTailAlign(&play->state.tha, max * sizeof(SSNode), ALIGNOF_MASK(SSNode));
|
||
|
||
ASSERT(nodeList->tbl != NULL, "psst->tbl != NULL", "../z_bgcheck.c", 1811);
|
||
|
||
nodeList->max = max;
|
||
nodeList->count = 0;
|
||
}
|
||
|
||
/**
|
||
* Reset DynaSSNodeList count
|
||
*/
|
||
void DynaSSNodeList_ResetCount(DynaSSNodeList* nodeList) {
|
||
nodeList->count = 0;
|
||
}
|
||
|
||
/**
|
||
* Get next available node index in DynaSSNodeList
|
||
* returns SS_NULL if list is full
|
||
*/
|
||
u16 DynaSSNodeList_GetNextNodeIdx(DynaSSNodeList* nodeList) {
|
||
u16 idx = nodeList->count++;
|
||
|
||
if (nodeList->max <= idx) {
|
||
return SS_NULL;
|
||
}
|
||
|
||
return idx;
|
||
}
|
||
|
||
/**
|
||
* original name: T_BGCheck_Vec3sToVec3f
|
||
*/
|
||
void BgCheck_Vec3sToVec3f(Vec3s* src, Vec3f* dst) {
|
||
dst->x = src->x;
|
||
dst->y = src->y;
|
||
dst->z = src->z;
|
||
}
|
||
|
||
/**
|
||
* original name: T_BGCheck_Vec3fToVec3s
|
||
*/
|
||
void BgCheck_Vec3fToVec3s(Vec3s* dst, Vec3f* src) {
|
||
dst->x = src->x;
|
||
dst->y = src->y;
|
||
dst->z = src->z;
|
||
}
|
||
|
||
/**
|
||
* Get CollisionPoly's lowest y point
|
||
*/
|
||
s16 CollisionPoly_GetMinY(CollisionPoly* poly, Vec3s* vtxList) {
|
||
//! @bug Due to rounding errors, some polys with a slight slope have a y normal of 1.0f/-1.0f. As such, this
|
||
//! optimization returns the wrong minimum y for a subset of these polys.
|
||
if (poly->normal.y == COLPOLY_SNORMAL(1.0f) || poly->normal.y == COLPOLY_SNORMAL(-1.0f)) {
|
||
return vtxList[COLPOLY_VTX_INDEX(poly->flags_vIA)].y;
|
||
} else {
|
||
s32 a = COLPOLY_VTX_INDEX(poly->flags_vIA);
|
||
s32 b = COLPOLY_VTX_INDEX(poly->flags_vIB);
|
||
s32 c = poly->vIC;
|
||
s16 min = vtxList[a].y;
|
||
|
||
if (min > vtxList[b].y) {
|
||
min = vtxList[b].y;
|
||
}
|
||
if (min < vtxList[c].y) {
|
||
return min;
|
||
}
|
||
return vtxList[c].y;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* CollisionPoly get unit normal
|
||
*/
|
||
void CollisionPoly_GetNormalF(CollisionPoly* poly, f32* nx, f32* ny, f32* nz) {
|
||
*nx = COLPOLY_GET_NORMAL(poly->normal.x);
|
||
*ny = COLPOLY_GET_NORMAL(poly->normal.y);
|
||
*nz = COLPOLY_GET_NORMAL(poly->normal.z);
|
||
}
|
||
|
||
/**
|
||
* Compute transform matrix mapping +y (up) to the collision poly's normal
|
||
*/
|
||
void func_80038A28(CollisionPoly* poly, f32 tx, f32 ty, f32 tz, MtxF* dest) {
|
||
f32 nx;
|
||
f32 ny;
|
||
f32 nz;
|
||
s32 pad;
|
||
f32 xx;
|
||
f32 zz;
|
||
f32 yz;
|
||
f32 xxInv;
|
||
f32 zzInv;
|
||
|
||
if (poly == NULL) {
|
||
return;
|
||
}
|
||
CollisionPoly_GetNormalF(poly, &nx, &ny, &nz);
|
||
|
||
xx = sqrtf(1.0f - SQ(nx));
|
||
if (!IS_ZERO(xx)) {
|
||
xxInv = 1.0f / xx;
|
||
zz = ny * xxInv;
|
||
yz = -(nz * xxInv);
|
||
} else {
|
||
zz = sqrtf(1.0f - SQ(ny));
|
||
if (1) {}
|
||
if (!IS_ZERO(zz)) {
|
||
zzInv = 1.0f / zz;
|
||
yz = nx * zzInv;
|
||
xx = -(nz * zzInv);
|
||
} else {
|
||
yz = 0.0f;
|
||
xx = 0.0f;
|
||
}
|
||
}
|
||
dest->xx = xx;
|
||
dest->yx = -nx * zz;
|
||
dest->zx = nx * yz;
|
||
dest->xy = nx;
|
||
dest->yy = ny;
|
||
dest->zy = nz;
|
||
dest->yz = yz;
|
||
dest->zz = zz;
|
||
dest->wx = 0.0f;
|
||
dest->wy = 0.0f;
|
||
dest->xz = 0.0f;
|
||
dest->wz = 0.0f;
|
||
dest->xw = tx;
|
||
dest->yw = ty;
|
||
dest->zw = tz;
|
||
dest->ww = 1.0f;
|
||
}
|
||
|
||
/**
|
||
* Calculate point distance from plane along normal
|
||
*/
|
||
f32 CollisionPoly_GetPointDistanceFromPlane(CollisionPoly* poly, Vec3f* point) {
|
||
return (poly->normal.x * point->x + poly->normal.y * point->y + poly->normal.z * point->z) * COLPOLY_NORMAL_FRAC +
|
||
poly->dist;
|
||
}
|
||
|
||
/**
|
||
* Get Poly Vertices
|
||
*/
|
||
void CollisionPoly_GetVertices(CollisionPoly* poly, Vec3s* vtxList, Vec3f* dest) {
|
||
BgCheck_Vec3sToVec3f(&vtxList[COLPOLY_VTX_INDEX(poly->flags_vIA)], &dest[0]);
|
||
BgCheck_Vec3sToVec3f(&vtxList[COLPOLY_VTX_INDEX(poly->flags_vIB)], &dest[1]);
|
||
BgCheck_Vec3sToVec3f(&vtxList[poly->vIC], &dest[2]);
|
||
}
|
||
|
||
/**
|
||
* Get vertices by bgId
|
||
* original name: T_Polygon_GetVertex_bg_ai
|
||
*/
|
||
void CollisionPoly_GetVerticesByBgId(CollisionPoly* poly, s32 bgId, CollisionContext* colCtx, Vec3f* dest) {
|
||
Vec3s* vtxList;
|
||
|
||
if (poly == NULL || bgId > BG_ACTOR_MAX || dest == NULL) {
|
||
PRINTF(VT_COL(RED, WHITE));
|
||
PRINTF(T("T_Polygon_GetVertex_bg_ai(): Error %d %d %d 引数が適切ではありません。処理を終了します。\n",
|
||
"T_Polygon_GetVertex_bg_ai(): Error %d %d %d Argument not appropriate. Processing terminated.\n"),
|
||
poly == NULL, bgId > BG_ACTOR_MAX, dest == NULL);
|
||
PRINTF(VT_RST);
|
||
|
||
if (dest != NULL) {
|
||
//! @bug: dest[2] x and y are not set to 0
|
||
dest[0].x = dest[0].y = dest[0].z = dest[1].x = dest[1].y = dest[1].z = dest[2].z = 0.0f;
|
||
}
|
||
} else {
|
||
if (bgId == BGCHECK_SCENE) {
|
||
vtxList = colCtx->colHeader->vtxList;
|
||
} else {
|
||
vtxList = colCtx->dyna.vtxList;
|
||
}
|
||
|
||
CollisionPoly_GetVertices(poly, vtxList, dest);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Checks if point (`x`,`z`) is within `chkDist` of `poly`, computing `yIntersect` if true
|
||
* Determinant max 300.0f
|
||
*/
|
||
s32 CollisionPoly_CheckYIntersectApprox1(CollisionPoly* poly, Vec3s* vtxList, f32 x, f32 z, f32* yIntersect,
|
||
f32 chkDist) {
|
||
static Vec3f polyVerts[3];
|
||
f32 nx;
|
||
f32 ny;
|
||
f32 nz;
|
||
Vec3s* vA;
|
||
Vec3s* vB;
|
||
Vec3s* vC;
|
||
|
||
vA = &vtxList[COLPOLY_VTX_INDEX(poly->flags_vIA)];
|
||
Math_Vec3s_ToVec3f(&polyVerts[0], vA);
|
||
vB = &vtxList[COLPOLY_VTX_INDEX(poly->flags_vIB)];
|
||
Math_Vec3s_ToVec3f(&polyVerts[1], vB);
|
||
vC = &vtxList[poly->vIC];
|
||
Math_Vec3s_ToVec3f(&polyVerts[2], vC);
|
||
|
||
nx = COLPOLY_GET_NORMAL(poly->normal.x);
|
||
ny = COLPOLY_GET_NORMAL(poly->normal.y);
|
||
nz = COLPOLY_GET_NORMAL(poly->normal.z);
|
||
|
||
return Math3D_TriChkPointParaYIntersectDist(&polyVerts[0], &polyVerts[1], &polyVerts[2], nx, ny, nz, poly->dist, z,
|
||
x, yIntersect, chkDist);
|
||
}
|
||
|
||
/**
|
||
* Checks if point (`x`,`z`) is within `chkDist` of `poly`, computing `yIntersect` if true
|
||
* Determinant max 0.0f (checks if on or within poly)
|
||
*/
|
||
s32 CollisionPoly_CheckYIntersect(CollisionPoly* poly, Vec3s* vtxList, f32 x, f32 z, f32* yIntersect, f32 chkDist) {
|
||
static Vec3f polyVerts[3];
|
||
f32 nx;
|
||
f32 ny;
|
||
f32 nz;
|
||
|
||
CollisionPoly_GetVertices(poly, vtxList, polyVerts);
|
||
CollisionPoly_GetNormalF(poly, &nx, &ny, &nz);
|
||
return Math3D_TriChkPointParaYIntersectInsideTri(&polyVerts[0], &polyVerts[1], &polyVerts[2], nx, ny, nz,
|
||
poly->dist, z, x, yIntersect, chkDist);
|
||
}
|
||
|
||
/**
|
||
* Checks if point (`x`,`z`) is within 1.0f of `poly`, computing `yIntersect` if true
|
||
* Determinant max 300.0f
|
||
*/
|
||
s32 CollisionPoly_CheckYIntersectApprox2(CollisionPoly* poly, Vec3s* vtxList, f32 x, f32 z, f32* yIntersect) {
|
||
return CollisionPoly_CheckYIntersectApprox1(poly, vtxList, x, z, yIntersect, 1.0f);
|
||
}
|
||
|
||
/**
|
||
* Checks if point (`y`,`z`) is within 1.0f of `poly`, computing `xIntersect` if true
|
||
* Determinant max 300.0f
|
||
*/
|
||
s32 CollisionPoly_CheckXIntersectApprox(CollisionPoly* poly, Vec3s* vtxList, f32 y, f32 z, f32* xIntersect) {
|
||
static Vec3f polyVerts[3];
|
||
f32 nx;
|
||
f32 ny;
|
||
f32 nz;
|
||
|
||
CollisionPoly_GetVertices(poly, vtxList, polyVerts);
|
||
CollisionPoly_GetNormalF(poly, &nx, &ny, &nz);
|
||
return Math3D_TriChkPointParaXIntersect(&polyVerts[0], &polyVerts[1], &polyVerts[2], nx, ny, nz, poly->dist, y, z,
|
||
xIntersect);
|
||
}
|
||
|
||
/**
|
||
* Checks if point (`x`,`y`) is within 1.0f of `poly`, computing `zIntersect` if true
|
||
* Determinant max 300.0f
|
||
*/
|
||
s32 CollisionPoly_CheckZIntersectApprox(CollisionPoly* poly, Vec3s* vtxList, f32 x, f32 y, f32* zIntersect) {
|
||
static Vec3f polyVerts[3];
|
||
f32 nx;
|
||
f32 ny;
|
||
f32 nz;
|
||
|
||
CollisionPoly_GetVertices(poly, vtxList, polyVerts);
|
||
CollisionPoly_GetNormalF(poly, &nx, &ny, &nz);
|
||
return Math3D_TriChkPointParaZIntersect(&polyVerts[0], &polyVerts[1], &polyVerts[2], nx, ny, nz, poly->dist, x, y,
|
||
zIntersect);
|
||
}
|
||
|
||
/**
|
||
* Test if travelling from `posA` to `posB` intersects `poly`
|
||
* returns true if an intersection occurs, else false
|
||
* returns `planeIntersect`, which is the point at which the line from `posA` to `posB` crosses `poly`'s plane
|
||
* if `chkOneFace` is true, return false (no intersection) when going through the poly from A to B is done in the
|
||
* normal's direction
|
||
*/
|
||
s32 CollisionPoly_LineVsPoly(CollisionPoly* poly, Vec3s* vtxList, Vec3f* posA, Vec3f* posB, Vec3f* planeIntersect,
|
||
s32 chkOneFace, f32 chkDist) {
|
||
static Vec3f polyVerts[3];
|
||
static Plane plane;
|
||
f32 planeDistA;
|
||
f32 planeDistB;
|
||
f32 planeDistDelta;
|
||
|
||
plane.originDist = poly->dist;
|
||
planeDistA =
|
||
(poly->normal.x * posA->x + poly->normal.y * posA->y + poly->normal.z * posA->z) * COLPOLY_NORMAL_FRAC +
|
||
plane.originDist;
|
||
planeDistB =
|
||
(poly->normal.x * posB->x + poly->normal.y * posB->y + poly->normal.z * posB->z) * COLPOLY_NORMAL_FRAC +
|
||
plane.originDist;
|
||
|
||
planeDistDelta = planeDistA - planeDistB;
|
||
if ((planeDistA >= 0.0f && planeDistB >= 0.0f) || (planeDistA < 0.0f && planeDistB < 0.0f) ||
|
||
(chkOneFace && planeDistA < 0.0f && planeDistB > 0.0f) || IS_ZERO(planeDistDelta)) {
|
||
return false;
|
||
}
|
||
|
||
CollisionPoly_GetNormalF(poly, &plane.normal.x, &plane.normal.y, &plane.normal.z);
|
||
CollisionPoly_GetVertices(poly, vtxList, polyVerts);
|
||
Math3D_LineSplitRatio(posA, posB, planeDistA / planeDistDelta, planeIntersect);
|
||
if ((fabsf(plane.normal.x) > 0.5f &&
|
||
Math3D_TriChkPointParaXDist(&polyVerts[0], &polyVerts[1], &polyVerts[2], &plane, planeIntersect->y,
|
||
planeIntersect->z, chkDist)) ||
|
||
(fabsf(plane.normal.y) > 0.5f &&
|
||
Math3D_TriChkPointParaYDist(&polyVerts[0], &polyVerts[1], &polyVerts[2], &plane, planeIntersect->z,
|
||
planeIntersect->x, chkDist)) ||
|
||
(fabsf(plane.normal.z) > 0.5f &&
|
||
Math3D_TriChkLineSegParaZDist(&polyVerts[0], &polyVerts[1], &polyVerts[2], &plane, planeIntersect->x,
|
||
planeIntersect->y, chkDist))) {
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Tests if sphere `center` `radius` intersects `poly`
|
||
*/
|
||
s32 CollisionPoly_SphVsPoly(CollisionPoly* poly, Vec3s* vtxList, Vec3f* center, f32 radius) {
|
||
static Sphere16 sphere;
|
||
static TriNorm tri;
|
||
Vec3f intersect;
|
||
|
||
CollisionPoly_GetVertices(poly, vtxList, tri.vtx);
|
||
CollisionPoly_GetNormalF(poly, &tri.plane.normal.x, &tri.plane.normal.y, &tri.plane.normal.z);
|
||
tri.plane.originDist = poly->dist;
|
||
sphere.center.x = center->x;
|
||
sphere.center.y = center->y;
|
||
sphere.center.z = center->z;
|
||
sphere.radius = radius;
|
||
return Math3D_TriVsSphIntersect(&sphere, &tri, &intersect);
|
||
}
|
||
|
||
/**
|
||
* Add poly to StaticLookup table
|
||
* Table is sorted by poly's smallest y vertex component
|
||
* `ssList` is the list to append a new poly to
|
||
* `polyList` is the CollisionPoly lookup list
|
||
* `vtxList` is the vertex lookup list
|
||
* `polyId` is the index of the poly in polyList to insert into the lookup table
|
||
*/
|
||
void StaticLookup_AddPolyToSSList(CollisionContext* colCtx, SSList* ssList, CollisionPoly* polyList, Vec3s* vtxList,
|
||
s16 polyId) {
|
||
SSNode* curNode;
|
||
SSNode* nextNode;
|
||
s32 polyYMin;
|
||
s16 curPolyId;
|
||
u16 newNodeId;
|
||
|
||
// if list is null
|
||
if (ssList->head == SS_NULL) {
|
||
SSNodeList_SetSSListHead(&colCtx->polyNodes, ssList, &polyId);
|
||
return;
|
||
}
|
||
|
||
polyYMin = CollisionPoly_GetMinY(&polyList[polyId], vtxList);
|
||
curNode = &colCtx->polyNodes.tbl[ssList->head];
|
||
curPolyId = curNode->polyId;
|
||
|
||
// if the poly being inserted has a lower y than the first poly
|
||
if (polyYMin < vtxList[COLPOLY_VTX_INDEX(polyList[curPolyId].flags_vIA)].y &&
|
||
polyYMin < vtxList[COLPOLY_VTX_INDEX(polyList[curPolyId].flags_vIB)].y &&
|
||
polyYMin < vtxList[polyList[curPolyId].vIC].y) {
|
||
SSNodeList_SetSSListHead(&colCtx->polyNodes, ssList, &polyId);
|
||
return;
|
||
}
|
||
while (true) {
|
||
// if at the end of the list
|
||
if (curNode->next == SS_NULL) {
|
||
s32 pad;
|
||
|
||
newNodeId = SSNodeList_GetNextNodeIdx(&colCtx->polyNodes);
|
||
SSNode_SetValue(&colCtx->polyNodes.tbl[newNodeId], &polyId, SS_NULL);
|
||
curNode->next = newNodeId;
|
||
return;
|
||
}
|
||
|
||
nextNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
curPolyId = nextNode->polyId;
|
||
|
||
// if the poly being inserted is lower than the next poly
|
||
if (polyYMin < vtxList[COLPOLY_VTX_INDEX(polyList[curPolyId].flags_vIA)].y &&
|
||
polyYMin < vtxList[COLPOLY_VTX_INDEX(polyList[curPolyId].flags_vIB)].y &&
|
||
polyYMin < vtxList[polyList[curPolyId].vIC].y) {
|
||
newNodeId = SSNodeList_GetNextNodeIdx(&colCtx->polyNodes);
|
||
SSNode_SetValue(&colCtx->polyNodes.tbl[newNodeId], &polyId, curNode->next);
|
||
curNode->next = newNodeId;
|
||
return;
|
||
}
|
||
curNode = nextNode;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Add CollisionPoly to StaticLookup list
|
||
*/
|
||
void StaticLookup_AddPoly(StaticLookup* lookup, CollisionContext* colCtx, CollisionPoly* polyList, Vec3s* vtxList,
|
||
s16 index) {
|
||
if (polyList[index].normal.y > COLPOLY_SNORMAL(0.5f)) {
|
||
StaticLookup_AddPolyToSSList(colCtx, &lookup->floor, polyList, vtxList, index);
|
||
} else if (polyList[index].normal.y < COLPOLY_SNORMAL(-0.8f)) {
|
||
StaticLookup_AddPolyToSSList(colCtx, &lookup->ceiling, polyList, vtxList, index);
|
||
} else {
|
||
StaticLookup_AddPolyToSSList(colCtx, &lookup->wall, polyList, vtxList, index);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Locates the closest static poly directly underneath `pos`, starting at list `ssList`
|
||
* returns yIntersect of the closest poly, or `yIntersectMin`
|
||
* stores the pointer of the closest poly to `outPoly`
|
||
* if BGCHECK_GROUND_CHECK_ON is set, ignore polys with a normal.y < 0 (from vertical walls to ceilings)
|
||
*/
|
||
f32 BgCheck_RaycastDownStaticList(CollisionContext* colCtx, u16 xpFlags, SSList* ssList, CollisionPoly** outPoly,
|
||
Vec3f* pos, f32 yIntersectMin, f32 chkDist, s32 groundChk) {
|
||
SSNode* curNode;
|
||
s32 polyId;
|
||
f32 result;
|
||
f32 yIntersect;
|
||
|
||
result = yIntersectMin;
|
||
if (ssList->head == SS_NULL) {
|
||
return result;
|
||
}
|
||
|
||
curNode = &colCtx->polyNodes.tbl[ssList->head];
|
||
|
||
while (true) {
|
||
polyId = curNode->polyId;
|
||
|
||
if (COLPOLY_VTX_CHECK_FLAGS_ANY(colCtx->colHeader->polyList[polyId].flags_vIA, xpFlags) ||
|
||
((groundChk & BGCHECK_GROUND_CHECK_ON) && colCtx->colHeader->polyList[polyId].normal.y < 0)) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
}
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
|
||
if (pos->y < colCtx->colHeader->vtxList[COLPOLY_VTX_INDEX(colCtx->colHeader->polyList[polyId].flags_vIA)].y &&
|
||
pos->y < colCtx->colHeader->vtxList[COLPOLY_VTX_INDEX(colCtx->colHeader->polyList[polyId].flags_vIB)].y &&
|
||
pos->y < colCtx->colHeader->vtxList[colCtx->colHeader->polyList[polyId].vIC].y) {
|
||
break;
|
||
}
|
||
|
||
if (CollisionPoly_CheckYIntersect(&colCtx->colHeader->polyList[polyId], colCtx->colHeader->vtxList, pos->x,
|
||
pos->z, &yIntersect, chkDist) == true) {
|
||
// if poly is closer to pos without going over
|
||
if (yIntersect < pos->y && result < yIntersect) {
|
||
result = yIntersect;
|
||
*outPoly = &colCtx->colHeader->polyList[polyId];
|
||
}
|
||
}
|
||
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
}
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Locates the closest static poly directly underneath `pos` within `lookup`.
|
||
* returns yIntersect of the closest poly, or `yIntersectMin`
|
||
* stores the pointer of the closest poly to `outPoly`
|
||
*/
|
||
f32 BgCheck_RaycastDownStatic(StaticLookup* lookup, CollisionContext* colCtx, u16 xpFlags, CollisionPoly** poly,
|
||
Vec3f* pos, u32 downChkFlags, f32 chkDist, f32 yIntersectMin) {
|
||
f32 yIntersect = yIntersectMin;
|
||
s32 groundChk;
|
||
|
||
if (downChkFlags & BGCHECK_RAYCAST_DOWN_CHECK_FLOORS) {
|
||
yIntersect = BgCheck_RaycastDownStaticList(colCtx, xpFlags, &lookup->floor, poly, pos, yIntersect, chkDist, 0);
|
||
}
|
||
|
||
if ((downChkFlags & BGCHECK_RAYCAST_DOWN_CHECK_WALLS) || (downChkFlags & BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE)) {
|
||
groundChk = 0;
|
||
if (downChkFlags & BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY) {
|
||
groundChk |= BGCHECK_GROUND_CHECK_ON;
|
||
}
|
||
yIntersect =
|
||
BgCheck_RaycastDownStaticList(colCtx, xpFlags, &lookup->wall, poly, pos, yIntersect, chkDist, groundChk);
|
||
}
|
||
|
||
if (downChkFlags & BGCHECK_RAYCAST_DOWN_CHECK_CEILINGS) {
|
||
groundChk = 0;
|
||
if (downChkFlags & BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY) {
|
||
groundChk |= BGCHECK_GROUND_CHECK_ON;
|
||
}
|
||
yIntersect =
|
||
BgCheck_RaycastDownStaticList(colCtx, xpFlags, &lookup->ceiling, poly, pos, yIntersect, chkDist, groundChk);
|
||
}
|
||
|
||
return yIntersect;
|
||
}
|
||
|
||
/**
|
||
* Compute wall displacement on `posX` and `posZ`
|
||
* sets `wallPolyPtr` to `poly` if `wallPolyPtr` is NULL or doesn't have `surfaceData[0] & 0x08000000` set
|
||
* returns true if `wallPolyPtr` was changed
|
||
* `invXZlength` is 1 / sqrt( sq(poly.normal.x) + sq(poly.normal.z) )
|
||
*/
|
||
s32 BgCheck_ComputeWallDisplacement(CollisionContext* colCtx, CollisionPoly* poly, f32* posX, f32* posZ, f32 nx, f32 ny,
|
||
f32 nz, f32 invXZlength, f32 planeDist, f32 radius, CollisionPoly** wallPolyPtr) {
|
||
CollisionPoly* wallPoly;
|
||
u32 surfaceData;
|
||
u32 hasFlag27;
|
||
f32 displacement = (radius - planeDist) * invXZlength;
|
||
|
||
*posX += displacement * nx;
|
||
*posZ += displacement * nz;
|
||
|
||
wallPoly = *wallPolyPtr;
|
||
if (wallPoly == NULL) {
|
||
*wallPolyPtr = poly;
|
||
return true;
|
||
}
|
||
|
||
surfaceData = colCtx->colHeader->surfaceTypeList[wallPoly->type].data[1];
|
||
hasFlag27 = surfaceData & 0x08000000 ? 1 : 0;
|
||
|
||
if (!hasFlag27) {
|
||
*wallPolyPtr = poly;
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Performs collision detection on static poly walls within `lookup` on sphere `pos`, `radius`
|
||
* returns true if a collision was detected
|
||
* `outX` `outZ` return the displaced x,z coordinates,
|
||
* `outPoly` returns the pointer to the nearest poly collided with, or NULL
|
||
*/
|
||
s32 BgCheck_SphVsStaticWall(StaticLookup* lookup, CollisionContext* colCtx, u16 xpFlags, f32* outX, f32* outZ,
|
||
Vec3f* pos, f32 radius, CollisionPoly** outPoly) {
|
||
Vec3f resultPos;
|
||
f32 zTemp;
|
||
f32 xTemp;
|
||
f32 planeDist;
|
||
f32 intersect;
|
||
s32 result;
|
||
CollisionPoly* curPoly;
|
||
CollisionPoly* polyList;
|
||
SSNode* curNode;
|
||
f32 invNormalXZ;
|
||
s32 polyId;
|
||
f32 normalXZ;
|
||
f32 nx;
|
||
f32 ny;
|
||
f32 nz;
|
||
s32 pad;
|
||
Vec3s* vtxList;
|
||
f32 temp_f16;
|
||
f32 zMin;
|
||
f32 zMax;
|
||
f32 xMin;
|
||
f32 xMax;
|
||
|
||
result = false;
|
||
if (lookup->wall.head == SS_NULL) {
|
||
return result;
|
||
}
|
||
resultPos = *pos;
|
||
|
||
polyList = colCtx->colHeader->polyList;
|
||
vtxList = colCtx->colHeader->vtxList;
|
||
curNode = &colCtx->polyNodes.tbl[lookup->wall.head];
|
||
|
||
while (true) {
|
||
polyId = curNode->polyId;
|
||
curPoly = &polyList[polyId];
|
||
if (pos->y < vtxList[COLPOLY_VTX_INDEX(curPoly->flags_vIA)].y &&
|
||
pos->y < vtxList[COLPOLY_VTX_INDEX(curPoly->flags_vIB)].y && pos->y < vtxList[curPoly->vIC].y) {
|
||
break;
|
||
}
|
||
|
||
nx = COLPOLY_GET_NORMAL(curPoly->normal.x);
|
||
ny = COLPOLY_GET_NORMAL(curPoly->normal.y);
|
||
nz = COLPOLY_GET_NORMAL(curPoly->normal.z);
|
||
normalXZ = sqrtf(SQ(nx) + SQ(nz));
|
||
planeDist = Math3D_DistPlaneToPos(nx, ny, nz, curPoly->dist, &resultPos);
|
||
if (radius < fabsf(planeDist) || COLPOLY_VTX_CHECK_FLAGS_ANY(curPoly->flags_vIA, xpFlags)) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
|
||
ASSERT(!IS_ZERO(normalXZ), "!IS_ZERO(ac_size)", "../z_bgcheck.c", 2854);
|
||
|
||
invNormalXZ = 1.0f / normalXZ;
|
||
temp_f16 = fabsf(nz) * invNormalXZ;
|
||
if (temp_f16 < 0.4f) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
|
||
// compute curPoly zMin/zMax
|
||
zTemp = vtxList[COLPOLY_VTX_INDEX(curPoly->flags_vIA)].z;
|
||
zMax = zMin = zTemp;
|
||
zTemp = vtxList[COLPOLY_VTX_INDEX(curPoly->flags_vIB)].z;
|
||
|
||
if (zTemp < zMin) {
|
||
zMin = zTemp;
|
||
} else if (zMax < zTemp) {
|
||
zMax = zTemp;
|
||
}
|
||
zTemp = vtxList[curPoly->vIC].z;
|
||
if (zTemp < zMin) {
|
||
zMin = zTemp;
|
||
} else if (zTemp > zMax) {
|
||
zMax = zTemp;
|
||
}
|
||
|
||
zMin -= radius;
|
||
zMax += radius;
|
||
|
||
if (resultPos.z < zMin || resultPos.z > zMax) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
if (CollisionPoly_CheckZIntersectApprox(curPoly, vtxList, resultPos.x, pos->y, &intersect)) {
|
||
f32 zIntersectDist = intersect - resultPos.z;
|
||
|
||
if (fabsf(zIntersectDist) <= radius / temp_f16) {
|
||
if (zIntersectDist * nz <= 4.0f) {
|
||
BgCheck_ComputeWallDisplacement(colCtx, curPoly, &resultPos.x, &resultPos.z, nx, ny, nz,
|
||
invNormalXZ, planeDist, radius, outPoly);
|
||
result = true;
|
||
}
|
||
}
|
||
}
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
}
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
}
|
||
|
||
curNode = &colCtx->polyNodes.tbl[lookup->wall.head];
|
||
|
||
while (true) {
|
||
polyId = curNode->polyId;
|
||
curPoly = &polyList[polyId];
|
||
if (pos->y < vtxList[COLPOLY_VTX_INDEX(curPoly->flags_vIA)].y &&
|
||
pos->y < vtxList[COLPOLY_VTX_INDEX(curPoly->flags_vIB)].y && pos->y < vtxList[curPoly->vIC].y) {
|
||
break;
|
||
}
|
||
|
||
nx = COLPOLY_GET_NORMAL(curPoly->normal.x);
|
||
ny = COLPOLY_GET_NORMAL(curPoly->normal.y);
|
||
nz = COLPOLY_GET_NORMAL(curPoly->normal.z);
|
||
normalXZ = sqrtf(SQ(nx) + SQ(nz));
|
||
planeDist = Math3D_DistPlaneToPos(nx, ny, nz, curPoly->dist, &resultPos);
|
||
if (radius < fabsf(planeDist) || COLPOLY_VTX_CHECK_FLAGS_ANY(curPoly->flags_vIA, xpFlags)) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
|
||
ASSERT(!IS_ZERO(normalXZ), "!IS_ZERO(ac_size)", "../z_bgcheck.c", 2964);
|
||
|
||
invNormalXZ = 1.0f / normalXZ;
|
||
temp_f16 = fabsf(nx) * invNormalXZ;
|
||
if (temp_f16 < 0.4f) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
|
||
// compute curPoly xMin/xMax
|
||
xTemp = vtxList[COLPOLY_VTX_INDEX(curPoly->flags_vIA)].x;
|
||
xMax = xMin = xTemp;
|
||
xTemp = vtxList[COLPOLY_VTX_INDEX(curPoly->flags_vIB)].x;
|
||
|
||
if (xTemp < xMin) {
|
||
xMin = xTemp;
|
||
} else if (xMax < xTemp) {
|
||
xMax = xTemp;
|
||
}
|
||
xTemp = vtxList[curPoly->vIC].x;
|
||
if (xTemp < xMin) {
|
||
xMin = xTemp;
|
||
} else if (xMax < xTemp) {
|
||
xMax = xTemp;
|
||
}
|
||
|
||
xMin -= radius;
|
||
xMax += radius;
|
||
|
||
if (resultPos.x < xMin || xMax < resultPos.x) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
if (CollisionPoly_CheckXIntersectApprox(curPoly, vtxList, pos->y, resultPos.z, &intersect)) {
|
||
f32 xIntersectDist = intersect - resultPos.x;
|
||
|
||
if (fabsf(xIntersectDist) <= radius / temp_f16) {
|
||
if (xIntersectDist * nx <= 4.0f) {
|
||
BgCheck_ComputeWallDisplacement(colCtx, curPoly, &resultPos.x, &resultPos.z, nx, ny, nz,
|
||
invNormalXZ, planeDist, radius, outPoly);
|
||
result = true;
|
||
}
|
||
}
|
||
}
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
}
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
}
|
||
|
||
*outX = resultPos.x;
|
||
*outZ = resultPos.z;
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Tests for collision with a static poly ceiling
|
||
* returns true if a collision occurs, else false
|
||
* `outPoly` returns the poly collided with
|
||
* `outY` returns the y coordinate needed to not collide with `outPoly`
|
||
*/
|
||
s32 BgCheck_CheckStaticCeiling(StaticLookup* lookup, u16 xpFlags, CollisionContext* colCtx, f32* outY, Vec3f* pos,
|
||
f32 checkHeight, CollisionPoly** outPoly) {
|
||
s32 result = false;
|
||
u16 nextId;
|
||
CollisionPoly* curPoly;
|
||
CollisionPoly* polyList;
|
||
f32 ceilingY;
|
||
Vec3s* vtxList;
|
||
SSNode* curNode;
|
||
s32 curPolyId;
|
||
|
||
if (lookup->ceiling.head == SS_NULL) {
|
||
return false;
|
||
}
|
||
curNode = &colCtx->polyNodes.tbl[lookup->ceiling.head];
|
||
polyList = colCtx->colHeader->polyList;
|
||
vtxList = colCtx->colHeader->vtxList;
|
||
|
||
*outY = pos->y;
|
||
|
||
while (true) {
|
||
f32 intersectDist;
|
||
f32 ny;
|
||
|
||
curPolyId = curNode->polyId;
|
||
if (COLPOLY_VTX_CHECK_FLAGS_ANY(colCtx->colHeader->polyList[curPolyId].flags_vIA, xpFlags)) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
curPoly = &polyList[curPolyId];
|
||
|
||
if (CollisionPoly_CheckYIntersectApprox2(curPoly, vtxList, pos->x, pos->z, &ceilingY)) {
|
||
intersectDist = ceilingY - *outY;
|
||
ny = COLPOLY_GET_NORMAL(curPoly->normal.y);
|
||
|
||
if (intersectDist > 0.0f && intersectDist < checkHeight && intersectDist * ny <= 0) {
|
||
*outY = ceilingY - checkHeight;
|
||
*outPoly = curPoly;
|
||
result = true;
|
||
}
|
||
}
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Tests if line `posA` to `posB` intersects with a static poly in list `ssList`. Uses polyCheckTbl
|
||
* returns true if such a poly exists, else false
|
||
* `outPoly` returns the pointer of the poly intersected
|
||
* `posB` and `outPos` returns the point of intersection with `outPoly`
|
||
* `outDistSq` returns the squared distance from `posA` to the point of intersect
|
||
*/
|
||
s32 BgCheck_CheckLineAgainstSSList(SSList* ssList, CollisionContext* colCtx, u16 xpFlags1, u16 xpFlags2, Vec3f* posA,
|
||
Vec3f* posB, Vec3f* outPos, CollisionPoly** outPoly, f32* outDistSq, f32 chkDist,
|
||
s32 bccFlags) {
|
||
SSNode* curNode;
|
||
u8* checkedPoly;
|
||
Vec3f polyIntersect;
|
||
CollisionPoly* polyList;
|
||
CollisionPoly* curPoly;
|
||
s32 result;
|
||
f32 minY;
|
||
f32 distSq;
|
||
|
||
result = false;
|
||
polyList = colCtx->colHeader->polyList;
|
||
if (ssList->head == SS_NULL) {
|
||
return result;
|
||
}
|
||
|
||
curNode = &colCtx->polyNodes.tbl[ssList->head];
|
||
while (true) {
|
||
s16 polyId = curNode->polyId;
|
||
|
||
checkedPoly = &colCtx->polyNodes.polyCheckTbl[polyId];
|
||
|
||
if (*checkedPoly == true || COLPOLY_VTX_CHECK_FLAGS_ANY(polyList[polyId].flags_vIA, xpFlags1) ||
|
||
!(xpFlags2 == 0 || COLPOLY_VTX_CHECK_FLAGS_ANY(polyList[polyId].flags_vIA, xpFlags2))) {
|
||
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
*checkedPoly = true;
|
||
curPoly = &polyList[polyId];
|
||
minY = CollisionPoly_GetMinY(curPoly, colCtx->colHeader->vtxList);
|
||
if (posA->y < minY && posB->y < minY) {
|
||
break;
|
||
}
|
||
if (CollisionPoly_LineVsPoly(curPoly, colCtx->colHeader->vtxList, posA, posB, &polyIntersect,
|
||
(bccFlags & BGCHECK_CHECK_ONE_FACE) != 0, chkDist)) {
|
||
distSq = Math3D_Vec3fDistSq(posA, &polyIntersect);
|
||
if (distSq < *outDistSq) {
|
||
|
||
*outDistSq = distSq;
|
||
*outPos = polyIntersect;
|
||
*posB = polyIntersect;
|
||
*outPoly = curPoly;
|
||
result = true;
|
||
}
|
||
}
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
}
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Tests if line `posA` to `posB` intersects with a static poly in `lookup`. Uses polyCheckTbl
|
||
* returns true if such a poly exists, else false
|
||
* `outPoly` returns the pointer of the poly intersected
|
||
* `posB` and `outPos` returns the point of intersection with `outPoly`
|
||
* `outDistSq` returns the squared distance from `posA` to the point of intersect
|
||
*/
|
||
s32 BgCheck_CheckLineInSubdivision(StaticLookup* lookup, CollisionContext* colCtx, u16 xpFlags1, u16 xpFlags2,
|
||
Vec3f* posA, Vec3f* posB, Vec3f* outPos, CollisionPoly** outPoly, f32 chkDist,
|
||
f32* outDistSq, u32 bccFlags) {
|
||
s32 result = false;
|
||
|
||
if ((bccFlags & BGCHECK_CHECK_FLOOR) && lookup->floor.head != SS_NULL) {
|
||
if (BgCheck_CheckLineAgainstSSList(&lookup->floor, colCtx, xpFlags1, xpFlags2, posA, posB, outPos, outPoly,
|
||
outDistSq, chkDist, bccFlags)) {
|
||
result = true;
|
||
}
|
||
}
|
||
|
||
if ((bccFlags & BGCHECK_CHECK_WALL) && lookup->wall.head != SS_NULL) {
|
||
if (BgCheck_CheckLineAgainstSSList(&lookup->wall, colCtx, xpFlags1, xpFlags2, posA, posB, outPos, outPoly,
|
||
outDistSq, chkDist, bccFlags)) {
|
||
result = true;
|
||
}
|
||
}
|
||
|
||
if ((bccFlags & BGCHECK_CHECK_CEILING) && lookup->ceiling.head != SS_NULL) {
|
||
if (BgCheck_CheckLineAgainstSSList(&lookup->ceiling, colCtx, xpFlags1, xpFlags2, posA, posB, outPos, outPoly,
|
||
outDistSq, chkDist, bccFlags)) {
|
||
result = true;
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Get first static poly intersecting sphere `center` `radius` from list `node`
|
||
* returns true if any poly intersects the sphere, else returns false
|
||
* `outPoly` returns the pointer of the first poly found that intersects
|
||
*/
|
||
s32 BgCheck_SphVsFirstStaticPolyList(SSNode* node, u16 xpFlags, CollisionContext* colCtx, Vec3f* center, f32 radius,
|
||
CollisionPoly** outPoly) {
|
||
CollisionPoly* polyList = colCtx->colHeader->polyList;
|
||
Vec3s* vtxList = colCtx->colHeader->vtxList;
|
||
|
||
while (true) {
|
||
u16 nextId;
|
||
s16 curPolyId = node->polyId;
|
||
CollisionPoly* curPoly = &polyList[curPolyId];
|
||
|
||
if (COLPOLY_VTX_CHECK_FLAGS_ANY(colCtx->colHeader->polyList[curPolyId].flags_vIA, xpFlags)) {
|
||
if (node->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
node = &colCtx->polyNodes.tbl[node->next];
|
||
continue;
|
||
}
|
||
}
|
||
|
||
if (center->y + radius < vtxList[COLPOLY_VTX_INDEX(curPoly->flags_vIA)].y &&
|
||
center->y + radius < vtxList[COLPOLY_VTX_INDEX(curPoly->flags_vIB)].y &&
|
||
center->y + radius < vtxList[curPoly->vIC].y) {
|
||
break;
|
||
}
|
||
|
||
if (CollisionPoly_SphVsPoly(curPoly, vtxList, center, radius)) {
|
||
*outPoly = curPoly;
|
||
return true;
|
||
}
|
||
if (node->next == SS_NULL) {
|
||
break;
|
||
}
|
||
node = &colCtx->polyNodes.tbl[node->next];
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Get first static poly intersecting sphere `center` `radius` within `lookup`
|
||
* returns true if any poly intersects the sphere, else false
|
||
* `outPoly` returns the first poly found that intersects
|
||
*/
|
||
s32 BgCheck_SphVsFirstStaticPoly(StaticLookup* lookup, u16 xpFlags, CollisionContext* colCtx, Vec3f* center, f32 radius,
|
||
CollisionPoly** outPoly, u16 bciFlags) {
|
||
if (lookup->floor.head != SS_NULL && !(bciFlags & BGCHECK_IGNORE_FLOOR) &&
|
||
BgCheck_SphVsFirstStaticPolyList(&colCtx->polyNodes.tbl[lookup->floor.head], xpFlags, colCtx, center, radius,
|
||
outPoly)) {
|
||
return true;
|
||
}
|
||
|
||
if (lookup->wall.head != SS_NULL && !(bciFlags & BGCHECK_IGNORE_WALL) &&
|
||
BgCheck_SphVsFirstStaticPolyList(&colCtx->polyNodes.tbl[lookup->wall.head], xpFlags, colCtx, center, radius,
|
||
outPoly)) {
|
||
return true;
|
||
}
|
||
|
||
if (lookup->ceiling.head != SS_NULL && !(bciFlags & BGCHECK_IGNORE_CEILING) &&
|
||
BgCheck_SphVsFirstStaticPolyList(&colCtx->polyNodes.tbl[lookup->ceiling.head], xpFlags, colCtx, center, radius,
|
||
outPoly)) {
|
||
return true;
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Get StaticLookup from `pos`
|
||
* Does not return NULL
|
||
*/
|
||
StaticLookup* BgCheck_GetNearestStaticLookup(CollisionContext* colCtx, StaticLookup* lookupTbl, Vec3f* pos) {
|
||
Vec3i sector;
|
||
s32 subdivAmountX;
|
||
|
||
BgCheck_GetStaticLookupIndicesFromPos(colCtx, pos, §or);
|
||
subdivAmountX = colCtx->subdivAmount.x;
|
||
return (sector.z * subdivAmountX) * colCtx->subdivAmount.y + lookupTbl + sector.x + sector.y * subdivAmountX;
|
||
}
|
||
|
||
/**
|
||
* Get StaticLookup from `pos`
|
||
* Returns NULL if just outside the mesh bounding box
|
||
*/
|
||
StaticLookup* BgCheck_GetStaticLookup(CollisionContext* colCtx, StaticLookup* lookupTbl, Vec3f* pos) {
|
||
Vec3i sector;
|
||
s32 subdivAmountX;
|
||
|
||
if (!BgCheck_PosInStaticBoundingBox(colCtx, pos)) {
|
||
return NULL;
|
||
}
|
||
BgCheck_GetStaticLookupIndicesFromPos(colCtx, pos, §or);
|
||
subdivAmountX = colCtx->subdivAmount.x;
|
||
return (sector.z * subdivAmountX) * colCtx->subdivAmount.y + lookupTbl + sector.x + sector.y * subdivAmountX;
|
||
}
|
||
|
||
/**
|
||
* Get StaticLookup subdivision indices from `pos`
|
||
* `sector` returns the subdivision x,y,z indices containing or is nearest to `pos`
|
||
*/
|
||
void BgCheck_GetStaticLookupIndicesFromPos(CollisionContext* colCtx, Vec3f* pos, Vec3i* sector) {
|
||
sector->x = (pos->x - colCtx->minBounds.x) * colCtx->subdivLengthInv.x;
|
||
sector->y = (pos->y - colCtx->minBounds.y) * colCtx->subdivLengthInv.y;
|
||
sector->z = (pos->z - colCtx->minBounds.z) * colCtx->subdivLengthInv.z;
|
||
|
||
if (sector->x < 0) {
|
||
sector->x = 0;
|
||
} else if (sector->x >= colCtx->subdivAmount.x) {
|
||
sector->x = colCtx->subdivAmount.x - 1;
|
||
}
|
||
|
||
if (sector->y < 0) {
|
||
sector->y = 0;
|
||
} else if (sector->y >= colCtx->subdivAmount.y) {
|
||
sector->y = colCtx->subdivAmount.y - 1;
|
||
}
|
||
|
||
if (sector->z < 0) {
|
||
sector->z = 0;
|
||
} else if (sector->z >= colCtx->subdivAmount.z) {
|
||
sector->z = colCtx->subdivAmount.z - 1;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Get negative bias subdivision indices
|
||
* decrements indices if `pos` is within BGCHECK_SUBDIV_OVERLAP units of the negative subdivision boundary
|
||
* `sx`, `sy`, `sz` returns the subdivision x, y, z indices
|
||
*/
|
||
void BgCheck_GetSubdivisionMinBounds(CollisionContext* colCtx, Vec3f* pos, s32* sx, s32* sy, s32* sz) {
|
||
f32 dx = pos->x - colCtx->minBounds.x;
|
||
f32 dy = pos->y - colCtx->minBounds.y;
|
||
f32 dz = pos->z - colCtx->minBounds.z;
|
||
|
||
*sx = dx * colCtx->subdivLengthInv.x;
|
||
*sy = dy * colCtx->subdivLengthInv.y;
|
||
*sz = dz * colCtx->subdivLengthInv.z;
|
||
|
||
if (((s32)dx % (s32)colCtx->subdivLength.x < BGCHECK_SUBDIV_OVERLAP) && (*sx > 0)) {
|
||
*sx -= 1;
|
||
}
|
||
|
||
if (((s32)dy % (s32)colCtx->subdivLength.y < BGCHECK_SUBDIV_OVERLAP) && (*sy > 0)) {
|
||
*sy -= 1;
|
||
}
|
||
|
||
if (((s32)dz % (s32)colCtx->subdivLength.z < BGCHECK_SUBDIV_OVERLAP) && (*sz > 0)) {
|
||
*sz -= 1;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Get positive bias subdivision indices
|
||
* increments indices if `pos` is within BGCHECK_SUBDIV_OVERLAP units of the positive subdivision boundary
|
||
* `sx`, `sy`, `sz` returns the subdivision x, y, z indices
|
||
*/
|
||
void BgCheck_GetSubdivisionMaxBounds(CollisionContext* colCtx, Vec3f* pos, s32* sx, s32* sy, s32* sz) {
|
||
f32 dx = pos->x - colCtx->minBounds.x;
|
||
f32 dy = pos->y - colCtx->minBounds.y;
|
||
f32 dz = pos->z - colCtx->minBounds.z;
|
||
|
||
*sx = dx * colCtx->subdivLengthInv.x;
|
||
*sy = dy * colCtx->subdivLengthInv.y;
|
||
*sz = dz * colCtx->subdivLengthInv.z;
|
||
|
||
if (((s32)colCtx->subdivLength.x - BGCHECK_SUBDIV_OVERLAP < (s32)dx % (s32)colCtx->subdivLength.x) &&
|
||
(*sx < colCtx->subdivAmount.x - 1)) {
|
||
*sx += 1;
|
||
}
|
||
|
||
if (((s32)colCtx->subdivLength.y - BGCHECK_SUBDIV_OVERLAP < (s32)dy % (s32)colCtx->subdivLength.y) &&
|
||
(*sy < colCtx->subdivAmount.y - 1)) {
|
||
*sy += 1;
|
||
}
|
||
|
||
if (((s32)colCtx->subdivLength.z - BGCHECK_SUBDIV_OVERLAP < (s32)dz % (s32)colCtx->subdivLength.z) &&
|
||
(*sz < colCtx->subdivAmount.z - 1)) {
|
||
*sz += 1;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Calculate the subdivision index bounding box for CollisionPoly `polyId`
|
||
* `subdivMinX`, `subdivMinY`, `subdivMinZ` returns the minimum subdivision x, y, z indices
|
||
* `subdivMaxX`, `subdivMaxY`, `subdivMaxZ` returns the maximum subdivision x, y, z indices
|
||
*/
|
||
void BgCheck_GetPolySubdivisionBounds(CollisionContext* colCtx, Vec3s* vtxList, CollisionPoly* polyList,
|
||
s32* subdivMinX, s32* subdivMinY, s32* subdivMinZ, s32* subdivMaxX,
|
||
s32* subdivMaxY, s32* subdivMaxZ, s16 polyId) {
|
||
u16* vtxDataTemp;
|
||
Vec3f minVtx;
|
||
Vec3f maxVtx;
|
||
|
||
f32 x;
|
||
f32 y;
|
||
f32 z;
|
||
|
||
Vec3s* vtx;
|
||
s16 vtxId = COLPOLY_VTX_INDEX(polyList[polyId].vtxData[0]);
|
||
|
||
Math_Vec3s_ToVec3f(&maxVtx, &vtxList[vtxId]);
|
||
Math_Vec3f_Copy(&minVtx, &maxVtx);
|
||
|
||
for (vtxDataTemp = polyList[polyId].vtxData + 1; vtxDataTemp < polyList[polyId].vtxData + 3; vtxDataTemp++) {
|
||
vtxId = COLPOLY_VTX_INDEX(*vtxDataTemp);
|
||
vtx = &vtxList[vtxId];
|
||
x = vtx->x;
|
||
y = vtx->y;
|
||
z = vtx->z;
|
||
|
||
if (minVtx.x > x) {
|
||
minVtx.x = x;
|
||
} else if (maxVtx.x < x) {
|
||
maxVtx.x = x;
|
||
}
|
||
|
||
if (minVtx.y > y) {
|
||
minVtx.y = y;
|
||
} else if (maxVtx.y < y) {
|
||
maxVtx.y = y;
|
||
}
|
||
|
||
if (minVtx.z > z) {
|
||
minVtx.z = z;
|
||
} else if (maxVtx.z < z) {
|
||
maxVtx.z = z;
|
||
}
|
||
}
|
||
BgCheck_GetSubdivisionMinBounds(colCtx, &minVtx, subdivMinX, subdivMinY, subdivMinZ);
|
||
BgCheck_GetSubdivisionMaxBounds(colCtx, &maxVtx, subdivMaxX, subdivMaxY, subdivMaxZ);
|
||
}
|
||
|
||
/**
|
||
* Test if poly `polyList`[`polyId`] intersects cube `min` `max`
|
||
* returns true if the poly intersects the cube, else false
|
||
*/
|
||
s32 BgCheck_PolyIntersectsSubdivision(Vec3f* min, Vec3f* max, CollisionPoly* polyList, Vec3s* vtxList, s16 polyId) {
|
||
f32 intersect;
|
||
Vec3f va2;
|
||
Vec3f vb2;
|
||
Vec3f vc2;
|
||
CollisionPoly* poly;
|
||
f32 nx;
|
||
f32 ny;
|
||
f32 nz;
|
||
f32 dist;
|
||
Vec3f va;
|
||
Vec3f vb;
|
||
Vec3f vc;
|
||
s32 flags[3];
|
||
|
||
flags[0] = flags[1] = 0;
|
||
poly = &polyList[polyId];
|
||
|
||
BgCheck_Vec3sToVec3f(&vtxList[COLPOLY_VTX_INDEX(poly->flags_vIA)], &va);
|
||
flags[0] = Math3D_PointRelativeToCubeFaces(&va, min, max);
|
||
if (flags[0] == 0) {
|
||
return true;
|
||
}
|
||
|
||
BgCheck_Vec3sToVec3f(&vtxList[COLPOLY_VTX_INDEX(poly->flags_vIB)], &vb);
|
||
flags[1] = Math3D_PointRelativeToCubeFaces(&vb, min, max);
|
||
if (flags[1] == 0) {
|
||
return true;
|
||
}
|
||
|
||
BgCheck_Vec3sToVec3f(&vtxList[poly->vIC], &vc);
|
||
flags[2] = Math3D_PointRelativeToCubeFaces(&vc, min, max);
|
||
if (flags[2] == 0) {
|
||
return true;
|
||
}
|
||
|
||
if (flags[0] & flags[1] & flags[2]) {
|
||
return false;
|
||
}
|
||
|
||
flags[0] |= Math3D_PointRelativeToCubeEdges(&va, min, max) << 8;
|
||
flags[1] |= Math3D_PointRelativeToCubeEdges(&vb, min, max) << 8;
|
||
flags[2] |= Math3D_PointRelativeToCubeEdges(&vc, min, max) << 8;
|
||
if (flags[0] & flags[1] & flags[2]) {
|
||
return false;
|
||
}
|
||
|
||
flags[0] |= Math3D_PointRelativeToCubeVertices(&va, min, max) << 0x18;
|
||
flags[1] |= Math3D_PointRelativeToCubeVertices(&vb, min, max) << 0x18;
|
||
flags[2] |= Math3D_PointRelativeToCubeVertices(&vc, min, max) << 0x18;
|
||
if (flags[0] & flags[1] & flags[2]) {
|
||
return false;
|
||
}
|
||
|
||
CollisionPoly_GetNormalF(poly, &nx, &ny, &nz);
|
||
dist = poly->dist;
|
||
|
||
if (Math3D_TriChkLineSegParaYIntersect(&va, &vb, &vc, nx, ny, nz, dist, min->z, min->x, &intersect, min->y,
|
||
max->y) ||
|
||
Math3D_TriChkLineSegParaYIntersect(&va, &vb, &vc, nx, ny, nz, dist, max->z, min->x, &intersect, min->y,
|
||
max->y) ||
|
||
Math3D_TriChkLineSegParaYIntersect(&va, &vb, &vc, nx, ny, nz, dist, min->z, max->x, &intersect, min->y,
|
||
max->y) ||
|
||
Math3D_TriChkLineSegParaYIntersect(&va, &vb, &vc, nx, ny, nz, dist, max->z, max->x, &intersect, min->y,
|
||
max->y)) {
|
||
return true;
|
||
}
|
||
if (Math3D_TriChkLineSegParaZIntersect(&va, &vb, &vc, nx, ny, nz, dist, min->x, min->y, &intersect, min->z,
|
||
max->z) ||
|
||
Math3D_TriChkLineSegParaZIntersect(&va, &vb, &vc, nx, ny, nz, dist, min->x, max->y, &intersect, min->z,
|
||
max->z) ||
|
||
Math3D_TriChkLineSegParaZIntersect(&va, &vb, &vc, nx, ny, nz, dist, max->x, min->y, &intersect, min->z,
|
||
max->z) ||
|
||
Math3D_TriChkLineSegParaZIntersect(&va, &vb, &vc, nx, ny, nz, dist, max->x, max->y, &intersect, min->z,
|
||
max->z)) {
|
||
return true;
|
||
}
|
||
if (Math3D_TriChkLineSegParaXIntersect(&va, &vb, &vc, nx, ny, nz, dist, min->y, min->z, &intersect, min->x,
|
||
max->x) ||
|
||
Math3D_TriChkLineSegParaXIntersect(&va, &vb, &vc, nx, ny, nz, dist, min->y, max->z, &intersect, min->x,
|
||
max->x) ||
|
||
Math3D_TriChkLineSegParaXIntersect(&va, &vb, &vc, nx, ny, nz, dist, max->y, min->z, &intersect, min->x,
|
||
max->x) ||
|
||
Math3D_TriChkLineSegParaXIntersect(&va, &vb, &vc, nx, ny, nz, dist, max->y, max->z, &intersect, min->x,
|
||
max->x)) {
|
||
return true;
|
||
}
|
||
|
||
BgCheck_Vec3sToVec3f(&vtxList[COLPOLY_VTX_INDEX(poly->flags_vIA)], &va2);
|
||
BgCheck_Vec3sToVec3f(&vtxList[COLPOLY_VTX_INDEX(poly->flags_vIB)], &vb2);
|
||
BgCheck_Vec3sToVec3f(&vtxList[poly->vIC], &vc2);
|
||
if (Math3D_LineVsCube(min, max, &va2, &vb2) || Math3D_LineVsCube(min, max, &vb2, &vc2) ||
|
||
Math3D_LineVsCube(min, max, &vc2, &va2)) {
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Initialize StaticLookup Table
|
||
* returns size of table, in bytes
|
||
*/
|
||
u32 BgCheck_InitializeStaticLookup(CollisionContext* colCtx, PlayState* play, StaticLookup* lookupTbl) {
|
||
Vec3s* vtxList;
|
||
CollisionPoly* polyList;
|
||
s32 polyMax;
|
||
s32 polyIdx;
|
||
s32 sx;
|
||
s32 sy;
|
||
s32 sz;
|
||
// subdivMin indices
|
||
s32 sxMin;
|
||
s32 syMin;
|
||
s32 szMin;
|
||
// subdivMax indices
|
||
s32 sxMax;
|
||
s32 syMax;
|
||
s32 szMax;
|
||
// subdiv min/max bounds for adding a poly
|
||
Vec3f curSubdivMin;
|
||
Vec3f curSubdivMax;
|
||
CollisionHeader* colHeader = colCtx->colHeader;
|
||
StaticLookup* lookupTblXY;
|
||
StaticLookup* lookupTblX;
|
||
StaticLookup* lookup;
|
||
s32 subdivAmountXY;
|
||
f32 subdivLengthX;
|
||
f32 subdivLengthY;
|
||
f32 subdivLengthZ;
|
||
|
||
for (lookupTblXY = lookupTbl;
|
||
lookupTblXY < (colCtx->subdivAmount.x * colCtx->subdivAmount.y * colCtx->subdivAmount.z + lookupTbl);
|
||
lookupTblXY++) {
|
||
lookupTblXY->floor.head = SS_NULL;
|
||
lookupTblXY->wall.head = SS_NULL;
|
||
lookupTblXY->ceiling.head = SS_NULL;
|
||
}
|
||
|
||
polyMax = colHeader->numPolygons;
|
||
vtxList = colHeader->vtxList;
|
||
polyList = colHeader->polyList;
|
||
subdivAmountXY = colCtx->subdivAmount.x * colCtx->subdivAmount.y;
|
||
subdivLengthX = colCtx->subdivLength.x + (2 * BGCHECK_SUBDIV_OVERLAP);
|
||
subdivLengthY = colCtx->subdivLength.y + (2 * BGCHECK_SUBDIV_OVERLAP);
|
||
subdivLengthZ = colCtx->subdivLength.z + (2 * BGCHECK_SUBDIV_OVERLAP);
|
||
|
||
for (polyIdx = 0; polyIdx < polyMax; polyIdx++) {
|
||
BgCheck_GetPolySubdivisionBounds(colCtx, vtxList, polyList, &sxMin, &syMin, &szMin, &sxMax, &syMax, &szMax,
|
||
polyIdx);
|
||
lookupTblXY = szMin * subdivAmountXY + lookupTbl;
|
||
curSubdivMin.z = (colCtx->subdivLength.z * szMin + colCtx->minBounds.z) - BGCHECK_SUBDIV_OVERLAP;
|
||
curSubdivMax.z = curSubdivMin.z + subdivLengthZ;
|
||
|
||
for (sz = szMin; sz < szMax + 1; sz++) {
|
||
lookupTblX = (colCtx->subdivAmount.x * syMin) + lookupTblXY;
|
||
curSubdivMin.y = (colCtx->subdivLength.y * syMin + colCtx->minBounds.y) - BGCHECK_SUBDIV_OVERLAP;
|
||
curSubdivMax.y = curSubdivMin.y + subdivLengthY;
|
||
|
||
for (sy = syMin; sy < syMax + 1; sy++) {
|
||
lookup = sxMin + lookupTblX;
|
||
curSubdivMin.x = (colCtx->subdivLength.x * sxMin + colCtx->minBounds.x) - BGCHECK_SUBDIV_OVERLAP;
|
||
curSubdivMax.x = curSubdivMin.x + subdivLengthX;
|
||
|
||
for (sx = sxMin; sx < sxMax + 1; sx++) {
|
||
if (BgCheck_PolyIntersectsSubdivision(&curSubdivMin, &curSubdivMax, polyList, vtxList, polyIdx)) {
|
||
StaticLookup_AddPoly(lookup, colCtx, polyList, vtxList, polyIdx);
|
||
}
|
||
curSubdivMin.x += colCtx->subdivLength.x;
|
||
curSubdivMax.x += colCtx->subdivLength.x;
|
||
lookup++;
|
||
}
|
||
curSubdivMin.y += colCtx->subdivLength.y;
|
||
curSubdivMax.y += colCtx->subdivLength.y;
|
||
lookupTblX += colCtx->subdivAmount.x;
|
||
}
|
||
curSubdivMin.z += colCtx->subdivLength.z;
|
||
curSubdivMax.z += colCtx->subdivLength.z;
|
||
lookupTblXY += subdivAmountXY;
|
||
}
|
||
}
|
||
return colCtx->polyNodes.count * sizeof(SSNode);
|
||
}
|
||
|
||
/**
|
||
* Is current scene a SPOT scene
|
||
*/
|
||
s32 BgCheck_IsSpotScene(PlayState* play) {
|
||
static s16 spotScenes[] = {
|
||
SCENE_HYRULE_FIELD, SCENE_KAKARIKO_VILLAGE, SCENE_GRAVEYARD, SCENE_ZORAS_RIVER,
|
||
SCENE_KOKIRI_FOREST, SCENE_SACRED_FOREST_MEADOW, SCENE_LAKE_HYLIA, SCENE_ZORAS_DOMAIN,
|
||
SCENE_ZORAS_FOUNTAIN, SCENE_GERUDO_VALLEY, SCENE_LOST_WOODS, SCENE_DESERT_COLOSSUS,
|
||
SCENE_GERUDOS_FORTRESS, SCENE_HAUNTED_WASTELAND, SCENE_HYRULE_CASTLE, SCENE_DEATH_MOUNTAIN_TRAIL,
|
||
SCENE_DEATH_MOUNTAIN_CRATER, SCENE_GORON_CITY, SCENE_LON_LON_RANCH,
|
||
};
|
||
s16* i;
|
||
|
||
for (i = spotScenes; i < spotScenes + ARRAY_COUNT(spotScenes); i++) {
|
||
if (play->sceneId == *i) {
|
||
return true;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
typedef struct BgCheckSceneMemEntry {
|
||
s16 sceneId;
|
||
u32 memSize;
|
||
} BgCheckSceneMemEntry;
|
||
|
||
/**
|
||
* Get custom scene memSize
|
||
*/
|
||
s32 BgCheck_TryGetCustomMemsize(s32 sceneId, u32* memSize) {
|
||
static BgCheckSceneMemEntry sceneMemList[] = {
|
||
{ SCENE_HYRULE_FIELD, 0xB798 }, { SCENE_GANONS_TOWER_COLLAPSE_EXTERIOR, 0x78C8 },
|
||
{ SCENE_GANON_BOSS, 0x70C8 }, { SCENE_SPIRIT_TEMPLE_BOSS, 0xACC8 },
|
||
{ SCENE_CHAMBER_OF_THE_SAGES, 0x70C8 }, { SCENE_SPIRIT_TEMPLE, 0x16CC8 },
|
||
{ SCENE_FIRE_TEMPLE, 0x198C8 }, { SCENE_GANONDORF_BOSS, 0x84C8 },
|
||
};
|
||
s32 i;
|
||
|
||
for (i = 0; i < ARRAY_COUNT(sceneMemList); i++) {
|
||
if (sceneId == sceneMemList[i].sceneId) {
|
||
*memSize = sceneMemList[i].memSize;
|
||
return true;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Compute subdivLength for scene mesh lookup, for a single dimension
|
||
*/
|
||
void BgCheck_SetSubdivisionDimension(f32 min, s32 subdivAmount, f32* max, f32* subdivLength, f32* subdivLengthInv) {
|
||
f32 length = (*max - min);
|
||
|
||
*subdivLength = (s32)(length / subdivAmount) + 1;
|
||
*subdivLength = CLAMP_MIN(*subdivLength, BGCHECK_SUBDIV_MIN);
|
||
*subdivLengthInv = 1.0f / *subdivLength;
|
||
|
||
*max = *subdivLength * subdivAmount + min;
|
||
}
|
||
|
||
typedef struct BgCheckSceneSubdivisionEntry {
|
||
s16 sceneId;
|
||
Vec3s subdivAmount;
|
||
s32 nodeListMax; // if -1, dynamically compute max nodes
|
||
} BgCheckSceneSubdivisionEntry;
|
||
|
||
/**
|
||
* Allocate CollisionContext
|
||
*/
|
||
void BgCheck_Allocate(CollisionContext* colCtx, PlayState* play, CollisionHeader* colHeader) {
|
||
static BgCheckSceneSubdivisionEntry sceneSubdivisionList[] = {
|
||
{ SCENE_SHADOW_TEMPLE, { 23, 7, 14 }, -1 },
|
||
{ SCENE_FOREST_TEMPLE, { 38, 1, 38 }, -1 },
|
||
};
|
||
u32 tblMax;
|
||
u32 memSize;
|
||
u32 lookupTblMemSize;
|
||
s32 customNodeListMax;
|
||
SSNodeList* nodeList;
|
||
u32 customMemSize;
|
||
s32 useCustomSubdivisions;
|
||
s32 i;
|
||
|
||
colCtx->colHeader = colHeader;
|
||
customNodeListMax = -1;
|
||
|
||
PRINTF(T("/*---------------- BGCheck バッファーメモリサイズ -------------*/\n",
|
||
"/*---------------- BGCheck Buffer Memory Size -------------*/\n"));
|
||
|
||
if ((R_SCENE_CAM_TYPE == SCENE_CAM_TYPE_FIXED_SHOP_VIEWPOINT) ||
|
||
(R_SCENE_CAM_TYPE == SCENE_CAM_TYPE_FIXED_TOGGLE_VIEWPOINT) || (R_SCENE_CAM_TYPE == SCENE_CAM_TYPE_FIXED) ||
|
||
(R_SCENE_CAM_TYPE == SCENE_CAM_TYPE_FIXED_MARKET)) {
|
||
if (play->sceneId == SCENE_STABLE) {
|
||
PRINTF(T("/* BGCheck LonLonサイズ %dbyte */\n", "/* BGCheck LonLon Size %dbyte */\n"), 0x3520);
|
||
colCtx->memSize = 0x3520;
|
||
} else {
|
||
PRINTF(T("/* BGCheck ミニサイズ %dbyte */\n", "/* BGCheck Mini Size %dbyte */\n"), 0x4E20);
|
||
colCtx->memSize = 0x4E20;
|
||
}
|
||
colCtx->dyna.polyNodesMax = 500;
|
||
colCtx->dyna.polyListMax = 256;
|
||
colCtx->dyna.vtxListMax = 256;
|
||
colCtx->subdivAmount.x = 2;
|
||
colCtx->subdivAmount.y = 2;
|
||
colCtx->subdivAmount.z = 2;
|
||
} else if (BgCheck_IsSpotScene(play) == true) {
|
||
colCtx->memSize = 0xF000;
|
||
PRINTF(T("/* BGCheck Spot用サイズ %dbyte */\n", "/* BGCheck Spot Size %dbyte */\n"), 0xF000);
|
||
colCtx->dyna.polyNodesMax = 1000;
|
||
colCtx->dyna.polyListMax = 512;
|
||
colCtx->dyna.vtxListMax = 512;
|
||
colCtx->subdivAmount.x = 16;
|
||
colCtx->subdivAmount.y = 4;
|
||
colCtx->subdivAmount.z = 16;
|
||
} else {
|
||
if (BgCheck_TryGetCustomMemsize(play->sceneId, &customMemSize)) {
|
||
colCtx->memSize = customMemSize;
|
||
} else {
|
||
colCtx->memSize = 0x1CC00;
|
||
}
|
||
PRINTF(T("/* BGCheck ノーマルサイズ %dbyte */\n", "/* BGCheck Normal Size %dbyte */\n"), colCtx->memSize);
|
||
colCtx->dyna.polyNodesMax = 1000;
|
||
colCtx->dyna.polyListMax = 512;
|
||
colCtx->dyna.vtxListMax = 512;
|
||
useCustomSubdivisions = false;
|
||
|
||
for (i = 0; i < ARRAY_COUNT(sceneSubdivisionList); i++) {
|
||
if (play->sceneId == sceneSubdivisionList[i].sceneId) {
|
||
colCtx->subdivAmount.x = sceneSubdivisionList[i].subdivAmount.x;
|
||
colCtx->subdivAmount.y = sceneSubdivisionList[i].subdivAmount.y;
|
||
colCtx->subdivAmount.z = sceneSubdivisionList[i].subdivAmount.z;
|
||
useCustomSubdivisions = true;
|
||
customNodeListMax = sceneSubdivisionList[i].nodeListMax;
|
||
}
|
||
}
|
||
if (!useCustomSubdivisions) {
|
||
colCtx->subdivAmount.x = 16;
|
||
colCtx->subdivAmount.y = 4;
|
||
colCtx->subdivAmount.z = 16;
|
||
}
|
||
}
|
||
colCtx->lookupTbl = THA_AllocTailAlign(&play->state.tha,
|
||
colCtx->subdivAmount.x * sizeof(StaticLookup) * colCtx->subdivAmount.y *
|
||
colCtx->subdivAmount.z,
|
||
ALIGNOF_MASK(StaticLookup));
|
||
if (colCtx->lookupTbl == NULL) {
|
||
LogUtils_HungupThread("../z_bgcheck.c", 4176);
|
||
}
|
||
colCtx->minBounds.x = colCtx->colHeader->minBounds.x;
|
||
colCtx->minBounds.y = colCtx->colHeader->minBounds.y;
|
||
colCtx->minBounds.z = colCtx->colHeader->minBounds.z;
|
||
colCtx->maxBounds.x = colCtx->colHeader->maxBounds.x;
|
||
colCtx->maxBounds.y = colCtx->colHeader->maxBounds.y;
|
||
colCtx->maxBounds.z = colCtx->colHeader->maxBounds.z;
|
||
BgCheck_SetSubdivisionDimension(colCtx->minBounds.x, colCtx->subdivAmount.x, &colCtx->maxBounds.x,
|
||
&colCtx->subdivLength.x, &colCtx->subdivLengthInv.x);
|
||
BgCheck_SetSubdivisionDimension(colCtx->minBounds.y, colCtx->subdivAmount.y, &colCtx->maxBounds.y,
|
||
&colCtx->subdivLength.y, &colCtx->subdivLengthInv.y);
|
||
BgCheck_SetSubdivisionDimension(colCtx->minBounds.z, colCtx->subdivAmount.z, &colCtx->maxBounds.z,
|
||
&colCtx->subdivLength.z, &colCtx->subdivLengthInv.z);
|
||
memSize = colCtx->subdivAmount.x * sizeof(StaticLookup) * colCtx->subdivAmount.y * colCtx->subdivAmount.z +
|
||
colCtx->colHeader->numPolygons * sizeof(u8) + colCtx->dyna.polyNodesMax * sizeof(SSNode) +
|
||
colCtx->dyna.polyListMax * sizeof(CollisionPoly) + colCtx->dyna.vtxListMax * sizeof(Vec3s) +
|
||
sizeof(CollisionContext);
|
||
if (customNodeListMax > 0) {
|
||
// tblMax is set without checking if customNodeListMax will result in a memory overflow
|
||
// this is a non-issue as long as sceneSubdivisionList.nodeListMax is -1
|
||
tblMax = customNodeListMax;
|
||
} else {
|
||
if (colCtx->memSize < memSize) {
|
||
LogUtils_HungupThread("../z_bgcheck.c", 4230);
|
||
}
|
||
tblMax = (colCtx->memSize - memSize) / sizeof(SSNode);
|
||
}
|
||
|
||
SSNodeList_Initialize(&colCtx->polyNodes);
|
||
SSNodeList_Alloc(play, &colCtx->polyNodes, tblMax, colCtx->colHeader->numPolygons);
|
||
|
||
lookupTblMemSize = BgCheck_InitializeStaticLookup(colCtx, play, colCtx->lookupTbl);
|
||
PRINTF(VT_FGCOL(GREEN));
|
||
PRINTF(T("/*---結局 BG使用サイズ %dbyte---*/\n", "/*---BG size used in the end %dbyte---*/\n"),
|
||
memSize + lookupTblMemSize);
|
||
PRINTF(VT_RST);
|
||
|
||
DynaPoly_Init(play, &colCtx->dyna);
|
||
DynaPoly_Alloc(play, &colCtx->dyna);
|
||
}
|
||
|
||
/**
|
||
* Get CollisionHeader
|
||
* original name: T_BGCheck_getBGDataInfo
|
||
*/
|
||
CollisionHeader* BgCheck_GetCollisionHeader(CollisionContext* colCtx, s32 bgId) {
|
||
if (bgId == BGCHECK_SCENE) {
|
||
return colCtx->colHeader;
|
||
}
|
||
if (bgId < 0 || bgId > BG_ACTOR_MAX) {
|
||
return NULL;
|
||
}
|
||
if (!(colCtx->dyna.bgActorFlags[bgId] & BGACTOR_IN_USE)) {
|
||
PRINTF(VT_COL(YELLOW, BLACK));
|
||
PRINTF(T("T_BGCheck_getBGDataInfo():そのbg_actor_indexは使われておりません。index=%d\n",
|
||
"T_BGCheck_getBGDataInfo(): That bg_actor_index is not in use. index=%d\n"));
|
||
PRINTF(VT_RST);
|
||
return NULL;
|
||
}
|
||
return colCtx->dyna.bgActors[bgId].colHeader;
|
||
}
|
||
|
||
/**
|
||
* Test if pos is near collision boundaries
|
||
*/
|
||
s32 BgCheck_PosInStaticBoundingBox(CollisionContext* colCtx, Vec3f* pos) {
|
||
if (pos->x < (colCtx->minBounds.x - BGCHECK_SUBDIV_OVERLAP) ||
|
||
(colCtx->maxBounds.x + BGCHECK_SUBDIV_OVERLAP) < pos->x ||
|
||
pos->y < (colCtx->minBounds.y - BGCHECK_SUBDIV_OVERLAP) ||
|
||
(colCtx->maxBounds.y + BGCHECK_SUBDIV_OVERLAP) < pos->y ||
|
||
pos->z < (colCtx->minBounds.z - BGCHECK_SUBDIV_OVERLAP) ||
|
||
(colCtx->maxBounds.z + BGCHECK_SUBDIV_OVERLAP) < pos->z) {
|
||
return false;
|
||
}
|
||
return true;
|
||
}
|
||
|
||
/**
|
||
* Raycast Downward
|
||
* If `actor` != null, bgcheck will be skipped for that actor
|
||
* returns the yIntersect of the nearest poly found directly below `pos`, or BGCHECK_Y_MIN if no floor detected
|
||
* returns the poly found in `outPoly`, and the bgId of the entity in `outBgId`
|
||
*/
|
||
f32 BgCheck_RaycastDownImpl(PlayState* play, CollisionContext* colCtx, u16 xpFlags, CollisionPoly** outPoly,
|
||
s32* outBgId, Vec3f* pos, Actor* actor, u32 downChkFlags, f32 chkDist) {
|
||
f32 yIntersectDyna;
|
||
s32* temp_a0;
|
||
StaticLookup* lookupTbl;
|
||
Vec3f checkPos;
|
||
StaticLookup* lookup;
|
||
DynaRaycastDown dynaRaycastDown;
|
||
f32 yIntersect;
|
||
|
||
*outBgId = BGCHECK_SCENE;
|
||
*outPoly = NULL;
|
||
lookupTbl = colCtx->lookupTbl;
|
||
yIntersect = BGCHECK_Y_MIN;
|
||
checkPos = *pos;
|
||
|
||
while (true) {
|
||
if (checkPos.y < colCtx->minBounds.y) {
|
||
break;
|
||
}
|
||
|
||
#if OOT_DEBUG
|
||
if (BgCheck_PosErrorCheck(&checkPos, "../z_bgcheck.c", 4410)) {
|
||
if (actor != NULL) {
|
||
PRINTF(T("こいつ,pself_actor->name %d\n", "This guy, pself_actor->name %d\n"), actor->id);
|
||
}
|
||
}
|
||
#endif
|
||
|
||
lookup = BgCheck_GetStaticLookup(colCtx, lookupTbl, &checkPos);
|
||
if (lookup == NULL) {
|
||
checkPos.y -= colCtx->subdivLength.y;
|
||
continue;
|
||
}
|
||
yIntersect =
|
||
BgCheck_RaycastDownStatic(lookup, colCtx, xpFlags, outPoly, pos, downChkFlags, chkDist, BGCHECK_Y_MIN);
|
||
if (yIntersect > BGCHECK_Y_MIN) {
|
||
break;
|
||
}
|
||
checkPos.y -= colCtx->subdivLength.y;
|
||
}
|
||
|
||
dynaRaycastDown.play = play;
|
||
dynaRaycastDown.colCtx = colCtx;
|
||
dynaRaycastDown.xpFlags = xpFlags;
|
||
dynaRaycastDown.resultPoly = outPoly;
|
||
dynaRaycastDown.yIntersect = yIntersect;
|
||
dynaRaycastDown.pos = pos;
|
||
dynaRaycastDown.bgId = outBgId;
|
||
dynaRaycastDown.actor = actor;
|
||
dynaRaycastDown.downChkFlags = downChkFlags;
|
||
dynaRaycastDown.chkDist = chkDist;
|
||
|
||
yIntersectDyna = BgCheck_RaycastDownDyna(&dynaRaycastDown);
|
||
|
||
if (yIntersect < yIntersectDyna) {
|
||
yIntersect = yIntersectDyna;
|
||
}
|
||
|
||
if (yIntersect != BGCHECK_Y_MIN && SurfaceType_IsSoft(colCtx, *outPoly, *outBgId)) {
|
||
yIntersect -= 1.0f;
|
||
}
|
||
return yIntersect;
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, ground check (UNUSED)
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_CameraRaycastDown1(CollisionContext* colCtx, CollisionPoly** outGroundPoly, Vec3f* pos) {
|
||
s32 bgId;
|
||
|
||
return BgCheck_RaycastDownImpl(NULL, colCtx, COLPOLY_IGNORE_CAMERA, outGroundPoly, &bgId, pos, NULL,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS |
|
||
BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY,
|
||
1.0f);
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, ground check
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_EntityRaycastDown1(CollisionContext* colCtx, CollisionPoly** outGroundPoly, Vec3f* pos) {
|
||
s32 bgId;
|
||
|
||
return BgCheck_RaycastDownImpl(NULL, colCtx, COLPOLY_IGNORE_ENTITY, outGroundPoly, &bgId, pos, NULL,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS |
|
||
BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY,
|
||
1.0f);
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, ground check
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_EntityRaycastDown2(PlayState* play, CollisionContext* colCtx, CollisionPoly** outGroundPoly, Vec3f* pos) {
|
||
s32 bgId;
|
||
|
||
return BgCheck_RaycastDownImpl(play, colCtx, COLPOLY_IGNORE_ENTITY, outGroundPoly, &bgId, pos, NULL,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS |
|
||
BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY,
|
||
1.0f);
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, ground check
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_EntityRaycastDown3(CollisionContext* colCtx, CollisionPoly** outGroundPoly, s32* bgId, Vec3f* pos) {
|
||
return BgCheck_RaycastDownImpl(NULL, colCtx, COLPOLY_IGNORE_ENTITY, outGroundPoly, bgId, pos, NULL,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS |
|
||
BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY,
|
||
1.0f);
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, ground check
|
||
* If `actor` != null, bgcheck will be skipped for that actor
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_EntityRaycastDown4(CollisionContext* colCtx, CollisionPoly** outGroundPoly, s32* bgId, Actor* actor,
|
||
Vec3f* pos) {
|
||
return BgCheck_RaycastDownImpl(NULL, colCtx, COLPOLY_IGNORE_ENTITY, outGroundPoly, bgId, pos, actor,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS |
|
||
BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY,
|
||
1.0f);
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, ground check
|
||
* If `actor` != null, bgcheck will be skipped for that actor
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_EntityRaycastDown5(PlayState* play, CollisionContext* colCtx, CollisionPoly** outGroundPoly, s32* bgId,
|
||
Actor* actor, Vec3f* pos) {
|
||
return BgCheck_RaycastDownImpl(play, colCtx, COLPOLY_IGNORE_ENTITY, outGroundPoly, bgId, pos, actor,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS |
|
||
BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY,
|
||
1.0f);
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, ground check
|
||
* If `actor` != null, bgcheck will be skipped for that actor
|
||
* `chkDist` is the distance beyond the poly's boundary where the check will still pass
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_EntityRaycastDown6(CollisionContext* colCtx, CollisionPoly** outGroundPoly, s32* bgId, Actor* actor,
|
||
Vec3f* pos, f32 chkDist) {
|
||
return BgCheck_RaycastDownImpl(NULL, colCtx, COLPOLY_IGNORE_ENTITY, outGroundPoly, bgId, pos, actor,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS |
|
||
BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY,
|
||
chkDist);
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, floor and exhaustive wall check (UNUSED)
|
||
* If `actor` != null, bgcheck will be skipped for that actor
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_EntityRaycastDown7(CollisionContext* colCtx, CollisionPoly** outPoly, s32* bgId, Actor* actor, Vec3f* pos) {
|
||
return BgCheck_RaycastDownImpl(NULL, colCtx, COLPOLY_IGNORE_ENTITY, outPoly, bgId, pos, actor,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS, 1.0f);
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, ground check
|
||
* `outGroundPoly` returns original value if no poly detected
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_AnyRaycastDown1(CollisionContext* colCtx, CollisionPoly* outGroundPoly, Vec3f* pos) {
|
||
CollisionPoly* checkResultPoly;
|
||
f32 result;
|
||
s32 bgId;
|
||
|
||
result = BgCheck_RaycastDownImpl(NULL, colCtx, COLPOLY_IGNORE_NONE, &checkResultPoly, &bgId, pos, NULL,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS |
|
||
BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY,
|
||
1.0f);
|
||
|
||
if (checkResultPoly != NULL) {
|
||
*outGroundPoly = *checkResultPoly;
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, ground check (UNUSED)
|
||
* `outGroundPoly` returns original value if no poly detected
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_AnyRaycastDown2(CollisionContext* colCtx, CollisionPoly* outGroundPoly, s32* bgId, Vec3f* pos) {
|
||
CollisionPoly* checkResultPoly;
|
||
f32 result = BgCheck_RaycastDownImpl(NULL, colCtx, COLPOLY_IGNORE_NONE, &checkResultPoly, bgId, pos, NULL,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS |
|
||
BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY,
|
||
1.0f);
|
||
|
||
if (checkResultPoly != NULL) {
|
||
*outGroundPoly = *checkResultPoly;
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, floor and exhaustive wall check
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_CameraRaycastDown2(CollisionContext* colCtx, CollisionPoly** outPoly, s32* bgId, Vec3f* pos) {
|
||
return BgCheck_RaycastDownImpl(NULL, colCtx, COLPOLY_IGNORE_CAMERA, outPoly, bgId, pos, NULL,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS, 1.0f);
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, exhaustive wall check (UNUSED)
|
||
* If `actor` != null, bgcheck will be skipped for that actor
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_EntityRaycastDownWalls(CollisionContext* colCtx, CollisionPoly** outPoly, s32* bgId, Actor* actor,
|
||
Vec3f* pos) {
|
||
return BgCheck_RaycastDownImpl(NULL, colCtx, COLPOLY_IGNORE_ENTITY, outPoly, bgId, pos, actor,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS, 1.0f);
|
||
}
|
||
|
||
/**
|
||
* Public raycast downward, floor and exhaustive wall check (UNUSED)
|
||
* returns yIntersect of the poly found, or BGCHECK_Y_MIN if no poly detected
|
||
*/
|
||
f32 BgCheck_EntityRaycastDown9(CollisionContext* colCtx, CollisionPoly** outPoly, s32* bgId, Vec3f* pos) {
|
||
return BgCheck_RaycastDownImpl(NULL, colCtx, COLPOLY_IGNORE_ENTITY, outPoly, bgId, pos, NULL,
|
||
BGCHECK_RAYCAST_DOWN_CHECK_WALLS | BGCHECK_RAYCAST_DOWN_CHECK_FLOORS, 1.0f);
|
||
}
|
||
|
||
/**
|
||
* Tests if moving from `posPrev` to `posNext` will collide with a "wall"
|
||
* `radius` is used to form a sphere for collision detection purposes
|
||
* `checkHeight` is the positive height above posNext to perform certain checks
|
||
* returns true if a collision is detected, else false
|
||
* `outPoly` returns the closest poly detected, while `outBgId` returns the poly owner
|
||
*/
|
||
s32 BgCheck_CheckWallImpl(CollisionContext* colCtx, u16 xpFlags, Vec3f* posResult, Vec3f* posNext, Vec3f* posPrev,
|
||
f32 radius, CollisionPoly** outPoly, s32* outBgId, Actor* actor, f32 checkHeight, u8 argA) {
|
||
StaticLookup* lookupTbl;
|
||
f32 temp_f0;
|
||
s32 result;
|
||
CollisionPoly* poly;
|
||
f32 dx, dy, dz; // change between posPrev to posNext
|
||
Vec3f sphCenter;
|
||
s32 dynaPolyCollision;
|
||
Vec3f posIntersect;
|
||
s32 bgId;
|
||
f32 temp_f0_2;
|
||
f32 f32temp;
|
||
f32 nx2, nz2;
|
||
Vec3f checkLineNext;
|
||
Vec3f checkLinePrev;
|
||
f32 n2XZDist;
|
||
f32 n3XZDist;
|
||
s32 bccFlags;
|
||
f32 nx;
|
||
f32 nz;
|
||
Vec3f posIntersect2;
|
||
s32 bgId2;
|
||
|
||
result = false;
|
||
*outBgId = BGCHECK_SCENE;
|
||
*outPoly = NULL;
|
||
lookupTbl = colCtx->lookupTbl;
|
||
*posResult = *posNext;
|
||
dx = posNext->x - posPrev->x;
|
||
dy = posNext->y - posPrev->y;
|
||
dz = posNext->z - posPrev->z;
|
||
|
||
#if OOT_DEBUG
|
||
if (BgCheck_PosErrorCheck(posNext, "../z_bgcheck.c", 4831) == true ||
|
||
BgCheck_PosErrorCheck(posPrev, "../z_bgcheck.c", 4832) == true) {
|
||
if (actor != NULL) {
|
||
PRINTF(T("こいつ,pself_actor->name %d\n", "This guy, pself_actor->name %d\n"), actor->id);
|
||
}
|
||
}
|
||
#endif
|
||
|
||
// if there's movement on the xz plane, and argA flag is 0,
|
||
if ((dx != 0.0f || dz != 0.0f) && (argA & 1) == 0) {
|
||
if ((checkHeight + dy) < 5.0f) {
|
||
//! @bug checkHeight is not applied to posPrev/posNext
|
||
result = BgCheck_CheckLineImpl(colCtx, xpFlags, COLPOLY_IGNORE_NONE, posPrev, posNext, &posIntersect, &poly,
|
||
&bgId, actor, 1.0f, BGCHECK_CHECK_ALL & ~BGCHECK_CHECK_CEILING);
|
||
if (result) {
|
||
f32 ny = COLPOLY_GET_NORMAL(poly->normal.y);
|
||
|
||
// if poly is floor, push result underneath the floor
|
||
if (ny > 0.5f) {
|
||
posResult->x = posIntersect.x;
|
||
if (checkHeight > 1.0f) {
|
||
posResult->y = posIntersect.y - 1.0f;
|
||
} else {
|
||
posResult->y = posIntersect.y - checkHeight;
|
||
}
|
||
posResult->z = posIntersect.z;
|
||
}
|
||
// poly is wall
|
||
else {
|
||
nx = COLPOLY_GET_NORMAL(poly->normal.x);
|
||
nz = COLPOLY_GET_NORMAL(poly->normal.z);
|
||
posResult->x = radius * nx + posIntersect.x;
|
||
posResult->y = radius * ny + posIntersect.y;
|
||
posResult->z = radius * nz + posIntersect.z;
|
||
}
|
||
*outPoly = poly;
|
||
*outBgId = bgId;
|
||
}
|
||
} else {
|
||
// if the radius is less than the distance travelled on the xz plane, also test for floor collisions
|
||
bccFlags = SQ(radius) < (SQ(dx) + SQ(dz))
|
||
? (BGCHECK_CHECK_ALL & ~BGCHECK_CHECK_CEILING)
|
||
: (BGCHECK_CHECK_ALL & ~BGCHECK_CHECK_FLOOR & ~BGCHECK_CHECK_CEILING);
|
||
|
||
// perform a straight line test to see if a line at posNext.y + checkHeight from posPrev.xz to posNext.xz
|
||
// passes through any wall and possibly floor polys
|
||
checkLineNext = *posNext;
|
||
checkLineNext.y += checkHeight;
|
||
checkLinePrev = *posPrev;
|
||
checkLinePrev.y = checkLineNext.y;
|
||
result = BgCheck_CheckLineImpl(colCtx, xpFlags, COLPOLY_IGNORE_NONE, &checkLinePrev, &checkLineNext,
|
||
&posIntersect, &poly, &bgId, actor, 1.0f, bccFlags);
|
||
|
||
if (result) {
|
||
nx2 = COLPOLY_GET_NORMAL(poly->normal.x);
|
||
nz2 = COLPOLY_GET_NORMAL(poly->normal.z);
|
||
n2XZDist = sqrtf(SQ(nx2) + SQ(nz2));
|
||
|
||
// if poly is not a "flat" floor or "flat" ceiling
|
||
if (!IS_ZERO(n2XZDist)) {
|
||
// normalize nx,nz and multiply each by the radius to go back to the other side of the wall
|
||
f32temp = 1.0f / n2XZDist;
|
||
temp_f0 = radius * f32temp;
|
||
posResult->x = temp_f0 * nx2 + posIntersect.x;
|
||
posResult->z = temp_f0 * nz2 + posIntersect.z;
|
||
*outPoly = poly;
|
||
*outBgId = bgId;
|
||
result = true;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
sphCenter = *posResult;
|
||
dynaPolyCollision = false;
|
||
sphCenter.y += checkHeight;
|
||
// test if sphere (sphCenter, radius) collides with a dynamic wall, displacing the x/z coordinates
|
||
if (BgCheck_SphVsDynaWall(colCtx, xpFlags, &posResult->x, &posResult->z, &sphCenter, radius, outPoly, outBgId,
|
||
actor)) {
|
||
result = true;
|
||
dynaPolyCollision = true;
|
||
sphCenter = *posResult;
|
||
sphCenter.y += checkHeight;
|
||
}
|
||
// test if sphere (sphCenter, radius) collides with a static wall, displacing the x/z coordinates
|
||
if (BgCheck_PosInStaticBoundingBox(colCtx, posNext) == true &&
|
||
// possible bug? if the sphere's radius is smaller than the distance to a subdivision boundary, some static
|
||
// polys will be missed
|
||
BgCheck_SphVsStaticWall(BgCheck_GetNearestStaticLookup(colCtx, lookupTbl, posResult), colCtx, xpFlags,
|
||
&posResult->x, &posResult->z, &sphCenter, radius, outPoly)) {
|
||
*outBgId = BGCHECK_SCENE;
|
||
result = true;
|
||
}
|
||
// if a collision with a dyna poly was detected
|
||
if (dynaPolyCollision == true || *outBgId != BGCHECK_SCENE) {
|
||
if (BgCheck_CheckLineImpl(colCtx, xpFlags, COLPOLY_IGNORE_NONE, posPrev, posResult, &posIntersect2, &poly,
|
||
&bgId2, actor, 1.0f, BGCHECK_CHECK_ONE_FACE | BGCHECK_CHECK_WALL)) {
|
||
f32 nx3 = COLPOLY_GET_NORMAL(poly->normal.x);
|
||
f32 nz3 = COLPOLY_GET_NORMAL(poly->normal.z);
|
||
|
||
n3XZDist = sqrtf(SQ(nx3) + SQ(nz3));
|
||
|
||
// if poly is not a "flat" floor or "flat" ceiling
|
||
if (!IS_ZERO(n3XZDist)) {
|
||
// normalize nx,nz and multiply each by the radius to go back to the other side of the wall
|
||
f32temp = 1.0f / n3XZDist;
|
||
temp_f0_2 = radius * f32temp;
|
||
posResult->x = temp_f0_2 * nx3 + posIntersect2.x;
|
||
posResult->z = temp_f0_2 * nz3 + posIntersect2.z;
|
||
*outPoly = poly;
|
||
*outBgId = bgId2;
|
||
result = true;
|
||
}
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if moving from `posPrev` to `posNext` will collide with a "wall"
|
||
* `radius` is used to form a sphere for collision detection purposes
|
||
* `checkHeight` is the positive height above posNext to perform certain checks
|
||
* returns true if a collision is detected, else false
|
||
* `outPoly` returns the closest poly detected
|
||
*/
|
||
s32 BgCheck_EntitySphVsWall1(CollisionContext* colCtx, Vec3f* posResult, Vec3f* posNext, Vec3f* posPrev, f32 radius,
|
||
CollisionPoly** outPoly, f32 checkHeight) {
|
||
s32 bgId;
|
||
|
||
return BgCheck_CheckWallImpl(colCtx, COLPOLY_IGNORE_ENTITY, posResult, posNext, posPrev, radius, outPoly, &bgId,
|
||
NULL, checkHeight, 0);
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if moving from `posPrev` to `posNext` will collide with a "wall"
|
||
* `radius` is used to form a sphere for collision detection purposes
|
||
* `checkHeight` is the positive height above posNext to perform certain checks
|
||
* returns true if a collision is detected, else false
|
||
* `outPoly` returns the closest poly detected, while `outBgId` returns the poly owner
|
||
*/
|
||
s32 BgCheck_EntitySphVsWall2(CollisionContext* colCtx, Vec3f* posResult, Vec3f* posNext, Vec3f* posPrev, f32 radius,
|
||
CollisionPoly** outPoly, s32* outBgId, f32 checkHeight) {
|
||
return BgCheck_CheckWallImpl(colCtx, COLPOLY_IGNORE_ENTITY, posResult, posNext, posPrev, radius, outPoly, outBgId,
|
||
NULL, checkHeight, 0);
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if moving from `posPrev` to `posNext` will collide with a "wall"
|
||
* `radius` is used to form a sphere for collision detection purposes
|
||
* `checkHeight` is the positive height above posNext to perform certain checks
|
||
* `actor` is the actor performing the check, allowing it to be skipped
|
||
* returns true if a collision is detected, else false
|
||
* `outPoly` returns the closest poly detected, while `outBgId` returns the poly owner
|
||
*/
|
||
s32 BgCheck_EntitySphVsWall3(CollisionContext* colCtx, Vec3f* posResult, Vec3f* posNext, Vec3f* posPrev, f32 radius,
|
||
CollisionPoly** outPoly, s32* outBgId, Actor* actor, f32 checkHeight) {
|
||
return BgCheck_CheckWallImpl(colCtx, COLPOLY_IGNORE_ENTITY, posResult, posNext, posPrev, radius, outPoly, outBgId,
|
||
actor, checkHeight, 0);
|
||
}
|
||
|
||
/***
|
||
* Public. Tests if moving from `posPrev` to `posNext` will collide with a "wall"
|
||
* Skips a check that occurs only when moving on the xz plane
|
||
* `radius` is used to form a sphere for collision detection purposes
|
||
* `checkHeight` is the positive height above posNext to perform certain checks
|
||
* `actor` is the actor performing the check, allowing it to be skipped
|
||
* returns true if a collision is detected, else false
|
||
* `outPoly` returns the closest poly detected, while `outBgId` returns the poly owner
|
||
*/
|
||
s32 BgCheck_EntitySphVsWall4(CollisionContext* colCtx, Vec3f* posResult, Vec3f* posNext, Vec3f* posPrev, f32 radius,
|
||
CollisionPoly** outPoly, s32* outBgId, Actor* actor, f32 checkHeight) {
|
||
return BgCheck_CheckWallImpl(colCtx, COLPOLY_IGNORE_ENTITY, posResult, posNext, posPrev, radius, outPoly, outBgId,
|
||
actor, checkHeight, 1);
|
||
}
|
||
|
||
/***
|
||
* Tests for collision with a ceiling poly
|
||
* `checkHeight` should be a positive value
|
||
* returns true if a collision occurs, else false
|
||
* `outPoly` returns the poly collided with, while `outBgId` returns the owner of the poly
|
||
* `outY` returns the y coordinate of pos needed to not collide with `outPoly`
|
||
*/
|
||
s32 BgCheck_CheckCeilingImpl(CollisionContext* colCtx, u16 xpFlags, f32* outY, Vec3f* pos, f32 checkHeight,
|
||
CollisionPoly** outPoly, s32* outBgId, Actor* actor) {
|
||
StaticLookup* lookupTbl;
|
||
StaticLookup* lookup;
|
||
s32 result;
|
||
Vec3f posTemp;
|
||
f32 tempY;
|
||
|
||
*outBgId = BGCHECK_SCENE;
|
||
*outY = pos->y;
|
||
|
||
#if OOT_DEBUG
|
||
if (BgCheck_PosErrorCheck(pos, "../z_bgcheck.c", 5206) == true) {
|
||
if (actor != NULL) {
|
||
PRINTF(T("こいつ,pself_actor->name %d\n", "This guy, pself_actor->name %d\n"), actor->id);
|
||
}
|
||
}
|
||
#endif
|
||
|
||
lookupTbl = colCtx->lookupTbl;
|
||
if (!BgCheck_PosInStaticBoundingBox(colCtx, pos)) {
|
||
return false;
|
||
}
|
||
|
||
lookup = BgCheck_GetNearestStaticLookup(colCtx, lookupTbl, pos);
|
||
result = BgCheck_CheckStaticCeiling(lookup, xpFlags, colCtx, outY, pos, checkHeight, outPoly);
|
||
|
||
posTemp = *pos;
|
||
posTemp.y = *outY;
|
||
tempY = *outY;
|
||
|
||
if (BgCheck_CheckDynaCeiling(colCtx, xpFlags, &tempY, &posTemp, checkHeight, outPoly, outBgId, actor)) {
|
||
*outY = tempY;
|
||
result = true;
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Tests for collision with any ceiling poly
|
||
* `checkHeight` must be a positive value
|
||
* returns true if a collision occurs, else false
|
||
* `outY` returns the displaced y coordinate needed to not collide with the poly
|
||
*/
|
||
s32 BgCheck_AnyCheckCeiling(CollisionContext* colCtx, f32* outY, Vec3f* pos, f32 checkHeight) {
|
||
CollisionPoly* poly;
|
||
s32 bgId;
|
||
|
||
return BgCheck_CheckCeilingImpl(colCtx, COLPOLY_IGNORE_NONE, outY, pos, checkHeight, &poly, &bgId, NULL);
|
||
}
|
||
|
||
/**
|
||
* Tests for collision with any entity solid ceiling poly
|
||
* `checkHeight` must be a positive value
|
||
* returns true if a collision occurs, else false
|
||
* `outY` returns the displaced y coordinate needed to not collide with the poly
|
||
*/
|
||
s32 BgCheck_EntityCheckCeiling(CollisionContext* colCtx, f32* outY, Vec3f* pos, f32 checkHeight,
|
||
CollisionPoly** outPoly, s32* outBgId, Actor* actor) {
|
||
return BgCheck_CheckCeilingImpl(colCtx, COLPOLY_IGNORE_ENTITY, outY, pos, checkHeight, outPoly, outBgId, actor);
|
||
}
|
||
|
||
/**
|
||
* Tests if a line from `posA` to `posB` intersects with a poly
|
||
* returns true if it does, else false
|
||
* `posB`? `posResult` returns the point of intersection
|
||
* `outPoly` returns the pointer to the intersected poly, while `outBgId` returns the entity the poly belongs to
|
||
*/
|
||
s32 BgCheck_CheckLineImpl(CollisionContext* colCtx, u16 xpFlags1, u16 xpFlags2, Vec3f* posA, Vec3f* posB,
|
||
Vec3f* posResult, CollisionPoly** outPoly, s32* outBgId, Actor* actor, f32 chkDist,
|
||
u32 bccFlags) {
|
||
StaticLookup* lookupTbl = colCtx->lookupTbl;
|
||
StaticLookup* iLookup;
|
||
s32 subdivMin[3];
|
||
s32 subdivMax[3];
|
||
s32 i;
|
||
s32 result;
|
||
f32 distSq;
|
||
Vec3f posBTemp = *posB;
|
||
Vec3f sectorMin;
|
||
Vec3f sectorMax;
|
||
|
||
*outBgId = BGCHECK_SCENE;
|
||
|
||
#if OOT_DEBUG
|
||
if (BgCheck_PosErrorCheck(posA, "../z_bgcheck.c", 5334) == true ||
|
||
BgCheck_PosErrorCheck(posB, "../z_bgcheck.c", 5335) == true) {
|
||
if (actor != NULL) {
|
||
PRINTF(T("こいつ,pself_actor->name %d\n", "This guy, pself_actor->name %d\n"), actor->id);
|
||
} else {
|
||
PRINTF(T("pself_actor == NULLで犯人不明\n", "pself_actor == NULL culprit unknown\n"));
|
||
}
|
||
}
|
||
#endif
|
||
|
||
BgCheck_ResetPolyCheckTbl(&colCtx->polyNodes, colCtx->colHeader->numPolygons);
|
||
BgCheck_GetStaticLookupIndicesFromPos(colCtx, posA, (Vec3i*)&subdivMin);
|
||
BgCheck_GetStaticLookupIndicesFromPos(colCtx, &posBTemp, (Vec3i*)&subdivMax);
|
||
*posResult = *posB;
|
||
result = false;
|
||
distSq = 1.0e38f;
|
||
*outPoly = NULL;
|
||
|
||
if (subdivMin[0] != subdivMax[0] || subdivMin[1] != subdivMax[1] || subdivMin[2] != subdivMax[2]) {
|
||
s32 k;
|
||
s32 temp_lo;
|
||
s32 j;
|
||
|
||
for (i = 0; i < 3; i++) {
|
||
if (subdivMax[i] < subdivMin[i]) {
|
||
j = subdivMax[i];
|
||
subdivMax[i] = subdivMin[i];
|
||
subdivMin[i] = j;
|
||
}
|
||
}
|
||
temp_lo = colCtx->subdivAmount.x * colCtx->subdivAmount.y;
|
||
iLookup = lookupTbl + subdivMin[2] * temp_lo;
|
||
sectorMin.z = subdivMin[2] * colCtx->subdivLength.z + colCtx->minBounds.z;
|
||
sectorMax.z = colCtx->subdivLength.z + sectorMin.z;
|
||
|
||
for (i = subdivMin[2]; i < subdivMax[2] + 1; i++) {
|
||
StaticLookup* jLookup = iLookup + subdivMin[1] * colCtx->subdivAmount.x;
|
||
|
||
sectorMin.y = subdivMin[1] * colCtx->subdivLength.y + colCtx->minBounds.y;
|
||
sectorMax.y = colCtx->subdivLength.y + sectorMin.y;
|
||
|
||
for (j = subdivMin[1]; j < subdivMax[1] + 1; j++) {
|
||
StaticLookup* lookup = jLookup + subdivMin[0];
|
||
|
||
sectorMin.x = subdivMin[0] * colCtx->subdivLength.x + colCtx->minBounds.x;
|
||
sectorMax.x = colCtx->subdivLength.x + sectorMin.x;
|
||
|
||
for (k = subdivMin[0]; k < subdivMax[0] + 1; k++) {
|
||
if (Math3D_LineVsCube(§orMin, §orMax, posA, &posBTemp) == true &&
|
||
BgCheck_CheckLineInSubdivision(lookup, colCtx, xpFlags1, xpFlags2, posA, &posBTemp, posResult,
|
||
outPoly, chkDist, &distSq, bccFlags)) {
|
||
result = true;
|
||
}
|
||
|
||
lookup++;
|
||
sectorMin.x += colCtx->subdivLength.x;
|
||
sectorMax.x += colCtx->subdivLength.x;
|
||
}
|
||
|
||
jLookup += colCtx->subdivAmount.x;
|
||
sectorMin.y += colCtx->subdivLength.y;
|
||
sectorMax.y += colCtx->subdivLength.y;
|
||
}
|
||
|
||
iLookup += temp_lo;
|
||
sectorMin.z += colCtx->subdivLength.z;
|
||
sectorMax.z += colCtx->subdivLength.z;
|
||
}
|
||
} else if (!BgCheck_PosInStaticBoundingBox(colCtx, posA)) {
|
||
return false;
|
||
} else {
|
||
result =
|
||
BgCheck_CheckLineInSubdivision(BgCheck_GetNearestStaticLookup(colCtx, lookupTbl, posA), colCtx, xpFlags1,
|
||
xpFlags2, posA, &posBTemp, posResult, outPoly, chkDist, &distSq, bccFlags);
|
||
if (result == true) {
|
||
distSq = Math3D_Vec3fDistSq(posResult, posA);
|
||
}
|
||
}
|
||
if ((bccFlags & BGCHECK_CHECK_DYNA) &&
|
||
BgCheck_CheckLineAgainstDyna(colCtx, xpFlags1, posA, &posBTemp, posResult, outPoly, &distSq, outBgId, actor,
|
||
chkDist, bccFlags)) {
|
||
result = true;
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Get bccFlags
|
||
*/
|
||
u32 BgCheck_GetBccFlags(s32 chkWall, s32 chkFloor, s32 chkCeil, s32 chkOneFace, s32 chkDyna) {
|
||
u32 result = 0;
|
||
|
||
if (chkWall) {
|
||
result = BGCHECK_CHECK_WALL;
|
||
}
|
||
if (chkFloor) {
|
||
result |= BGCHECK_CHECK_FLOOR;
|
||
}
|
||
if (chkCeil) {
|
||
result |= BGCHECK_CHECK_CEILING;
|
||
}
|
||
if (chkOneFace) {
|
||
result |= BGCHECK_CHECK_ONE_FACE;
|
||
}
|
||
if (chkDyna) {
|
||
result |= BGCHECK_CHECK_DYNA;
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if a line from `posA` to `posB` intersects with a poly
|
||
* returns true if it does, else false
|
||
*/
|
||
s32 BgCheck_CameraLineTest1(CollisionContext* colCtx, Vec3f* posA, Vec3f* posB, Vec3f* posResult,
|
||
CollisionPoly** outPoly, s32 chkWall, s32 chkFloor, s32 chkCeil, s32 chkOneFace,
|
||
s32* bgId) {
|
||
return BgCheck_CheckLineImpl(colCtx, COLPOLY_IGNORE_CAMERA, COLPOLY_IGNORE_NONE, posA, posB, posResult, outPoly,
|
||
bgId, NULL, 1.0f, BgCheck_GetBccFlags(chkWall, chkFloor, chkCeil, chkOneFace, true));
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if a line from `posA` to `posB` intersects with a poly
|
||
* returns true if it does, else false
|
||
*/
|
||
s32 BgCheck_CameraLineTest2(CollisionContext* colCtx, Vec3f* posA, Vec3f* posB, Vec3f* posResult,
|
||
CollisionPoly** outPoly, s32 chkWall, s32 chkFloor, s32 chkCeil, s32 chkOneFace,
|
||
s32* bgId) {
|
||
return BgCheck_CheckLineImpl(colCtx, COLPOLY_IGNORE_NONE, COLPOLY_IGNORE_CAMERA, posA, posB, posResult, outPoly,
|
||
bgId, NULL, 1.0f, BgCheck_GetBccFlags(chkWall, chkFloor, chkCeil, chkOneFace, true));
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if a line from `posA` to `posB` intersects with a poly
|
||
* returns true if it does, else false
|
||
*/
|
||
s32 BgCheck_EntityLineTest1(CollisionContext* colCtx, Vec3f* posA, Vec3f* posB, Vec3f* posResult,
|
||
CollisionPoly** outPoly, s32 chkWall, s32 chkFloor, s32 chkCeil, s32 chkOneFace,
|
||
s32* bgId) {
|
||
return BgCheck_CheckLineImpl(colCtx, COLPOLY_IGNORE_ENTITY, COLPOLY_IGNORE_NONE, posA, posB, posResult, outPoly,
|
||
bgId, NULL, 1.0f, BgCheck_GetBccFlags(chkWall, chkFloor, chkCeil, chkOneFace, true));
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if a line from `posA` to `posB` intersects with a poly
|
||
* returns true if it does, else false
|
||
*/
|
||
s32 BgCheck_EntityLineTest2(CollisionContext* colCtx, Vec3f* posA, Vec3f* posB, Vec3f* posResult,
|
||
CollisionPoly** outPoly, s32 chkWall, s32 chkFloor, s32 chkCeil, s32 chkOneFace, s32* bgId,
|
||
Actor* actor) {
|
||
return BgCheck_CheckLineImpl(colCtx, COLPOLY_IGNORE_ENTITY, COLPOLY_IGNORE_NONE, posA, posB, posResult, outPoly,
|
||
bgId, actor, 1.0f, BgCheck_GetBccFlags(chkWall, chkFloor, chkCeil, chkOneFace, true));
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if a line from `posA` to `posB` intersects with a poly
|
||
* returns true if it does, else false
|
||
*/
|
||
s32 BgCheck_EntityLineTest3(CollisionContext* colCtx, Vec3f* posA, Vec3f* posB, Vec3f* posResult,
|
||
CollisionPoly** outPoly, s32 chkWall, s32 chkFloor, s32 chkCeil, s32 chkOneFace, s32* bgId,
|
||
Actor* actor, f32 chkDist) {
|
||
return BgCheck_CheckLineImpl(colCtx, COLPOLY_IGNORE_ENTITY, COLPOLY_IGNORE_NONE, posA, posB, posResult, outPoly,
|
||
bgId, actor, chkDist,
|
||
BgCheck_GetBccFlags(chkWall, chkFloor, chkCeil, chkOneFace, true));
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if a line from `posA` to `posB` intersects with a poly
|
||
* returns true if it does, else false
|
||
*/
|
||
s32 BgCheck_ProjectileLineTest(CollisionContext* colCtx, Vec3f* posA, Vec3f* posB, Vec3f* posResult,
|
||
CollisionPoly** outPoly, s32 chkWall, s32 chkFloor, s32 chkCeil, s32 chkOneFace,
|
||
s32* bgId) {
|
||
return BgCheck_CheckLineImpl(colCtx, COLPOLY_IGNORE_PROJECTILES, COLPOLY_IGNORE_NONE, posA, posB, posResult,
|
||
outPoly, bgId, NULL, 1.0f,
|
||
BgCheck_GetBccFlags(chkWall, chkFloor, chkCeil, chkOneFace, true));
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if a line from `posA` to `posB` intersects with a poly
|
||
* returns true if it does, else false
|
||
*/
|
||
s32 BgCheck_AnyLineTest1(CollisionContext* colCtx, Vec3f* posA, Vec3f* posB, Vec3f* posResult, CollisionPoly** outPoly,
|
||
s32 chkOneFace) {
|
||
return BgCheck_AnyLineTest2(colCtx, posA, posB, posResult, outPoly, true, true, true, chkOneFace);
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if a line from `posA` to `posB` intersects with a poly
|
||
* returns true if it does, else false
|
||
*/
|
||
s32 BgCheck_AnyLineTest2(CollisionContext* colCtx, Vec3f* posA, Vec3f* posB, Vec3f* posResult, CollisionPoly** outPoly,
|
||
s32 chkWall, s32 chkFloor, s32 chkCeil, s32 chkOneFace) {
|
||
s32 bgId;
|
||
|
||
return BgCheck_CheckLineImpl(colCtx, COLPOLY_IGNORE_NONE, COLPOLY_IGNORE_NONE, posA, posB, posResult, outPoly,
|
||
&bgId, NULL, 1.0f, BgCheck_GetBccFlags(chkWall, chkFloor, chkCeil, chkOneFace, true));
|
||
}
|
||
|
||
/**
|
||
* Public. Tests if a line from `posA` to `posB` intersects with a poly
|
||
* returns true if it does, else false
|
||
*/
|
||
s32 BgCheck_AnyLineTest3(CollisionContext* colCtx, Vec3f* posA, Vec3f* posB, Vec3f* posResult, CollisionPoly** outPoly,
|
||
s32 chkWall, s32 chkFloor, s32 chkCeil, s32 chkOneFace, s32* bgId) {
|
||
return BgCheck_CheckLineImpl(colCtx, COLPOLY_IGNORE_NONE, COLPOLY_IGNORE_NONE, posA, posB, posResult, outPoly, bgId,
|
||
NULL, 1.0f, BgCheck_GetBccFlags(chkWall, chkFloor, chkCeil, chkOneFace, true));
|
||
}
|
||
|
||
/**
|
||
* Get first poly intersecting sphere `center` `radius`
|
||
* ignores `actor` dyna poly
|
||
* returns true if any poly intersects the sphere, else false
|
||
* `outPoly` returns the pointer of the first poly found that intersects
|
||
* `outBgId` returns the bgId of the entity that owns `outPoly`
|
||
*/
|
||
s32 BgCheck_SphVsFirstPolyImpl(CollisionContext* colCtx, u16 xpFlags, CollisionPoly** outPoly, s32* outBgId,
|
||
Vec3f* center, f32 radius, Actor* actor, u16 bciFlags) {
|
||
StaticLookup* lookup;
|
||
|
||
*outBgId = BGCHECK_SCENE;
|
||
|
||
#if OOT_DEBUG
|
||
if (BgCheck_PosErrorCheck(center, "../z_bgcheck.c", 5852) == true) {
|
||
if (actor != NULL) {
|
||
PRINTF(T("こいつ,pself_actor->name %d\n", "This guy, pself_actor->name %d\n"), actor->id);
|
||
}
|
||
}
|
||
#endif
|
||
|
||
lookup = BgCheck_GetStaticLookup(colCtx, colCtx->lookupTbl, center);
|
||
if (lookup == NULL) {
|
||
return false;
|
||
} else if (BgCheck_SphVsFirstStaticPoly(lookup, xpFlags, colCtx, center, radius, outPoly, bciFlags) ||
|
||
BgCheck_SphVsFirstDynaPoly(colCtx, xpFlags, outPoly, outBgId, center, radius, actor, bciFlags)) {
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Public get first poly intersecting sphere `center` `radius`
|
||
*/
|
||
s32 BgCheck_SphVsFirstPoly(CollisionContext* colCtx, Vec3f* center, f32 radius) {
|
||
CollisionPoly* poly;
|
||
s32 bgId;
|
||
|
||
return BgCheck_SphVsFirstPolyImpl(colCtx, COLPOLY_IGNORE_NONE, &poly, &bgId, center, radius, NULL,
|
||
BGCHECK_IGNORE_NONE);
|
||
}
|
||
|
||
/**
|
||
* Public get first wall poly intersecting sphere `center` `radius`
|
||
*/
|
||
s32 BgCheck_SphVsFirstWall(CollisionContext* colCtx, Vec3f* center, f32 radius) {
|
||
CollisionPoly* poly;
|
||
s32 bgId;
|
||
|
||
return BgCheck_SphVsFirstPolyImpl(colCtx, COLPOLY_IGNORE_NONE, &poly, &bgId, center, radius, NULL,
|
||
BGCHECK_IGNORE_FLOOR | BGCHECK_IGNORE_CEILING);
|
||
}
|
||
|
||
/**
|
||
* Init SSNodeList
|
||
*/
|
||
void SSNodeList_Initialize(SSNodeList* this) {
|
||
this->max = 0;
|
||
this->count = 0;
|
||
this->tbl = NULL;
|
||
this->polyCheckTbl = NULL;
|
||
}
|
||
|
||
/**
|
||
* Allocate SSNodeList
|
||
* tblMax is the number of SSNode records to allocate
|
||
* numPolys is the number of polygons defined within the CollisionHeader
|
||
*/
|
||
void SSNodeList_Alloc(PlayState* play, SSNodeList* this, s32 tblMax, s32 numPolys) {
|
||
this->max = tblMax;
|
||
this->count = 0;
|
||
this->tbl = THA_AllocTailAlign(&play->state.tha, tblMax * sizeof(SSNode), ALIGNOF_MASK(SSNode));
|
||
|
||
ASSERT(this->tbl != NULL, "this->short_slist_node_tbl != NULL", "../z_bgcheck.c", 5975);
|
||
|
||
this->polyCheckTbl = GAME_STATE_ALLOC(&play->state, numPolys, "../z_bgcheck.c", 5979);
|
||
|
||
ASSERT(this->polyCheckTbl != NULL, "this->polygon_check != NULL", "../z_bgcheck.c", 5981);
|
||
}
|
||
|
||
/**
|
||
* Get next SSNodeList SSNode
|
||
*/
|
||
SSNode* SSNodeList_GetNextNode(SSNodeList* this) {
|
||
SSNode* result = &this->tbl[this->count];
|
||
|
||
this->count++;
|
||
ASSERT(this->count < this->max, "this->short_slist_node_last_index < this->short_slist_node_size", "../z_bgcheck.c",
|
||
5998);
|
||
if (!(this->count < this->max)) {
|
||
return NULL;
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Get next SSNodeList SSNode index
|
||
*/
|
||
u16 SSNodeList_GetNextNodeIdx(SSNodeList* this) {
|
||
u16 new_index = this->count++;
|
||
|
||
ASSERT(new_index < this->max, "new_index < this->short_slist_node_size", "../z_bgcheck.c", 6021);
|
||
return new_index;
|
||
}
|
||
|
||
/**
|
||
* Initialize ScaleRotPos
|
||
*/
|
||
void ScaleRotPos_Initialize(ScaleRotPos* srp) {
|
||
srp->scale.x = srp->scale.y = srp->scale.z = 1.0f;
|
||
srp->pos.x = srp->pos.y = srp->pos.z = 0.0f;
|
||
srp->rot.x = srp->rot.y = srp->rot.z = 0;
|
||
}
|
||
|
||
/**
|
||
* Set ScaleRotPos
|
||
*/
|
||
void ScaleRotPos_SetValue(ScaleRotPos* srp, Vec3f* scale, Vec3s* rot, Vec3f* pos) {
|
||
srp->scale = *scale;
|
||
srp->rot = *rot;
|
||
srp->pos = *pos;
|
||
}
|
||
|
||
/**
|
||
* ScaleRotPos equality test
|
||
*/
|
||
s32 ScaleRotPos_Equals(ScaleRotPos* a, ScaleRotPos* b) {
|
||
if (a->scale.x != b->scale.x || a->scale.y != b->scale.y || a->scale.z != b->scale.z || a->rot.x != b->rot.x ||
|
||
a->rot.y != b->rot.y || a->rot.z != b->rot.z || a->pos.x != b->pos.x || a->pos.y != b->pos.y ||
|
||
a->pos.z != b->pos.z) {
|
||
return false;
|
||
}
|
||
return true;
|
||
}
|
||
|
||
/**
|
||
* Reset DynaLookup lists
|
||
*/
|
||
void DynaLookup_ResetLists(DynaLookup* dynaLookup) {
|
||
SSList_SetNull(&dynaLookup->ceiling);
|
||
SSList_SetNull(&dynaLookup->wall);
|
||
SSList_SetNull(&dynaLookup->floor);
|
||
}
|
||
|
||
/**
|
||
* Reset DynaLookup
|
||
*/
|
||
void DynaLookup_Reset(DynaLookup* dynaLookup) {
|
||
dynaLookup->polyStartIndex = 0;
|
||
DynaLookup_ResetLists(dynaLookup);
|
||
}
|
||
|
||
/**
|
||
* Reset vtxStartIndex
|
||
*/
|
||
void DynaLookup_ResetVtxStartIndex(u16* vtxStartIndex) {
|
||
*vtxStartIndex = 0;
|
||
}
|
||
|
||
/**
|
||
* Initialize BgActor
|
||
*/
|
||
void BgActor_Initialize(PlayState* play, BgActor* bgActor) {
|
||
bgActor->actor = NULL;
|
||
bgActor->colHeader = NULL;
|
||
ScaleRotPos_Initialize(&bgActor->prevTransform);
|
||
ScaleRotPos_Initialize(&bgActor->curTransform);
|
||
DynaLookup_Reset(&bgActor->dynaLookup);
|
||
DynaLookup_ResetVtxStartIndex(&bgActor->vtxStartIndex);
|
||
bgActor->boundingSphere.center.x = bgActor->boundingSphere.center.y = bgActor->boundingSphere.center.z = 0;
|
||
bgActor->boundingSphere.radius = 0;
|
||
}
|
||
|
||
/**
|
||
* setActor internal
|
||
*/
|
||
void BgActor_SetActor(BgActor* bgActor, Actor* actor, CollisionHeader* colHeader) {
|
||
bgActor->actor = actor;
|
||
bgActor->colHeader = colHeader;
|
||
bgActor->prevTransform.scale = actor->scale;
|
||
bgActor->prevTransform.rot = actor->shape.rot;
|
||
bgActor->prevTransform.rot.x--;
|
||
bgActor->prevTransform.pos = actor->world.pos;
|
||
bgActor->curTransform.scale = actor->scale;
|
||
bgActor->curTransform.rot = actor->shape.rot;
|
||
bgActor->curTransform.pos = actor->world.pos;
|
||
}
|
||
|
||
/**
|
||
* Test if the BgActor transform is the same
|
||
*/
|
||
s32 BgActor_IsTransformUnchanged(BgActor* bgActor) {
|
||
return ScaleRotPos_Equals(&bgActor->prevTransform, &bgActor->curTransform);
|
||
}
|
||
|
||
/**
|
||
* NULL polyList
|
||
*/
|
||
void DynaPoly_NullPolyList(CollisionPoly** polyList) {
|
||
*polyList = NULL;
|
||
}
|
||
|
||
/**
|
||
* Allocate dyna.polyList
|
||
*/
|
||
void DynaPoly_AllocPolyList(PlayState* play, CollisionPoly** polyList, s32 numPolys) {
|
||
*polyList = THA_AllocTailAlign(&play->state.tha, numPolys * sizeof(CollisionPoly), ALIGNOF_MASK(CollisionPoly));
|
||
ASSERT(*polyList != NULL, "ptbl->pbuf != NULL", "../z_bgcheck.c", 6247);
|
||
}
|
||
|
||
/**
|
||
* NULL vtxList
|
||
*/
|
||
void DynaPoly_NullVtxList(Vec3s** vtxList) {
|
||
*vtxList = NULL;
|
||
}
|
||
|
||
/**
|
||
* Allocate dyna.vtxList
|
||
*/
|
||
void DynaPoly_AllocVtxList(PlayState* play, Vec3s** vtxList, s32 numVtx) {
|
||
*vtxList = THA_AllocTailAlign(&play->state.tha, numVtx * sizeof(Vec3s), ALIGNOF_MASK(Vec3s));
|
||
ASSERT(*vtxList != NULL, "ptbl->pbuf != NULL", "../z_bgcheck.c", 6277);
|
||
}
|
||
|
||
/**
|
||
* Update BgActor's prevTransform
|
||
*/
|
||
void DynaPoly_SetBgActorPrevTransform(PlayState* play, BgActor* bgActor) {
|
||
bgActor->prevTransform = bgActor->curTransform;
|
||
}
|
||
|
||
/**
|
||
* Is BgActor Id
|
||
*/
|
||
s32 DynaPoly_IsBgIdBgActor(s32 bgId) {
|
||
if (bgId < 0 || bgId >= BG_ACTOR_MAX) {
|
||
return false;
|
||
}
|
||
return true;
|
||
}
|
||
|
||
/**
|
||
* Init DynaCollisionContext
|
||
*/
|
||
void DynaPoly_Init(PlayState* play, DynaCollisionContext* dyna) {
|
||
dyna->bitFlag = DYNAPOLY_INVALIDATE_LOOKUP;
|
||
DynaPoly_NullPolyList(&dyna->polyList);
|
||
DynaPoly_NullVtxList(&dyna->vtxList);
|
||
DynaSSNodeList_Initialize(play, &dyna->polyNodes);
|
||
}
|
||
|
||
/**
|
||
* Set DynaCollisionContext
|
||
*/
|
||
void DynaPoly_Alloc(PlayState* play, DynaCollisionContext* dyna) {
|
||
s32 i;
|
||
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
BgActor_Initialize(play, &dyna->bgActors[i]);
|
||
dyna->bgActorFlags[i] = 0;
|
||
}
|
||
DynaPoly_NullPolyList(&dyna->polyList);
|
||
DynaPoly_AllocPolyList(play, &dyna->polyList, dyna->polyListMax);
|
||
|
||
DynaPoly_NullVtxList(&dyna->vtxList);
|
||
DynaPoly_AllocVtxList(play, &dyna->vtxList, dyna->vtxListMax);
|
||
|
||
DynaSSNodeList_Initialize(play, &dyna->polyNodes);
|
||
DynaSSNodeList_Alloc(play, &dyna->polyNodes, dyna->polyNodesMax);
|
||
}
|
||
|
||
/**
|
||
* Set BgActor
|
||
* original name: DynaPolyInfo_setActor
|
||
*/
|
||
s32 DynaPoly_SetBgActor(PlayState* play, DynaCollisionContext* dyna, Actor* actor, CollisionHeader* colHeader) {
|
||
s32 bgId;
|
||
s32 foundSlot = false;
|
||
|
||
for (bgId = 0; bgId < BG_ACTOR_MAX; bgId++) {
|
||
if (!(dyna->bgActorFlags[bgId] & BGACTOR_IN_USE)) {
|
||
dyna->bgActorFlags[bgId] |= BGACTOR_IN_USE;
|
||
foundSlot = true;
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (!foundSlot) {
|
||
PRINTF(VT_FGCOL(RED));
|
||
PRINTF(T("DynaPolyInfo_setActor():ダイナミックポリゴン 空きインデックスはありません\n",
|
||
"DynaPolyInfo_setActor(): Dynamic polygon no free indexes\n"));
|
||
PRINTF(VT_RST);
|
||
return BG_ACTOR_MAX;
|
||
}
|
||
|
||
BgActor_SetActor(&dyna->bgActors[bgId], actor, colHeader);
|
||
dyna->bitFlag |= DYNAPOLY_INVALIDATE_LOOKUP;
|
||
|
||
dyna->bgActorFlags[bgId] &= ~BGACTOR_1;
|
||
PRINTF(VT_FGCOL(GREEN));
|
||
PRINTF("DynaPolyInfo_setActor():index %d\n", bgId);
|
||
PRINTF(VT_RST);
|
||
return bgId;
|
||
}
|
||
|
||
/**
|
||
* Gets the actor assigned to `bgId`
|
||
* possible orginal name: DynaPolyInfo_getActor
|
||
*/
|
||
DynaPolyActor* DynaPoly_GetActor(CollisionContext* colCtx, s32 bgId) {
|
||
if (!DynaPoly_IsBgIdBgActor(bgId) || !(colCtx->dyna.bgActorFlags[bgId] & BGACTOR_IN_USE) ||
|
||
colCtx->dyna.bgActorFlags[bgId] & BGACTOR_1) {
|
||
return NULL;
|
||
}
|
||
return (DynaPolyActor*)colCtx->dyna.bgActors[bgId].actor;
|
||
}
|
||
|
||
void DynaPoly_DisableCollision(PlayState* play, DynaCollisionContext* dyna, s32 bgId) {
|
||
if (DynaPoly_IsBgIdBgActor(bgId)) {
|
||
dyna->bgActorFlags[bgId] |= BGACTOR_COLLISION_DISABLED;
|
||
dyna->bitFlag |= DYNAPOLY_INVALIDATE_LOOKUP;
|
||
}
|
||
}
|
||
|
||
void DynaPoly_EnableCollision(PlayState* play, DynaCollisionContext* dyna, s32 bgId) {
|
||
if (DynaPoly_IsBgIdBgActor(bgId)) {
|
||
dyna->bgActorFlags[bgId] &= ~BGACTOR_COLLISION_DISABLED;
|
||
dyna->bitFlag |= DYNAPOLY_INVALIDATE_LOOKUP;
|
||
}
|
||
}
|
||
|
||
void DynaPoly_DisableCeilingCollision(PlayState* play, DynaCollisionContext* dyna, s32 bgId) {
|
||
if (DynaPoly_IsBgIdBgActor(bgId)) {
|
||
dyna->bgActorFlags[bgId] |= BGACTOR_CEILING_COLLISION_DISABLED;
|
||
dyna->bitFlag |= DYNAPOLY_INVALIDATE_LOOKUP;
|
||
}
|
||
}
|
||
|
||
void DynaPoly_EnableCeilingCollision(PlayState* play, DynaCollisionContext* dyna, s32 bgId) {
|
||
if (DynaPoly_IsBgIdBgActor(bgId)) {
|
||
dyna->bgActorFlags[bgId] &= ~BGACTOR_CEILING_COLLISION_DISABLED;
|
||
dyna->bitFlag |= DYNAPOLY_INVALIDATE_LOOKUP;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* original name: DynaPolyInfo_delReserve
|
||
*/
|
||
void DynaPoly_DeleteBgActor(PlayState* play, DynaCollisionContext* dyna, s32 bgId) {
|
||
DynaPolyActor* actor;
|
||
|
||
PRINTF(VT_FGCOL(GREEN));
|
||
PRINTF("DynaPolyInfo_delReserve():index %d\n", bgId);
|
||
PRINTF(VT_RST);
|
||
if (!DynaPoly_IsBgIdBgActor(bgId)) {
|
||
|
||
#if OOT_DEBUG
|
||
if (bgId == -1) {
|
||
PRINTF(VT_FGCOL(GREEN));
|
||
PRINTF(T("DynaPolyInfo_delReserve():削除されているはずの(?)\n"
|
||
"インデックス(== -1)のため,処理を中止します。\n",
|
||
"DynaPolyInfo_delReserve():The index that should have been deleted(?)\n"
|
||
" was(== -1), processing aborted.\n"));
|
||
PRINTF(VT_RST);
|
||
} else {
|
||
PRINTF(VT_FGCOL(RED));
|
||
PRINTF(T("DynaPolyInfo_delReserve():"
|
||
"確保していない/出来なかったインデックスの解放のため、処理を中止します。index == %d\n",
|
||
"DynaPolyInfo_delReserve():"
|
||
" Unable to deallocate index / index unallocated, processing aborted. index == %d\n"),
|
||
bgId);
|
||
PRINTF(VT_RST);
|
||
}
|
||
#endif
|
||
|
||
} else {
|
||
actor = DynaPoly_GetActor(&play->colCtx, bgId);
|
||
if (actor != NULL) {
|
||
|
||
actor->bgId = BGACTOR_NEG_ONE;
|
||
dyna->bgActors[bgId].actor = NULL;
|
||
dyna->bgActorFlags[bgId] |= BGACTOR_1;
|
||
}
|
||
}
|
||
}
|
||
|
||
void DynaPoly_InvalidateLookup(PlayState* play, DynaCollisionContext* dyna) {
|
||
dyna->bitFlag |= DYNAPOLY_INVALIDATE_LOOKUP;
|
||
}
|
||
|
||
/**
|
||
* original name: DynaPolyInfo_expandSRT
|
||
*/
|
||
void DynaPoly_AddBgActorToLookup(PlayState* play, DynaCollisionContext* dyna, s32 bgId, s32* vtxStartIndex,
|
||
s32* polyStartIndex) {
|
||
MtxF mtx;
|
||
Actor* actor;
|
||
s32 pad;
|
||
f32 radiusSq;
|
||
f32 numVtxInverse;
|
||
s32 i;
|
||
Vec3f pos;
|
||
Sphere16* sphere;
|
||
Vec3s* dVtxList;
|
||
Vec3s* point;
|
||
Vec3f newCenterPoint;
|
||
f32 newRadiusSq;
|
||
CollisionHeader* pbgdata;
|
||
Vec3f newVtx;
|
||
Vec3f vtxA;
|
||
Vec3f vtxB;
|
||
Vec3f vtxC;
|
||
Vec3f newNormal;
|
||
|
||
pbgdata = dyna->bgActors[bgId].colHeader;
|
||
sphere = &dyna->bgActors[bgId].boundingSphere;
|
||
actor = dyna->bgActors[bgId].actor;
|
||
dyna->bgActors[bgId].dynaLookup.polyStartIndex = *polyStartIndex;
|
||
dyna->bgActors[bgId].vtxStartIndex = *vtxStartIndex;
|
||
pos = actor->world.pos;
|
||
pos.y += actor->shape.yOffset * actor->scale.y;
|
||
|
||
ScaleRotPos_SetValue(&dyna->bgActors[bgId].curTransform, &actor->scale, &actor->shape.rot, &pos);
|
||
|
||
if (dyna->bgActorFlags[bgId] & BGACTOR_COLLISION_DISABLED) {
|
||
return;
|
||
}
|
||
|
||
#if OOT_DEBUG
|
||
if (!(dyna->polyListMax >= *polyStartIndex + pbgdata->numPolygons)) {
|
||
PRINTF(VT_FGCOL(RED));
|
||
PRINTF(T("DynaPolyInfo_expandSRT():polygon over %dが%dを越えるとダメ\n",
|
||
"DynaPolyInfo_expandSRT():polygon over do not use if %d exceeds %d\n"),
|
||
*polyStartIndex + pbgdata->numPolygons, dyna->polyListMax);
|
||
}
|
||
|
||
if (!(dyna->vtxListMax >= *vtxStartIndex + pbgdata->numVertices)) {
|
||
PRINTF(VT_FGCOL(RED));
|
||
PRINTF(T("DynaPolyInfo_expandSRT():vertex over %dが%dを越えるとダメ\n",
|
||
"DynaPolyInfo_expandSRT():vertex over do not use if %d exceeds %d\n"),
|
||
*vtxStartIndex + pbgdata->numVertices, dyna->vtxListMax);
|
||
}
|
||
|
||
ASSERT(dyna->polyListMax >= *polyStartIndex + pbgdata->numPolygons,
|
||
"pdyna_poly_info->poly_num >= *pstart_poly_index + pbgdata->poly_num", "../z_bgcheck.c", 6687);
|
||
ASSERT(dyna->vtxListMax >= *vtxStartIndex + pbgdata->numVertices,
|
||
"pdyna_poly_info->vert_num >= *pstart_vert_index + pbgdata->vtx_num", "../z_bgcheck.c", 6688);
|
||
#endif
|
||
|
||
if (!(dyna->bitFlag & DYNAPOLY_INVALIDATE_LOOKUP) &&
|
||
(BgActor_IsTransformUnchanged(&dyna->bgActors[bgId]) == true)) {
|
||
s32 pi;
|
||
|
||
for (pi = *polyStartIndex; pi < *polyStartIndex + pbgdata->numPolygons; pi++) {
|
||
CollisionPoly* poly = &dyna->polyList[pi];
|
||
s16 normalY = poly->normal.y;
|
||
|
||
if (normalY > COLPOLY_SNORMAL(0.5f)) {
|
||
s16 polyIndex = pi;
|
||
|
||
DynaSSNodeList_SetSSListHead(&dyna->polyNodes, &dyna->bgActors[bgId].dynaLookup.floor, &polyIndex);
|
||
} else if (normalY < COLPOLY_SNORMAL(-0.8f)) {
|
||
if (!(dyna->bgActorFlags[bgId] & BGACTOR_CEILING_COLLISION_DISABLED)) {
|
||
s16 polyIndex = pi;
|
||
|
||
DynaSSNodeList_SetSSListHead(&dyna->polyNodes, &dyna->bgActors[bgId].dynaLookup.ceiling,
|
||
&polyIndex);
|
||
}
|
||
} else {
|
||
s16 polyIndex = pi;
|
||
|
||
DynaSSNodeList_SetSSListHead(&dyna->polyNodes, &dyna->bgActors[bgId].dynaLookup.wall, &polyIndex);
|
||
}
|
||
}
|
||
|
||
*polyStartIndex += pbgdata->numPolygons;
|
||
*vtxStartIndex += pbgdata->numVertices;
|
||
return;
|
||
}
|
||
|
||
SkinMatrix_SetTranslateRotateYXZScale(
|
||
&mtx, dyna->bgActors[bgId].curTransform.scale.x, dyna->bgActors[bgId].curTransform.scale.y,
|
||
dyna->bgActors[bgId].curTransform.scale.z, dyna->bgActors[bgId].curTransform.rot.x,
|
||
dyna->bgActors[bgId].curTransform.rot.y, dyna->bgActors[bgId].curTransform.rot.z,
|
||
dyna->bgActors[bgId].curTransform.pos.x, dyna->bgActors[bgId].curTransform.pos.y,
|
||
dyna->bgActors[bgId].curTransform.pos.z);
|
||
|
||
numVtxInverse = 1.0f / pbgdata->numVertices;
|
||
newCenterPoint.x = newCenterPoint.y = newCenterPoint.z = 0.0f;
|
||
for (i = 0; i < pbgdata->numVertices; i++) {
|
||
Vec3f vtx;
|
||
Vec3f vtxT; // Vtx after mtx transform
|
||
s32 pad2;
|
||
|
||
Math_Vec3s_ToVec3f(&vtx, &pbgdata->vtxList[i]);
|
||
SkinMatrix_Vec3fMtxFMultXYZ(&mtx, &vtx, &vtxT);
|
||
BgCheck_Vec3fToVec3s(&dyna->vtxList[*vtxStartIndex + i], &vtxT);
|
||
|
||
if (i == 0) {
|
||
dyna->bgActors[bgId].minY = dyna->bgActors[bgId].maxY = vtxT.y;
|
||
} else if (vtxT.y < dyna->bgActors[bgId].minY) {
|
||
dyna->bgActors[bgId].minY = vtxT.y;
|
||
} else if (dyna->bgActors[bgId].maxY < vtxT.y) {
|
||
dyna->bgActors[bgId].maxY = vtxT.y;
|
||
}
|
||
newCenterPoint.x += vtxT.x;
|
||
newCenterPoint.y += vtxT.y;
|
||
newCenterPoint.z += vtxT.z;
|
||
}
|
||
|
||
newCenterPoint.x *= numVtxInverse;
|
||
newCenterPoint.y *= numVtxInverse;
|
||
newCenterPoint.z *= numVtxInverse;
|
||
sphere->center.x = newCenterPoint.x;
|
||
sphere->center.y = newCenterPoint.y;
|
||
sphere->center.z = newCenterPoint.z;
|
||
newRadiusSq = -SQ(10.0f);
|
||
|
||
for (i = 0; i < pbgdata->numVertices; i++) {
|
||
newVtx.x = dyna->vtxList[*vtxStartIndex + i].x;
|
||
newVtx.y = dyna->vtxList[*vtxStartIndex + i].y;
|
||
newVtx.z = dyna->vtxList[*vtxStartIndex + i].z;
|
||
radiusSq = Math3D_Vec3fDistSq(&newVtx, &newCenterPoint);
|
||
if (newRadiusSq < radiusSq) {
|
||
newRadiusSq = radiusSq;
|
||
}
|
||
}
|
||
|
||
sphere->radius = sqrtf(newRadiusSq) * 1.1f;
|
||
|
||
for (i = 0; i < pbgdata->numPolygons; i++) {
|
||
CollisionPoly* newPoly = &dyna->polyList[*polyStartIndex + i];
|
||
f32 newNormMagnitude;
|
||
|
||
*newPoly = pbgdata->polyList[i];
|
||
|
||
// Yeah, this is all kinds of fake, but my God, it matches.
|
||
newPoly->flags_vIA =
|
||
(COLPOLY_VTX_INDEX(newPoly->flags_vIA) + *vtxStartIndex) | COLPOLY_VTX_FLAGS_MASKED((*newPoly).flags_vIA);
|
||
newPoly->flags_vIB =
|
||
(COLPOLY_VTX_INDEX(newPoly->flags_vIB) + *vtxStartIndex) | COLPOLY_VTX_FLAGS_MASKED((*newPoly).flags_vIB);
|
||
newPoly->vIC = *vtxStartIndex + newPoly->vIC;
|
||
dVtxList = dyna->vtxList;
|
||
vtxA.x = dVtxList[(u32)COLPOLY_VTX_INDEX(newPoly->flags_vIA)].x;
|
||
vtxA.y = dVtxList[(u32)COLPOLY_VTX_INDEX(newPoly->flags_vIA)].y;
|
||
vtxA.z = dVtxList[(u32)COLPOLY_VTX_INDEX(newPoly->flags_vIA)].z;
|
||
vtxB.x = dVtxList[(u32)COLPOLY_VTX_INDEX(newPoly->flags_vIB)].x;
|
||
vtxB.y = dVtxList[(u32)COLPOLY_VTX_INDEX(newPoly->flags_vIB)].y;
|
||
vtxB.z = dVtxList[(u32)COLPOLY_VTX_INDEX(newPoly->flags_vIB)].z;
|
||
vtxC.x = dVtxList[newPoly->vIC].x;
|
||
vtxC.y = dVtxList[newPoly->vIC].y;
|
||
vtxC.z = dVtxList[newPoly->vIC].z;
|
||
Math3D_SurfaceNorm(&vtxA, &vtxB, &vtxC, &newNormal);
|
||
newNormMagnitude = Math3D_Vec3fMagnitude(&newNormal);
|
||
|
||
if (!IS_ZERO(newNormMagnitude)) {
|
||
newNormal.x *= (1.0f / newNormMagnitude);
|
||
newNormal.y *= (1.0f / newNormMagnitude);
|
||
newNormal.z *= (1.0f / newNormMagnitude);
|
||
newPoly->normal.x = COLPOLY_SNORMAL(newNormal.x);
|
||
newPoly->normal.y = COLPOLY_SNORMAL(newNormal.y);
|
||
newPoly->normal.z = COLPOLY_SNORMAL(newNormal.z);
|
||
}
|
||
|
||
newPoly->dist = -DOTXYZ(newNormal, dVtxList[(u32)COLPOLY_VTX_INDEX(newPoly->flags_vIA)]);
|
||
if (newNormal.y > 0.5f) {
|
||
s16 polyId = *polyStartIndex + i;
|
||
|
||
DynaSSNodeList_SetSSListHead(&dyna->polyNodes, &dyna->bgActors[bgId].dynaLookup.floor, &polyId);
|
||
} else if (newNormal.y < -0.8f) {
|
||
s16 polyId = *polyStartIndex + i;
|
||
|
||
DynaSSNodeList_SetSSListHead(&dyna->polyNodes, &dyna->bgActors[bgId].dynaLookup.ceiling, &polyId);
|
||
} else {
|
||
s16 polyId = *polyStartIndex + i;
|
||
|
||
DynaSSNodeList_SetSSListHead(&dyna->polyNodes, &dyna->bgActors[bgId].dynaLookup.wall, &polyId);
|
||
}
|
||
}
|
||
|
||
*polyStartIndex += pbgdata->numPolygons;
|
||
*vtxStartIndex += pbgdata->numVertices;
|
||
}
|
||
|
||
void DynaPoly_UnsetAllInteractFlags(PlayState* play, DynaCollisionContext* dyna, Actor* actor) {
|
||
DynaPolyActor* dynaActor;
|
||
s32 i;
|
||
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
if (dyna->bgActorFlags[i] & BGACTOR_IN_USE) {
|
||
dynaActor = DynaPoly_GetActor(&play->colCtx, i);
|
||
if (dynaActor != NULL && &dynaActor->actor == actor) {
|
||
DynaPolyActor_UnsetAllInteractFlags((DynaPolyActor*)actor);
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Original name: "DynaPolyInfo_setup"
|
||
*/
|
||
void DynaPoly_UpdateContext(PlayState* play, DynaCollisionContext* dyna) {
|
||
DynaPolyActor* actor;
|
||
s32 vtxStartIndex;
|
||
s32 polyStartIndex;
|
||
s32 i;
|
||
|
||
DynaSSNodeList_ResetCount(&dyna->polyNodes);
|
||
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
DynaLookup_ResetLists(&dyna->bgActors[i].dynaLookup);
|
||
}
|
||
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
if (dyna->bgActorFlags[i] & BGACTOR_1) {
|
||
// Initialize BgActor
|
||
PRINTF(VT_FGCOL(GREEN));
|
||
PRINTF(T("DynaPolyInfo_setup():削除 index=%d\n", "DynaPolyInfo_setup(): Delete index=%d\n"), i);
|
||
PRINTF(VT_RST);
|
||
|
||
dyna->bgActorFlags[i] = 0;
|
||
BgActor_Initialize(play, &dyna->bgActors[i]);
|
||
dyna->bitFlag |= DYNAPOLY_INVALIDATE_LOOKUP;
|
||
}
|
||
if (dyna->bgActors[i].actor != NULL && dyna->bgActors[i].actor->update == NULL) {
|
||
// Delete BgActor
|
||
PRINTF(VT_FGCOL(GREEN));
|
||
PRINTF(T("DynaPolyInfo_setup():削除 index=%d\n", "DynaPolyInfo_setup(): Delete index=%d\n"), i);
|
||
PRINTF(VT_RST);
|
||
actor = DynaPoly_GetActor(&play->colCtx, i);
|
||
if (actor == NULL) {
|
||
return;
|
||
}
|
||
actor->bgId = BGACTOR_NEG_ONE;
|
||
dyna->bgActorFlags[i] = 0;
|
||
|
||
BgActor_Initialize(play, &dyna->bgActors[i]);
|
||
dyna->bitFlag |= DYNAPOLY_INVALIDATE_LOOKUP;
|
||
}
|
||
}
|
||
vtxStartIndex = 0;
|
||
polyStartIndex = 0;
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
if (dyna->bgActorFlags[i] & BGACTOR_IN_USE) {
|
||
DynaPoly_AddBgActorToLookup(play, dyna, i, &vtxStartIndex, &polyStartIndex);
|
||
}
|
||
}
|
||
dyna->bitFlag &= ~DYNAPOLY_INVALIDATE_LOOKUP;
|
||
}
|
||
|
||
/**
|
||
* Update all BgActor's previous ScaleRotPos
|
||
*/
|
||
void DynaPoly_UpdateBgActorTransforms(PlayState* play, DynaCollisionContext* dyna) {
|
||
s32 i;
|
||
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
if (dyna->bgActorFlags[i] & BGACTOR_IN_USE) {
|
||
DynaPoly_SetBgActorPrevTransform(play, &dyna->bgActors[i]);
|
||
}
|
||
}
|
||
}
|
||
|
||
#define DYNA_RAYCAST_FLOORS 1
|
||
#define DYNA_RAYCAST_WALLS 2
|
||
#define DYNA_RAYCAST_CEILINGS 4
|
||
|
||
/**
|
||
* Performs a downward raycast check on a list of floor, wall, or ceiling dyna polys
|
||
* `listType` specifies the poly list type (e.g. DYNA_RAYCAST_FLOORS)
|
||
*/
|
||
f32 BgCheck_RaycastDownDynaList(DynaRaycastDown* dynaRaycastDown, u32 listType) {
|
||
CollisionPoly* polyList;
|
||
SSNode* curNode;
|
||
f32 result;
|
||
f32 yIntersect;
|
||
s16 id;
|
||
|
||
result = dynaRaycastDown->yIntersect;
|
||
if (dynaRaycastDown->ssList->head == SS_NULL) {
|
||
return result;
|
||
}
|
||
polyList = dynaRaycastDown->dyna->polyList;
|
||
curNode = &dynaRaycastDown->dyna->polyNodes.tbl[dynaRaycastDown->ssList->head];
|
||
|
||
while (true) {
|
||
id = curNode->polyId;
|
||
if (COLPOLY_VTX_CHECK_FLAGS_ANY(polyList[id].flags_vIA, dynaRaycastDown->xpFlags)) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dynaRaycastDown->dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
if ((listType & (DYNA_RAYCAST_WALLS | DYNA_RAYCAST_CEILINGS)) &&
|
||
(dynaRaycastDown->downChkFlags & BGCHECK_RAYCAST_DOWN_CHECK_GROUND_ONLY) &&
|
||
COLPOLY_GET_NORMAL(polyList[id].normal.y) < 0.0f) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dynaRaycastDown->dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
if (CollisionPoly_CheckYIntersectApprox1(&polyList[id], dynaRaycastDown->dyna->vtxList, dynaRaycastDown->pos->x,
|
||
dynaRaycastDown->pos->z, &yIntersect,
|
||
dynaRaycastDown->chkDist) == true &&
|
||
yIntersect < dynaRaycastDown->pos->y && result < yIntersect) {
|
||
result = yIntersect;
|
||
*dynaRaycastDown->resultPoly = &dynaRaycastDown->dyna->polyList[id];
|
||
}
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dynaRaycastDown->dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Performs a downward raycast check on dyna polys
|
||
* returns the yIntersect of the poly found, or BGCHECK_Y_MIN if no poly is found
|
||
*/
|
||
f32 BgCheck_RaycastDownDyna(DynaRaycastDown* dynaRaycastDown) {
|
||
s32 i;
|
||
f32 result;
|
||
f32 intersect2;
|
||
s32 i2;
|
||
s32 isPaused;
|
||
DynaPolyActor* dynaActor;
|
||
CollisionPoly* poly;
|
||
Vec3f polyVtx[3];
|
||
Vec3f polyNorm;
|
||
u32 polyIndex;
|
||
CollisionPoly* polyMin;
|
||
MtxF srpMtx;
|
||
f32 magnitude;
|
||
Vec3s* vtxList;
|
||
f32 polyDist;
|
||
Vec3f vtx;
|
||
f32 intersect;
|
||
ScaleRotPos* curTransform;
|
||
|
||
result = BGCHECK_Y_MIN;
|
||
*dynaRaycastDown->bgId = BGCHECK_SCENE;
|
||
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
if (!(dynaRaycastDown->colCtx->dyna.bgActorFlags[i] & BGACTOR_IN_USE)) {
|
||
continue;
|
||
}
|
||
|
||
if (dynaRaycastDown->actor == dynaRaycastDown->colCtx->dyna.bgActors[i].actor ||
|
||
dynaRaycastDown->pos->y < dynaRaycastDown->colCtx->dyna.bgActors[i].minY ||
|
||
!Math3D_XZInSphere(&dynaRaycastDown->colCtx->dyna.bgActors[i].boundingSphere, dynaRaycastDown->pos->x,
|
||
dynaRaycastDown->pos->z)) {
|
||
continue;
|
||
}
|
||
|
||
dynaRaycastDown->dyna = &dynaRaycastDown->colCtx->dyna;
|
||
if (dynaRaycastDown->downChkFlags & BGCHECK_RAYCAST_DOWN_CHECK_FLOORS) {
|
||
dynaRaycastDown->ssList = &dynaRaycastDown->colCtx->dyna.bgActors[i].dynaLookup.floor;
|
||
intersect2 = BgCheck_RaycastDownDynaList(dynaRaycastDown, DYNA_RAYCAST_FLOORS);
|
||
|
||
if (dynaRaycastDown->yIntersect < intersect2) {
|
||
dynaRaycastDown->yIntersect = intersect2;
|
||
*dynaRaycastDown->bgId = i;
|
||
result = intersect2;
|
||
}
|
||
}
|
||
if ((dynaRaycastDown->downChkFlags & BGCHECK_RAYCAST_DOWN_CHECK_WALLS) ||
|
||
(*dynaRaycastDown->resultPoly == NULL &&
|
||
(dynaRaycastDown->downChkFlags & BGCHECK_RAYCAST_DOWN_CHECK_WALLS_SIMPLE))) {
|
||
dynaRaycastDown->ssList = &dynaRaycastDown->colCtx->dyna.bgActors[i].dynaLookup.wall;
|
||
intersect2 = BgCheck_RaycastDownDynaList(dynaRaycastDown, DYNA_RAYCAST_WALLS);
|
||
|
||
if (dynaRaycastDown->yIntersect < intersect2) {
|
||
dynaRaycastDown->yIntersect = intersect2;
|
||
*dynaRaycastDown->bgId = i;
|
||
result = intersect2;
|
||
}
|
||
}
|
||
|
||
if (dynaRaycastDown->downChkFlags & BGCHECK_RAYCAST_DOWN_CHECK_CEILINGS) {
|
||
dynaRaycastDown->ssList = &dynaRaycastDown->colCtx->dyna.bgActors[i].dynaLookup.ceiling;
|
||
intersect2 = BgCheck_RaycastDownDynaList(dynaRaycastDown, DYNA_RAYCAST_CEILINGS);
|
||
|
||
if (dynaRaycastDown->yIntersect < intersect2) {
|
||
dynaRaycastDown->yIntersect = intersect2;
|
||
*dynaRaycastDown->bgId = i;
|
||
result = intersect2;
|
||
}
|
||
}
|
||
}
|
||
|
||
dynaActor = DynaPoly_GetActor(dynaRaycastDown->colCtx, *dynaRaycastDown->bgId);
|
||
if ((result != BGCHECK_Y_MIN) && (dynaActor != NULL) && (dynaRaycastDown->play != NULL)) {
|
||
isPaused = IS_PAUSED(&dynaRaycastDown->play->pauseCtx);
|
||
if (!isPaused && (dynaRaycastDown->colCtx->dyna.bgActorFlags[*dynaRaycastDown->bgId] & BGACTOR_1)) {
|
||
curTransform = &dynaRaycastDown->dyna->bgActors[*dynaRaycastDown->bgId].curTransform;
|
||
polyMin =
|
||
&dynaRaycastDown->dyna
|
||
->polyList[dynaRaycastDown->dyna->bgActors[*dynaRaycastDown->bgId].dynaLookup.polyStartIndex];
|
||
polyIndex = *dynaRaycastDown->resultPoly - polyMin;
|
||
poly = &dynaRaycastDown->dyna->bgActors[*dynaRaycastDown->bgId].colHeader->polyList[polyIndex];
|
||
|
||
SkinMatrix_SetTranslateRotateYXZScale(&srpMtx, curTransform->scale.x, curTransform->scale.y,
|
||
curTransform->scale.z, curTransform->rot.x, curTransform->rot.y,
|
||
curTransform->rot.z, curTransform->pos.x, curTransform->pos.y,
|
||
curTransform->pos.z);
|
||
|
||
vtxList = dynaRaycastDown->dyna->bgActors[*dynaRaycastDown->bgId].colHeader->vtxList;
|
||
|
||
for (i2 = 0; i2 < 3; i2++) {
|
||
s32 pad;
|
||
|
||
Math_Vec3s_ToVec3f(&vtx, &vtxList[COLPOLY_VTX_INDEX(poly->vtxData[i2])]);
|
||
SkinMatrix_Vec3fMtxFMultXYZ(&srpMtx, &vtx, &polyVtx[i2]);
|
||
}
|
||
|
||
Math3D_SurfaceNorm(&polyVtx[0], &polyVtx[1], &polyVtx[2], &polyNorm);
|
||
magnitude = Math3D_Vec3fMagnitude(&polyNorm);
|
||
|
||
if (!IS_ZERO(magnitude)) {
|
||
polyNorm.x *= 1.0f / magnitude;
|
||
polyNorm.y *= 1.0f / magnitude;
|
||
polyNorm.z *= 1.0f / magnitude;
|
||
polyDist = -DOTXYZ(polyNorm, polyVtx[0]);
|
||
if (Math3D_TriChkPointParaYIntersectInsideTri(
|
||
&polyVtx[0], &polyVtx[1], &polyVtx[2], polyNorm.x, polyNorm.y, polyNorm.z, polyDist,
|
||
dynaRaycastDown->pos->z, dynaRaycastDown->pos->x, &intersect, dynaRaycastDown->chkDist)) {
|
||
if (fabsf(intersect - result) < 1.0f) {
|
||
|
||
result = intersect;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Performs collision detection on a BgActor's wall polys on sphere `pos`, `radius`
|
||
* returns true if a collision was detected
|
||
* `outX` `outZ` return the displaced x,z coordinates
|
||
* `outPoly` returns the pointer to the nearest poly collided with, or NULL
|
||
* `outBgId` returns `bgId` if the poly SurfaceType's wall damage flag is not set, else ?
|
||
*/
|
||
s32 BgCheck_SphVsDynaWallInBgActor(CollisionContext* colCtx, u16 xpFlags, DynaCollisionContext* dyna, SSList* ssList,
|
||
f32* outX, f32* outZ, CollisionPoly** outPoly, s32* outBgId, Vec3f* pos, f32 radius,
|
||
s32 bgId) {
|
||
f32 temp;
|
||
f32 intersect;
|
||
s32 result = false;
|
||
CollisionPoly* poly;
|
||
SSNode* curNode;
|
||
f32 nx;
|
||
f32 ny;
|
||
f32 nz;
|
||
Vec3f resultPos;
|
||
s16 polyId;
|
||
f32 zTemp;
|
||
f32 xTemp;
|
||
f32 normalXZ;
|
||
f32 invNormalXZ;
|
||
f32 planeDist;
|
||
f32 temp_f18;
|
||
f32 zMin;
|
||
f32 zMax;
|
||
f32 xMin;
|
||
f32 xMax;
|
||
|
||
if (ssList->head == SS_NULL) {
|
||
return result;
|
||
}
|
||
|
||
resultPos = *pos;
|
||
curNode = &dyna->polyNodes.tbl[ssList->head];
|
||
|
||
while (true) {
|
||
polyId = curNode->polyId;
|
||
poly = &dyna->polyList[polyId];
|
||
CollisionPoly_GetNormalF(poly, &nx, &ny, &nz);
|
||
normalXZ = sqrtf(SQ(nx) + SQ(nz));
|
||
ASSERT(!IS_ZERO(normalXZ), "!IS_ZERO(ac_size)", "../z_bgcheck.c", 7382);
|
||
|
||
planeDist = Math3D_DistPlaneToPos(nx, ny, nz, poly->dist, &resultPos);
|
||
if (radius < fabsf(planeDist) || COLPOLY_VTX_CHECK_FLAGS_ANY(poly->flags_vIA, xpFlags)) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
invNormalXZ = 1.0f / normalXZ;
|
||
temp_f18 = fabsf(nz) * invNormalXZ;
|
||
if (temp_f18 < 0.4f) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
|
||
// compute poly zMin/zMax
|
||
zTemp = dyna->vtxList[COLPOLY_VTX_INDEX(poly->flags_vIA)].z;
|
||
zMax = zMin = zTemp;
|
||
|
||
zTemp = dyna->vtxList[COLPOLY_VTX_INDEX(poly->flags_vIB)].z;
|
||
if (zTemp < zMin) {
|
||
zMin = zTemp;
|
||
} else if (zTemp > zMax) {
|
||
zMax = zTemp;
|
||
}
|
||
|
||
zTemp = dyna->vtxList[poly->vIC].z;
|
||
if (zTemp < zMin) {
|
||
zMin = zTemp;
|
||
} else if (zMax < zTemp) {
|
||
zMax = zTemp;
|
||
}
|
||
|
||
zMin -= radius;
|
||
zMax += radius;
|
||
if (resultPos.z < zMin || zMax < resultPos.z) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
if (CollisionPoly_CheckZIntersectApprox(poly, dyna->vtxList, resultPos.x, pos->y, &intersect)) {
|
||
f32 zIntersectDist = intersect - resultPos.z;
|
||
|
||
if (fabsf(zIntersectDist) <= radius / temp_f18) {
|
||
if (zIntersectDist * nz <= 4.0f) {
|
||
if (BgCheck_ComputeWallDisplacement(colCtx, poly, &resultPos.x, &resultPos.z, nx, ny, nz,
|
||
invNormalXZ, planeDist, radius, outPoly)) {
|
||
*outBgId = bgId;
|
||
}
|
||
result = true;
|
||
}
|
||
}
|
||
}
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
}
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
}
|
||
|
||
curNode = &dyna->polyNodes.tbl[ssList->head];
|
||
while (true) {
|
||
polyId = curNode->polyId;
|
||
poly = &dyna->polyList[polyId];
|
||
CollisionPoly_GetNormalF(poly, &nx, &ny, &nz);
|
||
normalXZ = sqrtf(SQ(nx) + SQ(nz));
|
||
ASSERT(!IS_ZERO(normalXZ), "!IS_ZERO(ac_size)", "../z_bgcheck.c", 7489);
|
||
|
||
planeDist = Math3D_DistPlaneToPos(nx, ny, nz, poly->dist, &resultPos);
|
||
if (radius < fabsf(planeDist) || COLPOLY_VTX_CHECK_FLAGS_ANY(poly->flags_vIA, xpFlags)) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
|
||
invNormalXZ = 1.0f / normalXZ;
|
||
temp_f18 = fabsf(nx) * invNormalXZ;
|
||
if (temp_f18 < 0.4f) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
|
||
// compute poly xMin/xMax
|
||
xTemp = dyna->vtxList[COLPOLY_VTX_INDEX(poly->flags_vIA)].x;
|
||
xMax = xMin = xTemp;
|
||
xTemp = dyna->vtxList[COLPOLY_VTX_INDEX(poly->flags_vIB)].x;
|
||
|
||
if (xTemp < xMin) {
|
||
xMin = xTemp;
|
||
} else if (xMax < xTemp) {
|
||
xMax = xTemp;
|
||
}
|
||
xTemp = dyna->vtxList[poly->vIC].x;
|
||
if (xTemp < xMin) {
|
||
xMin = xTemp;
|
||
} else if (xMax < xTemp) {
|
||
xMax = xTemp;
|
||
}
|
||
|
||
xMin -= radius;
|
||
xMax += radius;
|
||
if (resultPos.x < xMin || xMax < resultPos.x) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
|
||
if (CollisionPoly_CheckXIntersectApprox(poly, dyna->vtxList, pos->y, resultPos.z, &intersect)) {
|
||
f32 xIntersectDist = intersect - resultPos.x;
|
||
|
||
if (fabsf(xIntersectDist) <= radius / temp_f18) {
|
||
if (xIntersectDist * nx <= 4.0f) {
|
||
if (BgCheck_ComputeWallDisplacement(colCtx, poly, &resultPos.x, &resultPos.z, nx, ny, nz,
|
||
invNormalXZ, planeDist, radius, outPoly)) {
|
||
*outBgId = bgId;
|
||
}
|
||
result = true;
|
||
}
|
||
}
|
||
}
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
}
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
}
|
||
*outX = resultPos.x;
|
||
*outZ = resultPos.z;
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Performs collision detection on all dyna poly walls using sphere `pos`, `radius`
|
||
* returns true if a collision was detected
|
||
* `outX` `outZ` return the displaced x,z coordinates
|
||
* `outPoly` returns the pointer to the nearest poly collided with, or NULL
|
||
* `outBgId` returns the index of the BgActor that owns `outPoly`
|
||
* If `actor` is not NULL, an BgActor bound to that actor will be ignored
|
||
*/
|
||
s32 BgCheck_SphVsDynaWall(CollisionContext* colCtx, u16 xpFlags, f32* outX, f32* outZ, Vec3f* pos, f32 radius,
|
||
CollisionPoly** outPoly, s32* outBgId, Actor* actor) {
|
||
Vec3f resultPos;
|
||
s32 result;
|
||
f32 r;
|
||
f32 dz;
|
||
f32 dx;
|
||
BgActor* bgActor;
|
||
s32 i;
|
||
|
||
result = false;
|
||
resultPos = *pos;
|
||
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
if (!(colCtx->dyna.bgActorFlags[i] & BGACTOR_IN_USE)) {
|
||
continue;
|
||
}
|
||
if ((colCtx->dyna.bgActors + i)->actor == actor) {
|
||
continue;
|
||
}
|
||
bgActor = &colCtx->dyna.bgActors[i];
|
||
|
||
if (bgActor->minY > resultPos.y || bgActor->maxY < resultPos.y) {
|
||
continue;
|
||
}
|
||
|
||
bgActor->boundingSphere.radius += (s16)radius;
|
||
|
||
r = bgActor->boundingSphere.radius;
|
||
dx = bgActor->boundingSphere.center.x - resultPos.x;
|
||
dz = bgActor->boundingSphere.center.z - resultPos.z;
|
||
if (SQ(r) < (SQ(dx) + SQ(dz)) || (!Math3D_XYInSphere(&bgActor->boundingSphere, resultPos.x, resultPos.y) &&
|
||
!Math3D_YZInSphere(&bgActor->boundingSphere, resultPos.y, resultPos.z))) {
|
||
bgActor->boundingSphere.radius -= (s16)radius;
|
||
continue;
|
||
}
|
||
bgActor->boundingSphere.radius -= (s16)radius;
|
||
if (BgCheck_SphVsDynaWallInBgActor(colCtx, xpFlags, &colCtx->dyna,
|
||
&(colCtx->dyna.bgActors + i)->dynaLookup.wall, outX, outZ, outPoly, outBgId,
|
||
&resultPos, radius, i)) {
|
||
resultPos.x = *outX;
|
||
resultPos.z = *outZ;
|
||
result = true;
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Tests for collision with a dyna poly ceiling, starting at `ssList`
|
||
* returns true if a collision occurs, else false
|
||
* `outPoly` returns the poly collided with
|
||
* `outY` returns the y coordinate needed to not collide with `outPoly`
|
||
*/
|
||
s32 BgCheck_CheckDynaCeilingList(CollisionContext* colCtx, u16 xpFlags, DynaCollisionContext* dyna, SSList* ssList,
|
||
f32* outY, Vec3f* pos, f32 checkHeight, CollisionPoly** outPoly) {
|
||
s16 polyId;
|
||
SSNode* curNode;
|
||
CollisionPoly* poly;
|
||
Vec3f testPos;
|
||
f32 ceilingY;
|
||
f32 sign;
|
||
f32 nx;
|
||
f32 ny;
|
||
f32 nz;
|
||
s32 result = false;
|
||
f32 intersectDist;
|
||
u16 padding;
|
||
|
||
if (ssList->head == SS_NULL) {
|
||
return false;
|
||
}
|
||
curNode = &dyna->polyNodes.tbl[ssList->head];
|
||
testPos = *pos;
|
||
|
||
while (true) {
|
||
polyId = curNode->polyId;
|
||
poly = &dyna->polyList[polyId];
|
||
if (COLPOLY_VTX_CHECK_FLAGS_ANY(poly->flags_vIA, xpFlags)) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
CollisionPoly_GetNormalF(poly, &nx, &ny, &nz);
|
||
if (checkHeight < Math3D_UDistPlaneToPos(nx, ny, nz, poly->dist, &testPos)) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
if (CollisionPoly_CheckYIntersectApprox2(poly, dyna->vtxList, testPos.x, testPos.z, &ceilingY)) {
|
||
intersectDist = ceilingY - testPos.y;
|
||
if (testPos.y < ceilingY && intersectDist < checkHeight && intersectDist * ny <= 0.0f) {
|
||
sign = (0.0f <= ny) ? 1.0f : -1.0f;
|
||
testPos.y = (sign * checkHeight) + ceilingY;
|
||
result = true;
|
||
*outPoly = poly;
|
||
}
|
||
}
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
*outY = testPos.y;
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Tests collision with a dyna poly ceiling
|
||
* returns true if a collision occurs, else false
|
||
* `outPoly` returns the poly collided with, while `outBgId` returns the id of the BgActor that owns the poly
|
||
* `outY` returns the y coordinate needed to not collide with `outPoly`, or `pos`.y + `chkDist` if no collision occurs
|
||
*/
|
||
s32 BgCheck_CheckDynaCeiling(CollisionContext* colCtx, u16 xpFlags, f32* outY, Vec3f* pos, f32 chkDist,
|
||
CollisionPoly** outPoly, s32* outBgId, Actor* actor) {
|
||
s32 i = 0;
|
||
s32 result = false;
|
||
f32 resultY;
|
||
f32 tempY = chkDist + pos->y;
|
||
BgActor* bgActor;
|
||
CollisionPoly* poly;
|
||
|
||
resultY = tempY;
|
||
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
if (!(colCtx->dyna.bgActorFlags[i] & BGACTOR_IN_USE)) {
|
||
continue;
|
||
}
|
||
if (actor == colCtx->dyna.bgActors[i].actor) {
|
||
continue;
|
||
}
|
||
if (!Math3D_XZInSphere(&colCtx->dyna.bgActors[i].boundingSphere, pos->x, pos->z)) {
|
||
continue;
|
||
}
|
||
if (BgCheck_CheckDynaCeilingList(colCtx, xpFlags, &colCtx->dyna, &colCtx->dyna.bgActors[i].dynaLookup.ceiling,
|
||
&tempY, pos, chkDist, &poly) == true &&
|
||
tempY < resultY) {
|
||
|
||
resultY = tempY;
|
||
*outPoly = poly;
|
||
*outBgId = i;
|
||
result = true;
|
||
}
|
||
}
|
||
*outY = resultY;
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Tests if DynaLineTest intersects with a poly
|
||
* returns true if a poly was intersected, else false
|
||
*/
|
||
s32 BgCheck_CheckLineAgainstBgActorSSList(DynaLineTest* dynaLineTest) {
|
||
f32 distSq;
|
||
s32 result;
|
||
CollisionPoly* curPoly;
|
||
SSNode* curNode;
|
||
Vec3f polyIntersect;
|
||
s16 polyId;
|
||
|
||
if (dynaLineTest->ssList->head == SS_NULL) {
|
||
return false;
|
||
}
|
||
|
||
curNode = &dynaLineTest->dyna->polyNodes.tbl[dynaLineTest->ssList->head];
|
||
result = false;
|
||
|
||
while (true) {
|
||
polyId = curNode->polyId;
|
||
curPoly = &dynaLineTest->dyna->polyList[polyId];
|
||
if (COLPOLY_VTX_CHECK_FLAGS_ANY(curPoly->flags_vIA, dynaLineTest->xpFlags)) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dynaLineTest->dyna->polyNodes.tbl[curNode->next];
|
||
}
|
||
} else {
|
||
if (CollisionPoly_LineVsPoly(curPoly, dynaLineTest->dyna->vtxList, dynaLineTest->posA, dynaLineTest->posB,
|
||
&polyIntersect, dynaLineTest->chkOneFace, dynaLineTest->chkDist)) {
|
||
distSq = Math3D_Vec3fDistSq(dynaLineTest->posA, &polyIntersect);
|
||
if (distSq < *dynaLineTest->distSq) {
|
||
*dynaLineTest->distSq = distSq;
|
||
*dynaLineTest->posResult = polyIntersect;
|
||
*dynaLineTest->posB = polyIntersect;
|
||
*dynaLineTest->resultPoly = curPoly;
|
||
result = true;
|
||
}
|
||
}
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dynaLineTest->dyna->polyNodes.tbl[curNode->next];
|
||
}
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Tests if line `posA` `posB` intersects with a dyna poly within BgActor `bgId`
|
||
* `distSq` is the maximum squared distance to check for a collision
|
||
* returns true if an intersection occurred, else false
|
||
* `posB`? and `posResult` return the point of intersection
|
||
* `outPoly` returns the poly intersected
|
||
* `distSq` returns the squared distance of the intersection
|
||
*/
|
||
s32 BgCheck_CheckLineAgainstBgActor(CollisionContext* colCtx, u16 xpFlags, Vec3f* posA, Vec3f* posB, Vec3f* posResult,
|
||
CollisionPoly** outPoly, f32* distSq, s32 bgId, f32 chkDist, s32 bccFlags) {
|
||
s32 result = false;
|
||
DynaLineTest dynaLineTest;
|
||
|
||
dynaLineTest.colCtx = colCtx;
|
||
dynaLineTest.xpFlags = xpFlags;
|
||
dynaLineTest.dyna = &colCtx->dyna;
|
||
dynaLineTest.posA = posA;
|
||
dynaLineTest.posB = posB;
|
||
dynaLineTest.posResult = posResult;
|
||
dynaLineTest.resultPoly = outPoly;
|
||
dynaLineTest.chkOneFace = (bccFlags & BGCHECK_CHECK_ONE_FACE) != 0;
|
||
dynaLineTest.distSq = distSq;
|
||
dynaLineTest.chkDist = chkDist;
|
||
|
||
dynaLineTest.ssList = &colCtx->dyna.bgActors[bgId].dynaLookup.wall;
|
||
if (bccFlags & BGCHECK_CHECK_WALL) {
|
||
if (BgCheck_CheckLineAgainstBgActorSSList(&dynaLineTest)) {
|
||
result = true;
|
||
}
|
||
}
|
||
dynaLineTest.ssList = &colCtx->dyna.bgActors[bgId].dynaLookup.floor;
|
||
if (bccFlags & BGCHECK_CHECK_FLOOR) {
|
||
if (BgCheck_CheckLineAgainstBgActorSSList(&dynaLineTest)) {
|
||
result = true;
|
||
}
|
||
}
|
||
dynaLineTest.ssList = &colCtx->dyna.bgActors[bgId].dynaLookup.ceiling;
|
||
if (bccFlags & BGCHECK_CHECK_CEILING) {
|
||
if (BgCheck_CheckLineAgainstBgActorSSList(&dynaLineTest)) {
|
||
result = true;
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Tests if line from `posA` to `posB` passes through a dyna poly.
|
||
* returns true if so, otherwise false
|
||
* `outPoly` returns the pointer of the poly intersected.
|
||
* `outBgId` returns the BgActor index of the poly
|
||
*/
|
||
s32 BgCheck_CheckLineAgainstDyna(CollisionContext* colCtx, u16 xpFlags, Vec3f* posA, Vec3f* posB, Vec3f* posResult,
|
||
CollisionPoly** outPoly, f32* distSq, s32* outBgId, Actor* actor, f32 chkDist,
|
||
s32 bccFlags) {
|
||
s32 pad;
|
||
s32 i;
|
||
s32 result = false;
|
||
Linef line;
|
||
f32 ay;
|
||
f32 by;
|
||
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
if (colCtx->dyna.bgActorFlags[i] & BGACTOR_IN_USE) {
|
||
if (actor != colCtx->dyna.bgActors[i].actor) {
|
||
ay = posA->y;
|
||
by = posB->y;
|
||
if (!(ay < colCtx->dyna.bgActors[i].minY) || !(by < colCtx->dyna.bgActors[i].minY)) {
|
||
if (!(colCtx->dyna.bgActors[i].maxY < ay) || !(colCtx->dyna.bgActors[i].maxY < by)) {
|
||
line.a = *posA;
|
||
line.b = *posB;
|
||
if (Math3D_LineVsSph(&colCtx->dyna.bgActors[i].boundingSphere, &line) != 0) {
|
||
if (BgCheck_CheckLineAgainstBgActor(colCtx, xpFlags, posA, posB, posResult, outPoly, distSq,
|
||
i, chkDist, bccFlags) == true) {
|
||
*outBgId = i;
|
||
result = true;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
/**
|
||
* Get first dyna poly intersecting sphere `center` `radius` from list `ssList`
|
||
* returns true if any poly intersects the sphere, else returns false
|
||
* `outPoly` returns the pointer of the first poly found that intersects
|
||
*/
|
||
s32 BgCheck_SphVsFirstDynaPolyList(CollisionContext* colCtx, u16 xpFlags, CollisionPoly** outPoly, Vec3f* center,
|
||
f32 radius, SSList* ssList) {
|
||
CollisionPoly* curPoly;
|
||
DynaCollisionContext* dyna;
|
||
SSNode* curNode;
|
||
s32 curPolyId;
|
||
|
||
if (ssList->head == SS_NULL) {
|
||
return false;
|
||
}
|
||
dyna = &colCtx->dyna;
|
||
curNode = &dyna->polyNodes.tbl[ssList->head];
|
||
while (true) {
|
||
curPolyId = curNode->polyId;
|
||
curPoly = &dyna->polyList[curPolyId];
|
||
if (COLPOLY_VTX_CHECK_FLAGS_ANY(curPoly->flags_vIA, xpFlags)) {
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
if (CollisionPoly_SphVsPoly(curPoly, dyna->vtxList, center, radius)) {
|
||
*outPoly = curPoly;
|
||
return true;
|
||
}
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
} else {
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
continue;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Get first dyna poly intersecting sphere `center` `radius` from BgActor `bgId`
|
||
* returns true if any poly intersects the sphere, else false
|
||
* `outPoly` returns the pointer of the first poly found that intersects
|
||
*/
|
||
s32 BgCheck_SphVsFirstDynaPolyInBgActor(CollisionContext* colCtx, u16 xpFlags, CollisionPoly** outPoly, Vec3f* center,
|
||
f32 radius, s32 bgId, u16 bciFlags) {
|
||
if ((bciFlags & BGCHECK_IGNORE_CEILING) == 0) {
|
||
if (BgCheck_SphVsFirstDynaPolyList(colCtx, xpFlags, outPoly, center, radius,
|
||
&colCtx->dyna.bgActors[bgId].dynaLookup.ceiling)) {
|
||
return true;
|
||
}
|
||
}
|
||
if ((bciFlags & BGCHECK_IGNORE_WALL) == 0) {
|
||
if (BgCheck_SphVsFirstDynaPolyList(colCtx, xpFlags, outPoly, center, radius,
|
||
&colCtx->dyna.bgActors[bgId].dynaLookup.wall)) {
|
||
return true;
|
||
}
|
||
}
|
||
if ((bciFlags & BGCHECK_IGNORE_FLOOR) == 0) {
|
||
if (BgCheck_SphVsFirstDynaPolyList(colCtx, xpFlags, outPoly, center, radius,
|
||
&colCtx->dyna.bgActors[bgId].dynaLookup.floor)) {
|
||
return true;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Gets first dyna poly intersecting sphere `center` `radius`
|
||
* returns true if poly detected, else false
|
||
* `outPoly` returns the first intersecting poly, while `outBgId` returns the BgActor index of that poly
|
||
*/
|
||
s32 BgCheck_SphVsFirstDynaPoly(CollisionContext* colCtx, u16 xpFlags, CollisionPoly** outPoly, s32* outBgId,
|
||
Vec3f* center, f32 radius, Actor* actor, u16 bciFlags) {
|
||
s32 i = 0;
|
||
Sphere16 testSphere;
|
||
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
if (!(colCtx->dyna.bgActorFlags[i] & BGACTOR_IN_USE)) {
|
||
continue;
|
||
}
|
||
if (colCtx->dyna.bgActors[i].actor == actor) {
|
||
continue;
|
||
}
|
||
testSphere.center.x = center->x;
|
||
testSphere.center.y = center->y;
|
||
testSphere.center.z = center->z;
|
||
testSphere.radius = radius;
|
||
if (!Math3D_SphVsSph(&testSphere, &colCtx->dyna.bgActors[i].boundingSphere)) {
|
||
continue;
|
||
}
|
||
if (BgCheck_SphVsFirstDynaPolyInBgActor(colCtx, xpFlags, outPoly, center, radius, i, bciFlags)) {
|
||
return true;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* SEGMENTED_TO_VIRTUAL CollisionHeader members
|
||
*/
|
||
void CollisionHeader_SegmentedToVirtual(CollisionHeader* colHeader) {
|
||
colHeader->vtxList = SEGMENTED_TO_VIRTUAL(colHeader->vtxList);
|
||
colHeader->polyList = SEGMENTED_TO_VIRTUAL(colHeader->polyList);
|
||
colHeader->surfaceTypeList = SEGMENTED_TO_VIRTUAL(colHeader->surfaceTypeList);
|
||
colHeader->bgCamList = SEGMENTED_TO_VIRTUAL(colHeader->bgCamList);
|
||
colHeader->waterBoxes = SEGMENTED_TO_VIRTUAL(colHeader->waterBoxes);
|
||
}
|
||
|
||
/**
|
||
* Convert CollisionHeader Segmented to Virtual addressing
|
||
*/
|
||
void CollisionHeader_GetVirtual(void* colHeader, CollisionHeader** dest) {
|
||
*dest = SEGMENTED_TO_VIRTUAL(colHeader);
|
||
CollisionHeader_SegmentedToVirtual(*dest);
|
||
}
|
||
|
||
/**
|
||
* SEGMENT_TO_VIRTUAL all active BgActor CollisionHeaders
|
||
*/
|
||
void func_800418D0(CollisionContext* colCtx, PlayState* play) {
|
||
DynaCollisionContext* dyna = &colCtx->dyna;
|
||
s32 i;
|
||
u16 flag;
|
||
|
||
for (i = 0; i < BG_ACTOR_MAX; i++) {
|
||
flag = dyna->bgActorFlags[i];
|
||
if ((flag & BGACTOR_IN_USE) && !(flag & BGACTOR_1)) {
|
||
Actor_SetObjectDependency(play, dyna->bgActors[i].actor);
|
||
CollisionHeader_SegmentedToVirtual(dyna->bgActors[i].colHeader);
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Reset SSNodeList polyCheckTbl
|
||
*/
|
||
void BgCheck_ResetPolyCheckTbl(SSNodeList* nodeList, s32 numPolys) {
|
||
u8* t;
|
||
|
||
for (t = nodeList->polyCheckTbl; t < nodeList->polyCheckTbl + numPolys; t++) {
|
||
*t = 0;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Get SurfaceType property set
|
||
*/
|
||
u32 SurfaceType_GetData(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId, s32 dataIdx) {
|
||
CollisionHeader* colHeader;
|
||
SurfaceType* surfaceTypes;
|
||
|
||
colHeader = BgCheck_GetCollisionHeader(colCtx, bgId);
|
||
if (colHeader == NULL || poly == NULL) {
|
||
return 0;
|
||
}
|
||
|
||
surfaceTypes = colHeader->surfaceTypeList;
|
||
if (surfaceTypes == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return 0;
|
||
}
|
||
|
||
return surfaceTypes[poly->type].data[dataIdx];
|
||
}
|
||
|
||
u32 SurfaceType_GetBgCamIndex(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 0) & 0xFF;
|
||
}
|
||
|
||
/**
|
||
* BgCam get setting of bgCam
|
||
*/
|
||
u16 BgCheck_GetBgCamSettingImpl(CollisionContext* colCtx, u32 bgCamIndex, s32 bgId) {
|
||
u16 camSetting;
|
||
CollisionHeader* colHeader;
|
||
BgCamInfo* bgCamList;
|
||
|
||
colHeader = BgCheck_GetCollisionHeader(colCtx, bgId);
|
||
if (colHeader == NULL) {
|
||
return CAM_SET_NONE;
|
||
}
|
||
|
||
bgCamList = colHeader->bgCamList;
|
||
camSetting = bgCamList[bgCamIndex].setting;
|
||
|
||
return camSetting;
|
||
}
|
||
|
||
/**
|
||
* BgCam Get the camera setting of bgCam
|
||
*/
|
||
u16 BgCheck_GetBgCamSetting(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
CollisionHeader* colHeader = BgCheck_GetCollisionHeader(colCtx, bgId);
|
||
BgCamInfo* bgCamList;
|
||
SurfaceType* surfaceTypes;
|
||
|
||
if (colHeader == NULL) {
|
||
return CAM_SET_NONE;
|
||
}
|
||
|
||
bgCamList = colHeader->bgCamList;
|
||
if (bgCamList == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return CAM_SET_NONE;
|
||
}
|
||
|
||
surfaceTypes = colHeader->surfaceTypeList;
|
||
if (surfaceTypes == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return CAM_SET_NONE;
|
||
}
|
||
|
||
return BgCheck_GetBgCamSettingImpl(colCtx, SurfaceType_GetBgCamIndex(colCtx, poly, bgId), bgId);
|
||
}
|
||
|
||
/**
|
||
* BgCam Get the total count of Vec3s data from bgCamFuncData
|
||
*/
|
||
u16 BgCheck_GetBgCamCountImpl(CollisionContext* colCtx, u32 bgCamIndex, s32 bgId) {
|
||
CollisionHeader* colHeader = BgCheck_GetCollisionHeader(colCtx, bgId);
|
||
BgCamInfo* bgCamList;
|
||
|
||
if (colHeader == NULL) {
|
||
return 0;
|
||
}
|
||
|
||
bgCamList = colHeader->bgCamList;
|
||
if (bgCamList == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return 0;
|
||
}
|
||
|
||
return bgCamList[bgCamIndex].count;
|
||
}
|
||
|
||
/**
|
||
* BgCam Get the total count of Vec3s data from bgCamFuncData
|
||
*/
|
||
u16 BgCheck_GetBgCamCount(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
CollisionHeader* colHeader = BgCheck_GetCollisionHeader(colCtx, bgId);
|
||
BgCamInfo* bgCamList;
|
||
SurfaceType* surfaceTypes;
|
||
|
||
if (colHeader == NULL) {
|
||
return 0;
|
||
}
|
||
|
||
bgCamList = colHeader->bgCamList;
|
||
if (bgCamList == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return 0;
|
||
}
|
||
|
||
surfaceTypes = colHeader->surfaceTypeList;
|
||
if (surfaceTypes == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return 0;
|
||
}
|
||
|
||
return BgCheck_GetBgCamCountImpl(colCtx, SurfaceType_GetBgCamIndex(colCtx, poly, bgId), bgId);
|
||
}
|
||
|
||
/**
|
||
* BgCam Get Vec3s data from bgCamFuncData
|
||
*/
|
||
Vec3s* BgCheck_GetBgCamFuncDataImpl(CollisionContext* colCtx, s32 bgCamIndex, s32 bgId) {
|
||
CollisionHeader* colHeader = BgCheck_GetCollisionHeader(colCtx, bgId);
|
||
BgCamInfo* bgCamList;
|
||
|
||
if (colHeader == NULL) {
|
||
return NULL;
|
||
}
|
||
|
||
bgCamList = colHeader->bgCamList;
|
||
if (bgCamList == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return NULL;
|
||
}
|
||
|
||
return (Vec3s*)SEGMENTED_TO_VIRTUAL(bgCamList[bgCamIndex].bgCamFuncData);
|
||
}
|
||
|
||
/**
|
||
* BgCam Get Vec3s data from bgCamFuncData
|
||
*/
|
||
Vec3s* BgCheck_GetBgCamFuncData(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
CollisionHeader* colHeader = BgCheck_GetCollisionHeader(colCtx, bgId);
|
||
BgCamInfo* bgCamList;
|
||
SurfaceType* surfaceTypes;
|
||
|
||
if (colHeader == NULL) {
|
||
return NULL;
|
||
}
|
||
|
||
bgCamList = colHeader->bgCamList;
|
||
if (bgCamList == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return NULL;
|
||
}
|
||
|
||
surfaceTypes = colHeader->surfaceTypeList;
|
||
if (surfaceTypes == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return NULL;
|
||
}
|
||
|
||
return BgCheck_GetBgCamFuncDataImpl(colCtx, SurfaceType_GetBgCamIndex(colCtx, poly, bgId), bgId);
|
||
}
|
||
|
||
u32 SurfaceType_GetExitIndex(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 0) >> 8 & 0x1F;
|
||
}
|
||
|
||
u32 SurfaceType_GetFloorType(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 0) >> 13 & 0x1F;
|
||
}
|
||
|
||
/**
|
||
* SurfaceType Get ? Property (& 0x001C 0000)
|
||
*/
|
||
u32 func_80041D70(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 0) >> 18 & 7;
|
||
}
|
||
|
||
u32 SurfaceType_GetWallType(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 0) >> 21 & 0x1F;
|
||
}
|
||
|
||
s32 SurfaceType_GetWallFlags(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return D_80119D90[SurfaceType_GetWallType(colCtx, poly, bgId)];
|
||
}
|
||
|
||
s32 SurfaceType_CheckWallFlag0(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return (SurfaceType_GetWallFlags(colCtx, poly, bgId) & WALL_FLAG_0) ? true : false;
|
||
}
|
||
|
||
s32 SurfaceType_CheckWallFlag1(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return (SurfaceType_GetWallFlags(colCtx, poly, bgId) & WALL_FLAG_1) ? true : false;
|
||
}
|
||
|
||
s32 SurfaceType_CheckWallFlag2(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return (SurfaceType_GetWallFlags(colCtx, poly, bgId) & WALL_FLAG_2) ? true : false;
|
||
}
|
||
|
||
u32 SurfaceType_GetFloorProperty2(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 0) >> 26 & 0xF;
|
||
}
|
||
|
||
u32 SurfaceType_GetFloorProperty(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 0) >> 26 & 0xF;
|
||
}
|
||
|
||
u32 SurfaceType_IsSoft(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 0) >> 30 & 1;
|
||
}
|
||
|
||
u32 SurfaceType_IsHorseBlocked(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 0) >> 31 & 1;
|
||
}
|
||
|
||
u32 SurfaceType_GetMaterial(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 1) & 0xF;
|
||
}
|
||
|
||
u16 SurfaceType_GetSfxOffset(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
s32 surfaceMaterial = SurfaceType_GetMaterial(colCtx, poly, bgId);
|
||
|
||
if ((surfaceMaterial < 0) || (surfaceMaterial >= ARRAY_COUNT(sSurfaceMaterialToSfxOffset))) {
|
||
return SURFACE_SFX_OFFSET_DIRT;
|
||
}
|
||
return sSurfaceMaterialToSfxOffset[surfaceMaterial];
|
||
}
|
||
|
||
u32 SurfaceType_GetFloorEffect(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 1) >> 4 & 3;
|
||
}
|
||
|
||
u32 SurfaceType_GetLightSetting(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 1) >> 6 & 0x1F;
|
||
}
|
||
|
||
u32 SurfaceType_GetEcho(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 1) >> 11 & 0x3F;
|
||
}
|
||
|
||
u32 SurfaceType_CanHookshot(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 1) >> 17 & 1;
|
||
}
|
||
|
||
/**
|
||
* CollisionPoly is ignored by entities
|
||
* Returns true if poly is ignored by entities, else false
|
||
*/
|
||
s32 SurfaceType_IsIgnoredByEntities(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
u32 flags;
|
||
|
||
if (BgCheck_GetCollisionHeader(colCtx, bgId) == NULL) {
|
||
return true;
|
||
}
|
||
flags = COLPOLY_VTX_CHECK_FLAGS_ANY(poly->flags_vIA, COLPOLY_IGNORE_ENTITY);
|
||
return !!flags;
|
||
}
|
||
|
||
/**
|
||
* CollisionPoly is ignored by projectiles
|
||
* Returns true if poly is ignored by projectiles, else false
|
||
*/
|
||
s32 SurfaceType_IsIgnoredByProjectiles(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
u32 flags;
|
||
|
||
if (BgCheck_GetCollisionHeader(colCtx, bgId) == NULL) {
|
||
return true;
|
||
}
|
||
flags = COLPOLY_VTX_CHECK_FLAGS_ANY(poly->flags_vIA, COLPOLY_IGNORE_PROJECTILES);
|
||
return !!flags;
|
||
}
|
||
|
||
/**
|
||
* Checks if poly is a floor conveyor
|
||
*
|
||
* A conveyor surface is enabled with non-zero speed.
|
||
* When enabled, the conveyor will exhibit two types of behaviour depending on the return value:
|
||
*
|
||
* If true, then it is a floor conveyor and will push player only while being stood on
|
||
* If false, then it is a water conveyor and will push player only while in water
|
||
*/
|
||
s32 SurfaceType_IsFloorConveyor(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
u32 flags;
|
||
|
||
if (BgCheck_GetCollisionHeader(colCtx, bgId) == NULL) {
|
||
return true;
|
||
}
|
||
flags = COLPOLY_VTX_CHECK_FLAGS_ANY(poly->flags_vIB, COLPOLY_IS_FLOOR_CONVEYOR);
|
||
return !!flags;
|
||
}
|
||
|
||
u32 SurfaceType_GetConveyorSpeed(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 1) >> 18 & 7;
|
||
}
|
||
|
||
/**
|
||
* returns a value between 0-63, representing 360 / 64 degrees of rotation
|
||
*/
|
||
u32 SurfaceType_GetConveyorDirection(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return SurfaceType_GetData(colCtx, poly, bgId, 1) >> 21 & 0x3F;
|
||
}
|
||
|
||
u32 func_80042108(CollisionContext* colCtx, CollisionPoly* poly, s32 bgId) {
|
||
return (SurfaceType_GetData(colCtx, poly, bgId, 1) & 0x08000000) ? 1 : 0;
|
||
}
|
||
|
||
/**
|
||
* Zora's Domain WaterBox in King Zora's Room
|
||
*/
|
||
WaterBox sZorasDomainWaterBox = { -348, 877, -1746, 553, 780, 0x2104 };
|
||
|
||
/**
|
||
* WaterBox's effective bounding box
|
||
*/
|
||
f32 sZorasDomainWaterBoxMinX = -348.0f;
|
||
f32 sZorasDomainWaterBoxMinY = 777.0f;
|
||
f32 sZorasDomainWaterBoxMinZ = -1746.0f;
|
||
f32 sZorasDomainWaterBoxMaxX = 205.0f;
|
||
f32 sZorasDomainWaterBoxMaxY = 977.0f;
|
||
f32 sZorasDomainWaterBoxMaxZ = -967.0f;
|
||
|
||
/**
|
||
* Public. Get the water surface at point (`x`, `ySurface`, `z`). `ySurface` doubles as position y input
|
||
* returns true if point is within the xz boundaries of an active water box, else false
|
||
* `ySurface` returns the water box's surface, while `outWaterBox` returns a pointer to the WaterBox
|
||
*/
|
||
s32 WaterBox_GetSurface1(PlayState* play, CollisionContext* colCtx, f32 x, f32 z, f32* ySurface,
|
||
WaterBox** outWaterBox) {
|
||
if (play->sceneId == SCENE_ZORAS_DOMAIN) {
|
||
s32 pad;
|
||
|
||
if (sZorasDomainWaterBoxMinX < x && x < sZorasDomainWaterBoxMaxX && sZorasDomainWaterBoxMinY < *ySurface &&
|
||
*ySurface < sZorasDomainWaterBoxMaxY && sZorasDomainWaterBoxMinZ < z && z < sZorasDomainWaterBoxMaxZ) {
|
||
*outWaterBox = &sZorasDomainWaterBox;
|
||
*ySurface = sZorasDomainWaterBox.ySurface;
|
||
return true;
|
||
}
|
||
}
|
||
return WaterBox_GetSurfaceImpl(play, colCtx, x, z, ySurface, outWaterBox);
|
||
}
|
||
|
||
/**
|
||
* Internal. Get the water surface at point (`x`, `ySurface`, `z`). `ySurface` doubles as position y input
|
||
* returns true if point is within the xz boundaries of an active water box, else false
|
||
* `ySurface` returns the water box's surface, while `outWaterBox` returns a pointer to the WaterBox
|
||
*/
|
||
s32 WaterBox_GetSurfaceImpl(PlayState* play, CollisionContext* colCtx, f32 x, f32 z, f32* ySurface,
|
||
WaterBox** outWaterBox) {
|
||
CollisionHeader* colHeader = colCtx->colHeader;
|
||
s32 room;
|
||
WaterBox* waterBox;
|
||
|
||
if (colHeader->numWaterBoxes == 0 || colHeader->waterBoxes == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return false;
|
||
}
|
||
|
||
for (waterBox = colHeader->waterBoxes; waterBox < colHeader->waterBoxes + colHeader->numWaterBoxes; waterBox++) {
|
||
room = WATERBOX_ROOM(waterBox->properties);
|
||
if (room == play->roomCtx.curRoom.num || room == WATERBOX_ROOM_ALL) {
|
||
if (!(waterBox->properties & WATERBOX_FLAG_19)) {
|
||
if (waterBox->xMin < x && x < waterBox->xMin + waterBox->xLength) {
|
||
if (waterBox->zMin < z && z < waterBox->zMin + waterBox->zLength) {
|
||
*outWaterBox = waterBox;
|
||
*ySurface = waterBox->ySurface;
|
||
return true;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Gets the first active WaterBox at `pos` with WATERBOX_FLAG_19 not set
|
||
* `surfaceChkDist` is the absolute y distance from the water surface to check
|
||
* returns the index of the waterbox found, or -1 if no waterbox is found
|
||
* `outWaterBox` returns the pointer to the waterbox found, or NULL if none is found
|
||
*/
|
||
s32 WaterBox_GetSurface2(PlayState* play, CollisionContext* colCtx, Vec3f* pos, f32 surfaceChkDist,
|
||
WaterBox** outWaterBox) {
|
||
CollisionHeader* colHeader = colCtx->colHeader;
|
||
s32 room;
|
||
s32 i;
|
||
WaterBox* waterBox;
|
||
WaterBox* waterBoxList = colHeader->waterBoxes; // unused, needed for matching
|
||
|
||
if (colHeader->numWaterBoxes == 0 || colHeader->waterBoxes == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
*outWaterBox = NULL;
|
||
return -1;
|
||
}
|
||
|
||
for (i = 0; i < colHeader->numWaterBoxes; i++) {
|
||
waterBox = &colHeader->waterBoxes[i];
|
||
|
||
room = WATERBOX_ROOM(waterBox->properties);
|
||
if (room == play->roomCtx.curRoom.num || room == WATERBOX_ROOM_ALL) {
|
||
if (!(waterBox->properties & WATERBOX_FLAG_19)) {
|
||
if (waterBox->xMin < pos->x && pos->x < waterBox->xMin + waterBox->xLength) {
|
||
if (waterBox->zMin < pos->z && pos->z < waterBox->zMin + waterBox->zLength) {
|
||
if (pos->y - surfaceChkDist < waterBox->ySurface &&
|
||
waterBox->ySurface < pos->y + surfaceChkDist) {
|
||
*outWaterBox = waterBox;
|
||
return i;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
*outWaterBox = NULL;
|
||
return -1;
|
||
}
|
||
|
||
/**
|
||
* WaterBox get BgCam index
|
||
*/
|
||
u32 WaterBox_GetBgCamIndex(CollisionContext* colCtx, WaterBox* waterBox) {
|
||
u32 bgCamIndex = waterBox->properties & 0xFF;
|
||
|
||
return bgCamIndex;
|
||
}
|
||
|
||
/**
|
||
* WaterBox get BgCam setting
|
||
*/
|
||
u16 WaterBox_GetBgCamSetting(CollisionContext* colCtx, WaterBox* waterBox) {
|
||
s32 bgCamIndex = WaterBox_GetBgCamIndex(colCtx, waterBox);
|
||
BgCamInfo* bgCamList = colCtx->colHeader->bgCamList;
|
||
|
||
if (bgCamList == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return CAM_SET_NONE;
|
||
}
|
||
|
||
return colCtx->colHeader->bgCamList[bgCamIndex].setting;
|
||
}
|
||
|
||
/**
|
||
* WaterBox get lighting settings
|
||
*/
|
||
u32 WaterBox_GetLightIndex(CollisionContext* colCtx, WaterBox* waterBox) {
|
||
u32 lightIndex = (waterBox->properties >> 8) & 0x1F;
|
||
|
||
return lightIndex;
|
||
}
|
||
|
||
/**
|
||
* Get the water surface at point (`x`, `ySurface`, `z`). `ySurface` doubles as position y input
|
||
* same as WaterBox_GetSurfaceImpl, but tests if WATERBOX_FLAG_19 is set
|
||
* returns true if point is within the xz boundaries of an active water box, else false
|
||
* `ySurface` returns the water box's surface, while `outWaterBox` returns a pointer to the WaterBox
|
||
*/
|
||
s32 func_800425B0(PlayState* play, CollisionContext* colCtx, f32 x, f32 z, f32* ySurface, WaterBox** outWaterBox) {
|
||
CollisionHeader* colHeader = colCtx->colHeader;
|
||
s32 room;
|
||
WaterBox* waterBox;
|
||
|
||
if (colHeader->numWaterBoxes == 0 || colHeader->waterBoxes == SEGMENTED_TO_VIRTUAL(NULL)) {
|
||
return false;
|
||
}
|
||
|
||
for (waterBox = colHeader->waterBoxes; waterBox < colHeader->waterBoxes + colHeader->numWaterBoxes; waterBox++) {
|
||
room = WATERBOX_ROOM(waterBox->properties);
|
||
if ((room == play->roomCtx.curRoom.num) || (room == WATERBOX_ROOM_ALL)) {
|
||
if (waterBox->properties & WATERBOX_FLAG_19) {
|
||
if (waterBox->xMin < x && x < (waterBox->xMin + waterBox->xLength)) {
|
||
if (waterBox->zMin < z && z < (waterBox->zMin + waterBox->zLength)) {
|
||
*outWaterBox = waterBox;
|
||
*ySurface = waterBox->ySurface;
|
||
return true;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Gets the `closestPoint` to `point` on the line formed from the intesection of planes `polyA` and `polyB`
|
||
* returns true if the `closestPoint` exists, else returns false
|
||
*/
|
||
s32 func_80042708(CollisionPoly* polyA, CollisionPoly* polyB, Vec3f* point, Vec3f* closestPoint) {
|
||
f32 n1X;
|
||
f32 n1Y;
|
||
f32 n1Z;
|
||
f32 n2X;
|
||
f32 n2Y;
|
||
f32 n2Z;
|
||
|
||
CollisionPoly_GetNormalF(polyA, &n1X, &n1Y, &n1Z);
|
||
CollisionPoly_GetNormalF(polyB, &n2X, &n2Y, &n2Z);
|
||
return Math3D_PlaneVsPlaneVsLineClosestPoint(n1X, n1Y, n1Z, polyA->dist, n2X, n2Y, n2Z, polyB->dist, point,
|
||
closestPoint);
|
||
}
|
||
|
||
/**
|
||
* Get the `closestPoint` to line (`pointA`, `pointB`) formed from the intersection of planes `polyA` and `polyB`
|
||
* returns true if the `closestPoint` exists, else returns false
|
||
*/
|
||
s32 func_800427B4(CollisionPoly* polyA, CollisionPoly* polyB, Vec3f* pointA, Vec3f* pointB, Vec3f* closestPoint) {
|
||
f32 n1X;
|
||
f32 n1Y;
|
||
f32 n1Z;
|
||
f32 n2X;
|
||
f32 n2Y;
|
||
f32 n2Z;
|
||
s32 result;
|
||
|
||
CollisionPoly_GetNormalF(polyA, &n1X, &n1Y, &n1Z);
|
||
CollisionPoly_GetNormalF(polyB, &n2X, &n2Y, &n2Z);
|
||
result = Math3D_PlaneVsLineSegClosestPoint(n1X, n1Y, n1Z, polyA->dist, n2X, n2Y, n2Z, polyB->dist, pointA, pointB,
|
||
closestPoint);
|
||
return result;
|
||
}
|
||
|
||
#if OOT_DEBUG
|
||
/**
|
||
* Draw a list of dyna polys, specified by `ssList`
|
||
*/
|
||
void BgCheck_DrawDynaPolyList(PlayState* play, CollisionContext* colCtx, DynaCollisionContext* dyna, SSList* ssList,
|
||
u8 r, u8 g, u8 b) {
|
||
s16 curPolyId;
|
||
CollisionPoly* poly;
|
||
SSNode* curNode;
|
||
Vec3f vA;
|
||
Vec3f vB;
|
||
Vec3f vC;
|
||
f32 nx;
|
||
f32 ny;
|
||
f32 nz;
|
||
|
||
if (ssList->head != SS_NULL) {
|
||
curNode = &dyna->polyNodes.tbl[ssList->head];
|
||
while (true) {
|
||
curPolyId = curNode->polyId;
|
||
poly = &dyna->polyList[curPolyId];
|
||
BgCheck_Vec3sToVec3f(COLPOLY_VTX_INDEX(poly->flags_vIA) + dyna->vtxList, &vA);
|
||
BgCheck_Vec3sToVec3f(COLPOLY_VTX_INDEX(poly->flags_vIB) + dyna->vtxList, &vB);
|
||
BgCheck_Vec3sToVec3f((s32)(poly->vIC) + dyna->vtxList, &vC);
|
||
if (AREG(26)) {
|
||
nx = COLPOLY_GET_NORMAL(poly->normal.x);
|
||
ny = COLPOLY_GET_NORMAL(poly->normal.y);
|
||
nz = COLPOLY_GET_NORMAL(poly->normal.z);
|
||
vA.x += AREG(26) * nx;
|
||
vA.y += AREG(26) * ny;
|
||
vA.z += AREG(26) * nz;
|
||
vB.x += AREG(26) * nx;
|
||
vB.y += AREG(26) * ny;
|
||
vB.z += AREG(26) * nz;
|
||
vC.x += AREG(26) * nx;
|
||
vC.y += AREG(26) * ny;
|
||
vC.z += AREG(26) * nz;
|
||
}
|
||
Collider_DrawPoly(play->state.gfxCtx, &vA, &vB, &vC, r, g, b);
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
}
|
||
curNode = &dyna->polyNodes.tbl[curNode->next];
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Draw a BgActor's dyna polys
|
||
* `bgId` is the BgActor index that should be drawn
|
||
*/
|
||
void BgCheck_DrawBgActor(PlayState* play, CollisionContext* colCtx, s32 bgId) {
|
||
if (AREG(21)) {
|
||
BgCheck_DrawDynaPolyList(play, colCtx, &colCtx->dyna, &colCtx->dyna.bgActors[bgId].dynaLookup.ceiling, 255, 0,
|
||
0);
|
||
}
|
||
if (AREG(22)) {
|
||
BgCheck_DrawDynaPolyList(play, colCtx, &colCtx->dyna, &colCtx->dyna.bgActors[bgId].dynaLookup.wall, 0, 255, 0);
|
||
}
|
||
if (AREG(23)) {
|
||
BgCheck_DrawDynaPolyList(play, colCtx, &colCtx->dyna, &colCtx->dyna.bgActors[bgId].dynaLookup.floor, 0, 0, 255);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Draw all dyna polys
|
||
*/
|
||
void BgCheck_DrawDynaCollision(PlayState* play, CollisionContext* colCtx) {
|
||
s32 bgId;
|
||
|
||
for (bgId = 0; bgId < BG_ACTOR_MAX; bgId++) {
|
||
|
||
if (!(colCtx->dyna.bgActorFlags[bgId] & BGACTOR_IN_USE)) {
|
||
continue;
|
||
}
|
||
BgCheck_DrawBgActor(play, colCtx, bgId);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Draw a static poly
|
||
*/
|
||
void BgCheck_DrawStaticPoly(PlayState* play, CollisionContext* colCtx, CollisionPoly* poly, u8 r, u8 g, u8 b) {
|
||
Vec3f vA;
|
||
Vec3f vB;
|
||
Vec3f vC;
|
||
f32 nx;
|
||
f32 ny;
|
||
f32 nz;
|
||
|
||
BgCheck_Vec3sToVec3f(COLPOLY_VTX_INDEX(poly->flags_vIA) + colCtx->colHeader->vtxList, &vA);
|
||
BgCheck_Vec3sToVec3f(COLPOLY_VTX_INDEX(poly->flags_vIB) + colCtx->colHeader->vtxList, &vB);
|
||
BgCheck_Vec3sToVec3f(poly->vIC + colCtx->colHeader->vtxList, &vC);
|
||
if (AREG(26) != 0) {
|
||
nx = COLPOLY_GET_NORMAL(poly->normal.x);
|
||
ny = COLPOLY_GET_NORMAL(poly->normal.y);
|
||
nz = COLPOLY_GET_NORMAL(poly->normal.z);
|
||
vA.x += AREG(26) * nx;
|
||
vA.y += AREG(26) * ny;
|
||
vA.z += AREG(26) * nz;
|
||
vB.x += AREG(26) * nx;
|
||
vB.y += AREG(26) * ny;
|
||
vB.z += AREG(26) * nz;
|
||
vC.x += AREG(26) * nx;
|
||
vC.y += AREG(26) * ny;
|
||
vC.z += AREG(26) * nz;
|
||
}
|
||
Collider_DrawPoly(play->state.gfxCtx, &vA, &vB, &vC, r, g, b);
|
||
}
|
||
|
||
/**
|
||
* Draw a list of static polys, specified by `ssList`
|
||
*/
|
||
void BgCheck_DrawStaticPolyList(PlayState* play, CollisionContext* colCtx, SSList* ssList, u8 r, u8 g, u8 b) {
|
||
SSNode* curNode;
|
||
CollisionPoly* polyList = colCtx->colHeader->polyList;
|
||
s16 curPolyId;
|
||
|
||
if (ssList->head != SS_NULL) {
|
||
curNode = &colCtx->polyNodes.tbl[ssList->head];
|
||
while (true) {
|
||
curPolyId = curNode->polyId;
|
||
BgCheck_DrawStaticPoly(play, colCtx, &polyList[curPolyId], r, g, b);
|
||
if (curNode->next == SS_NULL) {
|
||
break;
|
||
}
|
||
curNode = &colCtx->polyNodes.tbl[curNode->next];
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Draw scene collision
|
||
*/
|
||
void BgCheck_DrawStaticCollision(PlayState* play, CollisionContext* colCtx) {
|
||
Player* player = GET_PLAYER(play);
|
||
StaticLookup* lookup = BgCheck_GetNearestStaticLookup(colCtx, colCtx->lookupTbl, &player->actor.world.pos);
|
||
|
||
if (AREG(23) != 0) {
|
||
BgCheck_DrawStaticPolyList(play, colCtx, &lookup->floor, 0, 0, 255);
|
||
}
|
||
if (AREG(22) != 0) {
|
||
BgCheck_DrawStaticPolyList(play, colCtx, &lookup->wall, 0, 255, 0);
|
||
}
|
||
if (AREG(21) != 0) {
|
||
BgCheck_DrawStaticPolyList(play, colCtx, &lookup->ceiling, 255, 0, 0);
|
||
}
|
||
}
|
||
#endif
|