-- c99f979ad34f155fbeeea69b88bdc7458d89a21c by Derek Mauro <dmauro@google.com>: Remove a floating point division by zero test. This isn't testing behavior related to the library, and MSVC warns about it in opt mode. PiperOrigin-RevId: 285220804 -- 68b015491f0dbf1ab547994673281abd1f34cd4b by Gennadiy Rozental <rogeeff@google.com>: This CL introduces following changes to the class FlagImpl: * We eliminate the CommandLineFlagLocks struct. Instead callback guard and callback function are combined into a single CallbackData struct, while primary data lock is stored separately. * CallbackData member of class FlagImpl is initially set to be nullptr and is only allocated and initialized when a flag's callback is being set. For most flags we do not pay for the extra space and extra absl::Mutex now. * Primary data guard is stored in data_guard_ data member. This is a properly aligned character buffer of necessary size. During initialization of the flag we construct absl::Mutex in this space using placement new call. * We now avoid extra value copy after successful attempt to parse value out of string. Instead we swap flag's current value with tentative value we just produced. PiperOrigin-RevId: 285132636 -- ed45d118fb818969eb13094cf7827c885dfc562c by Tom Manshreck <shreck@google.com>: Change null-term* (and nul-term*) to NUL-term* in comments PiperOrigin-RevId: 285036610 -- 729619017944db895ce8d6d29c1995aa2e5628a5 by Derek Mauro <dmauro@google.com>: Use the Posix implementation of thread identity on MinGW. Some versions of MinGW suffer from thread_local bugs. PiperOrigin-RevId: 285022920 -- 39a25493503c76885bc3254c28f66a251c5b5bb0 by Greg Falcon <gfalcon@google.com>: Implementation detail change. Add further ABSL_NAMESPACE_BEGIN and _END annotation macros to files in Abseil. PiperOrigin-RevId: 285012012 GitOrigin-RevId: c99f979ad34f155fbeeea69b88bdc7458d89a21c Change-Id: I4c85d3704e45d11a9ac50d562f39640a6adbedc1
		
			
				
	
	
		
			326 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			326 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #ifndef ABSL_STRINGS_INTERNAL_STR_FORMAT_CHECKER_H_
 | |
| #define ABSL_STRINGS_INTERNAL_STR_FORMAT_CHECKER_H_
 | |
| 
 | |
| #include "absl/base/attributes.h"
 | |
| #include "absl/strings/internal/str_format/arg.h"
 | |
| #include "absl/strings/internal/str_format/extension.h"
 | |
| 
 | |
| // Compile time check support for entry points.
 | |
| 
 | |
| #ifndef ABSL_INTERNAL_ENABLE_FORMAT_CHECKER
 | |
| #if ABSL_HAVE_ATTRIBUTE(enable_if) && !defined(__native_client__)
 | |
| #define ABSL_INTERNAL_ENABLE_FORMAT_CHECKER 1
 | |
| #endif  // ABSL_HAVE_ATTRIBUTE(enable_if) && !defined(__native_client__)
 | |
| #endif  // ABSL_INTERNAL_ENABLE_FORMAT_CHECKER
 | |
| 
 | |
