diff --git a/bindings/nodejs/consts/keystone.js b/bindings/nodejs/consts/keystone.js index df6504d0622404f2e3b6c0b4b588296824e21c52..30927dd5fd61feafd5e0bc54bc9674c9b21d417e 100644 --- a/bindings/nodejs/consts/keystone.js +++ b/bindings/nodejs/consts/keystone.js @@ -54,17 +54,18 @@ module.exports.ERR_ASM_VARIANT_INVALID = 137 module.exports.ERR_ASM_EXPR_BRACKET = 138 module.exports.ERR_ASM_SYMBOL_MODIFIER = 139 module.exports.ERR_ASM_SYMBOL_REDEFINED = 140 -module.exports.ERR_ASM_RPAREN = 141 -module.exports.ERR_ASM_STAT_TOKEN = 142 -module.exports.ERR_ASM_UNSUPPORTED = 143 -module.exports.ERR_ASM_MACRO_TOKEN = 144 -module.exports.ERR_ASM_MACRO_PAREN = 145 -module.exports.ERR_ASM_MACRO_EQU = 146 -module.exports.ERR_ASM_MACRO_ARGS = 147 -module.exports.ERR_ASM_MACRO_LEVELS_EXCEED = 148 -module.exports.ERR_ASM_ESC_BACKSLASH = 149 -module.exports.ERR_ASM_ESC_OCTAL = 150 -module.exports.ERR_ASM_ESC_SEQUENCE = 151 +module.exports.ERR_ASM_SYMBOL_NOTFOUND = 141 +module.exports.ERR_ASM_RPAREN = 142 +module.exports.ERR_ASM_STAT_TOKEN = 143 +module.exports.ERR_ASM_UNSUPPORTED = 144 +module.exports.ERR_ASM_MACRO_TOKEN = 145 +module.exports.ERR_ASM_MACRO_PAREN = 146 +module.exports.ERR_ASM_MACRO_EQU = 147 +module.exports.ERR_ASM_MACRO_ARGS = 148 +module.exports.ERR_ASM_MACRO_LEVELS_EXCEED = 149 +module.exports.ERR_ASM_ESC_BACKSLASH = 150 +module.exports.ERR_ASM_ESC_OCTAL = 151 +module.exports.ERR_ASM_ESC_SEQUENCE = 152 module.exports.ERR_ASM_INVALIDOPERAND = 512 module.exports.ERR_ASM_MISSINGFEATURE = 513 module.exports.ERR_ASM_MNEMONICFAIL = 514 diff --git a/bindings/python/keystone/keystone_const.py b/bindings/python/keystone/keystone_const.py index 96162774621587081e17de1ba3c8b9fe8cd9911c..f0d22ff0845395ae067ed7bdc7cede8a25125316 100644 --- a/bindings/python/keystone/keystone_const.py +++ b/bindings/python/keystone/keystone_const.py @@ -54,17 +54,18 @@ KS_ERR_ASM_VARIANT_INVALID = 137 KS_ERR_ASM_EXPR_BRACKET = 138 KS_ERR_ASM_SYMBOL_MODIFIER = 139 KS_ERR_ASM_SYMBOL_REDEFINED = 140 -KS_ERR_ASM_RPAREN = 141 -KS_ERR_ASM_STAT_TOKEN = 142 -KS_ERR_ASM_UNSUPPORTED = 143 -KS_ERR_ASM_MACRO_TOKEN = 144 -KS_ERR_ASM_MACRO_PAREN = 145 -KS_ERR_ASM_MACRO_EQU = 146 -KS_ERR_ASM_MACRO_ARGS = 147 -KS_ERR_ASM_MACRO_LEVELS_EXCEED = 148 -KS_ERR_ASM_ESC_BACKSLASH = 149 -KS_ERR_ASM_ESC_OCTAL = 150 -KS_ERR_ASM_ESC_SEQUENCE = 151 +KS_ERR_ASM_SYMBOL_NOTFOUND = 141 +KS_ERR_ASM_RPAREN = 142 +KS_ERR_ASM_STAT_TOKEN = 143 +KS_ERR_ASM_UNSUPPORTED = 144 +KS_ERR_ASM_MACRO_TOKEN = 145 +KS_ERR_ASM_MACRO_PAREN = 146 +KS_ERR_ASM_MACRO_EQU = 147 +KS_ERR_ASM_MACRO_ARGS = 148 +KS_ERR_ASM_MACRO_LEVELS_EXCEED = 149 +KS_ERR_ASM_ESC_BACKSLASH = 150 +KS_ERR_ASM_ESC_OCTAL = 151 +KS_ERR_ASM_ESC_SEQUENCE = 152 KS_ERR_ASM_INVALIDOPERAND = 512 KS_ERR_ASM_MISSINGFEATURE = 513 KS_ERR_ASM_MNEMONICFAIL = 514 diff --git a/bindings/ruby/keystone_gem/lib/keystone/keystone_const.rb b/bindings/ruby/keystone_gem/lib/keystone/keystone_const.rb index 81dc47c9dcaf81752c90096de90304d1c50fc682..ce575ed5401dd229f084ec7c701362317d98c386 100644 --- a/bindings/ruby/keystone_gem/lib/keystone/keystone_const.rb +++ b/bindings/ruby/keystone_gem/lib/keystone/keystone_const.rb @@ -56,17 +56,18 @@ module Keystone KS_ERR_ASM_EXPR_BRACKET = 138 KS_ERR_ASM_SYMBOL_MODIFIER = 139 KS_ERR_ASM_SYMBOL_REDEFINED = 140 - KS_ERR_ASM_RPAREN = 141 - KS_ERR_ASM_STAT_TOKEN = 142 - KS_ERR_ASM_UNSUPPORTED = 143 - KS_ERR_ASM_MACRO_TOKEN = 144 - KS_ERR_ASM_MACRO_PAREN = 145 - KS_ERR_ASM_MACRO_EQU = 146 - KS_ERR_ASM_MACRO_ARGS = 147 - KS_ERR_ASM_MACRO_LEVELS_EXCEED = 148 - KS_ERR_ASM_ESC_BACKSLASH = 149 - KS_ERR_ASM_ESC_OCTAL = 150 - KS_ERR_ASM_ESC_SEQUENCE = 151 + KS_ERR_ASM_SYMBOL_NOTFOUND = 141 + KS_ERR_ASM_RPAREN = 142 + KS_ERR_ASM_STAT_TOKEN = 143 + KS_ERR_ASM_UNSUPPORTED = 144 + KS_ERR_ASM_MACRO_TOKEN = 145 + KS_ERR_ASM_MACRO_PAREN = 146 + KS_ERR_ASM_MACRO_EQU = 147 + KS_ERR_ASM_MACRO_ARGS = 148 + KS_ERR_ASM_MACRO_LEVELS_EXCEED = 149 + KS_ERR_ASM_ESC_BACKSLASH = 150 + KS_ERR_ASM_ESC_OCTAL = 151 + KS_ERR_ASM_ESC_SEQUENCE = 152 KS_ERR_ASM_INVALIDOPERAND = 512 KS_ERR_ASM_MISSINGFEATURE = 513 KS_ERR_ASM_MNEMONICFAIL = 514 diff --git a/include/keystone/keystone.h b/include/keystone/keystone.h index dfd776fee0b5950dc6d87bcef6dc38ff194d2741..1db837ffbe0659364ed3bd03fd5d9f37dcc87028 100644 --- a/include/keystone/keystone.h +++ b/include/keystone/keystone.h @@ -110,6 +110,7 @@ typedef enum ks_err { KS_ERR_ASM_EXPR_BRACKET, // brackets expression not supported on this target KS_ERR_ASM_SYMBOL_MODIFIER, // unexpected symbol modifier following '@' KS_ERR_ASM_SYMBOL_REDEFINED, // invalid symbol redefinition + KS_ERR_ASM_SYMBOL_NOTFOUND, // cannot find symbol KS_ERR_ASM_RPAREN, // expected ')' in parentheses expression KS_ERR_ASM_STAT_TOKEN, // unexpected token at start of statement KS_ERR_ASM_UNSUPPORTED, // unsupported token yet diff --git a/llvm/include/llvm/MC/MCAssembler.h b/llvm/include/llvm/MC/MCAssembler.h index ff069a8273ac6722036e44514d6474c883cae442..1e504fd5133d15e644ad5448c3d20fffb4e097e9 100644 --- a/llvm/include/llvm/MC/MCAssembler.h +++ b/llvm/include/llvm/MC/MCAssembler.h @@ -163,7 +163,7 @@ private: /// relocation. bool evaluateFixup(const MCAsmLayout &Layout, const MCFixup &Fixup, const MCFragment *DF, MCValue &Target, - uint64_t &Value) const; + uint64_t &Value, unsigned int &KsError) const; /// Check whether a fixup can be satisfied, or whether it needs to be relaxed /// (increased in size, in order to hold its value correctly). @@ -194,7 +194,7 @@ private: void finishLayout(MCAsmLayout &Layout); std::pair<uint64_t, bool> handleFixup(const MCAsmLayout &Layout, - MCFragment &F, const MCFixup &Fixup); + MCFragment &F, const MCFixup &Fixup, unsigned int &KsError); public: /// Compute the effective fragment size assuming it is laid out at the given @@ -265,10 +265,10 @@ public: /// Finish - Do final processing and write the object to the output stream. /// \p Writer is used for custom object writer (as the MCJIT does), /// if not specified it is automatically created from backend. - void Finish(); + void Finish(unsigned int &KsError); // Layout all section and prepare them for emission. - void layout(MCAsmLayout &Layout); + void layout(MCAsmLayout &Layout, unsigned int &KsError); // FIXME: This does not belong here. bool getSubsectionsViaSymbols() const { return SubsectionsViaSymbols; } diff --git a/llvm/include/llvm/MC/MCELFStreamer.h b/llvm/include/llvm/MC/MCELFStreamer.h index 0500861b404e799e06cc3b8931400a17fb2e855c..ebe439e141669dc91cd6dfd53cfdfcf45f319c5c 100644 --- a/llvm/include/llvm/MC/MCELFStreamer.h +++ b/llvm/include/llvm/MC/MCELFStreamer.h @@ -76,7 +76,7 @@ public: void EmitValueToAlignment(unsigned, int64_t, unsigned, unsigned) override; - void FinishImpl() override; + unsigned int FinishImpl() override; void EmitBundleAlignMode(unsigned AlignPow2) override; void EmitBundleLock(bool AlignToEnd) override; diff --git a/llvm/include/llvm/MC/MCObjectStreamer.h b/llvm/include/llvm/MC/MCObjectStreamer.h index 9d411a26cea40697dc2e9e80ff167dc43ed685a3..7ecc137748729d01b82b6780fcb69575f54eb8b6 100644 --- a/llvm/include/llvm/MC/MCObjectStreamer.h +++ b/llvm/include/llvm/MC/MCObjectStreamer.h @@ -137,7 +137,7 @@ public: bool EmitRelocDirective(const MCExpr &Offset, StringRef Name, const MCExpr *Expr, SMLoc Loc) override; void EmitFill(uint64_t NumBytes, uint8_t FillValue) override; - void FinishImpl() override; + unsigned int FinishImpl() override; /// Emit the absolute difference between two symbols if possible. /// diff --git a/llvm/include/llvm/MC/MCStreamer.h b/llvm/include/llvm/MC/MCStreamer.h index 509f5b49551c0662cc754bf9d1d6bddb6b6a0d76..262c9bada9c2e081de3e551ffa25d22b0bd3fb3b 100644 --- a/llvm/include/llvm/MC/MCStreamer.h +++ b/llvm/include/llvm/MC/MCStreamer.h @@ -742,9 +742,9 @@ public: void EmitRawText(const Twine &String); /// \brief Streamer specific finalization. - virtual void FinishImpl(); + virtual unsigned int FinishImpl(); /// \brief Finish emission of machine code. - void Finish(); + unsigned int Finish(); virtual bool mayHaveInstructions(MCSection &Sec) const { return true; } }; diff --git a/llvm/keystone/ks.cpp b/llvm/keystone/ks.cpp index fe3f0ba3d557dbd2f851afd9abd9f7c018b52eb0..ccef24ca866f63bae22040d51e42094ef8343747 100644 --- a/llvm/keystone/ks.cpp +++ b/llvm/keystone/ks.cpp @@ -94,6 +94,8 @@ const char *ks_strerror(ks_err code) return "Unexpected symbol modifier following '@' (KS_ERR_ASM_SYMBOL_MODIFIER)"; case KS_ERR_ASM_SYMBOL_REDEFINED: return "Invalid symbol redefined (KS_ERR_ASM_SYMBOL_REDEFINED)"; + case KS_ERR_ASM_SYMBOL_NOTFOUND: + return "Cannot find a symbol (KS_ERR_ASM_SYMBOL_NOTFOUND)"; case KS_ERR_ASM_RPAREN: // expected ')' in parentheses expression return "Expected ')' (KS_ERR_ASM_RPAREN)"; case KS_ERR_ASM_STAT_TOKEN: // unexpected token at start of statement diff --git a/llvm/lib/MC/MCAssembler.cpp b/llvm/lib/MC/MCAssembler.cpp index 2b4848730c08e9c961feb08f5851912fe3363cd4..ede069180b50fded184cccc5c913ecf0c0ba56b5 100644 --- a/llvm/lib/MC/MCAssembler.cpp +++ b/llvm/lib/MC/MCAssembler.cpp @@ -146,8 +146,10 @@ const MCSymbol *MCAssembler::getAtom(const MCSymbol &S) const { bool MCAssembler::evaluateFixup(const MCAsmLayout &Layout, const MCFixup &Fixup, const MCFragment *DF, - MCValue &Target, uint64_t &Value) const + MCValue &Target, uint64_t &Value, unsigned int &KsError) const { + KsError = 0; + // FIXME: This code has some duplication with recordRelocation. We should // probably merge the two into a single callback that tries to evaluate a // fixup and records a relocation if one is needed. @@ -192,6 +194,10 @@ bool MCAssembler::evaluateFixup(const MCAsmLayout &Layout, const MCSymbol &Sym = A->getSymbol(); if (Sym.isDefined()) Value += Layout.getSymbolOffset(Sym); + else { + KsError = KS_ERR_ASM_SYMBOL_NOTFOUND; + return false; + } } if (const MCSymbolRefExpr *B = Target.getSymB()) { const MCSymbol &Sym = B->getSymbol(); @@ -219,7 +225,6 @@ bool MCAssembler::evaluateFixup(const MCAsmLayout &Layout, Backend.processFixupValue(*this, Layout, Fixup, DF, Target, Value, IsResolved); - return IsResolved; } @@ -573,13 +578,17 @@ void MCAssembler::writeSectionData(const MCSection *Sec, std::pair<uint64_t, bool> MCAssembler::handleFixup(const MCAsmLayout &Layout, MCFragment &F, - const MCFixup &Fixup) { + const MCFixup &Fixup, unsigned int &KsError) { // Evaluate the fixup. MCValue Target; uint64_t FixedValue; bool IsPCRel = Backend.getFixupKindInfo(Fixup.getKind()).Flags & MCFixupKindInfo::FKF_IsPCRel; - if (!evaluateFixup(Layout, Fixup, &F, Target, FixedValue)) { + if (!evaluateFixup(Layout, Fixup, &F, Target, FixedValue, KsError)) { + if (KsError) { + // return a dummy value + return std::make_pair(0, false); + } // The fixup was unresolved, we need a relocation. Inform the object // writer of the relocation, and give it an opportunity to adjust the // fixup value if need be. @@ -589,7 +598,8 @@ std::pair<uint64_t, bool> MCAssembler::handleFixup(const MCAsmLayout &Layout, return std::make_pair(FixedValue, IsPCRel); } -void MCAssembler::layout(MCAsmLayout &Layout) { +void MCAssembler::layout(MCAsmLayout &Layout, unsigned int &KsError) +{ DEBUG_WITH_TYPE("mc-dump", { llvm::errs() << "assembler backend - pre-layout\n--\n"; dump(); }); @@ -657,7 +667,9 @@ void MCAssembler::layout(MCAsmLayout &Layout) { for (const MCFixup &Fixup : Fixups) { uint64_t FixedValue; bool IsPCRel; - std::tie(FixedValue, IsPCRel) = handleFixup(Layout, *F, Fixup); + std::tie(FixedValue, IsPCRel) = handleFixup(Layout, *F, Fixup, KsError); + if (KsError) + return; getBackend().applyFixup(Fixup, Contents.data(), Contents.size(), FixedValue, IsPCRel); } @@ -665,13 +677,14 @@ void MCAssembler::layout(MCAsmLayout &Layout) { } } -void MCAssembler::Finish() { +void MCAssembler::Finish(unsigned int &KsError) { // Create the layout object. MCAsmLayout Layout(*this); - layout(Layout); + layout(Layout, KsError); // Write the object file. - getWriter().writeObject(*this, Layout); + if (!KsError) + getWriter().writeObject(*this, Layout); } bool MCAssembler::fixupNeedsRelaxation(const MCFixup &Fixup, @@ -679,7 +692,8 @@ bool MCAssembler::fixupNeedsRelaxation(const MCFixup &Fixup, const MCAsmLayout &Layout) const { MCValue Target; uint64_t Value; - bool Resolved = evaluateFixup(Layout, Fixup, DF, Target, Value); + unsigned int KsError; + bool Resolved = evaluateFixup(Layout, Fixup, DF, Target, Value, KsError); return getBackend().fixupNeedsRelaxationAdvanced(Fixup, Resolved, Value, DF, Layout); } diff --git a/llvm/lib/MC/MCELFStreamer.cpp b/llvm/lib/MC/MCELFStreamer.cpp index 540125a56ed620dec13b1002ed1baace2f2635e9..f93129b38c4df75940fc5efe9a791e4e22d6b25c 100644 --- a/llvm/lib/MC/MCELFStreamer.cpp +++ b/llvm/lib/MC/MCELFStreamer.cpp @@ -623,14 +623,14 @@ void MCELFStreamer::EmitBundleUnlock() { Sec.setBundleLockState(MCSection::NotBundleLocked); } -void MCELFStreamer::FinishImpl() { +unsigned int MCELFStreamer::FinishImpl() { // Ensure the last section gets aligned if necessary. MCSection *CurSection = getCurrentSectionOnly(); setSectionAlignmentForBundling(getAssembler(), CurSection); EmitFrames(nullptr); - this->MCObjectStreamer::FinishImpl(); + return this->MCObjectStreamer::FinishImpl(); } MCStreamer *llvm::createELFStreamer(MCContext &Context, MCAsmBackend &MAB, diff --git a/llvm/lib/MC/MCObjectStreamer.cpp b/llvm/lib/MC/MCObjectStreamer.cpp index ae2403036e7449e69afe5e3d1a2e4b4bef89e322..4335af9476d30ac85981e5a4845052b831c333bb 100644 --- a/llvm/lib/MC/MCObjectStreamer.cpp +++ b/llvm/lib/MC/MCObjectStreamer.cpp @@ -501,7 +501,8 @@ void MCObjectStreamer::EmitFill(uint64_t NumBytes, uint8_t FillValue) { insert(new MCFillFragment(FillValue, NumBytes)); } -void MCObjectStreamer::FinishImpl() { +unsigned int MCObjectStreamer::FinishImpl() { + unsigned int KsError = 0; // If we are generating dwarf for assembly source files dump out the sections. //if (getContext().getGenDwarfForAssembly()) // MCGenDwarfInfo::Emit(this); @@ -510,5 +511,7 @@ void MCObjectStreamer::FinishImpl() { //MCDwarfLineTable::Emit(this, getAssembler().getDWARFLinetableParams()); flushPendingLabels(nullptr); - getAssembler().Finish(); + getAssembler().Finish(KsError); + + return KsError; } diff --git a/llvm/lib/MC/MCParser/AsmParser.cpp b/llvm/lib/MC/MCParser/AsmParser.cpp index 8378438f1a5dc150c193e9c28e9c25513ed17086..8166362157719ef3c988f32b978bbc4031da8f46 100644 --- a/llvm/lib/MC/MCParser/AsmParser.cpp +++ b/llvm/lib/MC/MCParser/AsmParser.cpp @@ -742,8 +742,8 @@ size_t AsmParser::Run(bool NoInitialTextSection, uint64_t Address, bool NoFinali // Finalize the output stream if there are no errors and if the client wants // us to. - if (!HadError && !NoFinalize) - Out.Finish(); + if (!KsError && !HadError && !NoFinalize) + KsError = Out.Finish(); //return HadError || getContext().hadError(); return count; @@ -1393,6 +1393,7 @@ bool AsmParser::parseStatement(ParseStatementInfo &Info, // Statements always start with an identifier or are a full line comment. AsmToken ID = getTok(); + //printf(">>> parseStatement:ID = %s\n", ID.getString().str().c_str()); SMLoc IDLoc = ID.getLoc(); StringRef IDVal; int64_t LocalLabelVal = -1; diff --git a/llvm/lib/MC/MCStreamer.cpp b/llvm/lib/MC/MCStreamer.cpp index 380155cf6c4dc2e14158d48754bd78969f7a64b7..e9e765f72f15eefc7d76c014870e0463a164af4c 100644 --- a/llvm/lib/MC/MCStreamer.cpp +++ b/llvm/lib/MC/MCStreamer.cpp @@ -612,7 +612,7 @@ void MCStreamer::EmitRawText(const Twine &T) { void MCStreamer::EmitWindowsUnwindTables() { } -void MCStreamer::Finish() { +unsigned int MCStreamer::Finish() { if (!DwarfFrameInfos.empty() && !DwarfFrameInfos.back().End) report_fatal_error("Unfinished frame!"); @@ -620,7 +620,7 @@ void MCStreamer::Finish() { if (TS) TS->finish(); - FinishImpl(); + return FinishImpl(); } void MCStreamer::EmitAssignment(MCSymbol *Symbol, const MCExpr *Value) { @@ -717,7 +717,7 @@ void MCStreamer::EmitCodeAlignment(unsigned ByteAlignment, void MCStreamer::emitValueToOffset(const MCExpr *Offset, unsigned char Value) {} void MCStreamer::EmitBundleAlignMode(unsigned AlignPow2) {} void MCStreamer::EmitBundleLock(bool AlignToEnd) {} -void MCStreamer::FinishImpl() {} +unsigned int MCStreamer::FinishImpl() { return 0; } void MCStreamer::EmitBundleUnlock() {} void MCStreamer::SwitchSection(MCSection *Section, const MCExpr *Subsection) {