Path: blob/main/contrib/llvm-project/llvm/lib/ExecutionEngine/RuntimeDyld/RuntimeDyldChecker.cpp
35266 views
//===--- RuntimeDyldChecker.cpp - RuntimeDyld tester framework --*- C++ -*-===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//78#include "llvm/ExecutionEngine/RuntimeDyldChecker.h"9#include "RuntimeDyldCheckerImpl.h"10#include "llvm/ADT/STLExtras.h"11#include "llvm/ADT/StringExtras.h"12#include "llvm/MC/MCAsmInfo.h"13#include "llvm/MC/MCContext.h"14#include "llvm/MC/MCDisassembler/MCDisassembler.h"15#include "llvm/MC/MCInst.h"16#include "llvm/MC/MCInstPrinter.h"17#include "llvm/MC/MCInstrInfo.h"18#include "llvm/MC/MCRegisterInfo.h"19#include "llvm/MC/MCSubtargetInfo.h"20#include "llvm/MC/MCTargetOptions.h"21#include "llvm/MC/TargetRegistry.h"22#include "llvm/Support/Endian.h"23#include "llvm/Support/MSVCErrorWorkarounds.h"24#include "llvm/Support/MemoryBuffer.h"25#include "llvm/Support/Path.h"26#include <cctype>27#include <memory>28#include <utility>2930#define DEBUG_TYPE "rtdyld"3132using namespace llvm;3334namespace {35struct TargetInfo {36const Target *TheTarget;37std::unique_ptr<MCSubtargetInfo> STI;38std::unique_ptr<MCRegisterInfo> MRI;39std::unique_ptr<MCAsmInfo> MAI;40std::unique_ptr<MCContext> Ctx;41std::unique_ptr<MCDisassembler> Disassembler;42std::unique_ptr<MCInstrInfo> MII;43std::unique_ptr<MCInstPrinter> InstPrinter;44};45} // anonymous namespace4647namespace llvm {4849// Helper class that implements the language evaluated by RuntimeDyldChecker.50class RuntimeDyldCheckerExprEval {51public:52RuntimeDyldCheckerExprEval(const RuntimeDyldCheckerImpl &Checker,53raw_ostream &ErrStream)54: Checker(Checker) {}5556bool evaluate(StringRef Expr) const {57// Expect equality expression of the form 'LHS = RHS'.58Expr = Expr.trim();59size_t EQIdx = Expr.find('=');6061ParseContext OutsideLoad(false);6263// Evaluate LHS.64StringRef LHSExpr = Expr.substr(0, EQIdx).rtrim();65StringRef RemainingExpr;66EvalResult LHSResult;67std::tie(LHSResult, RemainingExpr) =68evalComplexExpr(evalSimpleExpr(LHSExpr, OutsideLoad), OutsideLoad);69if (LHSResult.hasError())70return handleError(Expr, LHSResult);71if (RemainingExpr != "")72return handleError(Expr, unexpectedToken(RemainingExpr, LHSExpr, ""));7374// Evaluate RHS.75StringRef RHSExpr = Expr.substr(EQIdx + 1).ltrim();76EvalResult RHSResult;77std::tie(RHSResult, RemainingExpr) =78evalComplexExpr(evalSimpleExpr(RHSExpr, OutsideLoad), OutsideLoad);79if (RHSResult.hasError())80return handleError(Expr, RHSResult);81if (RemainingExpr != "")82return handleError(Expr, unexpectedToken(RemainingExpr, RHSExpr, ""));8384if (LHSResult.getValue() != RHSResult.getValue()) {85Checker.ErrStream << "Expression '" << Expr << "' is false: "86<< format("0x%" PRIx64, LHSResult.getValue())87<< " != " << format("0x%" PRIx64, RHSResult.getValue())88<< "\n";89return false;90}91return true;92}9394private:95// RuntimeDyldCheckerExprEval requires some context when parsing exprs. In96// particular, it needs to know whether a symbol is being evaluated in the97// context of a load, in which case we want the linker's local address for98// the symbol, or outside of a load, in which case we want the symbol's99// address in the remote target.100101struct ParseContext {102bool IsInsideLoad;103ParseContext(bool IsInsideLoad) : IsInsideLoad(IsInsideLoad) {}104};105106const RuntimeDyldCheckerImpl &Checker;107108enum class BinOpToken : unsigned {109Invalid,110Add,111Sub,112BitwiseAnd,113BitwiseOr,114ShiftLeft,115ShiftRight116};117118class EvalResult {119public:120EvalResult() : Value(0) {}121EvalResult(uint64_t Value) : Value(Value) {}122EvalResult(std::string ErrorMsg)123: Value(0), ErrorMsg(std::move(ErrorMsg)) {}124uint64_t getValue() const { return Value; }125bool hasError() const { return ErrorMsg != ""; }126const std::string &getErrorMsg() const { return ErrorMsg; }127128private:129uint64_t Value;130std::string ErrorMsg;131};132133StringRef getTokenForError(StringRef Expr) const {134if (Expr.empty())135return "";136137StringRef Token, Remaining;138if (isalpha(Expr[0]))139std::tie(Token, Remaining) = parseSymbol(Expr);140else if (isdigit(Expr[0]))141std::tie(Token, Remaining) = parseNumberString(Expr);142else {143unsigned TokLen = 1;144if (Expr.starts_with("<<") || Expr.starts_with(">>"))145TokLen = 2;146Token = Expr.substr(0, TokLen);147}148return Token;149}150151EvalResult unexpectedToken(StringRef TokenStart, StringRef SubExpr,152StringRef ErrText) const {153std::string ErrorMsg("Encountered unexpected token '");154ErrorMsg += getTokenForError(TokenStart);155if (SubExpr != "") {156ErrorMsg += "' while parsing subexpression '";157ErrorMsg += SubExpr;158}159ErrorMsg += "'";160if (ErrText != "") {161ErrorMsg += " ";162ErrorMsg += ErrText;163}164return EvalResult(std::move(ErrorMsg));165}166167bool handleError(StringRef Expr, const EvalResult &R) const {168assert(R.hasError() && "Not an error result.");169Checker.ErrStream << "Error evaluating expression '" << Expr170<< "': " << R.getErrorMsg() << "\n";171return false;172}173174std::pair<BinOpToken, StringRef> parseBinOpToken(StringRef Expr) const {175if (Expr.empty())176return std::make_pair(BinOpToken::Invalid, "");177178// Handle the two 2-character tokens.179if (Expr.starts_with("<<"))180return std::make_pair(BinOpToken::ShiftLeft, Expr.substr(2).ltrim());181if (Expr.starts_with(">>"))182return std::make_pair(BinOpToken::ShiftRight, Expr.substr(2).ltrim());183184// Handle one-character tokens.185BinOpToken Op;186switch (Expr[0]) {187default:188return std::make_pair(BinOpToken::Invalid, Expr);189case '+':190Op = BinOpToken::Add;191break;192case '-':193Op = BinOpToken::Sub;194break;195case '&':196Op = BinOpToken::BitwiseAnd;197break;198case '|':199Op = BinOpToken::BitwiseOr;200break;201}202203return std::make_pair(Op, Expr.substr(1).ltrim());204}205206EvalResult computeBinOpResult(BinOpToken Op, const EvalResult &LHSResult,207const EvalResult &RHSResult) const {208switch (Op) {209default:210llvm_unreachable("Tried to evaluate unrecognized operation.");211case BinOpToken::Add:212return EvalResult(LHSResult.getValue() + RHSResult.getValue());213case BinOpToken::Sub:214return EvalResult(LHSResult.getValue() - RHSResult.getValue());215case BinOpToken::BitwiseAnd:216return EvalResult(LHSResult.getValue() & RHSResult.getValue());217case BinOpToken::BitwiseOr:218return EvalResult(LHSResult.getValue() | RHSResult.getValue());219case BinOpToken::ShiftLeft:220return EvalResult(LHSResult.getValue() << RHSResult.getValue());221case BinOpToken::ShiftRight:222return EvalResult(LHSResult.getValue() >> RHSResult.getValue());223}224}225226// Parse a symbol and return a (string, string) pair representing the symbol227// name and expression remaining to be parsed.228std::pair<StringRef, StringRef> parseSymbol(StringRef Expr) const {229size_t FirstNonSymbol = Expr.find_first_not_of("0123456789"230"abcdefghijklmnopqrstuvwxyz"231"ABCDEFGHIJKLMNOPQRSTUVWXYZ"232":_.$");233return std::make_pair(Expr.substr(0, FirstNonSymbol),234Expr.substr(FirstNonSymbol).ltrim());235}236237// Evaluate a call to decode_operand. Decode the instruction operand at the238// given symbol and get the value of the requested operand.239// Returns an error if the instruction cannot be decoded, or the requested240// operand is not an immediate.241// On success, returns a pair containing the value of the operand, plus242// the expression remaining to be evaluated.243std::pair<EvalResult, StringRef> evalDecodeOperand(StringRef Expr) const {244if (!Expr.starts_with("("))245return std::make_pair(unexpectedToken(Expr, Expr, "expected '('"), "");246StringRef RemainingExpr = Expr.substr(1).ltrim();247StringRef Symbol;248std::tie(Symbol, RemainingExpr) = parseSymbol(RemainingExpr);249250if (!Checker.isSymbolValid(Symbol))251return std::make_pair(252EvalResult(("Cannot decode unknown symbol '" + Symbol + "'").str()),253"");254255// if there is an offset number expr256int64_t Offset = 0;257BinOpToken BinOp;258std::tie(BinOp, RemainingExpr) = parseBinOpToken(RemainingExpr);259switch (BinOp) {260case BinOpToken::Add: {261EvalResult Number;262std::tie(Number, RemainingExpr) = evalNumberExpr(RemainingExpr);263Offset = Number.getValue();264break;265}266case BinOpToken::Invalid:267break;268default:269return std::make_pair(270unexpectedToken(RemainingExpr, RemainingExpr,271"expected '+' for offset or ',' if no offset"),272"");273}274275if (!RemainingExpr.starts_with(","))276return std::make_pair(277unexpectedToken(RemainingExpr, RemainingExpr, "expected ','"), "");278RemainingExpr = RemainingExpr.substr(1).ltrim();279280EvalResult OpIdxExpr;281std::tie(OpIdxExpr, RemainingExpr) = evalNumberExpr(RemainingExpr);282if (OpIdxExpr.hasError())283return std::make_pair(OpIdxExpr, "");284285if (!RemainingExpr.starts_with(")"))286return std::make_pair(287unexpectedToken(RemainingExpr, RemainingExpr, "expected ')'"), "");288RemainingExpr = RemainingExpr.substr(1).ltrim();289290MCInst Inst;291uint64_t Size;292if (!decodeInst(Symbol, Inst, Size, Offset))293return std::make_pair(294EvalResult(("Couldn't decode instruction at '" + Symbol + "'").str()),295"");296297unsigned OpIdx = OpIdxExpr.getValue();298299auto printInst = [this](StringRef Symbol, MCInst Inst,300raw_string_ostream &ErrMsgStream) {301auto TT = Checker.getTripleForSymbol(Checker.getTargetFlag(Symbol));302auto TI = getTargetInfo(TT, Checker.getCPU(), Checker.getFeatures());303if (auto E = TI.takeError()) {304errs() << "Error obtaining instruction printer: "305<< toString(std::move(E)) << "\n";306return std::make_pair(EvalResult(ErrMsgStream.str()), "");307}308Inst.dump_pretty(ErrMsgStream, TI->InstPrinter.get());309return std::make_pair(EvalResult(ErrMsgStream.str()), "");310};311312if (OpIdx >= Inst.getNumOperands()) {313std::string ErrMsg;314raw_string_ostream ErrMsgStream(ErrMsg);315ErrMsgStream << "Invalid operand index '" << format("%i", OpIdx)316<< "' for instruction '" << Symbol317<< "'. Instruction has only "318<< format("%i", Inst.getNumOperands())319<< " operands.\nInstruction is:\n ";320321return printInst(Symbol, Inst, ErrMsgStream);322}323324const MCOperand &Op = Inst.getOperand(OpIdx);325if (!Op.isImm()) {326std::string ErrMsg;327raw_string_ostream ErrMsgStream(ErrMsg);328ErrMsgStream << "Operand '" << format("%i", OpIdx) << "' of instruction '"329<< Symbol << "' is not an immediate.\nInstruction is:\n ";330331return printInst(Symbol, Inst, ErrMsgStream);332}333334return std::make_pair(EvalResult(Op.getImm()), RemainingExpr);335}336337// Evaluate a call to next_pc.338// Decode the instruction at the given symbol and return the following program339// counter.340// Returns an error if the instruction cannot be decoded.341// On success, returns a pair containing the next PC, plus of the342// expression remaining to be evaluated.343std::pair<EvalResult, StringRef> evalNextPC(StringRef Expr,344ParseContext PCtx) const {345if (!Expr.starts_with("("))346return std::make_pair(unexpectedToken(Expr, Expr, "expected '('"), "");347StringRef RemainingExpr = Expr.substr(1).ltrim();348StringRef Symbol;349std::tie(Symbol, RemainingExpr) = parseSymbol(RemainingExpr);350351if (!Checker.isSymbolValid(Symbol))352return std::make_pair(353EvalResult(("Cannot decode unknown symbol '" + Symbol + "'").str()),354"");355356if (!RemainingExpr.starts_with(")"))357return std::make_pair(358unexpectedToken(RemainingExpr, RemainingExpr, "expected ')'"), "");359RemainingExpr = RemainingExpr.substr(1).ltrim();360361MCInst Inst;362uint64_t InstSize;363if (!decodeInst(Symbol, Inst, InstSize, 0))364return std::make_pair(365EvalResult(("Couldn't decode instruction at '" + Symbol + "'").str()),366"");367368uint64_t SymbolAddr = PCtx.IsInsideLoad369? Checker.getSymbolLocalAddr(Symbol)370: Checker.getSymbolRemoteAddr(Symbol);371372// ARM PC offset is 8 instead of 4, because it accounts for an additional373// prefetch instruction that increments PC even though it is implicit.374auto TT = Checker.getTripleForSymbol(Checker.getTargetFlag(Symbol));375uint64_t PCOffset = TT.getArch() == Triple::ArchType::arm ? 4 : 0;376377uint64_t NextPC = SymbolAddr + InstSize + PCOffset;378379return std::make_pair(EvalResult(NextPC), RemainingExpr);380}381382// Evaluate a call to stub_addr/got_addr.383// Look up and return the address of the stub for the given384// (<file name>, <section name>, <symbol name>) tuple.385// On success, returns a pair containing the stub address, plus the expression386// remaining to be evaluated.387std::pair<EvalResult, StringRef>388evalStubOrGOTAddr(StringRef Expr, ParseContext PCtx, bool IsStubAddr) const {389if (!Expr.starts_with("("))390return std::make_pair(unexpectedToken(Expr, Expr, "expected '('"), "");391StringRef RemainingExpr = Expr.substr(1).ltrim();392393// Handle file-name specially, as it may contain characters that aren't394// legal for symbols.395StringRef StubContainerName;396size_t ComaIdx = RemainingExpr.find(',');397StubContainerName = RemainingExpr.substr(0, ComaIdx).rtrim();398RemainingExpr = RemainingExpr.substr(ComaIdx).ltrim();399400if (!RemainingExpr.starts_with(","))401return std::make_pair(402unexpectedToken(RemainingExpr, Expr, "expected ','"), "");403RemainingExpr = RemainingExpr.substr(1).ltrim();404405StringRef Symbol;406std::tie(Symbol, RemainingExpr) = parseSymbol(RemainingExpr);407408// Parse optional parameter to filter by stub kind409StringRef KindNameFilter;410if (RemainingExpr.starts_with(",")) {411RemainingExpr = RemainingExpr.substr(1).ltrim();412size_t ClosingBracket = RemainingExpr.find(")");413KindNameFilter = RemainingExpr.substr(0, ClosingBracket);414RemainingExpr = RemainingExpr.substr(ClosingBracket);415}416417if (!RemainingExpr.starts_with(")"))418return std::make_pair(419unexpectedToken(RemainingExpr, Expr, "expected ')'"), "");420RemainingExpr = RemainingExpr.substr(1).ltrim();421422uint64_t StubAddr;423std::string ErrorMsg;424std::tie(StubAddr, ErrorMsg) =425Checker.getStubOrGOTAddrFor(StubContainerName, Symbol, KindNameFilter,426PCtx.IsInsideLoad, IsStubAddr);427428if (ErrorMsg != "")429return std::make_pair(EvalResult(ErrorMsg), "");430431return std::make_pair(EvalResult(StubAddr), RemainingExpr);432}433434std::pair<EvalResult, StringRef> evalSectionAddr(StringRef Expr,435ParseContext PCtx) const {436if (!Expr.starts_with("("))437return std::make_pair(unexpectedToken(Expr, Expr, "expected '('"), "");438StringRef RemainingExpr = Expr.substr(1).ltrim();439440// Handle file-name specially, as it may contain characters that aren't441// legal for symbols.442StringRef FileName;443size_t ComaIdx = RemainingExpr.find(',');444FileName = RemainingExpr.substr(0, ComaIdx).rtrim();445RemainingExpr = RemainingExpr.substr(ComaIdx).ltrim();446447if (!RemainingExpr.starts_with(","))448return std::make_pair(449unexpectedToken(RemainingExpr, Expr, "expected ','"), "");450RemainingExpr = RemainingExpr.substr(1).ltrim();451452StringRef SectionName;453size_t CloseParensIdx = RemainingExpr.find(')');454SectionName = RemainingExpr.substr(0, CloseParensIdx).rtrim();455RemainingExpr = RemainingExpr.substr(CloseParensIdx).ltrim();456457if (!RemainingExpr.starts_with(")"))458return std::make_pair(459unexpectedToken(RemainingExpr, Expr, "expected ')'"), "");460RemainingExpr = RemainingExpr.substr(1).ltrim();461462uint64_t StubAddr;463std::string ErrorMsg;464std::tie(StubAddr, ErrorMsg) = Checker.getSectionAddr(465FileName, SectionName, PCtx.IsInsideLoad);466467if (ErrorMsg != "")468return std::make_pair(EvalResult(ErrorMsg), "");469470return std::make_pair(EvalResult(StubAddr), RemainingExpr);471}472473// Evaluate an identifier expr, which may be a symbol, or a call to474// one of the builtin functions: get_insn_opcode or get_insn_length.475// Return the result, plus the expression remaining to be parsed.476std::pair<EvalResult, StringRef> evalIdentifierExpr(StringRef Expr,477ParseContext PCtx) const {478StringRef Symbol;479StringRef RemainingExpr;480std::tie(Symbol, RemainingExpr) = parseSymbol(Expr);481482// Check for builtin function calls.483if (Symbol == "decode_operand")484return evalDecodeOperand(RemainingExpr);485else if (Symbol == "next_pc")486return evalNextPC(RemainingExpr, PCtx);487else if (Symbol == "stub_addr")488return evalStubOrGOTAddr(RemainingExpr, PCtx, true);489else if (Symbol == "got_addr")490return evalStubOrGOTAddr(RemainingExpr, PCtx, false);491else if (Symbol == "section_addr")492return evalSectionAddr(RemainingExpr, PCtx);493494if (!Checker.isSymbolValid(Symbol)) {495std::string ErrMsg("No known address for symbol '");496ErrMsg += Symbol;497ErrMsg += "'";498if (Symbol.starts_with("L"))499ErrMsg += " (this appears to be an assembler local label - "500" perhaps drop the 'L'?)";501502return std::make_pair(EvalResult(ErrMsg), "");503}504505// The value for the symbol depends on the context we're evaluating in:506// Inside a load this is the address in the linker's memory, outside a507// load it's the address in the target processes memory.508uint64_t Value = PCtx.IsInsideLoad ? Checker.getSymbolLocalAddr(Symbol)509: Checker.getSymbolRemoteAddr(Symbol);510511// Looks like a plain symbol reference.512return std::make_pair(EvalResult(Value), RemainingExpr);513}514515// Parse a number (hexadecimal or decimal) and return a (string, string)516// pair representing the number and the expression remaining to be parsed.517std::pair<StringRef, StringRef> parseNumberString(StringRef Expr) const {518size_t FirstNonDigit = StringRef::npos;519if (Expr.starts_with("0x")) {520FirstNonDigit = Expr.find_first_not_of("0123456789abcdefABCDEF", 2);521if (FirstNonDigit == StringRef::npos)522FirstNonDigit = Expr.size();523} else {524FirstNonDigit = Expr.find_first_not_of("0123456789");525if (FirstNonDigit == StringRef::npos)526FirstNonDigit = Expr.size();527}528return std::make_pair(Expr.substr(0, FirstNonDigit),529Expr.substr(FirstNonDigit));530}531532// Evaluate a constant numeric expression (hexadecimal or decimal) and533// return a pair containing the result, and the expression remaining to be534// evaluated.535std::pair<EvalResult, StringRef> evalNumberExpr(StringRef Expr) const {536StringRef ValueStr;537StringRef RemainingExpr;538std::tie(ValueStr, RemainingExpr) = parseNumberString(Expr);539540if (ValueStr.empty() || !isdigit(ValueStr[0]))541return std::make_pair(542unexpectedToken(RemainingExpr, RemainingExpr, "expected number"), "");543uint64_t Value;544ValueStr.getAsInteger(0, Value);545return std::make_pair(EvalResult(Value), RemainingExpr);546}547548// Evaluate an expression of the form "(<expr>)" and return a pair549// containing the result of evaluating <expr>, plus the expression550// remaining to be parsed.551std::pair<EvalResult, StringRef> evalParensExpr(StringRef Expr,552ParseContext PCtx) const {553assert(Expr.starts_with("(") && "Not a parenthesized expression");554EvalResult SubExprResult;555StringRef RemainingExpr;556std::tie(SubExprResult, RemainingExpr) =557evalComplexExpr(evalSimpleExpr(Expr.substr(1).ltrim(), PCtx), PCtx);558if (SubExprResult.hasError())559return std::make_pair(SubExprResult, "");560if (!RemainingExpr.starts_with(")"))561return std::make_pair(562unexpectedToken(RemainingExpr, Expr, "expected ')'"), "");563RemainingExpr = RemainingExpr.substr(1).ltrim();564return std::make_pair(SubExprResult, RemainingExpr);565}566567// Evaluate an expression in one of the following forms:568// *{<number>}<expr>569// Return a pair containing the result, plus the expression remaining to be570// parsed.571std::pair<EvalResult, StringRef> evalLoadExpr(StringRef Expr) const {572assert(Expr.starts_with("*") && "Not a load expression");573StringRef RemainingExpr = Expr.substr(1).ltrim();574575// Parse read size.576if (!RemainingExpr.starts_with("{"))577return std::make_pair(EvalResult("Expected '{' following '*'."), "");578RemainingExpr = RemainingExpr.substr(1).ltrim();579EvalResult ReadSizeExpr;580std::tie(ReadSizeExpr, RemainingExpr) = evalNumberExpr(RemainingExpr);581if (ReadSizeExpr.hasError())582return std::make_pair(ReadSizeExpr, RemainingExpr);583uint64_t ReadSize = ReadSizeExpr.getValue();584if (ReadSize < 1 || ReadSize > 8)585return std::make_pair(EvalResult("Invalid size for dereference."), "");586if (!RemainingExpr.starts_with("}"))587return std::make_pair(EvalResult("Missing '}' for dereference."), "");588RemainingExpr = RemainingExpr.substr(1).ltrim();589590// Evaluate the expression representing the load address.591ParseContext LoadCtx(true);592EvalResult LoadAddrExprResult;593std::tie(LoadAddrExprResult, RemainingExpr) =594evalComplexExpr(evalSimpleExpr(RemainingExpr, LoadCtx), LoadCtx);595596if (LoadAddrExprResult.hasError())597return std::make_pair(LoadAddrExprResult, "");598599uint64_t LoadAddr = LoadAddrExprResult.getValue();600601// If there is no error but the content pointer is null then this is a602// zero-fill symbol/section.603if (LoadAddr == 0)604return std::make_pair(0, RemainingExpr);605606return std::make_pair(607EvalResult(Checker.readMemoryAtAddr(LoadAddr, ReadSize)),608RemainingExpr);609}610611// Evaluate a "simple" expression. This is any expression that _isn't_ an612// un-parenthesized binary expression.613//614// "Simple" expressions can be optionally bit-sliced. See evalSlicedExpr.615//616// Returns a pair containing the result of the evaluation, plus the617// expression remaining to be parsed.618std::pair<EvalResult, StringRef> evalSimpleExpr(StringRef Expr,619ParseContext PCtx) const {620EvalResult SubExprResult;621StringRef RemainingExpr;622623if (Expr.empty())624return std::make_pair(EvalResult("Unexpected end of expression"), "");625626if (Expr[0] == '(')627std::tie(SubExprResult, RemainingExpr) = evalParensExpr(Expr, PCtx);628else if (Expr[0] == '*')629std::tie(SubExprResult, RemainingExpr) = evalLoadExpr(Expr);630else if (isalpha(Expr[0]) || Expr[0] == '_')631std::tie(SubExprResult, RemainingExpr) = evalIdentifierExpr(Expr, PCtx);632else if (isdigit(Expr[0]))633std::tie(SubExprResult, RemainingExpr) = evalNumberExpr(Expr);634else635return std::make_pair(636unexpectedToken(Expr, Expr,637"expected '(', '*', identifier, or number"), "");638639if (SubExprResult.hasError())640return std::make_pair(SubExprResult, RemainingExpr);641642// Evaluate bit-slice if present.643if (RemainingExpr.starts_with("["))644std::tie(SubExprResult, RemainingExpr) =645evalSliceExpr(std::make_pair(SubExprResult, RemainingExpr));646647return std::make_pair(SubExprResult, RemainingExpr);648}649650// Evaluate a bit-slice of an expression.651// A bit-slice has the form "<expr>[high:low]". The result of evaluating a652// slice is the bits between high and low (inclusive) in the original653// expression, right shifted so that the "low" bit is in position 0 in the654// result.655// Returns a pair containing the result of the slice operation, plus the656// expression remaining to be parsed.657std::pair<EvalResult, StringRef>658evalSliceExpr(const std::pair<EvalResult, StringRef> &Ctx) const {659EvalResult SubExprResult;660StringRef RemainingExpr;661std::tie(SubExprResult, RemainingExpr) = Ctx;662663assert(RemainingExpr.starts_with("[") && "Not a slice expr.");664RemainingExpr = RemainingExpr.substr(1).ltrim();665666EvalResult HighBitExpr;667std::tie(HighBitExpr, RemainingExpr) = evalNumberExpr(RemainingExpr);668669if (HighBitExpr.hasError())670return std::make_pair(HighBitExpr, RemainingExpr);671672if (!RemainingExpr.starts_with(":"))673return std::make_pair(674unexpectedToken(RemainingExpr, RemainingExpr, "expected ':'"), "");675RemainingExpr = RemainingExpr.substr(1).ltrim();676677EvalResult LowBitExpr;678std::tie(LowBitExpr, RemainingExpr) = evalNumberExpr(RemainingExpr);679680if (LowBitExpr.hasError())681return std::make_pair(LowBitExpr, RemainingExpr);682683if (!RemainingExpr.starts_with("]"))684return std::make_pair(685unexpectedToken(RemainingExpr, RemainingExpr, "expected ']'"), "");686RemainingExpr = RemainingExpr.substr(1).ltrim();687688unsigned HighBit = HighBitExpr.getValue();689unsigned LowBit = LowBitExpr.getValue();690uint64_t Mask = ((uint64_t)1 << (HighBit - LowBit + 1)) - 1;691uint64_t SlicedValue = (SubExprResult.getValue() >> LowBit) & Mask;692return std::make_pair(EvalResult(SlicedValue), RemainingExpr);693}694695// Evaluate a "complex" expression.696// Takes an already evaluated subexpression and checks for the presence of a697// binary operator, computing the result of the binary operation if one is698// found. Used to make arithmetic expressions left-associative.699// Returns a pair containing the ultimate result of evaluating the700// expression, plus the expression remaining to be evaluated.701std::pair<EvalResult, StringRef>702evalComplexExpr(const std::pair<EvalResult, StringRef> &LHSAndRemaining,703ParseContext PCtx) const {704EvalResult LHSResult;705StringRef RemainingExpr;706std::tie(LHSResult, RemainingExpr) = LHSAndRemaining;707708// If there was an error, or there's nothing left to evaluate, return the709// result.710if (LHSResult.hasError() || RemainingExpr == "")711return std::make_pair(LHSResult, RemainingExpr);712713// Otherwise check if this is a binary expression.714BinOpToken BinOp;715std::tie(BinOp, RemainingExpr) = parseBinOpToken(RemainingExpr);716717// If this isn't a recognized expression just return.718if (BinOp == BinOpToken::Invalid)719return std::make_pair(LHSResult, RemainingExpr);720721// This is a recognized bin-op. Evaluate the RHS, then evaluate the binop.722EvalResult RHSResult;723std::tie(RHSResult, RemainingExpr) = evalSimpleExpr(RemainingExpr, PCtx);724725// If there was an error evaluating the RHS, return it.726if (RHSResult.hasError())727return std::make_pair(RHSResult, RemainingExpr);728729// This is a binary expression - evaluate and try to continue as a730// complex expr.731EvalResult ThisResult(computeBinOpResult(BinOp, LHSResult, RHSResult));732733return evalComplexExpr(std::make_pair(ThisResult, RemainingExpr), PCtx);734}735736bool decodeInst(StringRef Symbol, MCInst &Inst, uint64_t &Size,737int64_t Offset) const {738auto TT = Checker.getTripleForSymbol(Checker.getTargetFlag(Symbol));739auto TI = getTargetInfo(TT, Checker.getCPU(), Checker.getFeatures());740741if (auto E = TI.takeError()) {742errs() << "Error obtaining disassembler: " << toString(std::move(E))743<< "\n";744return false;745}746747StringRef SymbolMem = Checker.getSymbolContent(Symbol);748ArrayRef<uint8_t> SymbolBytes(SymbolMem.bytes_begin() + Offset,749SymbolMem.size() - Offset);750751MCDisassembler::DecodeStatus S =752TI->Disassembler->getInstruction(Inst, Size, SymbolBytes, 0, nulls());753754return (S == MCDisassembler::Success);755}756757Expected<TargetInfo> getTargetInfo(const Triple &TT, const StringRef &CPU,758const SubtargetFeatures &TF) const {759760auto TripleName = TT.str();761std::string ErrorStr;762const Target *TheTarget =763TargetRegistry::lookupTarget(TripleName, ErrorStr);764if (!TheTarget)765return make_error<StringError>("Error accessing target '" + TripleName +766"': " + ErrorStr,767inconvertibleErrorCode());768769std::unique_ptr<MCSubtargetInfo> STI(770TheTarget->createMCSubtargetInfo(TripleName, CPU, TF.getString()));771if (!STI)772return make_error<StringError>("Unable to create subtarget for " +773TripleName,774inconvertibleErrorCode());775776std::unique_ptr<MCRegisterInfo> MRI(TheTarget->createMCRegInfo(TripleName));777if (!MRI)778return make_error<StringError>("Unable to create target register info "779"for " +780TripleName,781inconvertibleErrorCode());782783MCTargetOptions MCOptions;784std::unique_ptr<MCAsmInfo> MAI(785TheTarget->createMCAsmInfo(*MRI, TripleName, MCOptions));786if (!MAI)787return make_error<StringError>("Unable to create target asm info " +788TripleName,789inconvertibleErrorCode());790791auto Ctx = std::make_unique<MCContext>(Triple(TripleName), MAI.get(),792MRI.get(), STI.get());793794std::unique_ptr<MCDisassembler> Disassembler(795TheTarget->createMCDisassembler(*STI, *Ctx));796if (!Disassembler)797return make_error<StringError>("Unable to create disassembler for " +798TripleName,799inconvertibleErrorCode());800801std::unique_ptr<MCInstrInfo> MII(TheTarget->createMCInstrInfo());802if (!MII)803return make_error<StringError>("Unable to create instruction info for" +804TripleName,805inconvertibleErrorCode());806807std::unique_ptr<MCInstPrinter> InstPrinter(TheTarget->createMCInstPrinter(808Triple(TripleName), 0, *MAI, *MII, *MRI));809if (!InstPrinter)810return make_error<StringError>(811"Unable to create instruction printer for" + TripleName,812inconvertibleErrorCode());813814return TargetInfo({TheTarget, std::move(STI), std::move(MRI),815std::move(MAI), std::move(Ctx), std::move(Disassembler),816std::move(MII), std::move(InstPrinter)});817}818};819} // namespace llvm820821RuntimeDyldCheckerImpl::RuntimeDyldCheckerImpl(822IsSymbolValidFunction IsSymbolValid, GetSymbolInfoFunction GetSymbolInfo,823GetSectionInfoFunction GetSectionInfo, GetStubInfoFunction GetStubInfo,824GetGOTInfoFunction GetGOTInfo, llvm::endianness Endianness, Triple TT,825StringRef CPU, SubtargetFeatures TF, raw_ostream &ErrStream)826: IsSymbolValid(std::move(IsSymbolValid)),827GetSymbolInfo(std::move(GetSymbolInfo)),828GetSectionInfo(std::move(GetSectionInfo)),829GetStubInfo(std::move(GetStubInfo)), GetGOTInfo(std::move(GetGOTInfo)),830Endianness(Endianness), TT(std::move(TT)), CPU(std::move(CPU)),831TF(std::move(TF)), ErrStream(ErrStream) {}832833bool RuntimeDyldCheckerImpl::check(StringRef CheckExpr) const {834CheckExpr = CheckExpr.trim();835LLVM_DEBUG(dbgs() << "RuntimeDyldChecker: Checking '" << CheckExpr836<< "'...\n");837RuntimeDyldCheckerExprEval P(*this, ErrStream);838bool Result = P.evaluate(CheckExpr);839(void)Result;840LLVM_DEBUG(dbgs() << "RuntimeDyldChecker: '" << CheckExpr << "' "841<< (Result ? "passed" : "FAILED") << ".\n");842return Result;843}844845bool RuntimeDyldCheckerImpl::checkAllRulesInBuffer(StringRef RulePrefix,846MemoryBuffer *MemBuf) const {847bool DidAllTestsPass = true;848unsigned NumRules = 0;849850std::string CheckExpr;851const char *LineStart = MemBuf->getBufferStart();852853// Eat whitespace.854while (LineStart != MemBuf->getBufferEnd() && isSpace(*LineStart))855++LineStart;856857while (LineStart != MemBuf->getBufferEnd() && *LineStart != '\0') {858const char *LineEnd = LineStart;859while (LineEnd != MemBuf->getBufferEnd() && *LineEnd != '\r' &&860*LineEnd != '\n')861++LineEnd;862863StringRef Line(LineStart, LineEnd - LineStart);864if (Line.starts_with(RulePrefix))865CheckExpr += Line.substr(RulePrefix.size()).str();866867// If there's a check expr string...868if (!CheckExpr.empty()) {869// ... and it's complete then run it, otherwise remove the trailer '\'.870if (CheckExpr.back() != '\\') {871DidAllTestsPass &= check(CheckExpr);872CheckExpr.clear();873++NumRules;874} else875CheckExpr.pop_back();876}877878// Eat whitespace.879LineStart = LineEnd;880while (LineStart != MemBuf->getBufferEnd() && isSpace(*LineStart))881++LineStart;882}883return DidAllTestsPass && (NumRules != 0);884}885886bool RuntimeDyldCheckerImpl::isSymbolValid(StringRef Symbol) const {887return IsSymbolValid(Symbol);888}889890uint64_t RuntimeDyldCheckerImpl::getSymbolLocalAddr(StringRef Symbol) const {891auto SymInfo = GetSymbolInfo(Symbol);892if (!SymInfo) {893logAllUnhandledErrors(SymInfo.takeError(), errs(), "RTDyldChecker: ");894return 0;895}896897if (SymInfo->isZeroFill())898return 0;899900return static_cast<uint64_t>(901reinterpret_cast<uintptr_t>(SymInfo->getContent().data()));902}903904uint64_t RuntimeDyldCheckerImpl::getSymbolRemoteAddr(StringRef Symbol) const {905auto SymInfo = GetSymbolInfo(Symbol);906if (!SymInfo) {907logAllUnhandledErrors(SymInfo.takeError(), errs(), "RTDyldChecker: ");908return 0;909}910911return SymInfo->getTargetAddress();912}913914uint64_t RuntimeDyldCheckerImpl::readMemoryAtAddr(uint64_t SrcAddr,915unsigned Size) const {916uintptr_t PtrSizedAddr = static_cast<uintptr_t>(SrcAddr);917assert(PtrSizedAddr == SrcAddr && "Linker memory pointer out-of-range.");918void *Ptr = reinterpret_cast<void*>(PtrSizedAddr);919920switch (Size) {921case 1:922return support::endian::read<uint8_t>(Ptr, Endianness);923case 2:924return support::endian::read<uint16_t>(Ptr, Endianness);925case 4:926return support::endian::read<uint32_t>(Ptr, Endianness);927case 8:928return support::endian::read<uint64_t>(Ptr, Endianness);929}930llvm_unreachable("Unsupported read size");931}932933StringRef RuntimeDyldCheckerImpl::getSymbolContent(StringRef Symbol) const {934auto SymInfo = GetSymbolInfo(Symbol);935if (!SymInfo) {936logAllUnhandledErrors(SymInfo.takeError(), errs(), "RTDyldChecker: ");937return StringRef();938}939return {SymInfo->getContent().data(), SymInfo->getContent().size()};940}941942TargetFlagsType RuntimeDyldCheckerImpl::getTargetFlag(StringRef Symbol) const {943auto SymInfo = GetSymbolInfo(Symbol);944if (!SymInfo) {945logAllUnhandledErrors(SymInfo.takeError(), errs(), "RTDyldChecker: ");946return TargetFlagsType{};947}948return SymInfo->getTargetFlags();949}950951Triple952RuntimeDyldCheckerImpl::getTripleForSymbol(TargetFlagsType Flag) const {953Triple TheTriple = TT;954955switch (TT.getArch()) {956case Triple::ArchType::arm:957if (~Flag & 0x1)958return TT;959TheTriple.setArchName((Twine("thumb") + TT.getArchName().substr(3)).str());960return TheTriple;961case Triple::ArchType::thumb:962if (Flag & 0x1)963return TT;964TheTriple.setArchName((Twine("arm") + TT.getArchName().substr(5)).str());965return TheTriple;966967default:968return TT;969}970}971972std::pair<uint64_t, std::string> RuntimeDyldCheckerImpl::getSectionAddr(973StringRef FileName, StringRef SectionName, bool IsInsideLoad) const {974975auto SecInfo = GetSectionInfo(FileName, SectionName);976if (!SecInfo) {977std::string ErrMsg;978{979raw_string_ostream ErrMsgStream(ErrMsg);980logAllUnhandledErrors(SecInfo.takeError(), ErrMsgStream,981"RTDyldChecker: ");982}983return std::make_pair(0, std::move(ErrMsg));984}985986// If this address is being looked up in "load" mode, return the content987// pointer, otherwise return the target address.988989uint64_t Addr = 0;990991if (IsInsideLoad) {992if (SecInfo->isZeroFill())993Addr = 0;994else995Addr = pointerToJITTargetAddress(SecInfo->getContent().data());996} else997Addr = SecInfo->getTargetAddress();998999return std::make_pair(Addr, "");1000}10011002std::pair<uint64_t, std::string> RuntimeDyldCheckerImpl::getStubOrGOTAddrFor(1003StringRef StubContainerName, StringRef SymbolName, StringRef StubKindFilter,1004bool IsInsideLoad, bool IsStubAddr) const {10051006assert((StubKindFilter.empty() || IsStubAddr) &&1007"Kind name filter only supported for stubs");1008auto StubInfo =1009IsStubAddr ? GetStubInfo(StubContainerName, SymbolName, StubKindFilter)1010: GetGOTInfo(StubContainerName, SymbolName);10111012if (!StubInfo) {1013std::string ErrMsg;1014{1015raw_string_ostream ErrMsgStream(ErrMsg);1016logAllUnhandledErrors(StubInfo.takeError(), ErrMsgStream,1017"RTDyldChecker: ");1018}1019return std::make_pair((uint64_t)0, std::move(ErrMsg));1020}10211022uint64_t Addr = 0;10231024if (IsInsideLoad) {1025if (StubInfo->isZeroFill())1026return std::make_pair((uint64_t)0, "Detected zero-filled stub/GOT entry");1027Addr = pointerToJITTargetAddress(StubInfo->getContent().data());1028} else1029Addr = StubInfo->getTargetAddress();10301031return std::make_pair(Addr, "");1032}10331034RuntimeDyldChecker::RuntimeDyldChecker(1035IsSymbolValidFunction IsSymbolValid, GetSymbolInfoFunction GetSymbolInfo,1036GetSectionInfoFunction GetSectionInfo, GetStubInfoFunction GetStubInfo,1037GetGOTInfoFunction GetGOTInfo, llvm::endianness Endianness, Triple TT,1038StringRef CPU, SubtargetFeatures TF, raw_ostream &ErrStream)1039: Impl(::std::make_unique<RuntimeDyldCheckerImpl>(1040std::move(IsSymbolValid), std::move(GetSymbolInfo),1041std::move(GetSectionInfo), std::move(GetStubInfo),1042std::move(GetGOTInfo), Endianness, std::move(TT), std::move(CPU),1043std::move(TF), ErrStream)) {}10441045RuntimeDyldChecker::~RuntimeDyldChecker() = default;10461047bool RuntimeDyldChecker::check(StringRef CheckExpr) const {1048return Impl->check(CheckExpr);1049}10501051bool RuntimeDyldChecker::checkAllRulesInBuffer(StringRef RulePrefix,1052MemoryBuffer *MemBuf) const {1053return Impl->checkAllRulesInBuffer(RulePrefix, MemBuf);1054}10551056std::pair<uint64_t, std::string>1057RuntimeDyldChecker::getSectionAddr(StringRef FileName, StringRef SectionName,1058bool LocalAddress) {1059return Impl->getSectionAddr(FileName, SectionName, LocalAddress);1060}106110621063