| namespace absl {
 | |
| ABSL_NAMESPACE_BEGIN
 | |
| namespace str_format_internal {
 | |
| 
 | |
| constexpr bool AllOf() { return true; }
 | |
| 
 | |
| template <typename... T>
 | |
| constexpr bool AllOf(bool b, T... t) {
 | |
|   return b && AllOf(t...);
 | |
| }
 | |
| 
 | |
| template <typename Arg>
 | |
| constexpr Conv ArgumentToConv() {
 | |
|   return decltype(str_format_internal::FormatConvertImpl(
 | |
|       std::declval<const Arg&>(), std::declval<const ConversionSpec&>(),
 | |
|       std::declval<FormatSinkImpl*>()))::kConv;
 | |
| }
 | |
| 
 | |
| #ifdef ABSL_INTERNAL_ENABLE_FORMAT_CHECKER
 | |
| 
 | |
| constexpr bool ContainsChar(const char* chars, char c) {
 | |
|   return *chars == c || (*chars && ContainsChar(chars + 1, c));
 | |
| }
 | |
| 
 | |
| // A constexpr compatible list of Convs.
 | |
| struct ConvList {
 | |
|   const Conv* array;
 | |
|   int count;
 | |
| 
 | |
|   // We do the bound check here to avoid having to do it on the callers.
 | |
|   // Returning an empty Conv has the same effect as short circuiting because it
 | |
|   // will never match any conversion.
 | |
|   constexpr Conv operator[](int i) const {
 | |
|     return i < count ? array[i] : Conv{};
 | |
|   }
 | |
| 
 | |
|   constexpr ConvList without_front() const {
 | |
|     return count != 0 ? ConvList{array + 1, count - 1} : *this;
 | |
|   }
 | |
| };
 | |
| 
 | |
| template <size_t count>
 | |
| struct ConvListT {
 | |
|   // Make sure the array has size > 0.
 | |
|   Conv list[count ? count : 1];
 | |
| };
 | |
| 
 | |
| constexpr char GetChar(string_view str, size_t index) {
 | |
|   return index < str.size() ? str[index] : char{};
 | |
| }
 | |
| 
 | |
| constexpr string_view ConsumeFront(string_view str, size_t len = 1) {
 | |
|   return len <= str.size() ? string_view(str.data() + len, str.size() - len)
 | |
|                            : string_view();
 | |
| }
 | |
| 
 | |
| constexpr string_view ConsumeAnyOf(string_view format, const char* chars) {
 | |
|   return ContainsChar(chars, GetChar(format, 0))
 | |
|              ? ConsumeAnyOf(ConsumeFront(format), chars)
 | |
|              : format;
 | |
| }
 | |
| 
 | |
| constexpr bool IsDigit(char c) { return c >= '0' && c <= '9'; }
 | |
| 
 | |
| // Helper class for the ParseDigits function.
 | |
| // It encapsulates the two return values we need there.
 | |
| struct Integer {
 | |
|   string_view format;
 | |
|   int value;
 | |
| 
 | |
|   // If the next character is a '$', consume it.
 | |
|   // Otherwise, make `this` an invalid positional argument.
 | |
|   constexpr Integer ConsumePositionalDollar() const {
 | |
|     return GetChar(format, 0) == '$' ? Integer{ConsumeFront(format), value}
 | |
|                                      : Integer{format, 0};
 | |
|   }
 | |
| };
 | |
| 
 | |
| constexpr Integer ParseDigits(string_view format, int value = 0) {
 | |
|   return IsDigit(GetChar(format, 0))
 | |
|              ? ParseDigits(ConsumeFront(format),
 | |
|                            10 * value + GetChar(format, 0) - '0')
 | |
|              : Integer{format, value};
 | |
| }
 | |
| 
 | |
| // Parse digits for a positional argument.
 | |
| // The parsing also consumes the '$'.
 | |
| constexpr Integer ParsePositional(string_view format) {
 | |
|   return ParseDigits(format).ConsumePositionalDollar();
 | |
| }
 | |
| 
 | |
| // Parses a single conversion specifier.
 | |
| // See ConvParser::Run() for post conditions.
 | |
| class ConvParser {
 | |
|   constexpr ConvParser SetFormat(string_view format) const {
 | |
|     return ConvParser(format, args_, error_, arg_position_, is_positional_);
 | |
|   }
 | |
| 
 | |
|   constexpr ConvParser SetArgs(ConvList args) const {
 | |
|     return ConvParser(format_, args, error_, arg_position_, is_positional_);
 | |
|   }
 | |
| 
 | |
|   constexpr ConvParser SetError(bool error) const {
 | |
|     return ConvParser(format_, args_, error_ || error, arg_position_,
 | |
|                       is_positional_);
 | |
|   }
 | |
| 
 | |
|   constexpr ConvParser SetArgPosition(int arg_position) const {
 | |
|     return ConvParser(format_, args_, error_, arg_position, is_positional_);
 | |
|   }
 | |
| 
 | |
|   // Consumes the next arg and verifies that it matches `conv`.
 | |
|   // `error_` is set if there is no next arg or if it doesn't match `conv`.
 | |
|   constexpr ConvParser ConsumeNextArg(char conv) const {
 | |
|     return SetArgs(args_.without_front()).SetError(!Contains(args_[0], conv));
 | |
|   }
 | |
| 
 | |
|   // Verify that positional argument `i.value` matches `conv`.
 | |
|   // `error_` is set if `i.value` is not a valid argument or if it doesn't
 | |
|   // match.
 | |
|   constexpr ConvParser VerifyPositional(Integer i, char conv) const {
 | |
|     return SetFormat(i.format).SetError(!Contains(args_[i.value - 1], conv));
 | |
|   }
 | |
| 
 | |
|   // Parse the position of the arg and store it in `arg_position_`.
 | |
|   constexpr ConvParser ParseArgPosition(Integer arg) const {
 | |
|     return SetFormat(arg.format).SetArgPosition(arg.value);
 | |
|   }
 | |
| 
 | |
|   // Consume the flags.
 | |
|   constexpr ConvParser ParseFlags() const {
 | |
|     return SetFormat(ConsumeAnyOf(format_, "-+ #0"));
 | |
|   }
 | |
| 
 | |
|   // Consume the width.
 | |
|   // If it is '*', we verify that it matches `args_`. `error_` is set if it
 | |
|   // doesn't match.
 | |
|   constexpr ConvParser ParseWidth() const {
 | |
|     return IsDigit(GetChar(format_, 0))
 | |
|                ? SetFormat(ParseDigits(format_).format)
 | |
|                : GetChar(format_, 0) == '*'
 | |
|                      ? is_positional_
 | |
|                            ? VerifyPositional(
 | |
|                                  ParsePositional(ConsumeFront(format_)), '*')
 | |
|                            : SetFormat(ConsumeFront(format_))
 | |
|                                  .ConsumeNextArg('*')
 | |
|                      : *this;
 | |
|   }
 | |
| 
 | |
|   // Consume the precision.
 | |
|   // If it is '*', we verify that it matches `args_`. `error_` is set if it
 | |
|   // doesn't match.
 | |
|   constexpr ConvParser ParsePrecision() const {
 | |
|     return GetChar(format_, 0) != '.'
 | |
|                ? *this
 | |
|                : GetChar(format_, 1) == '*'
 | |
|                      ? is_positional_
 | |
|                            ? VerifyPositional(
 | |
|                                  ParsePositional(ConsumeFront(format_, 2)), '*')
 | |
|                            : SetFormat(ConsumeFront(format_, 2))
 | |
|                                  .ConsumeNextArg('*')
 | |
|                      : SetFormat(ParseDigits(ConsumeFront(format_)).format);
 | |
|   }
 | |
| 
 | |
|   // Consume the length characters.
 | |
|   constexpr ConvParser ParseLength() const {
 | |
|     return SetFormat(ConsumeAnyOf(format_, "lLhjztq"));
 | |
|   }
 | |
| 
 | |
|   // Consume the conversion character and verify that it matches `args_`.
 | |
|   // `error_` is set if it doesn't match.
 | |
|   constexpr ConvParser ParseConversion() const {
 | |
|     return is_positional_
 | |
|                ? VerifyPositional({ConsumeFront(format_), arg_position_},
 | |
|                                   GetChar(format_, 0))
 | |
|                : ConsumeNextArg(GetChar(format_, 0))
 | |
|                      .SetFormat(ConsumeFront(format_));
 | |
|   }
 | |
| 
 | |
|   constexpr ConvParser(string_view format, ConvList args, bool error,
 | |
|                        int arg_position, bool is_positional)
 | |
|       : format_(format),
 | |
|         args_(args),
 | |
|         error_(error),
 | |
|         arg_position_(arg_position),
 | |
|         is_positional_(is_positional) {}
 | |
| 
 | |
|  public:
 | |
|   constexpr ConvParser(string_view format, ConvList args, bool is_positional)
 | |
|       : format_(format),
 | |
|         args_(args),
 | |
|         error_(false),
 | |
|         arg_position_(0),
 | |
|         is_positional_(is_positional) {}
 | |
| 
 | |
|   // Consume the whole conversion specifier.
 | |
|   // `format()` will be set to the character after the conversion character.
 | |
|   // `error()` will be set if any of the arguments do not match.
 | |
|   constexpr ConvParser Run() const {
 | |
|     return (is_positional_ ? ParseArgPosition(ParsePositional(format_)) : *this)
 | |
|         .ParseFlags()
 | |
|         .ParseWidth()
 | |
|         .ParsePrecision()
 | |
|         .ParseLength()
 | |
|         .ParseConversion();
 | |
|   }
 | |
| 
 | |
|   constexpr string_view format() const { return format_; }
 | |
|   constexpr ConvList args() const { return args_; }
 | |
|   constexpr bool error() const { return error_; }
 | |
|   constexpr bool is_positional() const { return is_positional_; }
 | |
| 
 | |
|  private:
 | |
|   string_view format_;
 | |
|   // Current list of arguments. If we are not in positional mode we will consume
 | |
|   // from the front.
 | |
|   ConvList args_;
 | |
|   bool error_;
 | |
|   // Holds the argument position of the conversion character, if we are in
 | |
|   // positional mode. Otherwise, it is unspecified.
 | |
|   int arg_position_;
 | |
|   // Whether we are in positional mode.
 | |
|   // It changes the behavior of '*' and where to find the converted argument.
 | |
|   bool is_positional_;
 | |
| };
 | |
| 
 | |
| // Parses a whole format expression.
 | |
| // See FormatParser::Run().
 | |
| class FormatParser {
 | |
|   static constexpr bool FoundPercent(string_view format) {
 | |
|     return format.empty() ||
 | |
|            (GetChar(format, 0) == '%' && GetChar(format, 1) != '%');
 | |
|   }
 | |
| 
 | |
|   // We use an inner function to increase the recursion limit.
 | |
|   // The inner function consumes up to `limit` characters on every run.
 | |
|   // This increases the limit from 512 to ~512*limit.
 | |
|   static constexpr string_view ConsumeNonPercentInner(string_view format,
 | |
|                                                       int limit = 20) {
 | |
|     return FoundPercent(format) || !limit
 | |
|                ? format
 | |
|                : ConsumeNonPercentInner(
 | |
|                      ConsumeFront(format, GetChar(format, 0) == '%' &&
 | |
|                                                   GetChar(format, 1) == '%'
 | |
|                                               ? 2
 | |
|                                               : 1),
 | |
|                      limit - 1);
 | |
|   }
 | |
| 
 | |
|   // Consume characters until the next conversion spec %.
 | |
|   // It skips %%.
 | |
|   static constexpr string_view ConsumeNonPercent(string_view format) {
 | |
|     return FoundPercent(format)
 | |
|                ? format
 | |
|                : ConsumeNonPercent(ConsumeNonPercentInner(format));
 | |
|   }
 | |
| 
 | |
|   static constexpr bool IsPositional(string_view format) {
 | |
|     return IsDigit(GetChar(format, 0)) ? IsPositional(ConsumeFront(format))
 | |
|                                        : GetChar(format, 0) == '$';
 | |
|   }
 | |
| 
 | |
|   constexpr bool RunImpl(bool is_positional) const {
 | |
|     // In non-positional mode we require all arguments to be consumed.
 | |
|     // In positional mode just reaching the end of the format without errors is
 | |
|     // enough.
 | |
|     return (format_.empty() && (is_positional || args_.count == 0)) ||
 | |
|            (!format_.empty() &&
 | |
|             ValidateArg(
 | |
|                 ConvParser(ConsumeFront(format_), args_, is_positional).Run()));
 | |
|   }
 | |
| 
 | |
|   constexpr bool ValidateArg(ConvParser conv) const {
 | |
|     return !conv.error() && FormatParser(conv.format(), conv.args())
 | |
|                                 .RunImpl(conv.is_positional());
 | |
|   }
 | |
| 
 | |
|  public:
 | |
|   constexpr FormatParser(string_view format, ConvList args)
 | |
|       : format_(ConsumeNonPercent(format)), args_(args) {}
 | |
| 
 | |
|   // Runs the parser for `format` and `args`.
 | |
|   // It verifies that the format is valid and that all conversion specifiers
 | |
|   // match the arguments passed.
 | |
|   // In non-positional mode it also verfies that all arguments are consumed.
 | |
|   constexpr bool Run() const {
 | |
|     return RunImpl(!format_.empty() && IsPositional(ConsumeFront(format_)));
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   string_view format_;
 | |
|   // Current list of arguments.
 | |
|   // If we are not in positional mode we will consume from the front and will
 | |
|   // have to be empty in the end.
 | |
|   ConvList args_;
 | |
| };
 | |
| 
 | |
| template <Conv... C>
 | |
| constexpr bool ValidFormatImpl(string_view format) {
 | |
|   return FormatParser(format,
 | |
|                       {ConvListT<sizeof...(C)>{{C...}}.list, sizeof...(C)})
 | |
|       .Run();
 | |
| }
 | |
| 
 | |
| #endif  // ABSL_INTERNAL_ENABLE_FORMAT_CHECKER
 | |
| 
 | |
| }  // namespace str_format_internal
 | |
| ABSL_NAMESPACE_END
 | |
| }  // namespace absl
 | |
| 
 | |
| #endif  // ABSL_STRINGS_INTERNAL_STR_FORMAT_CHECKER_H_
 |