P4C
The P4 Compiler
All Classes Namespaces Functions Variables Typedefs Enumerations Enumerator Friends Modules Pages
error_helper.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#ifndef LIB_ERROR_HELPER_H_
17#define LIB_ERROR_HELPER_H_
18
19#include <type_traits>
20
21#include <boost/format.hpp>
22
23#include "lib/error_message.h"
24#include "lib/source_file.h"
25#include "lib/stringify.h"
26
27namespace priv {
28
29// All these methods return std::string because this is the native format of boost::format
30// Position is printed at the beginning.
31static inline ErrorMessage error_helper(boost::format &f, ErrorMessage out) {
32 out.message = boost::str(f);
33 return out;
34}
35
36template <class... Args>
37auto error_helper(boost::format &f, ErrorMessage out, const char *t, Args &&...args) {
38 return error_helper(f % t, out, std::forward<Args>(args)...);
39}
40
41template <typename T, class... Args>
42auto error_helper(boost::format &f, ErrorMessage out, const T &t,
43 Args &&...args) -> std::enable_if_t<Util::has_toString_v<T>, ErrorMessage>;
44
45template <typename T, class... Args>
46auto error_helper(boost::format &f, ErrorMessage out, const T &t, Args &&...args)
47 -> std::enable_if_t<!Util::has_toString_v<T> && !std::is_pointer_v<T>, ErrorMessage>;
48
49template <typename T, class... Args>
50auto error_helper(boost::format &f, ErrorMessage out, const T *t, Args &&...args) {
51 // Contrary to bug_helper we do not want to show raw pointers to users in
52 // ordinary error messages. Therefore we explicitly delegate to
53 // reference-arg implementation here.
54 return error_helper(f, out, *t, std::forward<Args>(args)...);
55}
56
57template <class... Args>
58ErrorMessage error_helper(boost::format &f, ErrorMessage out, const Util::SourceInfo &info,
59 Args &&...args) {
60 if (info.isValid()) out.locations.push_back(info);
61 return error_helper(f % "", std::move(out), std::forward<Args>(args)...);
62}
63
64template <typename T>
65void maybeAddSourceInfo(ErrorMessage &out, const T &t) {
66 if constexpr (Util::has_SourceInfo_v<T>) {
67 auto info = t.getSourceInfo();
68 if (info.isValid()) out.locations.push_back(info);
69 }
70}
71
72template <typename T, class... Args>
73auto error_helper(boost::format &f, ErrorMessage out, const T &t, Args &&...args)
74 -> std::enable_if_t<!Util::has_toString_v<T> && !std::is_pointer_v<T>, ErrorMessage> {
75 maybeAddSourceInfo(out, t);
76 return error_helper(f % t, std::move(out), std::forward<Args>(args)...);
77}
78
79template <typename T, class... Args>
80auto error_helper(boost::format &f, ErrorMessage out, const T &t,
81 Args &&...args) -> std::enable_if_t<Util::has_toString_v<T>, ErrorMessage> {
82 maybeAddSourceInfo(out, t);
83 return error_helper(f % t.toString(), std::move(out), std::forward<Args>(args)...);
84}
85
86} // namespace priv
87
88// Most direct invocations of error_helper usually only reduce arguments
89template <class... Args>
90ErrorMessage error_helper(boost::format &f, Args &&...args) {
91 ErrorMessage msg;
92 return ::priv::error_helper(f, msg, std::forward<Args>(args)...);
93}
94
95// Invoked from ErrorReporter
96template <class... Args>
97ErrorMessage error_helper(boost::format &f, ErrorMessage msg, Args &&...args) {
98 return ::priv::error_helper(f, std::move(msg), std::forward<Args>(args)...);
99}
100
101// This overload exists for backwards compatibility
102template <class... Args>
103ErrorMessage error_helper(boost::format &f, const std::string &prefix, const Util::SourceInfo &info,
104 const std::string &suffix, Args &&...args) {
105 return ::priv::error_helper(f, ErrorMessage(prefix, info, suffix), std::forward<Args>(args)...);
106}
107
108#endif /* LIB_ERROR_HELPER_H_ */
Definition source_file.h:123
Definition error_message.h:36
std::vector< Util::SourceInfo > locations
Particular formatted message.
Definition error_message.h:42
std::string message
Typically error/warning type from catalog.
Definition error_message.h:41