P4C
The P4 Compiler
Loading...
Searching...
No Matches
error_reporter.h
1/*
2Copyright 2013-present Barefoot Networks, Inc.
3
4Licensed under the Apache License, Version 2.0 (the "License");
5you may not use this file except in compliance with the License.
6You may obtain a copy of the License at
7
8 http://www.apache.org/licenses/LICENSE-2.0
9
10Unless required by applicable law or agreed to in writing, software
11distributed under the License is distributed on an "AS IS" BASIS,
12WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13See the License for the specific language governing permissions and
14limitations under the License.
15*/
16
17#ifndef LIB_ERROR_REPORTER_H_
18#define LIB_ERROR_REPORTER_H_
19
20#include <iostream>
21#include <ostream>
22#include <set>
23#include <type_traits>
24#include <unordered_map>
25
26#include <boost/format.hpp>
27
28#include "absl/strings/str_format.h"
29#include "bug_helper.h"
30#include "error_catalog.h"
31#include "error_helper.h"
32#include "exceptions.h"
33
35enum class DiagnosticAction {
36 Ignore,
37 Info,
38 Warn,
39 Error
40};
41
42// Keeps track of compilation errors.
43// Errors are specified using the error() and warning() methods,
44// that use boost::format format strings, i.e.,
45// %1%, %2%, etc (starting at 1, not at 0).
46// Some compatibility for printf-style arguments is also supported.
48 protected:
49 unsigned int infoCount;
50 unsigned int warningCount;
51 unsigned int errorCount;
52 unsigned int maxErrorCount;
53
54 std::ostream *outputstream;
55
57 std::set<std::pair<int, const Util::SourceInfo>> errorTracker;
58
60 virtual void emit_message(const ErrorMessage &msg) {
61 *outputstream << msg.toString();
62 outputstream->flush();
63 }
64
65 virtual void emit_message(const ParserErrorMessage &msg) {
66 *outputstream << msg.toString();
67 outputstream->flush();
68 }
69
74 bool error_reported(int err, Util::SourceInfo source) {
75 if (!source.isValid()) return false;
76 auto p = errorTracker.emplace(err, source);
77 return !p.second; // if insertion took place, then we have not seen the error.
78 }
79
81 cstring get_error_name(int errorCode) { return ErrorCatalog::getCatalog().getName(errorCode); }
82
83 public:
85 : infoCount(0),
86 warningCount(0),
87 errorCount(0),
88 maxErrorCount(20),
89 defaultInfoDiagnosticAction(DiagnosticAction::Info),
90 defaultWarningDiagnosticAction(DiagnosticAction::Warn) {
91 outputstream = &std::cerr;
92 }
93
94 // error message for a bug
95 template <typename... Args>
96 std::string bug_message(const char *format, Args &&...args) {
97 boost::format fmt(format);
98 // FIXME: This will implicitly take location of the first argument having
99 // SourceInfo. Not sure if this always desireable or not.
100 return ::bug_helper(fmt, "", "", std::forward<Args>(args)...);
101 }
102
103 template <typename... Args>
104 std::string format_message(const char *format, Args &&...args) {
105 boost::format fmt(format);
106 return ::error_helper(fmt, std::forward<Args>(args)...).toString();
107 }
108
109 template <class T, typename = decltype(std::declval<T>()->getSourceInfo()), typename... Args>
110 void diagnose(DiagnosticAction action, const int errorCode, const char *format,
111 const char *suffix, T node, Args &&...args) {
112 if (!node || error_reported(errorCode, node->getSourceInfo())) return;
113
114 if (cstring name = get_error_name(errorCode))
115 diagnose(getDiagnosticAction(name, action), name, format, suffix, node,
116 std::forward<Args>(args)...);
117 else
118 diagnose(action, nullptr, format, suffix, node, std::forward<Args>(args)...);
119 }
120
121 template <typename... Args>
122 void diagnose(DiagnosticAction action, const int errorCode, const char *format,
123 const char *suffix, Args &&...args) {
124 if (cstring name = get_error_name(errorCode))
125 diagnose(getDiagnosticAction(name, action), name, format, suffix,
126 std::forward<Args>(args)...);
127 else
128 diagnose(action, nullptr, format, suffix, std::forward<Args>(args)...);
129 }
130
133 template <typename... Args>
134 void diagnose(DiagnosticAction action, const char *diagnosticName, const char *format,
135 const char *suffix, Args &&...args) {
136 if (action == DiagnosticAction::Ignore) return;
137
138 ErrorMessage::MessageType msgType = ErrorMessage::MessageType::None;
139 if (action == DiagnosticAction::Info) {
140 // Avoid burying errors in a pile of info messages:
141 // don't emit any more info messages if we've emitted errors.
142 if (errorCount > 0) return;
143
144 infoCount++;
145 msgType = ErrorMessage::MessageType::Info;
146 } else if (action == DiagnosticAction::Warn) {
147 // Avoid burying errors in a pile of warnings: don't emit any more warnings if we've
148 // emitted errors.
149 if (errorCount > 0) return;
150
151 warningCount++;
152 msgType = ErrorMessage::MessageType::Warning;
153 } else if (action == DiagnosticAction::Error) {
154 errorCount++;
155 msgType = ErrorMessage::MessageType::Error;
156 }
157
158 boost::format fmt(format);
159 ErrorMessage msg(msgType, diagnosticName ? diagnosticName : "", suffix);
160 msg = ::error_helper(fmt, msg, std::forward<Args>(args)...);
161 emit_message(msg);
162
163 if (errorCount > maxErrorCount)
164 FATAL_ERROR("Number of errors exceeded set maximum of %1%", maxErrorCount);
165 }
166
167 unsigned getErrorCount() const { return errorCount; }
168
169 unsigned getMaxErrorCount() const { return maxErrorCount; }
171 unsigned setMaxErrorCount(unsigned newMaxCount) {
172 auto r = maxErrorCount;
173 maxErrorCount = newMaxCount;
174 return r;
175 }
176
177 unsigned getWarningCount() const { return warningCount; }
178
179 unsigned getInfoCount() const { return infoCount; }
180
183 unsigned getDiagnosticCount() const { return errorCount + warningCount + infoCount; }
184
185 void setOutputStream(std::ostream *stream) { outputstream = stream; }
186
187 std::ostream *getOutputStream() const { return outputstream; }
188
191 template <typename T>
192 void parser_error(const Util::SourceInfo &location, const T &message) {
193 errorCount++;
194 std::stringstream ss;
195 ss << message;
196
197 emit_message(ParserErrorMessage(location, ss.str()));
198 }
199
206 template <typename... Args>
207 void parser_error(const Util::InputSources *sources, const char *fmt, Args &&...args) {
208 errorCount++;
209
210 Util::SourcePosition position = sources->getCurrentPosition();
211 position--;
212
213 // Unfortunately, we cannot go with statically checked format string
214 // here as it would require some changes to yyerror
215 std::string message;
216 if (!absl::FormatUntyped(&message, absl::UntypedFormatSpec(fmt),
217 {absl::FormatArg(args)...})) {
218 BUG("Failed to format string");
219 }
220
221 emit_message(ParserErrorMessage(Util::SourceInfo(sources, position), std::move(message)));
222 }
223
226 DiagnosticAction getDiagnosticAction(cstring diagnostic, DiagnosticAction defaultAction) {
227 // Actions for errors can never be overridden.
228 if (defaultAction == DiagnosticAction::Error) return defaultAction;
229 auto it = diagnosticActions.find(diagnostic);
230 if (it != diagnosticActions.end()) return it->second;
231 // if we're dealing with warnings and they have been globally modified
232 // (ignored or turned into errors), then return the global default
233 if (defaultAction == DiagnosticAction::Warn &&
234 defaultWarningDiagnosticAction != DiagnosticAction::Warn)
235 return defaultWarningDiagnosticAction;
236 return defaultAction;
237 }
238
240 void setDiagnosticAction(std::string_view diagnostic, DiagnosticAction action) {
241 diagnosticActions[cstring(diagnostic)] = action;
242 }
243
245 DiagnosticAction getDefaultWarningDiagnosticAction() { return defaultWarningDiagnosticAction; }
246
248 void setDefaultWarningDiagnosticAction(DiagnosticAction action) {
249 defaultWarningDiagnosticAction = action;
250 }
251
253 DiagnosticAction getDefaultInfoDiagnosticAction() { return defaultInfoDiagnosticAction; }
254
256 void setDefaultInfoDiagnosticAction(DiagnosticAction action) {
257 defaultInfoDiagnosticAction = action;
258 }
259
260 private:
262 DiagnosticAction defaultInfoDiagnosticAction;
263
265 DiagnosticAction defaultWarningDiagnosticAction;
266
268 std::unordered_map<cstring, DiagnosticAction> diagnosticActions;
269};
270
271#endif /* LIB_ERROR_REPORTER_H_ */
cstring getName(int errorCode)
retrieve the name for errorCode
Definition error_catalog.h:127
static ErrorCatalog & getCatalog()
Return the singleton object.
Definition error_catalog.h:101
Definition error_reporter.h:47
std::ostream * outputstream
the maximum number of errors that we print before fail
Definition error_reporter.h:54
void setDefaultWarningDiagnosticAction(DiagnosticAction action)
set the default diagnostic action for calls to warning().
Definition error_reporter.h:248
void setDefaultInfoDiagnosticAction(DiagnosticAction action)
set the default diagnostic action for calls to info().
Definition error_reporter.h:256
std::set< std::pair< int, const Util::SourceInfo > > errorTracker
Track errors or warnings that have already been issued for a particular source location.
Definition error_reporter.h:57
void setDiagnosticAction(std::string_view diagnostic, DiagnosticAction action)
Set the action to take for the given diagnostic.
Definition error_reporter.h:240
void diagnose(DiagnosticAction action, const char *diagnosticName, const char *format, const char *suffix, Args &&...args)
Definition error_reporter.h:134
bool error_reported(int err, Util::SourceInfo source)
Definition error_reporter.h:74
DiagnosticAction getDiagnosticAction(cstring diagnostic, DiagnosticAction defaultAction)
Definition error_reporter.h:226
void parser_error(const Util::InputSources *sources, const char *fmt, Args &&...args)
Definition error_reporter.h:207
DiagnosticAction getDefaultInfoDiagnosticAction()
Definition error_reporter.h:253
cstring get_error_name(int errorCode)
retrieve the format from the error catalog
Definition error_reporter.h:81
virtual void emit_message(const ErrorMessage &msg)
Output the message and flush the stream.
Definition error_reporter.h:60
void parser_error(const Util::SourceInfo &location, const T &message)
Definition error_reporter.h:192
unsigned setMaxErrorCount(unsigned newMaxCount)
set maxErrorCount to a the @newMaxCount threshold and return the previous value
Definition error_reporter.h:171
DiagnosticAction getDefaultWarningDiagnosticAction()
Definition error_reporter.h:245
unsigned getDiagnosticCount() const
Definition error_reporter.h:183
Definition source_file.h:271
Definition source_file.h:123
Definition source_file.h:53
Definition cstring.h:80
Definition error_message.h:36
Definition error_message.h:68