//===-- ResourceScriptParser.h ----------------------------------*- C++-*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===---------------------------------------------------------------------===// // // This defines the RC scripts parser. It takes a sequence of RC tokens // and then provides the method to parse the resources one by one. // //===---------------------------------------------------------------------===// #ifndef LLVM_TOOLS_LLVMRC_RESOURCESCRIPTPARSER_H #define LLVM_TOOLS_LLVMRC_RESOURCESCRIPTPARSER_H #include "ResourceScriptStmt.h" #include "ResourceScriptToken.h" #include "llvm/Support/Compiler.h" #include "llvm/Support/raw_ostream.h" #include #include namespace llvm { namespace opt { class InputArgList; } namespace rc { class RCParser { public: using LocIter = std::vector::iterator; using ParseType = Expected>; using ParseOptionType = Expected>; // Class describing a single failure of parser. class ParserError : public ErrorInfo { public: ParserError(const Twine &Expected, const LocIter CurLoc, const LocIter End); void log(raw_ostream &OS) const override { OS << CurMessage; } std::error_code convertToErrorCode() const override { return std::make_error_code(std::errc::invalid_argument); } const std::string &getMessage() const { return CurMessage; } static char ID; // Keep llvm::Error happy. private: std::string CurMessage; LocIter ErrorLoc, FileEnd; }; explicit RCParser(std::vector TokenList); // Reads and returns a single resource definition, or error message if any // occurred. ParseType parseSingleResource(); bool isEof() const; private: using Kind = RCToken::Kind; // Checks if the current parser state points to the token of type TokenKind. bool isNextTokenKind(Kind TokenKind) const; // These methods assume that the parser is not in EOF state. // Take a look at the current token. Do not fetch it. const RCToken &look() const; // Read the current token and advance the state by one token. const RCToken &read(); // Advance the state by one token, discarding the current token. void consume(); // The following methods try to read a single token, check if it has the // correct type and then parse it. // Each integer can be written as an arithmetic expression producing an // unsigned 32-bit integer. Expected readInt(); // Parse an integer. Expected readString(); // Parse a string. Expected readIdentifier(); // Parse an identifier. Expected readFilename(); // Parse a filename. Expected readIntOrString(); // Parse an integer or a string. Expected readTypeOrName(); // Parse an integer or an identifier. // Helper integer expression parsing methods. Expected parseIntExpr1(); Expected parseIntExpr2(); // Advance the state by one, discarding the current token. // If the discarded token had an incorrect type, fail. Error consumeType(Kind TokenKind); // Check the current token type. If it's TokenKind, discard it. // Return true if the parser consumed this token successfully. bool consumeOptionalType(Kind TokenKind); // Read at least MinCount, and at most MaxCount integers separated by // commas. The parser stops reading after fetching MaxCount integers // or after an error occurs. Whenever the parser reads a comma, it // expects an integer to follow. Expected> readIntsWithCommas(size_t MinCount, size_t MaxCount); // Read an unknown number of flags preceded by commas. Each correct flag // has an entry in FlagDesc array of length NumFlags. In case i-th // flag (0-based) has been read, the result is OR-ed with FlagValues[i]. // As long as parser has a comma to read, it expects to be fed with // a correct flag afterwards. Expected parseFlags(ArrayRef FlagDesc, ArrayRef FlagValues); // Reads a set of optional statements. These can change the behavior of // a number of resource types (e.g. STRINGTABLE, MENU or DIALOG) if provided // before the main block with the contents of the resource. // Usually, resources use a basic set of optional statements: // CHARACTERISTICS, LANGUAGE, VERSION // However, DIALOG and DIALOGEX extend this list by the following items: // CAPTION, CLASS, EXSTYLE, FONT, MENU, STYLE // UseExtendedStatements flag (off by default) allows the parser to read // the additional types of statements. // // Ref (to the list of all optional statements): // msdn.microsoft.com/en-us/library/windows/desktop/aa381002(v=vs.85).aspx enum class OptStmtType { BasicStmt, DialogStmt, DialogExStmt }; uint16_t parseMemoryFlags(uint16_t DefaultFlags); Expected parseOptionalStatements(OptStmtType StmtsType = OptStmtType::BasicStmt); // Read a single optional statement. Expected> parseSingleOptionalStatement(OptStmtType StmtsType = OptStmtType::BasicStmt); // Top-level resource parsers. ParseType parseLanguageResource(); ParseType parseAcceleratorsResource(); ParseType parseBitmapResource(); ParseType parseCursorResource(); ParseType parseDialogResource(bool IsExtended); ParseType parseIconResource(); ParseType parseHTMLResource(); ParseType parseMenuResource(); ParseType parseStringTableResource(); ParseType parseUserDefinedResource(IntOrString Type); ParseType parseVersionInfoResource(); // Helper DIALOG parser - a single control. Expected parseControl(); // Helper MENU parser. Expected parseMenuItemsList(); // Helper VERSIONINFO parser - read the contents of a single BLOCK statement, // from BEGIN to END. Expected> parseVersionInfoBlockContents(StringRef BlockName); // Helper VERSIONINFO parser - read either VALUE or BLOCK statement. Expected> parseVersionInfoStmt(); // Helper VERSIONINFO parser - read fixed VERSIONINFO statements. Expected parseVersionInfoFixed(); // Optional statement parsers. ParseOptionType parseLanguageStmt(); ParseOptionType parseCharacteristicsStmt(); ParseOptionType parseVersionStmt(); ParseOptionType parseCaptionStmt(); ParseOptionType parseClassStmt(); ParseOptionType parseExStyleStmt(); ParseOptionType parseFontStmt(OptStmtType DialogType); ParseOptionType parseStyleStmt(); // Raises an error. If IsAlreadyRead = false (default), this complains about // the token that couldn't be parsed. If the flag is on, this complains about // the correctly read token that makes no sense (that is, the current parser // state is beyond the erroneous token.) Error getExpectedError(const Twine &Message, bool IsAlreadyRead = false); std::vector Tokens; LocIter CurLoc; const LocIter End; }; } // namespace rc } // namespace llvm #endif