diff --git a/include/seastar/util/backtrace.hh b/include/seastar/util/backtrace.hh
index 5a4dbad54e..46355d3244 100644
--- a/include/seastar/util/backtrace.hh
+++ b/include/seastar/util/backtrace.hh
@@ -84,18 +84,18 @@ public:
using vector_type = boost::container::static_vector;
private:
vector_type _frames;
- size_t _hash;
- char _delimeter;
+ size_t _hash = 0;
private:
size_t calculate_hash() const noexcept;
public:
- simple_backtrace(vector_type f, char delimeter = ' ') noexcept : _frames(std::move(f)), _hash(calculate_hash()), _delimeter(delimeter) {}
- simple_backtrace(char delimeter = ' ') noexcept : simple_backtrace({}, delimeter) {}
+ simple_backtrace(vector_type f) noexcept : _frames(std::move(f)), _hash(calculate_hash()) {}
+ simple_backtrace() noexcept = default;
+ [[deprecated]] simple_backtrace(vector_type f, char) : simple_backtrace(std::move(f)) {}
+ [[deprecated]] simple_backtrace(char) {}
size_t hash() const noexcept { return _hash; }
- char delimeter() const noexcept { return _delimeter; }
- friend std::ostream& operator<<(std::ostream& out, const simple_backtrace&);
+ friend fmt::formatter;
bool operator==(const simple_backtrace& o) const noexcept {
return _hash == o._hash && _frames == o._frames;
@@ -116,7 +116,7 @@ public:
: _task_type(&ti)
{ }
- friend std::ostream& operator<<(std::ostream& out, const task_entry&);
+ friend fmt::formatter;
bool operator==(const task_entry& o) const noexcept {
return *_task_type == *o._task_type;
@@ -149,9 +149,8 @@ public:
~tasktrace();
size_t hash() const noexcept { return _hash; }
- char delimeter() const noexcept { return _main.delimeter(); }
- friend std::ostream& operator<<(std::ostream& out, const tasktrace&);
+ friend fmt::formatter;
bool operator==(const tasktrace& o) const noexcept;
@@ -182,10 +181,22 @@ struct hash {
}
-#if FMT_VERSION >= 90000
-template <> struct fmt::formatter : fmt::ostream_formatter {};
-template <> struct fmt::formatter : fmt::ostream_formatter {};
-#endif
+template <> struct fmt::formatter {
+ constexpr auto parse(format_parse_context& ctx) { return ctx.begin(); }
+ auto format(const seastar::frame&, fmt::format_context& ctx) const -> decltype(ctx.out());
+};
+template <> struct fmt::formatter {
+ constexpr auto parse(format_parse_context& ctx) { return ctx.begin(); }
+ auto format(const seastar::simple_backtrace&, fmt::format_context& ctx) const -> decltype(ctx.out());
+};
+template <> struct fmt::formatter {
+ constexpr auto parse(format_parse_context& ctx) { return ctx.begin(); }
+ auto format(const seastar::tasktrace&, fmt::format_context& ctx) const -> decltype(ctx.out());
+};
+template <> struct fmt::formatter {
+ constexpr auto parse(format_parse_context& ctx) { return ctx.begin(); }
+ auto format(const seastar::task_entry&, fmt::format_context& ctx) const -> decltype(ctx.out());
+};
namespace seastar {
diff --git a/src/util/backtrace.cc b/src/util/backtrace.cc
index cadf841e98..9b3a0bc944 100644
--- a/src/util/backtrace.cc
+++ b/src/util/backtrace.cc
@@ -33,6 +33,7 @@ module;
#include
#include
#include
+#include
#ifdef SEASTAR_MODULE
module seastar;
@@ -111,37 +112,23 @@ size_t simple_backtrace::calculate_hash() const noexcept {
}
std::ostream& operator<<(std::ostream& out, const frame& f) {
- if (!f.so->name.empty()) {
- out << f.so->name << "+";
- }
- out << format("0x{:x}", f.addr);
+ fmt::print(out, "{}", f);
return out;
}
std::ostream& operator<<(std::ostream& out, const simple_backtrace& b) {
- char delim[2] = {'\0', '\0'};
- for (auto f : b._frames) {
- out << delim << f;
- delim[0] = b.delimeter();
- }
+ fmt::print(out, "{}", b);
return out;
}
std::ostream& operator<<(std::ostream& out, const tasktrace& b) {
- out << b._main;
- for (auto&& e : b._prev) {
- out << "\n --------";
- std::visit(make_visitor([&] (const shared_backtrace& sb) {
- out << '\n' << sb;
- }, [&] (const task_entry& f) {
- out << "\n " << f;
- }), e);
- }
+ fmt::print(out, "{}", b);
return out;
}
std::ostream& operator<<(std::ostream& out, const task_entry& e) {
- return out << seastar::pretty_type_name(*e._task_type);
+ fmt::print(out, "{}", e);
+ return out;
}
tasktrace current_tasktrace() noexcept {
@@ -195,3 +182,43 @@ bool tasktrace::operator==(const tasktrace& o) const noexcept {
tasktrace::~tasktrace() {}
} // namespace seastar
+
+namespace fmt {
+
+auto formatter::format(const seastar::frame& f, format_context& ctx) const
+ -> decltype(ctx.out()) {
+ auto out = ctx.out();
+ if (!f.so->name.empty()) {
+ out = fmt::format_to(out, "{}+", f.so->name);
+ }
+ return fmt::format_to(out, "0x{:x}", f.addr);
+}
+
+auto formatter::format(const seastar::simple_backtrace& b, format_context& ctx) const
+ -> decltype(ctx.out()) {
+ return fmt::format_to(ctx.out(), "{}", fmt::join(b._frames, " "));
+}
+
+auto formatter::format(const seastar::tasktrace& b, format_context& ctx) const
+ -> decltype(ctx.out()) {
+ auto out = ctx.out();
+ out = fmt::format_to(out, "{}", b._main);
+ for (auto&& e : b._prev) {
+ out = fmt::format_to(out, "\n --------");
+ out = std::visit(seastar::make_visitor(
+ [&] (const seastar::shared_backtrace& sb) {
+ return fmt::format_to(out, "\n{}", sb);
+ },
+ [&] (const seastar::task_entry& f) {
+ return fmt::format_to(out, "\n {}", f);
+ }), e);
+ }
+ return out;
+}
+
+auto formatter::format(const seastar::task_entry& e, format_context& ctx) const
+ -> decltype(ctx.out()) {
+ return fmt::format_to(ctx.out(), "{}", seastar::pretty_type_name(*e._task_type));
+}
+
+}