|  | // Copyright 2021 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #ifndef QUICHE_QUIC_CORE_QUIC_CONNECTION_CONTEXT_H_ | 
|  | #define QUICHE_QUIC_CORE_QUIC_CONNECTION_CONTEXT_H_ | 
|  |  | 
|  | #include "absl/strings/str_format.h" | 
|  | #include "absl/strings/string_view.h" | 
|  | #include "quic/platform/api/quic_export.h" | 
|  | #include "common/platform/api/quiche_logging.h" | 
|  |  | 
|  | namespace quic { | 
|  |  | 
|  | // QuicConnectionTracer is responsible for emit trace messages for a single | 
|  | // QuicConnection. | 
|  | // QuicConnectionTracer is part of the QuicConnectionContext. | 
|  | class QUIC_EXPORT_PRIVATE QuicConnectionTracer { | 
|  | public: | 
|  | virtual ~QuicConnectionTracer() = default; | 
|  |  | 
|  | // Emit a trace message from a string literal. The trace may simply remember | 
|  | // the address of the literal in this function and read it at a later time. | 
|  | virtual void PrintLiteral(const char* literal) = 0; | 
|  |  | 
|  | // Emit a trace message from a string_view. Unlike PrintLiteral, this function | 
|  | // will not read |s| after it returns. | 
|  | virtual void PrintString(absl::string_view s) = 0; | 
|  |  | 
|  | // Emit a trace message from printf-style arguments. | 
|  | template <typename... Args> | 
|  | void Printf(const absl::FormatSpec<Args...>& format, const Args&... args) { | 
|  | std::string s = absl::StrFormat(format, args...); | 
|  | PrintString(s); | 
|  | } | 
|  | }; | 
|  |  | 
|  | // QuicConnectionContext is a per-QuicConnection context that includes | 
|  | // facilities useable by any part of a QuicConnection. A QuicConnectionContext | 
|  | // is owned by a QuicConnection. | 
|  | // | 
|  | // The 'top-level' QuicConnection functions are responsible for maintaining the | 
|  | // thread-local QuicConnectionContext pointer, such that any function called by | 
|  | // them(directly or indirectly) can access the context. | 
|  | // | 
|  | // Like QuicConnection, all facilities in QuicConnectionContext are assumed to | 
|  | // be called from a single thread at a time, they are NOT thread-safe. | 
|  | struct QUIC_EXPORT_PRIVATE QuicConnectionContext final { | 
|  | // Get the context on the current executing thread. nullptr if the current | 
|  | // function is not called from a 'top-level' QuicConnection function. | 
|  | static QuicConnectionContext* Current(); | 
|  |  | 
|  | std::unique_ptr<QuicConnectionTracer> tracer; | 
|  | }; | 
|  |  | 
|  | // QuicConnectionContextSwitcher is a RAII object used for maintaining the | 
|  | // thread-local QuicConnectionContext pointer. | 
|  | class QUIC_EXPORT_PRIVATE QuicConnectionContextSwitcher final { | 
|  | public: | 
|  | // The constructor switches from QuicConnectionContext::Current() to | 
|  | // |new_context|. | 
|  | explicit QuicConnectionContextSwitcher(QuicConnectionContext* new_context); | 
|  |  | 
|  | // The destructor switches from QuicConnectionContext::Current() back to the | 
|  | // old context. | 
|  | ~QuicConnectionContextSwitcher(); | 
|  |  | 
|  | private: | 
|  | QuicConnectionContext* old_context_; | 
|  | }; | 
|  |  | 
|  | // Emit a trace message from a string literal to the current tracer(if any). | 
|  | inline void QUIC_TRACELITERAL(const char* literal) { | 
|  | QuicConnectionContext* current = QuicConnectionContext::Current(); | 
|  | if (current && current->tracer) { | 
|  | current->tracer->PrintLiteral(literal); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Emit a trace message from a string_view to the current tracer(if any). | 
|  | inline void QUIC_TRACESTRING(absl::string_view s) { | 
|  | QuicConnectionContext* current = QuicConnectionContext::Current(); | 
|  | if (current && current->tracer) { | 
|  | current->tracer->PrintString(s); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Emit a trace message from printf-style arguments to the current tracer(if | 
|  | // any). | 
|  | template <typename... Args> | 
|  | void QUIC_TRACEPRINTF(const absl::FormatSpec<Args...>& format, | 
|  | const Args&... args) { | 
|  | QuicConnectionContext* current = QuicConnectionContext::Current(); | 
|  | if (current && current->tracer) { | 
|  | current->tracer->Printf(format, args...); | 
|  | } | 
|  | } | 
|  |  | 
|  | }  // namespace quic | 
|  |  | 
|  | #endif  // QUICHE_QUIC_CORE_QUIC_CONNECTION_CONTEXT_H_ |