blob: 20d8ff9624d9f21f6c18130bab9e26818d800c46 [file] [log] [blame]
Marek Sokolowski0b33df92017-08-18 18:24:17 +00001//===-- ResourceScriptParser.h ----------------------------------*- C++-*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===---------------------------------------------------------------------===//
9//
10// This defines the RC scripts parser. It takes a sequence of RC tokens
11// and then provides the method to parse the resources one by one.
12//
13//===---------------------------------------------------------------------===//
14
15#ifndef LLVM_TOOLS_LLVMRC_RESOURCESCRIPTPARSER_H
16#define LLVM_TOOLS_LLVMRC_RESOURCESCRIPTPARSER_H
17
18#include "ResourceScriptStmt.h"
19#include "ResourceScriptToken.h"
20
21#include "llvm/Support/Compiler.h"
22#include "llvm/Support/raw_ostream.h"
23
24#include <system_error>
25#include <vector>
26
27namespace llvm {
Zachary Turnere315d732017-10-11 20:12:09 +000028namespace opt {
29class InputArgList;
30}
Marek Sokolowski0b33df92017-08-18 18:24:17 +000031namespace rc {
32
33class RCParser {
34public:
35 using LocIter = std::vector<RCToken>::iterator;
36 using ParseType = Expected<std::unique_ptr<RCResource>>;
37 using ParseOptionType = Expected<std::unique_ptr<OptionalStmt>>;
38
39 // Class describing a single failure of parser.
40 class ParserError : public ErrorInfo<ParserError> {
41 public:
Zachary Turner5d1b2d32017-10-09 18:50:29 +000042 ParserError(const Twine &Expected, const LocIter CurLoc, const LocIter End);
Marek Sokolowski0b33df92017-08-18 18:24:17 +000043
44 void log(raw_ostream &OS) const override { OS << CurMessage; }
45 std::error_code convertToErrorCode() const override {
46 return std::make_error_code(std::errc::invalid_argument);
47 }
48 const std::string &getMessage() const { return CurMessage; }
49
50 static char ID; // Keep llvm::Error happy.
51
52 private:
53 std::string CurMessage;
54 LocIter ErrorLoc, FileEnd;
55 };
56
Zachary Turnere315d732017-10-11 20:12:09 +000057 explicit RCParser(std::vector<RCToken> TokenList);
Marek Sokolowski0b33df92017-08-18 18:24:17 +000058
59 // Reads and returns a single resource definition, or error message if any
60 // occurred.
61 ParseType parseSingleResource();
62
63 bool isEof() const;
64
65private:
66 using Kind = RCToken::Kind;
67
68 // Checks if the current parser state points to the token of type TokenKind.
69 bool isNextTokenKind(Kind TokenKind) const;
70
71 // These methods assume that the parser is not in EOF state.
72
73 // Take a look at the current token. Do not fetch it.
74 const RCToken &look() const;
75 // Read the current token and advance the state by one token.
76 const RCToken &read();
77 // Advance the state by one token, discarding the current token.
78 void consume();
79
80 // The following methods try to read a single token, check if it has the
81 // correct type and then parse it.
Marek Sokolowski59066482017-09-28 23:53:25 +000082 // Each integer can be written as an arithmetic expression producing an
83 // unsigned 32-bit integer.
Zachary Turner93bb30d2017-10-06 21:26:06 +000084 Expected<RCInt> readInt(); // Parse an integer.
Marek Sokolowski66c13b12017-08-28 22:58:31 +000085 Expected<StringRef> readString(); // Parse a string.
86 Expected<StringRef> readIdentifier(); // Parse an identifier.
Martin Storsjo76e573d2018-05-08 08:47:37 +000087 Expected<StringRef> readFilename(); // Parse a filename.
Marek Sokolowski66c13b12017-08-28 22:58:31 +000088 Expected<IntOrString> readIntOrString(); // Parse an integer or a string.
89 Expected<IntOrString> readTypeOrName(); // Parse an integer or an identifier.
Marek Sokolowski0b33df92017-08-18 18:24:17 +000090
Marek Sokolowski59066482017-09-28 23:53:25 +000091 // Helper integer expression parsing methods.
Martin Storsjoe96cf5f2018-12-05 13:22:56 +000092 Expected<IntWithNotMask> parseIntExpr1();
93 Expected<IntWithNotMask> parseIntExpr2();
Marek Sokolowski59066482017-09-28 23:53:25 +000094
Marek Sokolowski0b33df92017-08-18 18:24:17 +000095 // Advance the state by one, discarding the current token.
96 // If the discarded token had an incorrect type, fail.
97 Error consumeType(Kind TokenKind);
98
99 // Check the current token type. If it's TokenKind, discard it.
100 // Return true if the parser consumed this token successfully.
101 bool consumeOptionalType(Kind TokenKind);
102
103 // Read at least MinCount, and at most MaxCount integers separated by
104 // commas. The parser stops reading after fetching MaxCount integers
105 // or after an error occurs. Whenever the parser reads a comma, it
106 // expects an integer to follow.
Zachary Turner93bb30d2017-10-06 21:26:06 +0000107 Expected<SmallVector<RCInt, 8>> readIntsWithCommas(size_t MinCount,
108 size_t MaxCount);
Marek Sokolowski0b33df92017-08-18 18:24:17 +0000109
Marek Sokolowski66c13b12017-08-28 22:58:31 +0000110 // Read an unknown number of flags preceded by commas. Each correct flag
111 // has an entry in FlagDesc array of length NumFlags. In case i-th
Marek Sokolowskiafe86312017-09-29 17:46:32 +0000112 // flag (0-based) has been read, the result is OR-ed with FlagValues[i].
Marek Sokolowski66c13b12017-08-28 22:58:31 +0000113 // As long as parser has a comma to read, it expects to be fed with
114 // a correct flag afterwards.
Marek Sokolowskiafe86312017-09-29 17:46:32 +0000115 Expected<uint32_t> parseFlags(ArrayRef<StringRef> FlagDesc,
116 ArrayRef<uint32_t> FlagValues);
Marek Sokolowski66c13b12017-08-28 22:58:31 +0000117
Marek Sokolowski0b33df92017-08-18 18:24:17 +0000118 // Reads a set of optional statements. These can change the behavior of
119 // a number of resource types (e.g. STRINGTABLE, MENU or DIALOG) if provided
120 // before the main block with the contents of the resource.
121 // Usually, resources use a basic set of optional statements:
122 // CHARACTERISTICS, LANGUAGE, VERSION
123 // However, DIALOG and DIALOGEX extend this list by the following items:
124 // CAPTION, CLASS, EXSTYLE, FONT, MENU, STYLE
125 // UseExtendedStatements flag (off by default) allows the parser to read
126 // the additional types of statements.
127 //
128 // Ref (to the list of all optional statements):
129 // msdn.microsoft.com/en-us/library/windows/desktop/aa381002(v=vs.85).aspx
Zachary Turner44bde8d2017-10-06 20:51:20 +0000130 enum class OptStmtType { BasicStmt, DialogStmt, DialogExStmt };
131
Martin Storsjo370633f2018-05-15 06:35:29 +0000132 uint16_t parseMemoryFlags(uint16_t DefaultFlags);
133
Marek Sokolowski0b33df92017-08-18 18:24:17 +0000134 Expected<OptionalStmtList>
Zachary Turner44bde8d2017-10-06 20:51:20 +0000135 parseOptionalStatements(OptStmtType StmtsType = OptStmtType::BasicStmt);
Marek Sokolowski0b33df92017-08-18 18:24:17 +0000136
137 // Read a single optional statement.
138 Expected<std::unique_ptr<OptionalStmt>>
Zachary Turner44bde8d2017-10-06 20:51:20 +0000139 parseSingleOptionalStatement(OptStmtType StmtsType = OptStmtType::BasicStmt);
Marek Sokolowski0b33df92017-08-18 18:24:17 +0000140
141 // Top-level resource parsers.
142 ParseType parseLanguageResource();
Marek Sokolowski66c13b12017-08-28 22:58:31 +0000143 ParseType parseAcceleratorsResource();
Martin Storsjo31dc80f2018-05-07 20:27:37 +0000144 ParseType parseBitmapResource();
Marek Sokolowskif2e55892017-08-28 21:59:54 +0000145 ParseType parseCursorResource();
Marek Sokolowski7ca5fcc2017-08-29 16:49:59 +0000146 ParseType parseDialogResource(bool IsExtended);
Marek Sokolowski0b33df92017-08-18 18:24:17 +0000147 ParseType parseIconResource();
Marek Sokolowskif2e55892017-08-28 21:59:54 +0000148 ParseType parseHTMLResource();
Marek Sokolowski233d2b82017-08-28 23:46:30 +0000149 ParseType parseMenuResource();
Marek Sokolowski0b33df92017-08-18 18:24:17 +0000150 ParseType parseStringTableResource();
Marek Sokolowski0bce0412017-09-29 00:14:18 +0000151 ParseType parseUserDefinedResource(IntOrString Type);
Marek Sokolowski86b61382017-09-28 22:41:38 +0000152 ParseType parseVersionInfoResource();
Marek Sokolowski0b33df92017-08-18 18:24:17 +0000153
Marek Sokolowski7ca5fcc2017-08-29 16:49:59 +0000154 // Helper DIALOG parser - a single control.
155 Expected<Control> parseControl();
156
Marek Sokolowski233d2b82017-08-28 23:46:30 +0000157 // Helper MENU parser.
158 Expected<MenuDefinitionList> parseMenuItemsList();
159
Marek Sokolowski86b61382017-09-28 22:41:38 +0000160 // Helper VERSIONINFO parser - read the contents of a single BLOCK statement,
161 // from BEGIN to END.
162 Expected<std::unique_ptr<VersionInfoBlock>>
163 parseVersionInfoBlockContents(StringRef BlockName);
164 // Helper VERSIONINFO parser - read either VALUE or BLOCK statement.
165 Expected<std::unique_ptr<VersionInfoStmt>> parseVersionInfoStmt();
166 // Helper VERSIONINFO parser - read fixed VERSIONINFO statements.
167 Expected<VersionInfoResource::VersionInfoFixed> parseVersionInfoFixed();
168
Marek Sokolowski0b33df92017-08-18 18:24:17 +0000169 // Optional statement parsers.
170 ParseOptionType parseLanguageStmt();
171 ParseOptionType parseCharacteristicsStmt();
172 ParseOptionType parseVersionStmt();
Marek Sokolowski7ca5fcc2017-08-29 16:49:59 +0000173 ParseOptionType parseCaptionStmt();
Martin Storsjo65de7bd2018-05-15 19:21:28 +0000174 ParseOptionType parseClassStmt();
Martin Storsjocec32ed2018-11-29 12:17:39 +0000175 ParseOptionType parseExStyleStmt();
Zachary Turner44bde8d2017-10-06 20:51:20 +0000176 ParseOptionType parseFontStmt(OptStmtType DialogType);
Marek Sokolowski7ca5fcc2017-08-29 16:49:59 +0000177 ParseOptionType parseStyleStmt();
Marek Sokolowski0b33df92017-08-18 18:24:17 +0000178
179 // Raises an error. If IsAlreadyRead = false (default), this complains about
180 // the token that couldn't be parsed. If the flag is on, this complains about
181 // the correctly read token that makes no sense (that is, the current parser
182 // state is beyond the erroneous token.)
Zachary Turner5d1b2d32017-10-09 18:50:29 +0000183 Error getExpectedError(const Twine &Message, bool IsAlreadyRead = false);
Marek Sokolowski0b33df92017-08-18 18:24:17 +0000184
185 std::vector<RCToken> Tokens;
186 LocIter CurLoc;
187 const LocIter End;
188};
189
190} // namespace rc
191} // namespace llvm
192
193#endif