#include using namespace nx; void AciHeader::calculateSectionOffsets() { mFac.offset = align(mHeaderOffset, aci::kAciAlignSize) + align(sizeof(sAciHeader), aci::kAciAlignSize); mSac.offset = mFac.offset + align(mFac.size, aci::kAciAlignSize); mKc.offset = mSac.offset + align(mSac.size, aci::kAciAlignSize); } bool AciHeader::isEqual(const AciHeader & other) const { return (mHeaderOffset == other.mHeaderOffset) \ && (mType == other.mType) \ && (mIsProduction == other.mIsProduction) \ && (mIsUnqualifiedApproval == other.mIsUnqualifiedApproval) \ && (mAcidSize == other.mAcidSize) \ && (mProgramIdMin == other.mProgramIdMin) \ && (mProgramIdMax == other.mProgramIdMax) \ && (mProgramId == other.mProgramId) \ && (mFac == other.mFac) \ && (mSac == other.mSac) \ && (mKc == other.mKc); } void AciHeader::copyFrom(const AciHeader & other) { if (other.getSize()) { importBinary(other.getBytes(), other.getSize()); } else { mHeaderOffset = other.mHeaderOffset; mType = other.mType; mIsProduction = other.mIsProduction; mIsUnqualifiedApproval = other.mIsUnqualifiedApproval; mAcidSize = other.mAcidSize; mProgramIdMin = other.mProgramIdMin; mProgramIdMax = other.mProgramIdMax; mProgramId = other.mProgramId; mFac = other.mFac; mSac = other.mSac; mKc = other.mKc; } } AciHeader::AciHeader() { clear(); } AciHeader::AciHeader(const AciHeader & other) { importBinary(other.getBytes(), other.getSize()); } AciHeader::AciHeader(const byte_t * bytes, size_t len) { importBinary(bytes, len); } bool AciHeader::operator==(const AciHeader & other) const { return isEqual(other); } bool AciHeader::operator!=(const AciHeader & other) const { return !isEqual(other); } void AciHeader::operator=(const AciHeader & other) { this->importBinary(other.getBytes(), other.getSize()); } const byte_t * AciHeader::getBytes() const { return mBinaryBlob.getBytes(); } size_t AciHeader::getSize() const { return mBinaryBlob.getSize(); } void AciHeader::exportBinary() { mBinaryBlob.alloc(sizeof(sAciHeader)); sAciHeader* hdr = (sAciHeader*)mBinaryBlob.getBytes(); // set type switch (mType) { case (TYPE_ACI0): memcpy(hdr->signature, aci::kAciStructSig.c_str(), 4); break; case (TYPE_ACID): memcpy(hdr->signature, aci::kAciDescStructSig.c_str(), 4); break; default: throw fnd::Exception(kModuleName, "Unexpected ACI type"); } // set offset/size calculateSectionOffsets(); hdr->fac.offset = mFac.offset; hdr->fac.size = mFac.size; hdr->sac.offset = mSac.offset; hdr->sac.size = mSac.size; hdr->kc.offset = mKc.offset; hdr->kc.size = mKc.size; uint32_t flags = 0; if (mIsProduction) flags |= _BIT(aci::FLAG_PRODUCTION); if (mIsUnqualifiedApproval) flags |= _BIT(aci::FLAG_UNQUALIFIED_APPROVAL); hdr->flags = flags; if (mType == TYPE_ACI0) { // set program hdr->program_id_info.program_id = mProgramId; } else if (mType == TYPE_ACID) { mAcidSize = getAciSize(); hdr->size = mAcidSize; hdr->program_id_info.program_id_restrict.min = mProgramIdMin; hdr->program_id_info.program_id_restrict.max = mProgramIdMax; } } void AciHeader::importBinary(const byte_t * bytes, size_t len) { if (len < sizeof(sAciHeader)) { throw fnd::Exception(kModuleName, "ACI header too small"); } clear(); mBinaryBlob.alloc(sizeof(sAciHeader)); memcpy(mBinaryBlob.getBytes(), bytes, sizeof(sAciHeader)); sAciHeader* hdr = (sAciHeader*)mBinaryBlob.getBytes(); if (std::string(hdr->signature, 4) == aci::kAciStructSig) { mType = TYPE_ACI0; } else if (std::string(hdr->signature, 4) == aci::kAciDescStructSig) { mType = TYPE_ACID; } else { throw fnd::Exception(kModuleName, "ACI header corrupt"); } if (mType == TYPE_ACI0) { mProgramId = hdr->program_id_info.program_id.get(); mIsProduction = false; mIsUnqualifiedApproval = false; mAcidSize = 0; mProgramIdMin = 0; mProgramIdMax = 0; } else if (mType == TYPE_ACID) { mProgramId = 0; mIsProduction = _HAS_BIT(hdr->flags.get(), aci::FLAG_PRODUCTION); mIsUnqualifiedApproval = _HAS_BIT(hdr->flags.get(), aci::FLAG_UNQUALIFIED_APPROVAL); mAcidSize = hdr->size.get(); mProgramIdMin = hdr->program_id_info.program_id_restrict.min.get(); mProgramIdMax = hdr->program_id_info.program_id_restrict.max.get(); } // the header offset is the MIN(sac.offset, fac.offset, kc.offset) - sizeof(sHeader) mHeaderOffset = MAX(MIN(hdr->sac.offset.get(), MIN(hdr->fac.offset.get(), hdr->kc.offset.get())), align(sizeof(sAciHeader), aci::kAciAlignSize)) - align(sizeof(sAciHeader), aci::kAciAlignSize); mFac.offset = hdr->fac.offset.get() - mHeaderOffset; mFac.size = hdr->fac.size.get(); mSac.offset = hdr->sac.offset.get() - mHeaderOffset; mSac.size = hdr->sac.size.get(); mKc.offset = hdr->kc.offset.get() - mHeaderOffset; mKc.size = hdr->kc.size.get(); } void nx::AciHeader::clear() { mBinaryBlob.clear(); mHeaderOffset = 0; mType = TYPE_ACI0; mProgramId = 0; mProgramIdMin = 0; mProgramIdMax = 0; mAcidSize = 0; mIsProduction = false; mIsUnqualifiedApproval = false; mFac.offset = 0; mFac.size = 0; mSac.offset = 0; mSac.size = 0; mKc.offset = 0; mKc.size = 0; } size_t nx::AciHeader::getAciSize() const { return MAX(MAX(MAX(mSac.offset + mSac.size, mKc.offset + mKc.size), mFac.offset + mFac.size), sizeof(sAciHeader)); } size_t nx::AciHeader::getAcidSize() const { return mAcidSize; } /* void nx::AciHeader::setAcidSize(size_t size) { mAcidSize = size; } */ uint64_t nx::AciHeader::getProgramIdMin() const { return mProgramIdMin; } void nx::AciHeader::setProgramIdMin(uint64_t program_id) { mProgramIdMin = program_id; } uint64_t nx::AciHeader::getProgramIdMax() const { return mProgramIdMax; } void nx::AciHeader::setProgramIdMax(uint64_t program_id) { mProgramIdMax = program_id; } void nx::AciHeader::setHeaderOffset(size_t offset) { mHeaderOffset = offset; } AciHeader::AciType AciHeader::getAciType() const { return mType; } void AciHeader::setAciType(AciType type) { mType = type; } bool nx::AciHeader::isProduction() const { return mIsProduction; } void nx::AciHeader::setIsProduction(bool isProduction) { mIsProduction = isProduction; } bool nx::AciHeader::isUnqualifiedApproval() const { return mIsUnqualifiedApproval; } void nx::AciHeader::setIsUnqualifiedApproval(bool isUnqualifiedApproval) { mIsUnqualifiedApproval = isUnqualifiedApproval; } uint64_t AciHeader::getProgramId() const { return mProgramId; } void AciHeader::setProgramId(uint64_t program_id) { mProgramId = program_id; } const AciHeader::sSection & AciHeader::getFacPos() const { return mFac; } void AciHeader::setFacSize(size_t size) { mFac.size = size; } const AciHeader::sSection & AciHeader::getSacPos() const { return mSac; } void AciHeader::setSacSize(size_t size) { mSac.size = size; } const AciHeader::sSection & AciHeader::getKcPos() const { return mKc; } void AciHeader::setKcSize(size_t size) { mKc.size = size; }