botw/src/KingSystem/Event/evtOrderParam.cpp

186 lines
5.6 KiB
C++

#include "KingSystem/Event/evtOrderParam.h"
#include <basis/seadTypes.h>
#include <codec/seadHashCRC32.h>
#include <container/seadBuffer.h>
#include <heap/seadExpHeap.h>
#include <prim/seadSafeString.h>
#include "KingSystem/ActorSystem/actBaseProcLink.h"
namespace ksys::evt {
OrderParam::OrderParam(sead::ExpHeap* heap) {
mHeap = heap;
}
bool OrderParam::initialize(s32 entry_count) {
sead::FixedSafeString<0x100> error_message;
error_message.format("[%s] initialize(%d) is failed.", "ksys::evt::OrderParam", entry_count);
uninitialize();
if (entry_count == 0)
return true;
if (!mHeap)
return false;
if (entry_count < 1)
return false;
if (!mEntries.tryAllocBuffer(entry_count, mHeap))
return false;
// I think compiler is unrolling this loop
for (s32 i = 0; i != mEntries.size(); ++i) {
clearEntry(&mEntries[i]);
}
mEntryCount = 0;
mInitialized = true;
return true;
}
void OrderParam::uninitialize() {
for (s32 i = 0; i < mEntries.size(); i++) {
auto* e = &mEntries[i];
auto* name_ptr = e->mName;
if (name_ptr)
delete (sead::FixedSafeString<0x20>*)name_ptr;
auto* ptr = e->mPointer;
if (ptr) {
switch (e->mType) {
case OrderParamType::STRING:
delete (sead::FixedSafeString<0x40>*)ptr;
break;
case OrderParamType::INT:
case OrderParamType::INT_2:
delete (u32*)ptr;
break;
case OrderParamType::BYTE:
delete (char*)ptr;
break;
case OrderParamType::ACTOR:
delete (ksys::act::BaseProcLink*)ptr;
break;
case OrderParamType::ARRAY:
mHeap->free(ptr);
break;
default:
break;
}
}
clearEntry(e);
}
mEntries.freeBuffer();
mEntryCount = 0;
mInitialized = false;
}
// half done
void OrderParam::addParamActor(ksys::act::BaseProc& actor, sead::SafeString& name) {
u32 hash = sead::HashCRC32::calcStringHash(name.cstr());
s32 i;
for (i = 0; i < mEntries.size(); i++) {
if (mEntries[i].mHash == hash && mEntries[i].mType == OrderParamType::ACTOR) {
if (!mEntries[i].mPointer) {
break;
}
}
}
auto* entry = tryAlloc(OrderParamType::ACTOR, 0, name);
if (entry) {
auto* actor_ptr = (ksys::act::BaseProcLink*)entry->mPointer;
if (actor_ptr->acquire(&actor, false)) {
++this->mEntryCount;
}
}
}
bool OrderParam::getIntByName(const sead::SafeString& name, u32** out_ptr) {
return getPointerByName(name, out_ptr, nullptr, OrderParamType::INT);
}
bool OrderParam::getStringByName(const sead::SafeString& name, sead::SafeString** out_ptr) {
return getPointerByName(name, out_ptr, nullptr, OrderParamType::STRING);
}
bool OrderParam::getArrayByName(const sead::SafeString& name, void** out_ptr, u32* out_size) {
return getPointerByName(name, out_ptr, out_size, OrderParamType::ARRAY);
}
// This one also does not match
OrderParamEntry* OrderParam::tryAlloc(OrderParamType type, u32 size, sead::SafeString& name) {
sead::FixedSafeString<0x100> error_message;
error_message.format("[%s] tryAlloc_(%d, %d, %s) is failed.", "ksys::evt::OrderParam", type,
size, name.cstr());
for (s32 i = 0; i < mEntries.size(); i++) {
auto* e = &mEntries[i];
if (!e->mPointer) {
void** in_ptr = &(e->mPointer);
std::nothrow_t nothrow_t;
auto* heap = mHeap;
if (!heap)
return nullptr;
e->mName = new (heap, nothrow_t) sead::FixedSafeString<0x20>(name);
switch (type) {
case OrderParamType::STRING:
*in_ptr = new (heap, nothrow_t) sead::FixedSafeString<0x40>;
size = sizeof(sead::FixedSafeString<0x40>);
break;
case OrderParamType::INT:
case OrderParamType::INT_2:
*in_ptr = new (heap, nothrow_t) u32(0);
size = sizeof(u32);
break;
case OrderParamType::BYTE:
*in_ptr = new (heap, nothrow_t) char(0);
size = sizeof(char);
break;
case OrderParamType::ACTOR:
*in_ptr = new (heap, nothrow_t) ksys::act::BaseProcLink;
size = sizeof(ksys::act::BaseProcLink);
break;
case OrderParamType::ARRAY:
*in_ptr = new (heap, nothrow_t) char[size];
default:
break;
}
e->mSize = size;
if (e->mPointer) {
e->mHash = sead::HashCRC32::calcStringHash(e->mName->cstr());
e->mType = type;
return e;
}
if (e->mName)
delete e->mName;
// clearEntry(e);
*e = {};
return nullptr;
// auto* entry = mAllocArray+i;
// if (*in_ptr) {
//} else {
//}
}
}
return nullptr;
}
void* OrderParam::getPointerByName(const sead::SafeString& name, u32* out_size,
OrderParamType type) const {
const u32 hash = sead::HashCRC32::calcStringHash(name);
for (s32 i = 0; i < mEntries.size(); i++) {
if (mEntries[i].mHash == hash && mEntries[i].mType == type) {
if (out_size)
*out_size = mEntries[i].mSize;
return mEntries[i].mPointer;
}
}
return nullptr;
}
} // namespace ksys::evt