P4C
The P4 Compiler
Loading...
Searching...
No Matches
log.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_LOG_H_
18#define LIB_LOG_H_
19
20#include <functional>
21#include <iostream>
22#include <set>
23#include <sstream>
24#include <vector>
25
26#include "config.h"
27#include "indent.h"
28#include "stringify.h"
29
30#ifndef __GNUC__
31#define __attribute__(X)
32#endif
33
34namespace Log {
35namespace Detail {
36// The global verbosity level.
37extern int verbosity;
38
39// A cache of the maximum log level requested for any file.
40extern int maximumLogLevel;
41
42// Used to restrict logging to a specific IR context.
43extern bool enableLoggingGlobally;
44extern bool enableLoggingInContext; // if enableLoggingGlobally is true, this is ignored.
45
46// Look up the log level of @file.
47int fileLogLevel(const char *file);
48std::ostream &fileLogOutput(const char *file);
49
50// A utility class used to prepend file and log level information to logging output.
51// also controls indent control and locking for multithreaded use
53 const char *fn;
54 int level;
55 static int ostream_xalloc;
56 static void setup_ostream_xalloc(std::ostream &);
57 friend std::ostream &operator<<(std::ostream &, const OutputLogPrefix &);
58 friend std::ostream &clearPrefix(std::ostream &out);
59#ifdef MULTITHREAD
60 struct lock_t;
61 mutable lock_t *lock = nullptr;
62#endif // MULTITHREAD
63 public:
64 OutputLogPrefix(const char *f, int l) : fn(f), level(l) {}
66 static void indent(std::ostream &out);
67};
68
69void addInvalidateCallback(void (*)(void));
70std::ostream &clearPrefix(std::ostream &out);
71} // namespace Detail
72
73inline std::ostream &endl(std::ostream &out) {
74 out << std::endl;
75 Detail::OutputLogPrefix::indent(out);
76 return out;
77}
78using IndentCtl::indent;
80using IndentCtl::unindent;
81
82inline bool fileLogLevelIsAtLeast(const char *file, int level) {
83 // If there's no file with a log level of at least @level, we don't need to do
84 // the more expensive per-file check.
85 if (Detail::maximumLogLevel < level) {
86 return false;
87 }
88
89 return Detail::fileLogLevel(file) >= level;
90}
91
92// Process @spec and update the log level requested for the appropriate file.
93void addDebugSpec(const char *spec);
94
95inline bool verbose() { return Detail::verbosity > 0; }
96inline int verbosity() { return Detail::verbosity; }
97inline bool enableLogging() {
98 return Detail::enableLoggingGlobally || Detail::enableLoggingInContext;
99}
100void increaseVerbosity();
101
102} // namespace Log
103
104#ifndef MAX_LOGGING_LEVEL
105// can be set on build command line and disables higher logging levels at compile time
106#define MAX_LOGGING_LEVEL 10
107#endif
108
109// NOLINTBEGIN(bugprone-macro-parentheses)
110#define LOGGING(N) \
111 ((N) <= MAX_LOGGING_LEVEL && ::Log::fileLogLevelIsAtLeast(__FILE__, N) && \
112 ::Log::enableLogging())
113#define LOGN(N, X) \
114 (LOGGING(N) ? ::Log::Detail::fileLogOutput(__FILE__) \
115 << ::Log::Detail::OutputLogPrefix(__FILE__, N) << X \
116 << ::Log::Detail::clearPrefix << std::endl \
117 : std::clog)
118#define LOG1(X) LOGN(1, X)
119#define LOG2(X) LOGN(2, X)
120#define LOG3(X) LOGN(3, X)
121#define LOG4(X) LOGN(4, X)
122#define LOG5(X) LOGN(5, X)
123#define LOG6(X) LOGN(6, X)
124#define LOG7(X) LOGN(7, X)
125#define LOG8(X) LOGN(8, X)
126#define LOG9(X) LOGN(9, X)
127
128#define LOGN_UNINDENT(N) \
129 (LOGGING(N) ? ::Log::Detail::fileLogOutput(__FILE__) << IndentCtl::unindent : std::clog)
130#define LOG1_UNINDENT LOGN_UNINDENT(1)
131#define LOG2_UNINDENT LOGN_UNINDENT(2)
132#define LOG3_UNINDENT LOGN_UNINDENT(3)
133#define LOG4_UNINDENT LOGN_UNINDENT(4)
134#define LOG5_UNINDENT LOGN_UNINDENT(5)
135#define LOG6_UNINDENT LOGN_UNINDENT(6)
136#define LOG7_UNINDENT LOGN_UNINDENT(7)
137#define LOG8_UNINDENT LOGN_UNINDENT(8)
138#define LOG9_UNINDENT LOGN_UNINDENT(9)
139
140#define LOG_FEATURE(TAG, N, X) \
141 ((N) <= MAX_LOGGING_LEVEL && ::Log::fileLogLevelIsAtLeast(TAG, N) \
142 ? ::Log::Detail::fileLogOutput(TAG) \
143 << ::Log::Detail::OutputLogPrefix(TAG, N) << X << std::endl \
144 : std::clog)
145
146#define P4C_ERROR(X) (std::clog << "ERROR: " << X << std::endl)
147#define P4C_WARNING(X) (::Log::verbose() ? std::clog << "WARNING: " << X << std::endl : std::clog)
148#define ERRWARN(C, X) ((C) ? P4C_ERROR(X) : P4C_WARNING(X))
149// NOLINTEND(bugprone-macro-parentheses)
150
151static inline std::ostream &operator<<(std::ostream &out,
152 std::function<std::ostream &(std::ostream &)> fn) {
153 return fn(out);
154}
155
159template <class Cont>
160std::ostream &format_container(std::ostream &out, const Cont &container, char lbrace, char rbrace) {
161 std::vector<std::string> elems;
162 bool foundnl = false;
163 for (auto &el : container) {
164 std::stringstream tmp;
165 tmp << el;
166 elems.emplace_back(tmp.str());
167 if (!foundnl) foundnl = elems.back().find('\n') != std::string::npos;
168 }
169 if (foundnl) {
170 for (auto &el : elems) {
171 out << Log::endl << Log::indent;
172 for (auto &ch : el) {
173 if (ch == '\n')
174 out << Log::endl;
175 else
176 out << ch;
177 }
178 out << Log::unindent;
179 }
180 } else {
181 const char *sep = " ";
182 out << lbrace;
183 for (auto &el : elems) {
184 out << sep << el;
185 sep = ", ";
186 }
187 out << (sep + 1) << rbrace;
188 }
189
190 return out;
191}
192
193template <class T>
194std::ostream &operator<<(std::ostream &out, const std::vector<T> &vec) {
195 return format_container(out, vec, '[', ']');
196}
197
198template <class T>
199std::ostream &operator<<(std::ostream &out, const std::set<T> &set) {
200 return format_container(out, set, '(', ')');
201}
202
203#endif /* LIB_LOG_H_ */
Definition indent.h:89
Definition log.h:52