From: Giuseppe Ottaviano Date: Thu, 18 Dec 2014 02:19:00 +0000 (-0800) Subject: Disable implicit conversions from std::string for non-char* Range X-Git-Tag: v0.22.0~89 X-Git-Url: http://demsky.eecs.uci.edu/git/?a=commitdiff_plain;h=c1a244da6daacd948fdbda1d0770372623101fb7;p=folly.git Disable implicit conversions from std::string for non-char* Range Summary: `Range` has an implicit constructors from strings for any `Iter`, however such constructors are invalid (compilation fails) if `Iter` is not `[const] char *`. This can be an issue for overload resolution: for example struct IsString { bool operator()(folly::Range) { return false; } bool operator()(folly::StringPiece) { return true; } }; IsString()(std::string()); fails to compile because the overload is ambiguous, even if the conversion to `ByteRange` is invalid. This patch disables all the invalid constructors from `[const] char*`, `std::string`, and `fbstring`. Test Plan: fbconfig -r folly && fbmake runtests_opt Reviewed By: philipp@fb.com Subscribers: folly-diffs@ FB internal diff: D1746899 Signature: t1:1746899:1418868361:50784c4993df0bd96eeb62c09c659d5e53964d9b --- diff --git a/folly/Range.h b/folly/Range.h index f61c76e9..dccb92d0 100644 --- a/folly/Range.h +++ b/folly/Range.h @@ -121,6 +121,23 @@ value_before(Iter i) { return *--i; } +/* + * Use IsCharPointer::type to enable const char* or char*. + * Use IsCharPointer::const_type to enable only const char*. + */ +template struct IsCharPointer {}; + +template <> +struct IsCharPointer { + typedef int type; +}; + +template <> +struct IsCharPointer { + typedef int const_type; + typedef int type; +}; + } // namespace detail /** @@ -176,19 +193,19 @@ public: : b_(start), e_(start + size) { } #if FOLLY_HAVE_CONSTEXPR_STRLEN - // Works only for Range + template ::type = 0> constexpr /* implicit */ Range(Iter str) : b_(str), e_(str + strlen(str)) {} #else - // Works only for Range + template ::type = 0> /* implicit */ Range(Iter str) : b_(str), e_(str + strlen(str)) {} #endif - // Works only for Range + template ::const_type = 0> /* implicit */ Range(const std::string& str) : b_(str.data()), e_(b_ + str.size()) {} - // Works only for Range + template ::const_type = 0> Range(const std::string& str, std::string::size_type startFrom) { if (UNLIKELY(startFrom > str.size())) { throw std::out_of_range("index out of range"); @@ -196,7 +213,7 @@ public: b_ = str.data() + startFrom; e_ = str.data() + str.size(); } - // Works only for Range + template ::const_type = 0> Range(const std::string& str, std::string::size_type startFrom, std::string::size_type size) { @@ -210,6 +227,7 @@ public: e_ = b_ + size; } } + template ::type = 0> Range(const Range& str, size_t startFrom, size_t size) { @@ -223,10 +241,10 @@ public: e_ = b_ + size; } } - // Works only for Range + template ::const_type = 0> /* implicit */ Range(const fbstring& str) : b_(str.data()), e_(b_ + str.size()) { } - // Works only for Range + template ::const_type = 0> Range(const fbstring& str, fbstring::size_type startFrom) { if (UNLIKELY(startFrom > str.size())) { throw std::out_of_range("index out of range"); @@ -234,7 +252,7 @@ public: b_ = str.data() + startFrom; e_ = str.data() + str.size(); } - // Works only for Range + template ::const_type = 0> Range(const fbstring& str, fbstring::size_type startFrom, fbstring::size_type size) { if (UNLIKELY(startFrom > str.size())) { @@ -358,10 +376,10 @@ public: assert(b_ < e_); return detail::value_before(e_); } - // Works only for Range + // Works only for Range and Range std::string str() const { return std::string(b_, size()); } std::string toString() const { return str(); } - // Works only for Range + // Works only for Range and Range fbstring fbstr() const { return fbstring(b_, size()); } fbstring toFbstring() const { return fbstr(); } @@ -369,7 +387,7 @@ public: return const_range_type(*this); }; - // Works only for Range (and Range) + // Works only for Range and Range int compare(const const_range_type& o) const { const size_type tsize = this->size(); const size_type osize = o.size(); @@ -399,7 +417,7 @@ public: return b_[i]; } - // Works only for Range + // Works only for Range and Range uint32_t hash() const { // Taken from fbi/nstring.h: // Quick and dirty bernstein hash...fine for short ascii strings diff --git a/folly/test/RangeTest.cpp b/folly/test/RangeTest.cpp index a8f2c615..29ea85bf 100644 --- a/folly/test/RangeTest.cpp +++ b/folly/test/RangeTest.cpp @@ -805,6 +805,16 @@ TEST(StringPiece, split_step_with_process_range_delimiter_additional_args) { EXPECT_TRUE(p.empty()); } +TEST(StringPiece, NoInvalidImplicitConversions) { + struct IsString { + bool operator()(folly::Range) { return false; } + bool operator()(folly::StringPiece) { return true; } + }; + + std::string s = "hello"; + EXPECT_TRUE(IsString()(s)); +} + TEST(qfind, UInt32_Ranges) { vector a({1, 2, 3, 260, 5}); vector b({2, 3, 4});