2017-07-17 08:26:19 +00:00
|
|
|
#include <cstdio>
|
|
|
|
#include <crypto/aes.h>
|
|
|
|
#include <fnd/io.h>
|
2017-08-05 13:09:50 +00:00
|
|
|
#include <fnd/MemoryBlob.h>
|
2017-07-17 08:26:19 +00:00
|
|
|
#include <nx/NXCrypto.h>
|
|
|
|
#include <nx/NpdmBinary.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
|
|
|
|
const std::string kInstructionType[2] = { "32Bit", "64Bit" };
|
|
|
|
const std::string kProcAddrSpace[4] = { "Unknown", "64Bit", "32Bit", "32Bit no reserved" };
|
|
|
|
const std::string kAciType[2] = { "ACI0", "ACID" };
|
|
|
|
const std::string kMiscFlag[15] = { "EnableDebug", "ForceDebug", "bit2", "bit3", "bit4", "bit5", "bit6", "bit7", "bit8", "bit9", "bit10", "bit11", "bit12", "bit13", "bit14"};
|
|
|
|
const std::string kFsaFlag[64] =
|
|
|
|
{
|
|
|
|
"ApplicationInfo",
|
|
|
|
"BootModeControl",
|
|
|
|
"Calibration",
|
|
|
|
"SystemSaveData",
|
|
|
|
"GameCard",
|
|
|
|
"SaveDataBackUp",
|
|
|
|
"SaveDataManagement",
|
|
|
|
"BisAllRaw",
|
|
|
|
"GameCardRaw",
|
|
|
|
"GameCardPrivate",
|
|
|
|
"SetTime",
|
|
|
|
"ContentManager",
|
|
|
|
"ImageManager",
|
|
|
|
"CreateSaveData",
|
|
|
|
"SystemSaveDataManagement",
|
|
|
|
"BisFileSystem",
|
|
|
|
"SystemUpdate",
|
|
|
|
"SaveDataMeta",
|
|
|
|
"DeviceSaveData",
|
|
|
|
"SettingsControl",
|
|
|
|
"Bit20",
|
|
|
|
"Bit21",
|
|
|
|
"Bit22",
|
|
|
|
"Bit23",
|
|
|
|
"Bit24",
|
|
|
|
"Bit25",
|
|
|
|
"Bit26",
|
|
|
|
"Bit27",
|
|
|
|
"Bit28",
|
|
|
|
"Bit29",
|
|
|
|
"Bit30",
|
|
|
|
"Bit31",
|
|
|
|
"Bit32",
|
|
|
|
"Bit33",
|
|
|
|
"Bit34",
|
|
|
|
"Bit35",
|
|
|
|
"Bit36",
|
|
|
|
"Bit37",
|
|
|
|
"Bit38",
|
|
|
|
"Bit39",
|
|
|
|
"Bit40",
|
|
|
|
"Bit41",
|
|
|
|
"Bit42",
|
|
|
|
"Bit43",
|
|
|
|
"Bit44",
|
|
|
|
"Bit45",
|
|
|
|
"Bit46",
|
|
|
|
"Bit47",
|
|
|
|
"Bit48",
|
|
|
|
"Bit49",
|
|
|
|
"Bit50",
|
|
|
|
"Bit51",
|
|
|
|
"Bit52",
|
|
|
|
"Bit53",
|
|
|
|
"Bit54",
|
|
|
|
"Bit55",
|
|
|
|
"Bit56",
|
|
|
|
"Bit57",
|
|
|
|
"Bit58",
|
|
|
|
"Bit59",
|
|
|
|
"Bit60",
|
|
|
|
"Bit61",
|
|
|
|
"Debug",
|
|
|
|
"FullPermission"
|
|
|
|
};
|
|
|
|
const std::string kSysCall[0x80] =
|
|
|
|
{
|
|
|
|
"svc00",
|
|
|
|
"SetHeapSize",
|
|
|
|
"SetMemoryPermission",
|
|
|
|
"SetMemoryAttribute",
|
|
|
|
"MapMemory",
|
|
|
|
"UnmapMemory",
|
|
|
|
"QueryMemory",
|
|
|
|
"ExitProcess",
|
|
|
|
"CreateThread",
|
|
|
|
"StartThread",
|
|
|
|
"ExitThread",
|
|
|
|
"SleepThread",
|
|
|
|
"GetThreadPriority",
|
|
|
|
"SetThreadPriority",
|
|
|
|
"GetThreadCoreMask",
|
|
|
|
"SetThreadCoreMask",
|
|
|
|
"GetCurrentProcessorNumber",
|
|
|
|
"SignalEvent",
|
|
|
|
"ClearEvent",
|
|
|
|
"MapSharedMemory",
|
|
|
|
"UnmapSharedMemory",
|
|
|
|
"CreateTransferMemory",
|
|
|
|
"CloseHandle",
|
|
|
|
"ResetSignal",
|
|
|
|
"WaitSynchronization",
|
|
|
|
"CancelSynchronization",
|
|
|
|
"ArbitrateLock",
|
|
|
|
"ArbitrateUnlock",
|
|
|
|
"WaitProcessWideKeyAtomic",
|
|
|
|
"SignalProcessWideKey",
|
|
|
|
"GetSystemTick",
|
|
|
|
"ConnectToNamedPort",
|
|
|
|
"SendSyncRequestLight",
|
|
|
|
"SendSyncRequest",
|
|
|
|
"SendSyncRequestWithUserBuffer",
|
|
|
|
"SendAsyncRequestWithUserBuffer",
|
|
|
|
"GetProcessId",
|
|
|
|
"GetThreadId",
|
|
|
|
"Break",
|
|
|
|
"OutputDebugString",
|
|
|
|
"ReturnFromException",
|
|
|
|
"GetInfo",
|
|
|
|
"FlushEntireDataCache",
|
|
|
|
"FlushDataCache",
|
|
|
|
"MapPhysicalMemory",
|
|
|
|
"UnmapPhysicalMemory",
|
|
|
|
"svc2E",
|
|
|
|
"GetLastThreadInfo",
|
|
|
|
"GetResourceLimitLimitValue",
|
|
|
|
"GetResourceLimitCurrentValue",
|
|
|
|
"SetThreadActivity",
|
|
|
|
"GetThreadContext3",
|
|
|
|
"svc34",
|
|
|
|
"svc35",
|
|
|
|
"svc36",
|
|
|
|
"svc37",
|
|
|
|
"svc38",
|
|
|
|
"svc39",
|
|
|
|
"svc3A",
|
|
|
|
"svc3B",
|
|
|
|
"DumpInfo",
|
|
|
|
"svc3D",
|
|
|
|
"svc3E",
|
|
|
|
"svc3F",
|
|
|
|
"CreateSession",
|
|
|
|
"AcceptSession",
|
|
|
|
"ReplyAndReceiveLight",
|
|
|
|
"ReplyAndReceive",
|
|
|
|
"ReplyAndReceiveWithUserBuffer",
|
|
|
|
"CreateEvent",
|
|
|
|
"svc46",
|
|
|
|
"svc47",
|
|
|
|
"svc48",
|
|
|
|
"svc49",
|
|
|
|
"svc4A",
|
|
|
|
"svc4B",
|
|
|
|
"svc4C",
|
|
|
|
"SleepSystem",
|
|
|
|
"ReadWriteRegister",
|
|
|
|
"SetProcessActivity",
|
|
|
|
"CreateSharedMemory",
|
|
|
|
"MapTransferMemory",
|
|
|
|
"UnmapTransferMemory",
|
|
|
|
"CreateInterruptEvent",
|
|
|
|
"QueryPhysicalAddress",
|
|
|
|
"QueryIoMapping",
|
|
|
|
"CreateDeviceAddressSpace",
|
|
|
|
"AttachDeviceAddressSpace",
|
|
|
|
"DetachDeviceAddressSpace",
|
|
|
|
"MapDeviceAddressSpaceByForce",
|
|
|
|
"MapDeviceAddressSpaceAligned",
|
|
|
|
"MapDeviceAddressSpace",
|
|
|
|
"UnmapDeviceAddressSpace",
|
|
|
|
"InvalidateProcessDataCache",
|
|
|
|
"StoreProcessDataCache",
|
|
|
|
"FlushProcessDataCache",
|
|
|
|
"DebugActiveProcess",
|
|
|
|
"BreakDebugProcess",
|
|
|
|
"TerminateDebugProcess",
|
|
|
|
"GetDebugEvent",
|
|
|
|
"ContinueDebugEvent",
|
|
|
|
"GetProcessList",
|
|
|
|
"GetThreadList",
|
|
|
|
"GetDebugThreadContext",
|
|
|
|
"SetDebugThreadContext",
|
|
|
|
"QueryDebugProcessMemory",
|
|
|
|
"ReadDebugProcessMemory",
|
|
|
|
"WriteDebugProcessMemory",
|
|
|
|
"SetHardwareBreakPoint",
|
|
|
|
"GetDebugThreadParam",
|
|
|
|
"svc6E",
|
|
|
|
"svc6F",
|
|
|
|
"CreatePort",
|
|
|
|
"ManageNamedPort",
|
|
|
|
"ConnectToPort",
|
|
|
|
"SetProcessMemoryPermission",
|
|
|
|
"MapProcessMemory",
|
|
|
|
"UnmapProcessMemory",
|
|
|
|
"QueryProcessMemory",
|
|
|
|
"MapProcessCodeMemory",
|
|
|
|
"UnmapProcessCodeMemory",
|
|
|
|
"CreateProcess",
|
|
|
|
"StartProcess",
|
|
|
|
"TerminateProcess",
|
|
|
|
"GetProcessInfo",
|
|
|
|
"CreateResourceLimit",
|
|
|
|
"SetResourceLimitLimitValue",
|
|
|
|
"CallSecureMonitor"
|
|
|
|
};
|
|
|
|
|
|
|
|
const std::string kMemMapPerm[2] = { "RW", "RO" };
|
|
|
|
const std::string kMemMapType[2] = { "Io", "Static" };
|
|
|
|
|
2017-07-21 10:30:16 +00:00
|
|
|
const std::string kAcidTarget[2] = { "Development", "Production" };
|
|
|
|
|
2017-07-17 08:26:19 +00:00
|
|
|
void displayNpdmHeader(const nx::NpdmHeader& hdr)
|
|
|
|
{
|
|
|
|
printf("[NPDM HEADER]\n");
|
|
|
|
printf(" Process Architecture Params:\n");
|
|
|
|
printf(" Ins. Type: %s\n", kInstructionType[hdr.getInstructionType()].c_str());
|
|
|
|
printf(" Addr Space: %s\n", kProcAddrSpace[hdr.getProcAddressSpaceType()].c_str());
|
|
|
|
printf(" Main Thread Params:\n");
|
|
|
|
printf(" Priority: %d\n", hdr.getMainThreadPriority());
|
|
|
|
printf(" CpuId: %d\n", hdr.getMainThreadCpuId());
|
|
|
|
printf(" StackSize: 0x%x\n", hdr.getMainThreadStackSize());
|
|
|
|
printf(" TitleInfo:\n");
|
|
|
|
printf(" Version: v%" PRIu32 "\n", hdr.getVersion());
|
|
|
|
printf(" Name: %s\n", hdr.getName().c_str());
|
|
|
|
if (hdr.getProductCode().length())
|
|
|
|
{
|
|
|
|
printf(" ProductCode: %s\n", hdr.getProductCode().c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void displayAciHdr(const nx::AciHeader& aci)
|
|
|
|
{
|
|
|
|
printf("[Access Control Info]\n");
|
|
|
|
printf(" ACI Type: %s\n", kAciType[aci.getAciType()].c_str());
|
|
|
|
if (aci.getAciType() == nx::AciBinary::TYPE_ACI0)
|
|
|
|
{
|
|
|
|
printf(" ProgramID: %016" PRIx64 "\n", aci.getProgramId());
|
|
|
|
}
|
|
|
|
else if (aci.getAciType() == nx::AciBinary::TYPE_ACID)
|
|
|
|
{
|
2017-07-21 10:30:16 +00:00
|
|
|
|
|
|
|
printf(" ACID Size: %" PRIx64 "\n", aci.getAcidSize());
|
|
|
|
printf(" Target: %s\n", kAcidTarget[aci.isProduction()].c_str());
|
|
|
|
printf(" ProgramID Restriction\n");
|
|
|
|
printf(" Min: %016" PRIx64 "\n", aci.getProgramIdMin());
|
|
|
|
printf(" Max: %016" PRIx64 "\n", aci.getProgramIdMax());
|
2017-07-17 08:26:19 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void displayFac(const nx::FacBinary& fac)
|
|
|
|
{
|
|
|
|
printf("[FS Access Control]\n");
|
|
|
|
printf(" Format Version: %d\n", fac.getFormatVersion());
|
|
|
|
|
|
|
|
if (fac.getFsaRightsList().getSize())
|
|
|
|
{
|
|
|
|
printf(" FS Rights:\n");
|
|
|
|
for (size_t i = 0; i < fac.getFsaRightsList().getSize(); i++)
|
|
|
|
{
|
|
|
|
if (i % 10 == 0)
|
|
|
|
{
|
|
|
|
printf("%s ", i != 0 ? "\n" : "");
|
|
|
|
}
|
|
|
|
printf("%s%s", kFsaFlag[fac.getFsaRightsList()[i]].c_str(), fac.getFsaRightsList()[i] != fac.getFsaRightsList().atBack() ? ", " : "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf(" FS Rights: NONE\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fac.getContentOwnerIdList().getSize())
|
|
|
|
{
|
|
|
|
printf(" Content Owner IDs:\n");
|
|
|
|
for (size_t i = 0; i < fac.getContentOwnerIdList().getSize(); i++)
|
|
|
|
{
|
|
|
|
printf(" 0x%08x\n", fac.getContentOwnerIdList()[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (fac.getSaveDataOwnerIdList().getSize())
|
|
|
|
{
|
|
|
|
printf(" Save Data Owner IDs:\n");
|
|
|
|
for (size_t i = 0; i < fac.getSaveDataOwnerIdList().getSize(); i++)
|
|
|
|
{
|
|
|
|
printf(" 0x%08x\n", fac.getSaveDataOwnerIdList()[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void displaySac(const nx::SacBinary& sac)
|
|
|
|
{
|
|
|
|
printf("[Service Access Control]\n");
|
|
|
|
printf(" Service List:\n");
|
|
|
|
for (size_t i = 0; i < sac.getServiceList().getSize(); i++)
|
|
|
|
{
|
|
|
|
if (i % 10 == 0)
|
|
|
|
{
|
|
|
|
printf("%s ", i != 0 ? "\n" : "");
|
|
|
|
}
|
|
|
|
printf("%s%s%s", sac.getServiceList()[i].getName().c_str(), sac.getServiceList()[i].isServer() ? "(isSrv)" : "", sac.getServiceList()[i] != sac.getServiceList().atBack() ? ", " : "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void displayKernelCap(const nx::KcBinary& kern)
|
|
|
|
{
|
|
|
|
printf("[Kernel Capabilities]\n");
|
|
|
|
if (kern.getThreadInfo().isSet())
|
|
|
|
{
|
|
|
|
nx::ThreadInfoHandler threadInfo = kern.getThreadInfo();
|
|
|
|
printf(" Thread Priority:\n");
|
|
|
|
printf(" Min: %d\n", threadInfo.getMinPriority());
|
|
|
|
printf(" Max: %d\n", threadInfo.getMaxPriority());
|
|
|
|
printf(" CpuId:\n");
|
|
|
|
printf(" Min: %d\n", threadInfo.getMinCpuId());
|
|
|
|
printf(" Max: %d\n", threadInfo.getMaxCpuId());
|
|
|
|
}
|
|
|
|
if (kern.getSystemCalls().isSet())
|
|
|
|
{
|
2018-03-22 05:26:22 +00:00
|
|
|
fnd::List<uint8_t> syscalls = kern.getSystemCalls().getSystemCalls();
|
2017-07-17 08:26:19 +00:00
|
|
|
printf(" SystemCalls:");
|
|
|
|
printf("\n ");
|
|
|
|
size_t lineLen = 0;
|
|
|
|
for (size_t i = 0; i < syscalls.getSize(); i++)
|
|
|
|
{
|
|
|
|
if (lineLen > 60)
|
|
|
|
{
|
|
|
|
lineLen = 0;
|
|
|
|
printf("\n ");
|
|
|
|
}
|
|
|
|
printf("%s%s", kSysCall[syscalls[i]].c_str(), syscalls[i] != syscalls.atBack() ? ", " : "\n");
|
|
|
|
lineLen += kSysCall[syscalls[i]].length();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (kern.getMemoryMaps().isSet())
|
|
|
|
{
|
|
|
|
fnd::List<nx::MemoryMappingHandler::sMemoryMapping> maps = kern.getMemoryMaps().getMemoryMaps();
|
|
|
|
fnd::List<nx::MemoryMappingHandler::sMemoryMapping> ioMaps = kern.getMemoryMaps().getIoMemoryMaps();
|
|
|
|
|
|
|
|
printf(" MemoryMaps:\n");
|
|
|
|
for (size_t i = 0; i < maps.getSize(); i++)
|
|
|
|
{
|
2018-03-22 05:26:22 +00:00
|
|
|
printf(" 0x%016" PRIx64 " - 0x%016" PRIx64 " (perm=%s) (type=%s)\n", (uint64_t)maps[i].addr << 12, ((uint64_t)(maps[i].addr + maps[i].size) << 12) - 1, kMemMapPerm[maps[i].perm].c_str(), kMemMapType[maps[i].type].c_str());
|
2017-07-17 08:26:19 +00:00
|
|
|
}
|
|
|
|
//printf(" IoMaps:\n");
|
|
|
|
for (size_t i = 0; i < ioMaps.getSize(); i++)
|
|
|
|
{
|
2018-03-22 05:26:22 +00:00
|
|
|
printf(" 0x%016" PRIx64 " - 0x%016" PRIx64 " (perm=%s) (type=%s)\n", (uint64_t)ioMaps[i].addr << 12, ((uint64_t)(ioMaps[i].addr + ioMaps[i].size) << 12) - 1, kMemMapPerm[ioMaps[i].perm].c_str(), kMemMapType[ioMaps[i].type].c_str());
|
2017-07-17 08:26:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (kern.getInterupts().isSet())
|
|
|
|
{
|
2018-03-22 05:26:22 +00:00
|
|
|
fnd::List<uint16_t> interupts = kern.getInterupts().getInteruptList();
|
2017-07-17 08:26:19 +00:00
|
|
|
printf(" Interupts Flags:\n");
|
2018-03-22 05:26:22 +00:00
|
|
|
for (uint32_t i = 0; i < interupts.getSize(); i++)
|
2017-07-17 08:26:19 +00:00
|
|
|
{
|
|
|
|
if (i % 10 == 0)
|
|
|
|
{
|
|
|
|
printf("%s ", i != 0 ? "\n" : "");
|
|
|
|
}
|
|
|
|
printf("0x%x%s", interupts[i], interupts[i] != interupts.atBack() ? ", " : "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (kern.getMiscParams().isSet())
|
|
|
|
{
|
|
|
|
printf(" ProgramType: %d\n", kern.getMiscParams().getProgramType());
|
|
|
|
}
|
|
|
|
if (kern.getKernelVersion().isSet())
|
|
|
|
{
|
|
|
|
printf(" Kernel Version: %d.%d\n", kern.getKernelVersion().getVerMajor(), kern.getKernelVersion().getVerMinor());
|
|
|
|
}
|
|
|
|
if (kern.getHandleTableSize().isSet())
|
|
|
|
{
|
|
|
|
printf(" Handle Table Size: 0x%x\n", kern.getHandleTableSize().getHandleTableSize());
|
|
|
|
}
|
|
|
|
if (kern.getMiscFlags().isSet())
|
|
|
|
{
|
|
|
|
fnd::List<nx::MiscFlagsHandler::Flags> flagList = kern.getMiscFlags().getFlagList();
|
|
|
|
|
|
|
|
printf(" Misc Flags:\n");
|
2018-03-22 05:26:22 +00:00
|
|
|
for (uint32_t i = 0; i < flagList.getSize(); i++)
|
2017-07-17 08:26:19 +00:00
|
|
|
{
|
|
|
|
if (i % 10 == 0)
|
|
|
|
{
|
|
|
|
printf("%s ", i != 0 ? "\n" : "");
|
|
|
|
}
|
|
|
|
printf("%s%s", kMiscFlag[flagList[i]].c_str(), flagList[i] != flagList.atBack() ? ", " : "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int main(int argc, char** argv)
|
|
|
|
{
|
|
|
|
if (argc < 2)
|
|
|
|
{
|
|
|
|
printf("usage: npdmtool <file>\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
fnd::MemoryBlob file;
|
|
|
|
fnd::io::readFile(argv[1], file);
|
|
|
|
|
|
|
|
// import
|
|
|
|
nx::NpdmBinary npdm;
|
|
|
|
npdm.importBinary(file.getBytes(), file.getSize());
|
|
|
|
|
|
|
|
// npdm binary
|
|
|
|
displayNpdmHeader(npdm);
|
|
|
|
|
|
|
|
// aci binary
|
|
|
|
displayAciHdr(npdm.getAci());
|
|
|
|
displayFac(npdm.getAci().getFac());
|
|
|
|
displaySac(npdm.getAci().getSac());
|
|
|
|
displayKernelCap(npdm.getAci().getKc());
|
|
|
|
|
|
|
|
// acid binary
|
|
|
|
displayAciHdr(npdm.getAcid());
|
|
|
|
displayFac(npdm.getAcid().getFac());
|
|
|
|
displaySac(npdm.getAcid().getSac());
|
|
|
|
displayKernelCap(npdm.getAcid().getKc());
|
|
|
|
|
|
|
|
} catch (const fnd::Exception& e)
|
|
|
|
{
|
|
|
|
printf("%s\n", e.what());
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|