-- f34cd235a12ad0ee1fea3a1ee5a427272dc2b285 by Abseil Team <absl-team@google.com>: Migrates uses of deprecated map types to recommended types. PiperOrigin-RevId: 309945156 -- e3410a47ad32c0775b6911610bc47b22938decad by Matthew Brown <matthewbr@google.com>: Internal Change PiperOrigin-RevId: 309856021 -- a58cfa25e0bb59e7fa9647ac1aae65eaccff0086 by Greg Falcon <gfalcon@google.com>: Internal change. PiperOrigin-RevId: 309804612 -- cdc5ec310035fbe25f496bda283fe655d94d7769 by Mark Barolak <mbar@google.com>: Standardize the header comments for friend functions in cord.h PiperOrigin-RevId: 309779073 -- fe61602701be795e54477b0fdbf5ffc1df12a6b7 by Samuel Benzaquen <sbenza@google.com>: Implement %f natively for any input. It evaluates the input at runtime and allocates stack space accordingly. This removes a potential fallback into snprintf, improves performance, and removes all memory allocations in this formatting path. PiperOrigin-RevId: 309752501 -- 79e2a24f3f959e8b06ddf1d440bbabbd5f89b5b7 by Greg Falcon <gfalcon@google.com>: Add a Cord::swap() method. Many other Abseil types already provide this, but it was missing here. We already provided a two-argument free function form of `swap()`, but that API is better suited for generic code. The swap member function is a better API when the types are known. PiperOrigin-RevId: 309751740 -- 85cdf60024f153fb4fcb7fe68ed2b14b9faf119d by Derek Mauro <dmauro@google.com>: Cleanup uses of "linker initialized" SpinLocks PiperOrigin-RevId: 309581867 -- 9e5443bfcec4b94056b13c75326576e987ab88fb by Matt Kulukundis <kfm@google.com>: Clarify intended mixing properties of `absl::Hash` PiperOrigin-RevId: 309520174 -- a0630f0827b67f217aaeae68a448fe4c1101e17d by Greg Falcon <gfalcon@google.com>: Comment out a test in Emscripten to sidestep `long double` issues. PiperOrigin-RevId: 309482953 GitOrigin-RevId: f34cd235a12ad0ee1fea3a1ee5a427272dc2b285 Change-Id: Icce0c9d547117374d596b9d684e4054ddd118669
		
			
				
	
	
		
			463 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			463 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
#ifndef ABSL_STRINGS_INTERNAL_STR_FORMAT_ARG_H_
 | 
						|
#define ABSL_STRINGS_INTERNAL_STR_FORMAT_ARG_H_
 | 
						|
 | 
						|
#include <string.h>
 | 
						|
#include <wchar.h>
 | 
						|
 | 
						|
#include <cstdio>
 | 
						|
#include <iomanip>
 | 
						|
#include <limits>
 | 
						|
#include <memory>
 | 
						|
#include <sstream>
 | 
						|
#include <string>
 | 
						|
#include <type_traits>
 | 
						|
 | 
						|
#include "absl/base/port.h"
 | 
						|
#include "absl/meta/type_traits.h"
 | 
						|
#include "absl/numeric/int128.h"
 | 
						|
#include "absl/strings/internal/str_format/extension.h"
 | 
						|
#include "absl/strings/string_view.h"
 | 
						|
 | 
						|
