Line | Branch | Exec | Source |
---|---|---|---|
1 | // Copyright Contributors to the OpenVDB Project | ||
2 | // SPDX-License-Identifier: MPL-2.0 | ||
3 | |||
4 | /// @file compiler/Logger.h | ||
5 | /// | ||
6 | /// @authors Richard Jones | ||
7 | /// | ||
8 | /// @brief Logging system to collect errors and warnings throughout the | ||
9 | /// different stages of parsing and compilation. | ||
10 | /// | ||
11 | |||
12 | #ifndef OPENVDB_AX_COMPILER_LOGGER_HAS_BEEN_INCLUDED | ||
13 | #define OPENVDB_AX_COMPILER_LOGGER_HAS_BEEN_INCLUDED | ||
14 | |||
15 | #include "../ast/AST.h" | ||
16 | |||
17 | #include <openvdb/version.h> | ||
18 | |||
19 | #include <functional> | ||
20 | #include <string> | ||
21 | #include <unordered_map> | ||
22 | |||
23 | class TestLogger; | ||
24 | |||
25 | namespace openvdb { | ||
26 | OPENVDB_USE_VERSION_NAMESPACE | ||
27 | namespace OPENVDB_VERSION_NAME { | ||
28 | |||
29 | namespace ax { | ||
30 | |||
31 | /// @brief Logger for collecting errors and warnings that occur during AX | ||
32 | /// compilation. | ||
33 | /// | ||
34 | /// @details Error and warning output can be customised using the function | ||
35 | /// pointer arguments. These require a function that takes the formatted error | ||
36 | /// or warning string and handles the output, returning void. | ||
37 | /// e.g. | ||
38 | /// void streamCerr(const std::string& message) { | ||
39 | /// std::cerr << message << std::endl; | ||
40 | /// } | ||
41 | /// | ||
42 | /// The Logger handles formatting of messages, tracking of number of errors or | ||
43 | /// warnings and retrieval of errored lines of code to be printed if needed. | ||
44 | /// Use of the Logger to track new errors or warnings can be done either with | ||
45 | /// the line/column numbers directly (e.g during lexing and parsing where the | ||
46 | /// code is being iterated through) or referring to the AST node using its | ||
47 | /// position in the Tree (e.g. during codegen where only the AST node is known | ||
48 | /// directly, not the corresponding line/column numbers). To find the line or | ||
49 | /// column numbers for events logged using AST nodes, the Logger stores a map | ||
50 | /// of Node* to line and column numbers. This must be populated e.g. during | ||
51 | /// parsing, to allow resolution of code locations when they are not | ||
52 | /// explicitly available. The Logger also stores a pointer to the AST Tree | ||
53 | /// that these nodes belong to and the code used to create it. | ||
54 | /// | ||
55 | /// @warning The logger is not thread safe. A unique instance of the Logger | ||
56 | /// should be used for unique invocations of ax pipelines. | ||
57 | class OPENVDB_AX_API Logger | ||
58 | { | ||
59 | public: | ||
60 | using Ptr = std::shared_ptr<Logger>; | ||
61 | |||
62 | using CodeLocation = std::pair<size_t, size_t>; | ||
63 | using OutputFunction = std::function<void(const std::string&)>; | ||
64 | |||
65 | /// @brief Construct a Logger with optional error and warning output | ||
66 | /// functions, defaults stream errors to std::cerr and suppress warnings | ||
67 | /// @param errors Optional error output function | ||
68 | /// @param warnings Optional warning output function | ||
69 | Logger(const OutputFunction& errors = | ||
70 | ✗ | [](const std::string& msg){ | |
71 | std::cerr << msg << std::endl; | ||
72 | ✗ | }, | |
73 | const OutputFunction& warnings = [](const std::string&){}); | ||
74 | ~Logger(); | ||
75 | |||
76 | /// @brief Log a compiler error and its offending code location. If the | ||
77 | /// offending location is (0,0), the message is treated as not having an | ||
78 | /// associated code location. | ||
79 | /// @param message The error message | ||
80 | /// @param lineCol The line/column number of the offending code | ||
81 | /// @return true if can continue to capture future messages. | ||
82 | bool error(const std::string& message, const CodeLocation& lineCol = CodeLocation(0,0)); | ||
83 | |||
84 | /// @brief Log a compiler error using the offending AST node. Used in AST | ||
85 | /// traversal. | ||
86 | /// @param message The error message | ||
87 | /// @param node The offending AST node causing the error | ||
88 | /// @return true if can continue to capture future messages. | ||
89 | bool error(const std::string& message, const ax::ast::Node* node); | ||
90 | |||
91 | /// @brief Log a compiler warning and its offending code location. If the | ||
92 | /// offending location is (0,0), the message is treated as not having an | ||
93 | /// associated code location. | ||
94 | /// @param message The warning message | ||
95 | /// @param lineCol The line/column number of the offending code | ||
96 | /// @return true if can continue to capture future messages. | ||
97 | bool warning(const std::string& message, const CodeLocation& lineCol = CodeLocation(0,0)); | ||
98 | |||
99 | /// @brief Log a compiler warning using the offending AST node. Used in AST | ||
100 | /// traversal. | ||
101 | /// @param message The warning message | ||
102 | /// @param node The offending AST node causing the warning | ||
103 | /// @return true if can continue to capture future messages. | ||
104 | bool warning(const std::string& message, const ax::ast::Node* node); | ||
105 | |||
106 | /// | ||
107 | |||
108 | /// @brief Returns the number of errors that have been encountered | ||
109 |
16/29✗ Branch 0 not taken.
✓ Branch 1 taken 7938 times.
✓ Branch 2 taken 9 times.
✓ Branch 3 taken 547 times.
✓ Branch 4 taken 4373 times.
✓ Branch 5 taken 1539 times.
✗ Branch 6 not taken.
✓ Branch 7 taken 3 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 3 times.
✗ Branch 11 not taken.
✓ Branch 13 taken 2 times.
✗ Branch 14 not taken.
✓ Branch 16 taken 1 times.
✗ Branch 17 not taken.
✓ Branch 19 taken 1 times.
✗ Branch 20 not taken.
✓ Branch 22 taken 1 times.
✗ Branch 23 not taken.
✓ Branch 25 taken 1 times.
✗ Branch 26 not taken.
✓ Branch 28 taken 1 times.
✗ Branch 29 not taken.
✓ Branch 31 taken 1 times.
✗ Branch 32 not taken.
✓ Branch 34 taken 1 times.
✗ Branch 35 not taken.
✓ Branch 37 taken 1 times.
✗ Branch 38 not taken.
|
19866 | inline size_t errors() const { return mNumErrors; } |
110 | /// @brief Returns the number of warnings that have been encountered | ||
111 |
11/22✓ Branch 1 taken 4 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 3 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 3 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 2 times.
✗ Branch 11 not taken.
✓ Branch 13 taken 1 times.
✗ Branch 14 not taken.
✓ Branch 16 taken 1 times.
✗ Branch 17 not taken.
✓ Branch 19 taken 1 times.
✗ Branch 20 not taken.
✓ Branch 22 taken 1 times.
✗ Branch 23 not taken.
✓ Branch 25 taken 1 times.
✗ Branch 26 not taken.
✓ Branch 28 taken 1 times.
✗ Branch 29 not taken.
✓ Branch 31 taken 1 times.
✗ Branch 32 not taken.
|
1587 | inline size_t warnings() const { return mNumWarnings; } |
112 | |||
113 | /// @brief Returns true if an error has been found, false otherwise | ||
114 |
5/10✓ Branch 1 taken 282 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 1 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 1 times.
✗ Branch 11 not taken.
✓ Branch 13 taken 1 times.
✗ Branch 14 not taken.
|
286 | inline bool hasError() const { return this->errors() > 0; } |
115 | /// @brief Returns true if a warning has been found, false otherwise | ||
116 |
6/12✓ Branch 1 taken 1 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 1 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 1 times.
✗ Branch 11 not taken.
✓ Branch 13 taken 1 times.
✗ Branch 14 not taken.
✓ Branch 16 taken 1 times.
✗ Branch 17 not taken.
|
6 | inline bool hasWarning() const { return this->warnings() > 0; } |
117 | /// @brief Returns true if it has errored and the max errors has been hit | ||
118 | 2206 | inline bool atErrorLimit() const { | |
119 |
4/4✓ Branch 1 taken 896 times.
✓ Branch 2 taken 1310 times.
✓ Branch 4 taken 285 times.
✓ Branch 5 taken 611 times.
|
2206 | return this->getMaxErrors() > 0 && this->errors() >= this->getMaxErrors(); |
120 | } | ||
121 | |||
122 | /// @brief Clear the tree-code mapping and reset the number of errors/warnings | ||
123 | /// @note The tree-code mapping must be repopulated to retrieve line and | ||
124 | /// column numbers during AST traversal i.e. code generation. The | ||
125 | /// openvdb::ax::ast::parse() function does this for a given input code | ||
126 | /// string. | ||
127 | void clear(); | ||
128 | |||
129 | /// @brief Set any warnings that are encountered to be promoted to errors | ||
130 | /// @param warnAsError If true, warnings will be treated as errors | ||
131 | void setWarningsAsErrors(const bool warnAsError = false); | ||
132 | /// @brief Returns if warning are promoted to errors | ||
133 | bool getWarningsAsErrors() const; | ||
134 | |||
135 | /// @brief Sets the maximum number of errors that are allowed before | ||
136 | /// compilation should exit. | ||
137 | /// @note The logger will continue to increment the error counter beyond | ||
138 | /// this value but, once reached, it will not invoke the error callback. | ||
139 | /// @param maxErrors The number of allowed errors | ||
140 | void setMaxErrors(const size_t maxErrors = 0); | ||
141 | /// @brief Returns the number of allowed errors | ||
142 | size_t getMaxErrors() const; | ||
143 | |||
144 | /// Error/warning formatting options | ||
145 | |||
146 | /// @brief Set whether the output should number the errors/warnings | ||
147 | /// @param numbered If true, messages will be numbered | ||
148 | void setNumberedOutput(const bool numbered = true); | ||
149 | /// @brief Number of spaces to indent every new line before the message is formatted | ||
150 | void setIndent(const size_t ident = 0); | ||
151 | /// @brief Set a prefix for each warning message | ||
152 | void setErrorPrefix(const char* prefix = "error: "); | ||
153 | /// @brief Set a prefix for each warning message | ||
154 | void setWarningPrefix(const char* prefix = "warning: "); | ||
155 | /// @brief Set whether the output should include the offending line of code | ||
156 | /// @param print If true, offending lines of code will be appended to the | ||
157 | /// output message | ||
158 | void setPrintLines(const bool print = true); | ||
159 | |||
160 | /// @brief Returns whether the messages will be numbered | ||
161 | bool getNumberedOutput() const; | ||
162 | /// @brief Returns the number of spaces to be printed before every new line | ||
163 | size_t getIndent() const; | ||
164 | /// @brief Returns the prefix for each error message | ||
165 | const char* getErrorPrefix() const; | ||
166 | /// @brief Returns the prefix for each warning message | ||
167 | const char* getWarningPrefix() const; | ||
168 | /// @brief Returns whether the messages will include the line of offending code | ||
169 | bool getPrintLines() const; | ||
170 | |||
171 | /// @brief Set the source code that lines can be printed from if an error or | ||
172 | /// warning is raised | ||
173 | /// @param code The AX code as a c-style string | ||
174 | void setSourceCode(const char* code); | ||
175 | |||
176 | /// These functions are only to be used during parsing to allow line and | ||
177 | /// column number retrieval during later stages of compilation when working | ||
178 | /// solely with an AST | ||
179 | |||
180 | /// @brief Set the AST source tree which will be used as reference for the | ||
181 | /// locations of nodes when resolving line and column numbers during AST | ||
182 | /// traversal | ||
183 | /// @note To be used just by ax::parse before any AST modifications to | ||
184 | /// ensure traversal of original source tree is possible, when adding | ||
185 | /// messages using Node* which may correspond to modified trees | ||
186 | /// @param tree Pointer to const AST | ||
187 | void setSourceTree(openvdb::ax::ast::Tree::ConstPtr tree); | ||
188 | |||
189 | /// @brief Add a node to the code location map | ||
190 | /// @param node Pointer to AST node | ||
191 | /// @param location Line and column number in code | ||
192 | void addNodeLocation(const ax::ast::Node* node, const CodeLocation& location); | ||
193 | |||
194 | // forward declaration | ||
195 | struct Settings; | ||
196 | struct SourceCode; | ||
197 | |||
198 | private: | ||
199 | |||
200 | friend class ::TestLogger; | ||
201 | |||
202 | OutputFunction mErrorOutput; | ||
203 | OutputFunction mWarningOutput; | ||
204 | |||
205 | size_t mNumErrors; | ||
206 | size_t mNumWarnings; | ||
207 | |||
208 | std::unique_ptr<Settings> mSettings; | ||
209 | |||
210 | // components needed for verbose error info i.e. line/column numbers and | ||
211 | // lines from source code | ||
212 | std::unique_ptr<SourceCode> mCode; | ||
213 | ax::ast::Tree::ConstPtr mTreePtr; | ||
214 | std::unordered_map<const ax::ast::Node*, CodeLocation> mNodeToLineColMap; | ||
215 | }; | ||
216 | |||
217 | } // namespace ax | ||
218 | } // namespace OPENVDB_VERSION_NAME | ||
219 | } // namespace openvdb | ||
220 | |||
221 | #endif // OPENVDB_AX_COMPILER_LOGGER_HAS_BEEN_INCLUDED | ||
222 | |||
223 |