/* * Copyright (c) Meta Platforms, Inc. and affiliates. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #pragma once #include #include #include #include "oi/OICache.h" #include "oi/OICodeGen.h" #include "oi/OICompiler.h" #include "oi/OIParser.h" #include "oi/SymbolService.h" #include "oi/TrapInfo.h" #include "oi/TreeBuilder.h" #include "oi/X86InstDefs.h" namespace oi::detail { class OIDebugger { OIDebugger(const OICodeGen::Config&, OICompiler::Config, TreeBuilder::Config); public: OIDebugger(pid_t, const OICodeGen::Config&, OICompiler::Config, TreeBuilder::Config); OIDebugger(std::filesystem::path, const OICodeGen::Config&, OICompiler::Config, TreeBuilder::Config); bool segmentInit(void); bool stopTarget(void); bool interruptTarget(void); bool compileCode(); bool processTargetData(); bool executeCode(pid_t); void setDataSegmentSize(size_t); void restoreState(void); bool segConfigExists(void) const { return segConfig.existingConfig; }; enum oidMode { OID_MODE_THREAD, OID_MODE_FUNC }; void setMode(oidMode newMode) { mode = newMode; }; bool targetAttach(void); enum processTrapRet { OID_ERR, OID_CONT, OID_DONE }; OIDebugger::processTrapRet processTrap(pid_t, bool = true, bool = true); bool contTargetThread(bool detach = true) const; bool isGlobalDataProbeEnabled(void) const; static uint64_t singlestepInst(pid_t, struct user_regs_struct&); static bool singleStepFunc(pid_t, uint64_t); bool parseScript(std::istream& script); bool patchFunctions(); void stopAll(); bool removeTraps(pid_t); bool removeTrap(pid_t, const trapInfo&); void enableDrgn(); bool unmapSegments(bool deleteSegConf = false); bool isInterrupted(void) const { return oidShouldExit; }; void setCacheBasePath(std::filesystem::path basePath) { if (std::filesystem::exists(basePath.parent_path()) && !std::filesystem::exists(basePath)) { // Create cachedir if parent directory exists // TODO if returning false here, throw an error std::filesystem::create_directory(basePath); } cache.basePath = std::move(basePath); } void setCacheRemoteEnabled(bool upload, bool download) { cache.enableUpload = upload; cache.enableDownload = download; cache.abortOnLoadFail = download && !upload; } bool validateCache() { if ((cache.enableUpload || cache.enableDownload) && !cache.isEnabled()) { LOG(ERROR) << "Cache download/upload option specified when cache is " "disabled - aborting!"; return false; } return true; } void setHardDisableDrgn(bool val) { symbols->setHardDisableDrgn(val); } void setStrict(bool val) { treeBuilderConfig.strict = val; } bool uploadCache() { return std::all_of( std::begin(pdata), std::end(pdata), [this](const auto& req) { return std::all_of( std::begin(req.args), std::end(req.args), [this, &req](const auto& arg) { return cache.upload(irequest{req.type, req.func, arg}); }); }); } bool downloadCache() { return std::all_of( std::begin(pdata), std::end(pdata), [this](const auto& req) { return std::all_of( std::begin(req.args), std::end(req.args), [this, &req](const auto& arg) { return cache.download(irequest{req.type, req.func, arg}); }); }); }; std::pair getTreeBuilderTyping() { assert(pdata.numReqs() == 1); auto [type, th, _] = typeInfos.at(pdata.getReq().getReqForArg()); return {type, th}; }; std::map getPaddingInfo() { assert(pdata.numReqs() == 1); return std::get<2>(typeInfos.at(pdata.getReq().getReqForArg())); } void setCustomCodeFile(std::filesystem::path newCCT) { customCodeFile = std::move(newCCT); } private: bool debug = false; pid_t traceePid{}; uint64_t objectAddr{}; oidMode mode{OID_MODE_THREAD}; enum class SegType { text, data }; enum class StatusType { sleep, traced, running, zombie, dead, diskSleep, stopped, other, bad }; static OIDebugger::StatusType getTaskState(pid_t pid); static std::string taskStateToString(OIDebugger::StatusType); size_t dataSegSize{1 << 20}; size_t textSegSize{(1 << 22) + (1 << 20)}; std::vector threadList; ParseData pdata{}; uint64_t replayInstsCurIdx{}; bool oidShouldExit{false}; uint64_t count{}; bool sigIntHandlerActive{false}; const int sizeofInt3 = 1; const int sizeofUd2 = 2; const int replayInstSize = 512; bool trapsRemoved{false}; std::shared_ptr symbols; OICache cache; /* * Map address of valid INT3 instruction to metadata for that interrupt. * It MUST be an ordered map (std::map) to handle overlapping traps. */ std::map> activeTraps; std::unordered_map> threadTrapState; std::unordered_map replayInstMap; std::unordered_map< irequest, std::tuple>> typeInfos; template std::optional remoteSyscall(Args...); bool setupLogFile(void); bool cleanupLogFile(void); using ObjectAddrMap = std::unordered_map, std::shared_ptr>, uintptr_t>; ObjectAddrMap remoteObjAddrs{}; bool setupSegment(SegType); bool unmapSegment(SegType); bool writeTargetMemory(void*, void*, size_t) const; bool readTargetMemory(void*, void*, size_t) const; std::optional> locateJitCodeStart(const irequest&, const OICompiler::RelocResult::SymTable&); bool writePrologue(const prequest&, const OICompiler::RelocResult::SymTable&); bool readInstFromTarget(uintptr_t, uint8_t*, size_t); void createSegmentConfigFile(void); void deleteSegmentConfig(bool); std::optional> makeTrapInfo(const prequest&, const trapType, const uint64_t); bool functionPatch(const prequest&); bool canProcessTrapForThread(pid_t) const; bool replayTrappedInstr(const trapInfo&, pid_t, struct user_regs_struct&, struct user_fpregs_struct&) const; bool locateObjectsAddresses(const trapInfo&, struct user_regs_struct&); processTrapRet processFuncTrap(const trapInfo&, pid_t, struct user_regs_struct&, struct user_fpregs_struct&); processTrapRet processJitCodeRet(const trapInfo&, pid_t); bool processGlobal(const std::string&); static void dumpRegs(const char*, pid_t, struct user_regs_struct*); std::optional nextReplayInstrAddr(const trapInfo&); static int getExtendedWaitEventType(int); static bool isExtendedWait(int); void dumpAlltaskStates(void); std::optional> findRetLocs(FuncDesc&); bool contTargetThread(pid_t, unsigned long = 0) const; OICompiler::Config compilerConfig{}; const OICodeGen::Config& generatorConfig; TreeBuilder::Config treeBuilderConfig{}; std::optional generateCode(const irequest&); std::fstream segmentConfigFile; std::filesystem::path segConfigFilePath; std::filesystem::path customCodeFile; struct c { uintptr_t textSegBase{}; size_t textSegSize{}; uintptr_t constStart{}; uintptr_t jitCodeStart{}; uintptr_t replayInstBase{}; bool existingConfig{false}; uintptr_t dataSegBase{}; size_t dataSegSize{}; uintptr_t cookie{}; int logFile{}; } segConfig{}; /* * The first 3 words of the data segment contain: * 1. The OID identifier a.k.a. "magic id", 01DE8 in hex * 2. A random value (cookie) to make sure that the data * segment we are reading from was not populated in * an older run. * 3. The size of the data segment as written by the JIT-ed * code. */ struct DataHeader { uintptr_t magicId; uintptr_t cookie; uintptr_t size; uintptr_t timeTakenNs; /* * Flexible Array Member are not standard in C++, but this is * exactly what we need for the `data` field. These pragmas * disable the pedantic warnings, so the compiler stops yelling at us. * We want the header to be the size of the fields above. This is * important for the `decodeTargetData` method, to give the right size * to `folly::ByteRange range(…)`. */ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wpedantic" uint8_t data[]; #pragma GCC diagnostic pop }; bool decodeTargetData(const DataHeader&, std::vector&) const; static constexpr size_t prologueLength = 64; static constexpr size_t constLength = 64; }; } // namespace oi::detail