namespace absl {
 | 
						|
ABSL_NAMESPACE_BEGIN
 | 
						|
 | 
						|
class Cord;
 | 
						|
class FormatCountCapture;
 | 
						|
class FormatSink;
 | 
						|
 | 
						|
namespace str_format_internal {
 | 
						|
 | 
						|
template <typename T, typename = void>
 | 
						|
struct HasUserDefinedConvert : std::false_type {};
 | 
						|
 | 
						|
template <typename T>
 | 
						|
struct HasUserDefinedConvert<T, void_t<decltype(AbslFormatConvert(
 | 
						|
                                    std::declval<const T&>(),
 | 
						|
                                    std::declval<const FormatConversionSpec&>(),
 | 
						|
                                    std::declval<FormatSink*>()))>>
 | 
						|
    : std::true_type {};
 | 
						|
 | 
						|
template <typename T>
 | 
						|
class StreamedWrapper;
 | 
						|
 | 
						|
// If 'v' can be converted (in the printf sense) according to 'conv',
 | 
						|
// then convert it, appending to `sink` and return `true`.
 | 
						|
// Otherwise fail and return `false`.
 | 
						|
 | 
						|
// Raw pointers.
 | 
						|
struct VoidPtr {
 | 
						|
  VoidPtr() = default;
 | 
						|
  template <typename T,
 | 
						|
            decltype(reinterpret_cast<uintptr_t>(std::declval<T*>())) = 0>
 | 
						|
  VoidPtr(T* ptr)  // NOLINT
 | 
						|
      : value(ptr ? reinterpret_cast<uintptr_t>(ptr) : 0) {}
 | 
						|
  uintptr_t value;
 | 
						|
};
 | 
						|
 | 
						|
template <FormatConversionCharSet C>
 | 
						|
struct ArgConvertResult {
 | 
						|
  bool value;
 | 
						|
};
 | 
						|
 | 
						|
template <FormatConversionCharSet C>
 | 
						|
constexpr FormatConversionCharSet ExtractCharSet(ArgConvertResult<C>) {
 | 
						|
  return C;
 | 
						|
}
 | 
						|
 | 
						|
using StringConvertResult =
 | 
						|
    ArgConvertResult<FormatConversionCharSetInternal::s>;
 | 
						|
ArgConvertResult<FormatConversionCharSetInternal::p> FormatConvertImpl(
 | 
						|
    VoidPtr v, FormatConversionSpecImpl conv, FormatSinkImpl* sink);
 | 
						|
 | 
						|
// Strings.
 | 
						|
StringConvertResult FormatConvertImpl(const std::string& v,
 | 
						|
                                      FormatConversionSpecImpl conv,
 | 
						|
                                      FormatSinkImpl* sink);
 | 
						|
StringConvertResult FormatConvertImpl(string_view v,
 | 
						|
                                      FormatConversionSpecImpl conv,
 | 
						|
                                      FormatSinkImpl* sink);
 | 
						|
ArgConvertResult<FormatConversionCharSetUnion(
 | 
						|
    FormatConversionCharSetInternal::s, FormatConversionCharSetInternal::p)>
 | 
						|
FormatConvertImpl(const char* v, const FormatConversionSpecImpl conv,
 | 
						|
                  FormatSinkImpl* sink);
 | 
						|
 | 
						|
template <class AbslCord, typename std::enable_if<std::is_same<
 | 
						|
                              AbslCord, absl::Cord>::value>::type* = nullptr>
 | 
						|
StringConvertResult FormatConvertImpl(const AbslCord& value,
 | 
						|
                                      FormatConversionSpecImpl conv,
 | 
						|
                                      FormatSinkImpl* sink) {
 | 
						|
  if (conv.conversion_char() != FormatConversionCharInternal::s) {
 | 
						|
    return {false};
 | 
						|
  }
 | 
						|
 | 
						|
  bool is_left = conv.has_left_flag();
 | 
						|
  size_t space_remaining = 0;
 | 
						|
 | 
						|
  int width = conv.width();
 | 
						|
  if (width >= 0) space_remaining = width;
 | 
						|
 | 
						|
  size_t to_write = value.size();
 | 
						|
 | 
						|
  int precision = conv.precision();
 | 
						|
  if (precision >= 0)
 | 
						|
    to_write = (std::min)(to_write, static_cast<size_t>(precision));
 | 
						|
 | 
						|
  space_remaining = Excess(to_write, space_remaining);
 | 
						|
 | 
						|
  if (space_remaining > 0 && !is_left) sink->Append(space_remaining, ' ');
 | 
						|
 | 
						|
  for (string_view piece : value.Chunks()) {
 | 
						|
    if (piece.size() > to_write) {
 | 
						|
      piece.remove_suffix(piece.size() - to_write);
 | 
						|
      to_write = 0;
 | 
						|
    } else {
 | 
						|
      to_write -= piece.size();
 | 
						|
    }
 | 
						|
    sink->Append(piece);
 | 
						|
    if (to_write == 0) {
 | 
						|
      break;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (space_remaining > 0 && is_left) sink->Append(space_remaining, ' ');
 | 
						|
  return {true};
 | 
						|
}
 | 
						|
 | 
						|
using IntegralConvertResult = ArgConvertResult<FormatConversionCharSetUnion(
 | 
						|
    FormatConversionCharSetInternal::c,
 | 
						|
    FormatConversionCharSetInternal::kNumeric,
 | 
						|
    FormatConversionCharSetInternal::kStar)>;
 | 
						|
using FloatingConvertResult =
 | 
						|
    ArgConvertResult<FormatConversionCharSetInternal::kFloating>;
 | 
						|
 | 
						|
// Floats.
 | 
						|
FloatingConvertResult FormatConvertImpl(float v, FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
FloatingConvertResult FormatConvertImpl(double v, FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
FloatingConvertResult FormatConvertImpl(long double v,
 | 
						|
                                        FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
 | 
						|
// Chars.
 | 
						|
IntegralConvertResult FormatConvertImpl(char v, FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
IntegralConvertResult FormatConvertImpl(signed char v,
 | 
						|
                                        FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
IntegralConvertResult FormatConvertImpl(unsigned char v,
 | 
						|
                                        FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
 | 
						|
// Ints.
 | 
						|
IntegralConvertResult FormatConvertImpl(short v,  // NOLINT
 | 
						|
                                        FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
IntegralConvertResult FormatConvertImpl(unsigned short v,  // NOLINT
 | 
						|
                                        FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
IntegralConvertResult FormatConvertImpl(int v, FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
IntegralConvertResult FormatConvertImpl(unsigned v,
 | 
						|
                                        FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
IntegralConvertResult FormatConvertImpl(long v,  // NOLINT
 | 
						|
                                        FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
IntegralConvertResult FormatConvertImpl(unsigned long v,  // NOLINT
 | 
						|
                                        FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
IntegralConvertResult FormatConvertImpl(long long v,  // NOLINT
 | 
						|
                                        FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
IntegralConvertResult FormatConvertImpl(unsigned long long v,  // NOLINT
 | 
						|
                                        FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
IntegralConvertResult FormatConvertImpl(int128 v, FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
IntegralConvertResult FormatConvertImpl(uint128 v,
 | 
						|
                                        FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink);
 | 
						|
template <typename T, enable_if_t<std::is_same<T, bool>::value, int> = 0>
 | 
						|
IntegralConvertResult FormatConvertImpl(T v, FormatConversionSpecImpl conv,
 | 
						|
                                        FormatSinkImpl* sink) {
 | 
						|
  return FormatConvertImpl(static_cast<int>(v), conv, sink);
 | 
						|
}
 | 
						|
 | 
						|
// We provide this function to help the checker, but it is never defined.
 | 
						|
// FormatArgImpl will use the underlying Convert functions instead.
 | 
						|
template <typename T>
 | 
						|
typename std::enable_if<std::is_enum<T>::value &&
 | 
						|
                            !HasUserDefinedConvert<T>::value,
 | 
						|
                        IntegralConvertResult>::type
 | 
						|
FormatConvertImpl(T v, FormatConversionSpecImpl conv, FormatSinkImpl* sink);
 | 
						|
 | 
						|
template <typename T>
 | 
						|
StringConvertResult FormatConvertImpl(const StreamedWrapper<T>& v,
 | 
						|
                                      FormatConversionSpecImpl conv,
 | 
						|
                                      FormatSinkImpl* out) {
 | 
						|
  std::ostringstream oss;
 | 
						|
  oss << v.v_;
 | 
						|
  if (!oss) return {false};
 | 
						|
  return str_format_internal::FormatConvertImpl(oss.str(), conv, out);
 | 
						|
}
 | 
						|
 | 
						|
// Use templates and dependent types to delay evaluation of the function
 | 
						|
// until after FormatCountCapture is fully defined.
 | 
						|
struct FormatCountCaptureHelper {
 | 
						|
  template <class T = int>
 | 
						|
  static ArgConvertResult<FormatConversionCharSetInternal::n> ConvertHelper(
 | 
						|
      const FormatCountCapture& v, FormatConversionSpecImpl conv,
 | 
						|
      FormatSinkImpl* sink) {
 | 
						|
    const absl::enable_if_t<sizeof(T) != 0, FormatCountCapture>& v2 = v;
 | 
						|
 | 
						|
    if (conv.conversion_char() !=
 | 
						|
        str_format_internal::FormatConversionCharInternal::n) {
 | 
						|
      return {false};
 | 
						|
    }
 | 
						|
    *v2.p_ = static_cast<int>(sink->size());
 | 
						|
    return {true};
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
template <class T = int>
 | 
						|
ArgConvertResult<FormatConversionCharSetInternal::n> FormatConvertImpl(
 | 
						|
    const FormatCountCapture& v, FormatConversionSpecImpl conv,
 | 
						|
    FormatSinkImpl* sink) {
 | 
						|
  return FormatCountCaptureHelper::ConvertHelper(v, conv, sink);
 | 
						|
}
 | 
						|
 | 
						|
// Helper friend struct to hide implementation details from the public API of
 | 
						|
// FormatArgImpl.
 | 
						|
struct FormatArgImplFriend {
 | 
						|
  template <typename Arg>
 | 
						|
  static bool ToInt(Arg arg, int* out) {
 | 
						|
    // A value initialized FormatConversionSpecImpl has a `none` conv, which
 | 
						|
    // tells the dispatcher to run the `int` conversion.
 | 
						|
    return arg.dispatcher_(arg.data_, {}, out);
 | 
						|
  }
 | 
						|
 | 
						|
  template <typename Arg>
 | 
						|
  static bool Convert(Arg arg, FormatConversionSpecImpl conv,
 | 
						|
                      FormatSinkImpl* out) {
 | 
						|
    return arg.dispatcher_(arg.data_, conv, out);
 | 
						|
  }
 | 
						|
 | 
						|
  template <typename Arg>
 | 
						|
  static typename Arg::Dispatcher GetVTablePtrForTest(Arg arg) {
 | 
						|
    return arg.dispatcher_;
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
// A type-erased handle to a format argument.
 | 
						|
class FormatArgImpl {
 | 
						|
 private:
 | 
						|
  enum { kInlinedSpace = 8 };
 | 
						|
 | 
						|
  using VoidPtr = str_format_internal::VoidPtr;
 | 
						|
 | 
						|
  union Data {
 | 
						|
    const void* ptr;
 | 
						|
    const volatile void* volatile_ptr;
 | 
						|
    char buf[kInlinedSpace];
 | 
						|
  };
 | 
						|
 | 
						|
  using Dispatcher = bool (*)(Data, FormatConversionSpecImpl, void* out);
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  struct store_by_value
 | 
						|
      : std::integral_constant<bool, (sizeof(T) <= kInlinedSpace) &&
 | 
						|
                                         (std::is_integral<T>::value ||
 | 
						|
                                          std::is_floating_point<T>::value ||
 | 
						|
                                          std::is_pointer<T>::value ||
 | 
						|
                                          std::is_same<VoidPtr, T>::value)> {};
 | 
						|
 | 
						|
  enum StoragePolicy { ByPointer, ByVolatilePointer, ByValue };
 | 
						|
  template <typename T>
 | 
						|
  struct storage_policy
 | 
						|
      : std::integral_constant<StoragePolicy,
 | 
						|
                               (std::is_volatile<T>::value
 | 
						|
                                    ? ByVolatilePointer
 | 
						|
                                    : (store_by_value<T>::value ? ByValue
 | 
						|
                                                                : ByPointer))> {
 | 
						|
  };
 | 
						|
 | 
						|
  // To reduce the number of vtables we will decay values before hand.
 | 
						|
  // Anything with a user-defined Convert will get its own vtable.
 | 
						|
  // For everything else:
 | 
						|
  //   - Decay char* and char arrays into `const char*`
 | 
						|
  //   - Decay any other pointer to `const void*`
 | 
						|
  //   - Decay all enums to their underlying type.
 | 
						|
  //   - Decay function pointers to void*.
 | 
						|
  template <typename T, typename = void>
 | 
						|
  struct DecayType {
 | 
						|
    static constexpr bool kHasUserDefined =
 | 
						|
        str_format_internal::HasUserDefinedConvert<T>::value;
 | 
						|
    using type = typename std::conditional<
 | 
						|
        !kHasUserDefined && std::is_convertible<T, const char*>::value,
 | 
						|
        const char*,
 | 
						|
        typename std::conditional<!kHasUserDefined &&
 | 
						|
                                      std::is_convertible<T, VoidPtr>::value,
 | 
						|
                                  VoidPtr, const T&>::type>::type;
 | 
						|
  };
 | 
						|
  template <typename T>
 | 
						|
  struct DecayType<T,
 | 
						|
                   typename std::enable_if<
 | 
						|
                       !str_format_internal::HasUserDefinedConvert<T>::value &&
 | 
						|
                       std::is_enum<T>::value>::type> {
 | 
						|
    using type = typename std::underlying_type<T>::type;
 | 
						|
  };
 | 
						|
 | 
						|
 public:
 | 
						|
  template <typename T>
 | 
						|
  explicit FormatArgImpl(const T& value) {
 | 
						|
    using D = typename DecayType<T>::type;
 | 
						|
    static_assert(
 | 
						|
        std::is_same<D, const T&>::value || storage_policy<D>::value == ByValue,
 | 
						|
        "Decayed types must be stored by value");
 | 
						|
    Init(static_cast<D>(value));
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  friend struct str_format_internal::FormatArgImplFriend;
 | 
						|
  template <typename T, StoragePolicy = storage_policy<T>::value>
 | 
						|
  struct Manager;
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  struct Manager<T, ByPointer> {
 | 
						|
    static Data SetValue(const T& value) {
 | 
						|
      Data data;
 | 
						|
      data.ptr = std::addressof(value);
 | 
						|
      return data;
 | 
						|
    }
 | 
						|
 | 
						|
    static const T& Value(Data arg) { return *static_cast<const T*>(arg.ptr); }
 | 
						|
  };
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  struct Manager<T, ByVolatilePointer> {
 | 
						|
    static Data SetValue(const T& value) {
 | 
						|
      Data data;
 | 
						|
      data.volatile_ptr = &value;
 | 
						|
      return data;
 | 
						|
    }
 | 
						|
 | 
						|
    static const T& Value(Data arg) {
 | 
						|
      return *static_cast<const T*>(arg.volatile_ptr);
 | 
						|
    }
 | 
						|
  };
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  struct Manager<T, ByValue> {
 | 
						|
    static Data SetValue(const T& value) {
 | 
						|
      Data data;
 | 
						|
      memcpy(data.buf, &value, sizeof(value));
 | 
						|
      return data;
 | 
						|
    }
 | 
						|
 | 
						|
    static T Value(Data arg) {
 | 
						|
      T value;
 | 
						|
      memcpy(&value, arg.buf, sizeof(T));
 | 
						|
      return value;
 | 
						|
    }
 | 
						|
  };
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  void Init(const T& value) {
 | 
						|
    data_ = Manager<T>::SetValue(value);
 | 
						|
    dispatcher_ = &Dispatch<T>;
 | 
						|
  }
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  static int ToIntVal(const T& val) {
 | 
						|
    using CommonType = typename std::conditional<std::is_signed<T>::value,
 | 
						|
                                                 int64_t, uint64_t>::type;
 | 
						|
    if (static_cast<CommonType>(val) >
 | 
						|
        static_cast<CommonType>((std::numeric_limits<int>::max)())) {
 | 
						|
      return (std::numeric_limits<int>::max)();
 | 
						|
    } else if (std::is_signed<T>::value &&
 | 
						|
               static_cast<CommonType>(val) <
 | 
						|
                   static_cast<CommonType>((std::numeric_limits<int>::min)())) {
 | 
						|
      return (std::numeric_limits<int>::min)();
 | 
						|
    }
 | 
						|
    return static_cast<int>(val);
 | 
						|
  }
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  static bool ToInt(Data arg, int* out, std::true_type /* is_integral */,
 | 
						|
                    std::false_type) {
 | 
						|
    *out = ToIntVal(Manager<T>::Value(arg));
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  static bool ToInt(Data arg, int* out, std::false_type,
 | 
						|
                    std::true_type /* is_enum */) {
 | 
						|
    *out = ToIntVal(static_cast<typename std::underlying_type<T>::type>(
 | 
						|
        Manager<T>::Value(arg)));
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  static bool ToInt(Data, int*, std::false_type, std::false_type) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  static bool Dispatch(Data arg, FormatConversionSpecImpl spec, void* out) {
 | 
						|
    // A `none` conv indicates that we want the `int` conversion.
 | 
						|
    if (ABSL_PREDICT_FALSE(spec.conversion_char() ==
 | 
						|
                           FormatConversionCharInternal::kNone)) {
 | 
						|
      return ToInt<T>(arg, static_cast<int*>(out), std::is_integral<T>(),
 | 
						|
                      std::is_enum<T>());
 | 
						|
    }
 | 
						|
 | 
						|
    return str_format_internal::FormatConvertImpl(
 | 
						|
               Manager<T>::Value(arg), spec, static_cast<FormatSinkImpl*>(out))
 | 
						|
        .value;
 | 
						|
  }
 | 
						|
 | 
						|
  Data data_;
 | 
						|
  Dispatcher dispatcher_;
 | 
						|
};
 | 
						|
 | 
						|
#define ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(T, E)                     \
 | 
						|
  E template bool FormatArgImpl::Dispatch<T>(Data, FormatConversionSpecImpl, \
 | 
						|
                                             void*)
 | 
						|
 | 
						|
#define ABSL_INTERNAL_FORMAT_DISPATCH_OVERLOADS_EXPAND_(...)                   \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(str_format_internal::VoidPtr,     \
 | 
						|
                                             __VA_ARGS__);                     \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(bool, __VA_ARGS__);               \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(char, __VA_ARGS__);               \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(signed char, __VA_ARGS__);        \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(unsigned char, __VA_ARGS__);      \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(short, __VA_ARGS__); /* NOLINT */ \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(unsigned short,      /* NOLINT */ \
 | 
						|
                                             __VA_ARGS__);                     \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(int, __VA_ARGS__);                \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(unsigned int, __VA_ARGS__);       \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(long, __VA_ARGS__); /* NOLINT */  \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(unsigned long,      /* NOLINT */  \
 | 
						|
                                             __VA_ARGS__);                     \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(long long, /* NOLINT */           \
 | 
						|
                                             __VA_ARGS__);                     \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(unsigned long long, /* NOLINT */  \
 | 
						|
                                             __VA_ARGS__);                     \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(int128, __VA_ARGS__);             \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(uint128, __VA_ARGS__);            \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(float, __VA_ARGS__);              \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(double, __VA_ARGS__);             \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(long double, __VA_ARGS__);        \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(const char*, __VA_ARGS__);        \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(std::string, __VA_ARGS__);        \
 | 
						|
  ABSL_INTERNAL_FORMAT_DISPATCH_INSTANTIATE_(string_view, __VA_ARGS__)
 | 
						|
 | 
						|
ABSL_INTERNAL_FORMAT_DISPATCH_OVERLOADS_EXPAND_(extern);
 | 
						|
 | 
						|
 | 
						|
}  // namespace str_format_internal
 | 
						|
ABSL_NAMESPACE_END
 | 
						|
}  // namespace absl
 | 
						|
 | 
						|
#endif  // ABSL_STRINGS_INTERNAL_STR_FORMAT_ARG_H_
 |