Skip to content

Commit 4010ffb

Browse files
committed
Work around boost::format requiring deprecated allocate
Signed-off-by: Vladimir Still <[email protected]>
1 parent 8a59817 commit 4010ffb

File tree

10 files changed

+92
-46
lines changed

10 files changed

+92
-46
lines changed

backends/p4tools/common/lib/logging.h

+5-6
Original file line numberDiff line numberDiff line change
@@ -6,18 +6,17 @@
66
#include <string>
77
#include <utility>
88

9-
#include <boost/format.hpp>
10-
9+
#include "lib/boost_format_compat.h"
1110
#include "lib/log.h"
1211

1312
namespace P4::P4Tools {
1413

1514
/// Helper function for @printFeature
16-
inline std::string logHelper(boost::format &f) { return f.str(); }
15+
inline BoostFormatCompat &logHelper(BoostFormatCompat &f) { return f; }
1716

1817
/// Helper function for @printFeature
1918
template <class T, class... Args>
20-
std::string logHelper(boost::format &f, T &&t, Args &&...args) {
19+
BoostFormatCompat &logHelper(BoostFormatCompat &f, T &&t, Args &&...args) {
2120
return logHelper(f % std::forward<T>(t), std::forward<Args>(args)...);
2221
}
2322

@@ -32,8 +31,8 @@ void printFeature(const std::string &label, int level, const std::string &fmt,
3231
return;
3332
}
3433

35-
boost::format f(fmt);
36-
LOG_FEATURE(label.c_str(), level, logHelper(f, std::forward<Arguments>(args)...));
34+
BoostFormatCompat f(fmt.c_str());
35+
LOG_FEATURE(label.c_str(), level, logHelper(f, std::forward<Arguments>(args)...).str());
3736
}
3837

3938
/// Helper functions that prints strings associated with basic tool information.

frontends/p4/typeChecking/typeCheckTypes.cpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -331,13 +331,13 @@ static bool checkEnumValueInitializer(const IR::Type_Bits *type, const IR::Expre
331331
std::string extraMsg;
332332
if (!type->isSigned && constant->value < low) {
333333
extraMsg =
334-
str(boost::format(
334+
str(BoostFormatCompat(
335335
"the value %1% is negative, but the underlying type %2% is unsigned") %
336336
constant->value % type->toString());
337337
} else {
338338
extraMsg =
339-
str(boost::format("the value %1% requires %2% bits but the underlying "
340-
"%3% type %4% only contains %5% bits") %
339+
str(BoostFormatCompat("the value %1% requires %2% bits but the underlying "
340+
"%3% type %4% only contains %5% bits") %
341341
constant->value % required % (type->isSigned ? "signed" : "unsigned") %
342342
type->toString() % type->size);
343343
}

frontends/p4/typeChecking/typeConstraints.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -124,7 +124,7 @@ std::string TypeConstraint::localError(Explain *explainer) const {
124124
if (errFormat.isNullOrEmpty()) return "";
125125

126126
std::string message, explanation;
127-
boost::format fmt = boost::format(errFormat);
127+
BoostFormatCompat fmt(errFormat);
128128
switch (errArguments.size()) {
129129
case 0:
130130
message = boost::str(fmt);

frontends/p4/typeChecking/typeConstraints.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -92,7 +92,7 @@ class TypeConstraint : public IHasDbPrint, public ICastable {
9292
/// composed in reverse order, from bottom to top. The top of the stack
9393
/// has no 'derivedFrom' field, and it contains the actual source
9494
/// position where the analysis started.
95-
boost::format fmt(format);
95+
BoostFormatCompat fmt(format);
9696
return reportErrorImpl(subst,
9797
" ---- Actual error:\n" +
9898
::P4::error_helper(fmt, std::forward<Args>(args)...).toString());

frontends/parsers/parserDriver.cpp

+2-3
Original file line numberDiff line numberDiff line change
@@ -6,15 +6,14 @@
66
#include <sstream>
77
#include <string_view>
88

9-
#include <boost/format.hpp>
10-
119
#include "frontends/common/constantFolding.h"
1210
#include "frontends/common/options.h"
1311
#include "frontends/parsers/p4/p4AnnotationLexer.hpp"
1412
#include "frontends/parsers/p4/p4lexer.hpp"
1513
#include "frontends/parsers/p4/p4parser.hpp"
1614
#include "frontends/parsers/v1/v1lexer.hpp"
1715
#include "frontends/parsers/v1/v1parser.hpp"
16+
#include "lib/boost_format_compat.h"
1817
#include "lib/error.h"
1918

2019
#ifdef HAVE_LIBBOOST_IOSTREAMS
@@ -108,7 +107,7 @@ void AbstractParserDriver::onParseError(const Util::SourceInfo &location,
108107
auto &context = BaseCompileContext::get();
109108
if (message == unexpectedIdentifierError) {
110109
context.errorReporter().parser_error(
111-
location, boost::format("%s \"%s\"") % unexpectedIdentifierError % lastIdentifier);
110+
location, BoostFormatCompat("%s \"%s\"") % unexpectedIdentifierError % lastIdentifier);
112111
} else {
113112
context.errorReporter().parser_error(location, message);
114113
}

lib/boost_format_compat.h

+50
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
/*
2+
Licensed under the Apache License, Version 2.0 (the "License");
3+
you may not use this file except in compliance with the License.
4+
You may obtain a copy of the License at
5+
6+
http://www.apache.org/licenses/LICENSE-2.0
7+
8+
Unless required by applicable law or agreed to in writing, software
9+
distributed under the License is distributed on an "AS IS" BASIS,
10+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
11+
See the License for the specific language governing permissions and
12+
limitations under the License.
13+
*/
14+
15+
#ifndef LIB_BOOST_FORMAT_COMPAT_H_
16+
#define LIB_BOOST_FORMAT_COMPAT_H_
17+
18+
#include <boost/format.hpp>
19+
20+
namespace P4 {
21+
22+
/// Allocator wrapper that adds the overload removed in C++20 for the sake of Boost.
23+
template <typename T>
24+
struct BoostCompatAllocator : std::allocator<T> {
25+
/// Inherit constructors.
26+
using std::allocator<T>::allocator;
27+
28+
BoostCompatAllocator() = default;
29+
/// Explictly make it possible to construct our allocator from std::allocator.
30+
/// Some parts of libstdc++ 9 require that for some reason.
31+
BoostCompatAllocator(const std::allocator<T> &other) : std::allocator<T>(other) {}
32+
BoostCompatAllocator(std::allocator<T> &&other) : std::allocator<T>(std::move(other)) {}
33+
34+
/// Make sure the base class'es allocate is visible.
35+
using std::allocator<T>::allocate;
36+
37+
/// This one was removed in C++20, but boost::format < 1.74 needs it.
38+
T *allocate(typename std::allocator<T>::size_type n, const void *) { return allocate(n); }
39+
};
40+
41+
/// Boost < 1.74 is not compatible with C++20 as it expect deprecated
42+
/// allocator::allocate(size_type, void*) to exist. To work around that, we use a simple wrapper
43+
/// over std::allocator that adds this overload back. All uses of boost::format in P4C should use
44+
// this type instead.
45+
using BoostFormatCompat =
46+
boost::basic_format<char, std::char_traits<char>, BoostCompatAllocator<char>>;
47+
48+
} // namespace P4
49+
50+
#endif // LIB_BOOST_FORMAT_COMPAT_H_

lib/bug_helper.h

+9-10
Original file line numberDiff line numberDiff line change
@@ -23,9 +23,8 @@ limitations under the License.
2323
#include <string_view>
2424
#include <utility>
2525

26-
#include <boost/format.hpp>
27-
2826
#include "absl/strings/str_cat.h"
27+
#include "boost_format_compat.h"
2928
#include "cstring.h"
3029
#include "source_file.h"
3130
#include "stringify.h"
@@ -60,27 +59,27 @@ std::pair<std::string_view, std::string> maybeAddSourceInfo(const T &t, std::str
6059
return {"", ""};
6160
}
6261

63-
static inline std::string bug_helper(boost::format &f, std::string_view position,
62+
static inline std::string bug_helper(BoostFormatCompat &f, std::string_view position,
6463
std::string_view tail) {
6564
return absl::StrCat(position, position.empty() ? "" : ": ", boost::str(f), "\n", tail);
6665
}
6766

6867
template <typename T, class... Args>
69-
auto bug_helper(boost::format &f, std::string_view position, std::string_view tail, const T *t,
68+
auto bug_helper(BoostFormatCompat &f, std::string_view position, std::string_view tail, const T *t,
7069
Args &&...args);
7170

7271
template <typename T, class... Args>
73-
auto bug_helper(boost::format &f, std::string_view position, std::string_view tail, const T &t,
72+
auto bug_helper(BoostFormatCompat &f, std::string_view position, std::string_view tail, const T &t,
7473
Args &&...args) -> std::enable_if_t<!std::is_pointer_v<T>, std::string>;
7574

7675
template <class... Args>
77-
std::string bug_helper(boost::format &f, std::string_view position, std::string_view tail,
76+
std::string bug_helper(BoostFormatCompat &f, std::string_view position, std::string_view tail,
7877
const char *t, Args &&...args) {
7978
return bug_helper(f % t, position, tail, std::forward<Args>(args)...);
8079
}
8180

8281
template <class... Args>
83-
std::string bug_helper(boost::format &f, std::string_view position, std::string_view tail,
82+
std::string bug_helper(BoostFormatCompat &f, std::string_view position, std::string_view tail,
8483
const Util::SourceInfo &info, Args &&...args) {
8584
auto [outPos, outTail] = detail::getPositionTail(info, position, tail);
8685
return bug_helper(f % "", outPos, outTail, std::forward<Args>(args)...);
@@ -105,7 +104,7 @@ std::ostream &operator<<(std::ostream &os, const DbprintDispatchPtr<T> &dispatch
105104
}
106105

107106
template <typename T, class... Args>
108-
auto bug_helper(boost::format &f, std::string_view position, std::string_view tail, const T *t,
107+
auto bug_helper(BoostFormatCompat &f, std::string_view position, std::string_view tail, const T *t,
109108
Args &&...args) {
110109
if (t == nullptr) return bug_helper(f, position, tail, std::forward<Args>(args)...);
111110

@@ -132,7 +131,7 @@ std::ostream &operator<<(std::ostream &os, const DbprintDispatchRef<T> &dispatch
132131
}
133132

134133
template <typename T, class... Args>
135-
auto bug_helper(boost::format &f, std::string_view position, std::string_view tail, const T &t,
134+
auto bug_helper(BoostFormatCompat &f, std::string_view position, std::string_view tail, const T &t,
136135
Args &&...args) -> std::enable_if_t<!std::is_pointer_v<T>, std::string> {
137136
auto [outPos, outTail] = maybeAddSourceInfo(t, position, tail);
138137
return bug_helper(f % DbprintDispatchRef<T>{t}, outPos, outTail, std::forward<Args>(args)...);
@@ -141,7 +140,7 @@ auto bug_helper(boost::format &f, std::string_view position, std::string_view ta
141140

142141
// Most direct invocations of bug_helper usually only reduce arguments
143142
template <class... Args>
144-
std::string bug_helper(boost::format &f, std::string_view position, std::string_view tail,
143+
std::string bug_helper(BoostFormatCompat &f, std::string_view position, std::string_view tail,
145144
Args &&...args) {
146145
return detail::bug_helper(f, position, tail, std::forward<Args>(args)...);
147146
}

lib/error_helper.h

+16-16
Original file line numberDiff line numberDiff line change
@@ -16,9 +16,9 @@ limitations under the License.
1616
#ifndef LIB_ERROR_HELPER_H_
1717
#define LIB_ERROR_HELPER_H_
1818

19-
#include <type_traits>
19+
#include <lib/boost_format_compat.h>
2020

21-
#include <boost/format.hpp>
21+
#include <type_traits>
2222

2323
#include "lib/error_message.h"
2424
#include "lib/source_file.h"
@@ -29,34 +29,34 @@ namespace priv {
2929

3030
// All these methods return std::string because this is the native format of boost::format
3131
// Position is printed at the beginning.
32-
static inline ErrorMessage error_helper(boost::format &f, ErrorMessage out) {
32+
static inline ErrorMessage error_helper(BoostFormatCompat &f, ErrorMessage out) {
3333
out.message = boost::str(f);
3434
return out;
3535
}
3636

3737
template <class... Args>
38-
auto error_helper(boost::format &f, ErrorMessage out, const char *t, Args &&...args) {
38+
auto error_helper(BoostFormatCompat &f, ErrorMessage out, const char *t, Args &&...args) {
3939
return error_helper(f % t, out, std::forward<Args>(args)...);
4040
}
4141

4242
template <typename T, class... Args>
43-
auto error_helper(boost::format &f, ErrorMessage out, const T &t,
44-
Args &&...args) -> std::enable_if_t<Util::has_toString_v<T>, ErrorMessage>;
43+
auto error_helper(BoostFormatCompat &f, ErrorMessage out, const T &t, Args &&...args)
44+
-> std::enable_if_t<Util::has_toString_v<T>, ErrorMessage>;
4545

4646
template <typename T, class... Args>
47-
auto error_helper(boost::format &f, ErrorMessage out, const T &t, Args &&...args)
47+
auto error_helper(BoostFormatCompat &f, ErrorMessage out, const T &t, Args &&...args)
4848
-> std::enable_if_t<!Util::has_toString_v<T> && !std::is_pointer_v<T>, ErrorMessage>;
4949

5050
template <typename T, class... Args>
51-
auto error_helper(boost::format &f, ErrorMessage out, const T *t, Args &&...args) {
51+
auto error_helper(BoostFormatCompat &f, ErrorMessage out, const T *t, Args &&...args) {
5252
// Contrary to bug_helper we do not want to show raw pointers to users in
5353
// ordinary error messages. Therefore we explicitly delegate to
5454
// reference-arg implementation here.
5555
return error_helper(f, out, *t, std::forward<Args>(args)...);
5656
}
5757

5858
template <class... Args>
59-
ErrorMessage error_helper(boost::format &f, ErrorMessage out, const Util::SourceInfo &info,
59+
ErrorMessage error_helper(BoostFormatCompat &f, ErrorMessage out, const Util::SourceInfo &info,
6060
Args &&...args) {
6161
if (info.isValid()) out.locations.push_back(info);
6262
return error_helper(f % "", std::move(out), std::forward<Args>(args)...);
@@ -71,15 +71,15 @@ void maybeAddSourceInfo(ErrorMessage &out, const T &t) {
7171
}
7272

7373
template <typename T, class... Args>
74-
auto error_helper(boost::format &f, ErrorMessage out, const T &t, Args &&...args)
74+
auto error_helper(BoostFormatCompat &f, ErrorMessage out, const T &t, Args &&...args)
7575
-> std::enable_if_t<!Util::has_toString_v<T> && !std::is_pointer_v<T>, ErrorMessage> {
7676
maybeAddSourceInfo(out, t);
7777
return error_helper(f % t, std::move(out), std::forward<Args>(args)...);
7878
}
7979

8080
template <typename T, class... Args>
81-
auto error_helper(boost::format &f, ErrorMessage out, const T &t,
82-
Args &&...args) -> std::enable_if_t<Util::has_toString_v<T>, ErrorMessage> {
81+
auto error_helper(BoostFormatCompat &f, ErrorMessage out, const T &t, Args &&...args)
82+
-> std::enable_if_t<Util::has_toString_v<T>, ErrorMessage> {
8383
maybeAddSourceInfo(out, t);
8484
return error_helper(f % t.toString(), std::move(out), std::forward<Args>(args)...);
8585
}
@@ -88,21 +88,21 @@ auto error_helper(boost::format &f, ErrorMessage out, const T &t,
8888

8989
// Most direct invocations of error_helper usually only reduce arguments
9090
template <class... Args>
91-
ErrorMessage error_helper(boost::format &f, Args &&...args) {
91+
ErrorMessage error_helper(BoostFormatCompat &f, Args &&...args) {
9292
ErrorMessage msg;
9393
return priv::error_helper(f, msg, std::forward<Args>(args)...);
9494
}
9595

9696
// Invoked from ErrorReporter
9797
template <class... Args>
98-
ErrorMessage error_helper(boost::format &f, ErrorMessage msg, Args &&...args) {
98+
ErrorMessage error_helper(BoostFormatCompat &f, ErrorMessage msg, Args &&...args) {
9999
return priv::error_helper(f, std::move(msg), std::forward<Args>(args)...);
100100
}
101101

102102
// This overload exists for backwards compatibility
103103
template <class... Args>
104-
ErrorMessage error_helper(boost::format &f, const std::string &prefix, const Util::SourceInfo &info,
105-
const std::string &suffix, Args &&...args) {
104+
ErrorMessage error_helper(BoostFormatCompat &f, const std::string &prefix,
105+
const Util::SourceInfo &info, const std::string &suffix, Args &&...args) {
106106
return priv::error_helper(f, ErrorMessage(prefix, info, suffix), std::forward<Args>(args)...);
107107
}
108108

lib/error_reporter.h

+4-5
Original file line numberDiff line numberDiff line change
@@ -23,9 +23,8 @@ limitations under the License.
2323
#include <type_traits>
2424
#include <unordered_map>
2525

26-
#include <boost/format.hpp>
27-
2826
#include "absl/strings/str_format.h"
27+
#include "boost_format_compat.h"
2928
#include "bug_helper.h"
3029
#include "error_catalog.h"
3130
#include "error_helper.h"
@@ -96,15 +95,15 @@ class ErrorReporter {
9695
// error message for a bug
9796
template <typename... Args>
9897
std::string bug_message(const char *format, Args &&...args) {
99-
boost::format fmt(format);
98+
BoostFormatCompat fmt(format);
10099
// FIXME: This will implicitly take location of the first argument having
101100
// SourceInfo. Not sure if this always desireable or not.
102101
return ::P4::bug_helper(fmt, "", "", std::forward<Args>(args)...);
103102
}
104103

105104
template <typename... Args>
106105
std::string format_message(const char *format, Args &&...args) {
107-
boost::format fmt(format);
106+
BoostFormatCompat fmt(format);
108107
return ::P4::error_helper(fmt, std::forward<Args>(args)...).toString();
109108
}
110109

@@ -157,7 +156,7 @@ class ErrorReporter {
157156
msgType = ErrorMessage::MessageType::Error;
158157
}
159158

160-
boost::format fmt(format);
159+
BoostFormatCompat fmt(format);
161160
ErrorMessage msg(msgType, diagnosticName ? diagnosticName : "", suffix);
162161
msg = ::P4::error_helper(fmt, msg, std::forward<Args>(args)...);
163162
emit_message(msg);

lib/exceptions.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,7 @@ class P4CExceptionBase : public std::exception {
7676
template <typename... Args>
7777
explicit P4CExceptionBase(const char *format, Args &&...args) {
7878
traceCreation();
79-
boost::format fmt(format);
79+
BoostFormatCompat fmt(format);
8080
// FIXME: This will implicitly take location of the first argument having
8181
// SourceInfo. Not sure if this always desireable or not.
8282
message = ::P4::bug_helper(fmt, "", "", std::forward<Args>(args)...);

0 commit comments

Comments
 (0)