39#ifndef OR_TOOLS_BASE_DUMP_VARS_H_
40#define OR_TOOLS_BASE_DUMP_VARS_H_
49#include "absl/container/inlined_vector.h"
53#define DUMP_FOR_EACH_N0(F)
54#define DUMP_FOR_EACH_N1(F, a) F(a)
55#define DUMP_FOR_EACH_N2(F, a, ...) F(a) DUMP_FOR_EACH_N1(F, __VA_ARGS__)
56#define DUMP_FOR_EACH_N3(F, a, ...) F(a) DUMP_FOR_EACH_N2(F, __VA_ARGS__)
57#define DUMP_FOR_EACH_N4(F, a, ...) F(a) DUMP_FOR_EACH_N3(F, __VA_ARGS__)
58#define DUMP_FOR_EACH_N5(F, a, ...) F(a) DUMP_FOR_EACH_N4(F, __VA_ARGS__)
59#define DUMP_FOR_EACH_N6(F, a, ...) F(a) DUMP_FOR_EACH_N5(F, __VA_ARGS__)
60#define DUMP_FOR_EACH_N7(F, a, ...) F(a) DUMP_FOR_EACH_N6(F, __VA_ARGS__)
61#define DUMP_FOR_EACH_N8(F, a, ...) F(a) DUMP_FOR_EACH_N7(F, __VA_ARGS__)
62#define DUMP_FOR_EACH_N9(F, a, ...) F(a) DUMP_FOR_EACH_N8(F, __VA_ARGS__)
63#define DUMP_FOR_EACH_N10(F, a, ...) F(a) DUMP_FOR_EACH_N9(F, __VA_ARGS__)
64#define DUMP_FOR_EACH_N11(F, a, ...) F(a) DUMP_FOR_EACH_N10(F, __VA_ARGS__)
65#define DUMP_FOR_EACH_N12(F, a, ...) F(a) DUMP_FOR_EACH_N11(F, __VA_ARGS__)
67#define DUMP_CONCATENATE(x, y) x##y
68#define DUMP_FOR_EACH_(N, F, ...) \
69 DUMP_CONCATENATE(DUMP_FOR_EACH_N, N)(F __VA_OPT__(, __VA_ARGS__))
71#define DUMP_NARG(...) DUMP_NARG_(__VA_OPT__(__VA_ARGS__, ) DUMP_RSEQ_N())
72#define DUMP_NARG_(...) DUMP_ARG_N(__VA_ARGS__)
73#define DUMP_ARG_N(_1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, N, ...) N
74#define DUMP_RSEQ_N() 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0
75#define DUMP_FOR_EACH(F, ...) \
76 DUMP_FOR_EACH_(DUMP_NARG(__VA_ARGS__), F __VA_OPT__(, __VA_ARGS__))
78#define DUMP_VARS(...) DUMP_VARS_WITH_BINDINGS((), __VA_ARGS__)
81#define DUMP_STRINGIZE(a) #a,
82#define DUMP_STRINGIFY(...) DUMP_FOR_EACH(DUMP_STRINGIZE, __VA_ARGS__)
85#define DUMP_IDENTITY(...) __VA_ARGS__
87#define DUMP_RM_PARENS(...) DUMP_IDENTITY __VA_ARGS__
89#define DUMP_GEN_ONE_BINDING(a) , &a = a
90#define DUMP_GEN_BINDING(binding) \
91 &DUMP_FOR_EACH(DUMP_GEN_ONE_BINDING, DUMP_RM_PARENS(binding))
93#define DUMP_VARS_WITH_BINDINGS(binding, ...) \
94 ::operations_research::base::internal_dump_vars::make_dump_vars<>( \
95 ::operations_research::base::internal_dump_vars::DumpNames{ \
96 DUMP_STRINGIFY(__VA_ARGS__)}, \
97 [DUMP_GEN_BINDING(binding)]( \
98 ::std::ostream& os, const ::std::string& field_sep, \
99 const ::std::string& kv_sep, \
100 const ::operations_research::base::internal_dump_vars::DumpNames& \
102 ::operations_research::base::internal_dump_vars::print_fields{ \
104 .field_sep = field_sep, \
116 const ::absl::InlinedVector<T, 8>& vec) {
118 os << ::std::to_string(it) <<
',';
125std::ostream&
operator<<(std::ostream& os, const ::std::vector<T>& vec) {
127 os << ::std::to_string(it) <<
',';
133std::ostream&
operator<<(std::ostream& os, const ::std::optional<T>& opt) {
135 os << ::std::to_string(opt.value());
142template <
typename T,
typename U>
144 const ::util_intops::StrongVector<T, U>& vec) {
146 os << ::std::to_string(it) <<
',';
161 template <
class T1,
class T2,
class... Ts>
162 void operator()(
const T1& t1,
const T2& t2,
const Ts&... ts) {
177 explicit Dump(const ::std::string&& field_sep, const ::std::string&& kv_sep,
179 : field_sep_(::
std::move(field_sep)),
180 kv_sep_(::
std::move(kv_sep)),
181 names_(::
std::move(names)),
182 f_(::
std::move(f)) {}
184 ::std::string
str()
const {
185 ::std::ostringstream oss;
190 template <
class... N>
192 return Dump<F>(::std::string{field_sep_}, ::std::string{kv_sep_},
197 field_sep_ = ::std::move(field_sep);
201 Dump&
sep(::std::string&& field_sep, ::std::string&& kv_sep) {
202 field_sep_ = ::std::move(field_sep);
203 kv_sep_ = ::std::move(kv_sep);
208 dump.print_fields_(os);
213 void print_fields_(::std::ostream& os)
const {
214 f_(os, field_sep_, kv_sep_, names_);
217 ::std::string field_sep_;
218 ::std::string kv_sep_;
227 " = ", ::std::move(names), ::std::move(f));
::std::string str() const
Dump< F > as(N &&... names) const
friend::std::ostream & operator<<(::std::ostream &os, const Dump &dump)
Dump & sep(::std::string &&field_sep, ::std::string &&kv_sep)
Dump & sep(::std::string &&field_sep)
Dump(const ::std::string &&field_sep, const ::std::string &&kv_sep, DumpNames &&names, F f)
Dump< F > make_dump_vars(DumpNames &&names, F f)
std::ostream & operator<<(std::ostream &os, const ::absl::InlinedVector< T, 8 > &vec)
needed by routing
::std::vector<::std::string > DumpNames
const ::std::string & kv_sep
const ::std::string & field_sep
void operator()(const T1 &t1, const T2 &t2, const Ts &... ts)
void operator()(const T &t)