mirror of
https://github.com/jakcron/nstool.git
synced 2025-01-25 08:51:06 +00:00
285 lines
6.3 KiB
C++
285 lines
6.3 KiB
C++
#include <nx/NpdmHeader.h>
|
|
|
|
|
|
|
|
nx::NpdmHeader::NpdmHeader()
|
|
{
|
|
clear();
|
|
}
|
|
|
|
nx::NpdmHeader::NpdmHeader(const NpdmHeader & other)
|
|
{
|
|
copyFrom(other);
|
|
}
|
|
|
|
nx::NpdmHeader::NpdmHeader(const byte_t * bytes, size_t len)
|
|
{
|
|
importBinary(bytes, len);
|
|
}
|
|
|
|
bool nx::NpdmHeader::operator==(const NpdmHeader & other) const
|
|
{
|
|
return isEqual(other);
|
|
}
|
|
|
|
bool nx::NpdmHeader::operator!=(const NpdmHeader & other) const
|
|
{
|
|
return isEqual(other);
|
|
}
|
|
|
|
void nx::NpdmHeader::operator=(const NpdmHeader & other)
|
|
{
|
|
copyFrom(other);
|
|
}
|
|
|
|
const byte_t * nx::NpdmHeader::getBytes() const
|
|
{
|
|
return mBinaryBlob.getBytes();
|
|
}
|
|
|
|
size_t nx::NpdmHeader::getSize() const
|
|
{
|
|
return mBinaryBlob.getSize();
|
|
}
|
|
|
|
void nx::NpdmHeader::calculateOffsets()
|
|
{
|
|
mAcidPos.offset = align(sizeof(sNpdmHeader), kNpdmAlignSize);
|
|
mAciPos.offset = mAcidPos.offset + align(mAcidPos.size, kNpdmAlignSize);
|
|
}
|
|
|
|
bool nx::NpdmHeader::isEqual(const NpdmHeader & other) const
|
|
{
|
|
return (mInstructionType == other.mInstructionType) \
|
|
&& (mProcAddressSpaceType == other.mProcAddressSpaceType) \
|
|
&& (mMainThreadPriority == other.mMainThreadPriority) \
|
|
&& (mMainThreadCpuId == other.mMainThreadCpuId) \
|
|
&& (mVersion == other.mVersion) \
|
|
&& (mMainThreadStackSize == other.mMainThreadStackSize) \
|
|
&& (mName == other.mName) \
|
|
&& (mProductCode == other.mProductCode) \
|
|
&& (mAciPos == other.mAciPos) \
|
|
&& (mAcidPos == other.mAcidPos);
|
|
}
|
|
|
|
void nx::NpdmHeader::copyFrom(const NpdmHeader & other)
|
|
{
|
|
if (other.getSize())
|
|
{
|
|
importBinary(other.getBytes(), other.getSize());
|
|
}
|
|
else
|
|
{
|
|
mInstructionType = other.mInstructionType;
|
|
mProcAddressSpaceType = other.mProcAddressSpaceType;
|
|
mMainThreadPriority = other.mMainThreadPriority;
|
|
mMainThreadCpuId = other.mMainThreadCpuId;
|
|
mVersion = other.mVersion;
|
|
mMainThreadStackSize = other.mMainThreadStackSize;
|
|
mName = other.mName;
|
|
mProductCode = other.mProductCode;
|
|
mAciPos = other.mAciPos;
|
|
mAcidPos = other.mAcidPos;
|
|
}
|
|
}
|
|
|
|
void nx::NpdmHeader::exportBinary()
|
|
{
|
|
mBinaryBlob.alloc(sizeof(sNpdmHeader));
|
|
sNpdmHeader* hdr = (sNpdmHeader*)mBinaryBlob.getBytes();
|
|
|
|
hdr->set_signature(kNpdmStructSig.c_str());
|
|
byte_t flag = ((byte_t)(mInstructionType & 1) | (byte_t)((mProcAddressSpaceType & 3) << 1)) & 0xf;
|
|
hdr->set_flags(flag);
|
|
hdr->set_main_thread_priority(mMainThreadPriority);
|
|
hdr->set_main_thread_cpu_id(mMainThreadCpuId);
|
|
hdr->set_version(mVersion);
|
|
hdr->set_main_thread_stack_size(mMainThreadStackSize);
|
|
hdr->set_name(mName.c_str());
|
|
hdr->set_product_code(mProductCode.c_str());
|
|
|
|
calculateOffsets();
|
|
hdr->aci().set_offset(mAciPos.offset);
|
|
hdr->aci().set_size(mAciPos.size);
|
|
hdr->acid().set_offset(mAcidPos.offset);
|
|
hdr->acid().set_size(mAcidPos.size);
|
|
}
|
|
|
|
void nx::NpdmHeader::importBinary(const byte_t * bytes, size_t len)
|
|
{
|
|
if (len < sizeof(sNpdmHeader))
|
|
{
|
|
throw fnd::Exception(kModuleName, "NPDM header too small");
|
|
}
|
|
|
|
clear();
|
|
|
|
mBinaryBlob.alloc(sizeof(sNpdmHeader));
|
|
memcpy(mBinaryBlob.getBytes(), bytes, mBinaryBlob.getSize());
|
|
sNpdmHeader* hdr = (sNpdmHeader*)mBinaryBlob.getBytes();
|
|
|
|
if (memcmp(kNpdmStructSig.c_str(), hdr->signature(), 4) != 0)
|
|
{
|
|
throw fnd::Exception(kModuleName, "NPDM header corrupt");
|
|
}
|
|
|
|
byte_t flag = hdr->flags() & 0xf;
|
|
mInstructionType = (InstructionType)(flag & 1);
|
|
mProcAddressSpaceType = (ProcAddrSpaceType)((flag >> 1) & 3);
|
|
mMainThreadPriority = hdr->main_thread_priority();
|
|
mMainThreadCpuId = hdr->main_thread_cpu_id();
|
|
mVersion = hdr->version();
|
|
mMainThreadStackSize = hdr->main_thread_stack_size();
|
|
mName = std::string(hdr->name(), kNameMaxLen);
|
|
if (mName[0] == '\0')
|
|
{
|
|
mName.clear();
|
|
}
|
|
mProductCode = std::string(hdr->product_code(), kProductCodeMaxLen);
|
|
if (mProductCode[0] == '\0')
|
|
{
|
|
mProductCode.clear();
|
|
}
|
|
mAciPos.offset = hdr->aci().offset();
|
|
mAciPos.size = hdr->aci().size();
|
|
mAcidPos.offset = hdr->acid().offset();
|
|
mAcidPos.size = hdr->acid().size();
|
|
}
|
|
|
|
void nx::NpdmHeader::clear()
|
|
{
|
|
mBinaryBlob.clear();
|
|
mInstructionType = INSTR_64BIT;
|
|
mProcAddressSpaceType = ADDR_SPACE_64BIT;
|
|
mMainThreadPriority = 0;
|
|
mMainThreadCpuId = 0;
|
|
mVersion = 0;
|
|
mMainThreadStackSize = 0;
|
|
mName.clear();
|
|
mProductCode.clear();
|
|
mAciPos.offset = 0;
|
|
mAciPos.size = 0;
|
|
mAcidPos.offset = 0;
|
|
mAcidPos.size = 0;
|
|
}
|
|
|
|
size_t nx::NpdmHeader::getNpdmSize() const
|
|
{
|
|
return MAX(mAcidPos.offset + mAcidPos.size, mAciPos.offset + mAciPos.size);
|
|
}
|
|
|
|
nx::NpdmHeader::InstructionType nx::NpdmHeader::getInstructionType() const
|
|
{
|
|
return mInstructionType;
|
|
}
|
|
|
|
void nx::NpdmHeader::setInstructionType(InstructionType type)
|
|
{
|
|
mInstructionType = type;
|
|
}
|
|
|
|
nx::NpdmHeader::ProcAddrSpaceType nx::NpdmHeader::getProcAddressSpaceType() const
|
|
{
|
|
return mProcAddressSpaceType;
|
|
}
|
|
|
|
void nx::NpdmHeader::setProcAddressSpaceType(ProcAddrSpaceType type)
|
|
{
|
|
mProcAddressSpaceType = type;
|
|
}
|
|
|
|
byte_t nx::NpdmHeader::getMainThreadPriority() const
|
|
{
|
|
return mMainThreadPriority;
|
|
}
|
|
|
|
void nx::NpdmHeader::setMainThreadPriority(byte_t priority)
|
|
{
|
|
if (priority > kMaxPriority)
|
|
{
|
|
throw fnd::Exception(kModuleName, "Illegal main thread priority (range 0-63)");
|
|
}
|
|
|
|
mMainThreadPriority = priority;
|
|
}
|
|
|
|
byte_t nx::NpdmHeader::getMainThreadCpuId() const
|
|
{
|
|
return mMainThreadCpuId;
|
|
}
|
|
|
|
void nx::NpdmHeader::setMainThreadCpuId(byte_t core_num)
|
|
{
|
|
mMainThreadCpuId = core_num;
|
|
}
|
|
|
|
uint32_t nx::NpdmHeader::getVersion() const
|
|
{
|
|
return mVersion;
|
|
}
|
|
|
|
void nx::NpdmHeader::setVersion(uint32_t version)
|
|
{
|
|
mVersion = version;
|
|
}
|
|
|
|
uint32_t nx::NpdmHeader::getMainThreadStackSize() const
|
|
{
|
|
return mMainThreadStackSize;
|
|
}
|
|
|
|
void nx::NpdmHeader::setMainThreadStackSize(uint32_t size)
|
|
{
|
|
mMainThreadStackSize = size;
|
|
}
|
|
|
|
const std::string & nx::NpdmHeader::getName() const
|
|
{
|
|
return mName;
|
|
}
|
|
|
|
void nx::NpdmHeader::setName(const std::string & name)
|
|
{
|
|
if (name.length() > kNameMaxLen)
|
|
{
|
|
throw fnd::Exception(kModuleName, "Name is too long");
|
|
}
|
|
|
|
mName = name;
|
|
}
|
|
|
|
const std::string & nx::NpdmHeader::getProductCode() const
|
|
{
|
|
return mProductCode;
|
|
}
|
|
|
|
void nx::NpdmHeader::setProductCode(const std::string & product_code)
|
|
{
|
|
if (product_code.length() > kProductCodeMaxLen)
|
|
{
|
|
throw fnd::Exception(kModuleName, "Product Code is too long");
|
|
}
|
|
|
|
mProductCode = product_code;
|
|
}
|
|
|
|
const nx::NpdmHeader::sSection & nx::NpdmHeader::getAciPos() const
|
|
{
|
|
return mAciPos;
|
|
}
|
|
|
|
void nx::NpdmHeader::setAciSize(size_t size)
|
|
{
|
|
mAciPos.size = size;
|
|
}
|
|
|
|
const nx::NpdmHeader::sSection & nx::NpdmHeader::getAcidPos() const
|
|
{
|
|
return mAcidPos;
|
|
}
|
|
|
|
void nx::NpdmHeader::setAcidSize(size_t size)
|
|
{
|
|
mAcidPos.size = size;
|
|
}
|