2022-03-21 21:51:23 -04:00
|
|
|
#include "global.h"
|
|
|
|
#include "vt.h"
|
|
|
|
|
2022-05-13 18:43:55 -04:00
|
|
|
#include "soh/frame_interpolation.h"
|
|
|
|
|
2022-03-21 21:51:23 -04:00
|
|
|
EffectSsInfo sEffectSsInfo = { 0 }; // "EffectSS2Info"
|
|
|
|
|
|
|
|
void EffectSs_InitInfo(GlobalContext* globalCtx, s32 tableSize) {
|
|
|
|
u32 i;
|
|
|
|
EffectSs* effectSs;
|
|
|
|
EffectSsOverlay* overlay;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_COUNT(gEffectSsOverlayTable); i++) {
|
|
|
|
overlay = &gEffectSsOverlayTable[i];
|
|
|
|
osSyncPrintf("effect index %3d:size=%6dbyte romsize=%6dbyte\n", i,
|
|
|
|
(uintptr_t)overlay->vramEnd - (uintptr_t)overlay->vramStart, overlay->vromEnd - overlay->vromStart);
|
|
|
|
}
|
|
|
|
|
|
|
|
sEffectSsInfo.table =
|
2022-07-05 19:29:34 -04:00
|
|
|
GAMESTATE_ALLOC_MC(&globalCtx->state, tableSize * sizeof(EffectSs));
|
|
|
|
ASSERT(sEffectSsInfo.table != NULL);
|
2022-03-21 21:51:23 -04:00
|
|
|
|
|
|
|
sEffectSsInfo.searchStartIndex = 0;
|
|
|
|
sEffectSsInfo.tableSize = tableSize;
|
|
|
|
|
|
|
|
for (effectSs = &sEffectSsInfo.table[0]; effectSs < &sEffectSsInfo.table[sEffectSsInfo.tableSize]; effectSs++) {
|
|
|
|
EffectSs_Reset(effectSs);
|
|
|
|
}
|
|
|
|
|
|
|
|
overlay = &gEffectSsOverlayTable[0];
|
|
|
|
for (i = 0; i < ARRAY_COUNT(gEffectSsOverlayTable); i++) {
|
|
|
|
overlay->loadedRamAddr = NULL;
|
|
|
|
overlay++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void EffectSs_ClearAll(GlobalContext* globalCtx) {
|
|
|
|
u32 i;
|
|
|
|
EffectSs* effectSs;
|
|
|
|
EffectSsOverlay* overlay;
|
|
|
|
void* addr;
|
|
|
|
|
|
|
|
sEffectSsInfo.table = NULL;
|
|
|
|
sEffectSsInfo.searchStartIndex = 0;
|
|
|
|
sEffectSsInfo.tableSize = 0;
|
|
|
|
|
|
|
|
// This code doesn't actually work, since table was just set to NULL and tableSize to 0
|
|
|
|
for (effectSs = &sEffectSsInfo.table[0]; effectSs < &sEffectSsInfo.table[sEffectSsInfo.tableSize]; effectSs++) {
|
|
|
|
EffectSs_Delete(effectSs);
|
|
|
|
}
|
|
|
|
|
|
|
|
overlay = &gEffectSsOverlayTable[0];
|
|
|
|
for (i = 0; i < ARRAY_COUNT(gEffectSsOverlayTable); i++) {
|
|
|
|
addr = overlay->loadedRamAddr;
|
|
|
|
|
|
|
|
if (addr != NULL) {
|
2022-07-05 19:29:34 -04:00
|
|
|
ZELDA_ARENA_FREE_DEBUG(addr);
|
2022-03-21 21:51:23 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
overlay->loadedRamAddr = NULL;
|
|
|
|
overlay++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void EffectSs_Delete(EffectSs* effectSs) {
|
|
|
|
if (effectSs->flags & 2) {
|
|
|
|
Audio_StopSfxByPos(&effectSs->pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (effectSs->flags & 4) {
|
|
|
|
Audio_StopSfxByPos(&effectSs->vec);
|
|
|
|
}
|
|
|
|
|
|
|
|
EffectSs_Reset(effectSs);
|
|
|
|
}
|
|
|
|
|
|
|
|
void EffectSs_Reset(EffectSs* effectSs) {
|
|
|
|
u32 i;
|
|
|
|
|
|
|
|
effectSs->type = EFFECT_SS_TYPE_MAX;
|
|
|
|
effectSs->accel.x = effectSs->accel.y = effectSs->accel.z = 0;
|
|
|
|
effectSs->velocity.x = effectSs->velocity.y = effectSs->velocity.z = 0;
|
|
|
|
effectSs->vec.x = effectSs->vec.y = effectSs->vec.z = 0;
|
|
|
|
effectSs->pos.x = effectSs->pos.y = effectSs->pos.z = 0;
|
|
|
|
effectSs->life = -1;
|
|
|
|
effectSs->flags = 0;
|
|
|
|
effectSs->priority = 128;
|
|
|
|
effectSs->draw = NULL;
|
|
|
|
effectSs->update = NULL;
|
|
|
|
effectSs->gfx = NULL;
|
|
|
|
effectSs->actor = NULL;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_COUNT(effectSs->regs); i++) {
|
|
|
|
effectSs->regs[i] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s32 EffectSs_FindSlot(s32 priority, s32* pIndex) {
|
|
|
|
s32 foundFree;
|
|
|
|
s32 i;
|
|
|
|
|
|
|
|
if (sEffectSsInfo.searchStartIndex >= sEffectSsInfo.tableSize) {
|
|
|
|
sEffectSsInfo.searchStartIndex = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search for a free slot
|
|
|
|
i = sEffectSsInfo.searchStartIndex;
|
|
|
|
foundFree = false;
|
|
|
|
while (true) {
|
|
|
|
if (sEffectSsInfo.table[i].life == -1) {
|
|
|
|
foundFree = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
|
|
|
|
if (i >= sEffectSsInfo.tableSize) {
|
|
|
|
i = 0; // Loop around the whole table
|
|
|
|
}
|
|
|
|
|
|
|
|
// After a full loop, break out
|
|
|
|
if (i == sEffectSsInfo.searchStartIndex) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (foundFree == true) {
|
|
|
|
*pIndex = i;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If all slots are in use, search for a slot with a lower priority
|
|
|
|
// Note that a lower priority is representend by a higher value
|
|
|
|
i = sEffectSsInfo.searchStartIndex;
|
|
|
|
while (true) {
|
|
|
|
// Equal priority should only be considered "lower" if flag 0 is set
|
|
|
|
if ((priority <= sEffectSsInfo.table[i].priority) &&
|
|
|
|
!((priority == sEffectSsInfo.table[i].priority) && (sEffectSsInfo.table[i].flags & 1))) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
|
|
|
|
if (i >= sEffectSsInfo.tableSize) {
|
|
|
|
i = 0; // Loop around the whole table
|
|
|
|
}
|
|
|
|
|
|
|
|
// After a full loop, return 1 to indicate that we failed to find a suitable slot
|
|
|
|
if (i == sEffectSsInfo.searchStartIndex) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*pIndex = i;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EffectSs_Insert(GlobalContext* globalCtx, EffectSs* effectSs) {
|
|
|
|
s32 index;
|
|
|
|
|
|
|
|
if (FrameAdvance_IsEnabled(globalCtx) != true) {
|
|
|
|
if (EffectSs_FindSlot(effectSs->priority, &index) == 0) {
|
|
|
|
sEffectSsInfo.searchStartIndex = index + 1;
|
|
|
|
sEffectSsInfo.table[index] = *effectSs;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// original name: "EffectSoftSprite2_makeEffect"
|
|
|
|
void EffectSs_Spawn(GlobalContext* globalCtx, s32 type, s32 priority, void* initParams) {
|
|
|
|
s32 index;
|
|
|
|
u32 overlaySize;
|
|
|
|
EffectSsOverlay* overlayEntry;
|
|
|
|
EffectSsInit* initInfo;
|
|
|
|
|
|
|
|
overlayEntry = &gEffectSsOverlayTable[type];
|
|
|
|
|
2022-07-05 19:29:34 -04:00
|
|
|
ASSERT(type < EFFECT_SS_TYPE_MAX);
|
2022-03-21 21:51:23 -04:00
|
|
|
|
|
|
|
if (EffectSs_FindSlot(priority, &index) != 0) {
|
|
|
|
// Abort because we couldn't find a suitable slot to add this effect in
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
sEffectSsInfo.searchStartIndex = index + 1;
|
|
|
|
overlaySize = (uintptr_t)overlayEntry->vramEnd - (uintptr_t)overlayEntry->vramStart;
|
|
|
|
|
|
|
|
if (overlayEntry->vramStart == NULL) {
|
|
|
|
// "Not an overlay"
|
|
|
|
osSyncPrintf("EffectSoftSprite2_makeEffect():オーバーレイではありません。\n");
|
|
|
|
initInfo = overlayEntry->initInfo;
|
|
|
|
} else {
|
|
|
|
if (overlayEntry->loadedRamAddr == NULL) {
|
2022-07-05 19:29:34 -04:00
|
|
|
overlayEntry->loadedRamAddr = ZELDA_ARENA_MALLOC_RDEBUG(overlaySize);
|
2022-03-21 21:51:23 -04:00
|
|
|
|
|
|
|
if (overlayEntry->loadedRamAddr == NULL) {
|
|
|
|
osSyncPrintf(VT_FGCOL(RED));
|
|
|
|
// "The memory of %d byte cannot be secured. Therefore, the program cannot be loaded.
|
|
|
|
// What a dangerous situation! Naturally, effects will not produced either."
|
|
|
|
osSyncPrintf("EffectSoftSprite2_makeEffect():zelda_malloc_r()により,%"
|
|
|
|
"dbyteのメモリ確保ができま\nせん。そのため、プログラムのロードも\n出来ません。ただいま危険"
|
|
|
|
"な状態です!\nもちろん,エフェクトも出ません。\n",
|
|
|
|
overlaySize);
|
|
|
|
osSyncPrintf(VT_RST);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Overlay_Load(overlayEntry->vromStart, overlayEntry->vromEnd, overlayEntry->vramStart, overlayEntry->vramEnd,
|
|
|
|
overlayEntry->loadedRamAddr);
|
|
|
|
|
|
|
|
osSyncPrintf(VT_FGCOL(GREEN));
|
|
|
|
osSyncPrintf("EFFECT SS OVL:SegRom %08x %08x, Seg %08x %08x, RamStart %08x, type: %d\n",
|
|
|
|
overlayEntry->vromStart, overlayEntry->vromEnd, overlayEntry->vramStart, overlayEntry->vramEnd,
|
|
|
|
overlayEntry->loadedRamAddr, type);
|
|
|
|
osSyncPrintf(VT_RST);
|
|
|
|
}
|
|
|
|
|
|
|
|
initInfo = (void*)(uintptr_t)((overlayEntry->initInfo != NULL)
|
|
|
|
? (void*)((uintptr_t)overlayEntry->initInfo -
|
|
|
|
((intptr_t)overlayEntry->vramStart - (intptr_t)overlayEntry->loadedRamAddr))
|
|
|
|
: NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (initInfo->init == NULL) {
|
|
|
|
// "Effects have already been loaded, but the constructor is NULL so the addition will not occur.
|
|
|
|
// Please fix this. (Waste of memory) %08x %d"
|
|
|
|
osSyncPrintf("EffectSoftSprite2_makeEffect():すでにエフェクトはロード済みで\nすが,"
|
|
|
|
"コンストラクターがNULLなので追加をやめます。\n直してください。(メモリーの無駄) %08x %d\n",
|
|
|
|
initInfo, type);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete the previous effect in the slot, in case the slot wasn't free
|
|
|
|
EffectSs_Delete(&sEffectSsInfo.table[index]);
|
|
|
|
|
|
|
|
sEffectSsInfo.table[index].type = type;
|
|
|
|
sEffectSsInfo.table[index].priority = priority;
|
2022-05-13 18:43:55 -04:00
|
|
|
sEffectSsInfo.table[index].epoch++;
|
2022-03-21 21:51:23 -04:00
|
|
|
|
|
|
|
if (initInfo->init(globalCtx, index, &sEffectSsInfo.table[index], initParams) == 0) {
|
|
|
|
osSyncPrintf(VT_FGCOL(GREEN));
|
|
|
|
// "Construction failed for some reason. The constructor returned an error.
|
|
|
|
// Ceasing effect addition."
|
|
|
|
osSyncPrintf("EffectSoftSprite2_makeEffect():"
|
|
|
|
"何らかの理由でコンストラクト失敗。コンストラクターがエラーを返しました。エフェクトの追加を中"
|
|
|
|
"止します。\n");
|
|
|
|
osSyncPrintf(VT_RST);
|
|
|
|
EffectSs_Reset(&sEffectSsInfo.table[index]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void EffectSs_Update(GlobalContext* globalCtx, s32 index) {
|
|
|
|
EffectSs* effectSs = &sEffectSsInfo.table[index];
|
|
|
|
|
|
|
|
if (effectSs->update != NULL) {
|
|
|
|
effectSs->velocity.x += effectSs->accel.x;
|
|
|
|
effectSs->velocity.y += effectSs->accel.y;
|
|
|
|
effectSs->velocity.z += effectSs->accel.z;
|
|
|
|
|
|
|
|
effectSs->pos.x += effectSs->velocity.x;
|
|
|
|
effectSs->pos.y += effectSs->velocity.y;
|
|
|
|
effectSs->pos.z += effectSs->velocity.z;
|
|
|
|
|
|
|
|
effectSs->update(globalCtx, index, effectSs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void EffectSs_UpdateAll(GlobalContext* globalCtx) {
|
|
|
|
s32 i;
|
|
|
|
|
|
|
|
for (i = 0; i < sEffectSsInfo.tableSize; i++) {
|
|
|
|
if (sEffectSsInfo.table[i].life > -1) {
|
|
|
|
sEffectSsInfo.table[i].life--;
|
|
|
|
|
|
|
|
if (sEffectSsInfo.table[i].life < 0) {
|
|
|
|
EffectSs_Delete(&sEffectSsInfo.table[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sEffectSsInfo.table[i].life > -1) {
|
|
|
|
EffectSs_Update(globalCtx, i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void EffectSs_Draw(GlobalContext* globalCtx, s32 index) {
|
|
|
|
EffectSs* effectSs = &sEffectSsInfo.table[index];
|
|
|
|
|
|
|
|
if (effectSs->draw != NULL) {
|
2022-05-13 18:43:55 -04:00
|
|
|
FrameInterpolation_RecordOpenChild(effectSs, effectSs->epoch);
|
2022-03-21 21:51:23 -04:00
|
|
|
effectSs->draw(globalCtx, index, effectSs);
|
2022-05-13 18:43:55 -04:00
|
|
|
FrameInterpolation_RecordCloseChild();
|
2022-03-21 21:51:23 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// original name: "EffectSoftSprite2_disp"
|
|
|
|
void EffectSs_DrawAll(GlobalContext* globalCtx) {
|
|
|
|
Lights* lights = LightContext_NewLights(&globalCtx->lightCtx, globalCtx->state.gfxCtx);
|
|
|
|
s32 i;
|
|
|
|
|
|
|
|
Lights_BindAll(lights, globalCtx->lightCtx.listHead, NULL);
|
|
|
|
Lights_Draw(lights, globalCtx->state.gfxCtx);
|
|
|
|
|
|
|
|
for (i = 0; i < sEffectSsInfo.tableSize; i++) {
|
|
|
|
if (sEffectSsInfo.table[i].life > -1) {
|
|
|
|
if ((sEffectSsInfo.table[i].pos.x > 32000.0f) || (sEffectSsInfo.table[i].pos.x < -32000.0f) ||
|
|
|
|
(sEffectSsInfo.table[i].pos.y > 32000.0f) || (sEffectSsInfo.table[i].pos.y < -32000.0f) ||
|
|
|
|
(sEffectSsInfo.table[i].pos.z > 32000.0f) || (sEffectSsInfo.table[i].pos.z < -32000.0f)) {
|
|
|
|
osSyncPrintf(VT_FGCOL(RED));
|
|
|
|
// "Since the position is outside the area, delete it.
|
|
|
|
// Effect label No. %d: Please respond by the program.
|
|
|
|
// Here is ==> pos (%f, %f, %f) and the label is in z_effect_soft_sprite_dlftbls.decl."
|
|
|
|
osSyncPrintf("EffectSoftSprite2_disp():位置が領域外のため "
|
|
|
|
"削除します。エフェクトラベルNo.%d:プログラムの方で対応をお願いします。ここです ==> "
|
|
|
|
"pos(%f, %f, %f)で、ラベルはz_effect_soft_sprite_dlftbls.declにあります。\n",
|
|
|
|
sEffectSsInfo.table[i].type, sEffectSsInfo.table[i].pos.x, sEffectSsInfo.table[i].pos.y,
|
|
|
|
sEffectSsInfo.table[i].pos.z);
|
|
|
|
osSyncPrintf(VT_FGCOL(GREEN));
|
|
|
|
// "If you are using pos for something else, consult me."
|
|
|
|
osSyncPrintf("もし、posを別のことに使っている場合相談に応じます。\n");
|
|
|
|
osSyncPrintf(VT_RST);
|
|
|
|
|
|
|
|
EffectSs_Delete(&sEffectSsInfo.table[i]);
|
|
|
|
} else {
|
|
|
|
EffectSs_Draw(globalCtx, i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s16 func_80027DD4(s16 arg0, s16 arg1, s32 arg2) {
|
|
|
|
s16 ret = (arg2 == 0) ? arg1 : (arg0 + (s32)((arg1 - arg0) / (f32)arg2));
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
s16 func_80027E34(s16 arg0, s16 arg1, f32 arg2) {
|
|
|
|
return (arg1 - arg0) * arg2 + arg0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u8 func_80027E84(u8 arg0, u8 arg1, f32 arg2) {
|
|
|
|
return arg2 * ((f32)arg1 - (f32)arg0) + arg0;
|
|
|
|
}
|