|  | //===-- 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/StringSaver.h" | 
|  | #include "llvm/Support/raw_ostream.h" | 
|  |  | 
|  | #include <system_error> | 
|  | #include <vector> | 
|  |  | 
|  | namespace llvm { | 
|  | namespace rc { | 
|  |  | 
|  | class RCParser { | 
|  | public: | 
|  | using LocIter = std::vector<RCToken>::iterator; | 
|  | using ParseType = Expected<std::unique_ptr<RCResource>>; | 
|  | using ParseOptionType = Expected<std::unique_ptr<OptionalStmt>>; | 
|  |  | 
|  | // Class describing a single failure of parser. | 
|  | class ParserError : public ErrorInfo<ParserError> { | 
|  | 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<RCToken> 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<RCInt> readInt();               // Parse an integer. | 
|  | Expected<StringRef> readString();        // Parse a string. | 
|  | Expected<StringRef> readIdentifier();    // Parse an identifier. | 
|  | Expected<StringRef> readFilename();      // Parse a filename. | 
|  | Expected<IntOrString> readIntOrString(); // Parse an integer or a string. | 
|  | Expected<IntOrString> readTypeOrName();  // Parse an integer or an identifier. | 
|  |  | 
|  | // Helper integer expression parsing methods. | 
|  | Expected<IntWithNotMask> parseIntExpr1(); | 
|  | Expected<IntWithNotMask> 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<SmallVector<RCInt, 8>> 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<uint32_t> parseFlags(ArrayRef<StringRef> FlagDesc, | 
|  | ArrayRef<uint32_t> 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<OptionalStmtList> | 
|  | parseOptionalStatements(OptStmtType StmtsType = OptStmtType::BasicStmt); | 
|  |  | 
|  | // Read a single optional statement. | 
|  | Expected<std::unique_ptr<OptionalStmt>> | 
|  | 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 parseMenuExResource(); | 
|  | ParseType parseStringTableResource(); | 
|  | ParseType parseUserDefinedResource(IntOrString Type); | 
|  | ParseType parseVersionInfoResource(); | 
|  |  | 
|  | // Helper DIALOG parser - a single control. | 
|  | Expected<Control> parseControl(); | 
|  |  | 
|  | // Helper MENU parser. | 
|  | Expected<MenuDefinitionList> parseMenuItemsList(); | 
|  |  | 
|  | // Helper MENUEX parser. | 
|  | Expected<MenuDefinitionList> parseMenuExItemsList(); | 
|  |  | 
|  | // Helper VERSIONINFO parser - read the contents of a single BLOCK statement, | 
|  | // from BEGIN to END. | 
|  | Expected<std::unique_ptr<VersionInfoBlock>> | 
|  | parseVersionInfoBlockContents(StringRef BlockName); | 
|  | // Helper VERSIONINFO parser - read either VALUE or BLOCK statement. | 
|  | Expected<std::unique_ptr<VersionInfoStmt>> parseVersionInfoStmt(); | 
|  | // Helper VERSIONINFO parser - read fixed VERSIONINFO statements. | 
|  | Expected<VersionInfoResource::VersionInfoFixed> parseVersionInfoFixed(); | 
|  |  | 
|  | // Optional statement parsers. | 
|  | ParseOptionType parseLanguageStmt(); | 
|  | ParseOptionType parseCharacteristicsStmt(); | 
|  | ParseOptionType parseVersionStmt(); | 
|  | ParseOptionType parseCaptionStmt(); | 
|  | ParseOptionType parseClassStmt(); | 
|  | ParseOptionType parseExStyleStmt(); | 
|  | ParseOptionType parseFontStmt(OptStmtType DialogType); | 
|  | ParseOptionType parseStyleStmt(); | 
|  | ParseOptionType parseMenuStmt(); | 
|  |  | 
|  | // 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<RCToken> Tokens; | 
|  | LocIter CurLoc; | 
|  | const LocIter End; | 
|  |  | 
|  | BumpPtrAllocator Alloc; | 
|  | StringSaver Saver{Alloc}; | 
|  | }; | 
|  |  | 
|  | } // namespace rc | 
|  | } // namespace llvm | 
|  |  | 
|  | #endif |