P4C
The P4 Compiler
Loading...
Searching...
No Matches
bug_helper.h
1/*
2Licensed under the Apache License, Version 2.0 (the "License");
3you may not use this file except in compliance with the License.
4You may obtain a copy of the License at
5
6 http://www.apache.org/licenses/LICENSE-2.0
7
8Unless required by applicable law or agreed to in writing, software
9distributed under the License is distributed on an "AS IS" BASIS,
10WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
11See the License for the specific language governing permissions and
12limitations under the License.
13*/
14
15/* -*-c++-*- */
16
17#ifndef LIB_BUG_HELPER_H_
18#define LIB_BUG_HELPER_H_
19
20#include <sstream>
21#include <string>
22#include <string_view>
23#include <utility>
24
25#include <boost/format.hpp>
26
27#include "absl/strings/str_cat.h"
28#include "cstring.h"
29#include "source_file.h"
30#include "stringify.h"
31
32namespace detail {
33
34static inline std::pair<std::string_view, std::string> getPositionTail(const Util::SourceInfo &info,
35 std::string_view position,
36 std::string_view tail) {
37 std::string_view posString = info.toPositionString().string_view();
38 std::string outTail(tail);
39 if (position.empty()) {
40 position = posString;
41 } else {
42 outTail.append(posString);
43 if (!posString.empty()) outTail.append("\n");
44 }
45 outTail += info.toSourceFragment();
46
47 return std::pair(position, outTail);
48}
49
50template <typename T>
51std::pair<std::string_view, std::string> maybeAddSourceInfo(const T &t, std::string_view position,
52 std::string_view tail) {
53 if constexpr (Util::has_SourceInfo_v<T>)
54 return getPositionTail(t.getSourceInfo(), position, tail);
55
56 (void)position;
57 (void)tail;
58 return {"", ""};
59}
60
61static inline std::string bug_helper(boost::format &f, std::string_view position,
62 std::string_view tail) {
63 return absl::StrCat(position, position.empty() ? "" : ": ", boost::str(f), "\n", tail);
64}
65
66template <typename T, class... Args>
67auto bug_helper(boost::format &f, std::string_view position, std::string_view tail, const T *t,
68 Args &&...args);
69
70template <typename T, class... Args>
71auto bug_helper(boost::format &f, std::string_view position, std::string_view tail, const T &t,
72 Args &&...args) -> std::enable_if_t<!std::is_pointer_v<T>, std::string>;
73
74template <class... Args>
75std::string bug_helper(boost::format &f, std::string_view position, std::string_view tail,
76 const char *t, Args &&...args) {
77 return bug_helper(f % t, position, tail, std::forward<Args>(args)...);
78}
79
80template <class... Args>
81std::string bug_helper(boost::format &f, std::string_view position, std::string_view tail,
82 const Util::SourceInfo &info, Args &&...args) {
83 auto [outPos, outTail] = detail::getPositionTail(info, position, tail);
84 return bug_helper(f % "", outPos, outTail, std::forward<Args>(args)...);
85}
86
87template <typename T, class... Args>
88auto bug_helper(boost::format &f, std::string_view position, std::string_view tail, const T *t,
89 Args &&...args) {
90 if (t == nullptr) return bug_helper(f, position, tail, std::forward<Args>(args)...);
91
92 auto [outPos, outTail] = maybeAddSourceInfo(*t, position, tail);
93 // We define operator<< for all T's that have either dbprint() or toString()
94 // methods (in stringify.h). Note, however, that these overloads are
95 // provided in the global namespace, not namespace of the T itself and
96 // therefore cannot be found via ADL. As a result, name lookup for the `str
97 // << t` below succeeds as it is encloding namespace for `detail`. However,
98 // `f % t` will fail as operator<< will be called from some
99 // `boost::io::detail` namespace for which global namespace is not a direct
100 // enclosing one and therefore the lookup for operator<< will fail.
101 std::stringstream str;
102 str << t;
103 return bug_helper(f % str.str(), outPos, outTail, std::forward<Args>(args)...);
104}
105
106template <typename T, class... Args>
107auto bug_helper(boost::format &f, std::string_view position, std::string_view tail, const T &t,
108 Args &&...args) -> std::enable_if_t<!std::is_pointer_v<T>, std::string> {
109 auto [outPos, outTail] = maybeAddSourceInfo(t, position, tail);
110 std::stringstream str;
111 str << t;
112 return bug_helper(f % str.str(), outPos, outTail, std::forward<Args>(args)...);
113}
114} // namespace detail
115
116// Most direct invocations of bug_helper usually only reduce arguments
117template <class... Args>
118std::string bug_helper(boost::format &f, std::string_view position, std::string_view tail,
119 Args &&...args) {
120 return detail::bug_helper(f, position, tail, std::forward<Args>(args)...);
121}
122
123#endif /* LIB_BUG_HELPER_H_ */
Definition source_file.h:123