From d75a0450ad60b992207961cdd29d1385f2969d84 Mon Sep 17 00:00:00 2001 From: Marshall Clow Date: Mon, 20 May 2019 21:56:51 +0000 Subject: [PATCH] Ensure that hash uses char_traits. Fixes PR#41876. Reviewed as https://reviews.llvm.org/D61954 llvm-svn: 361201 --- libcxx/include/string | 22 +++---- .../basic.string.hash/char_type_hash.fail.cpp | 66 +++++++++++++++++++ 2 files changed, 75 insertions(+), 13 deletions(-) create mode 100644 libcxx/test/std/strings/basic.string.hash/char_type_hash.fail.cpp diff --git a/libcxx/include/string b/libcxx/include/string index 3b01c4104a98..8d4e13cf5273 100644 --- a/libcxx/include/string +++ b/libcxx/include/string @@ -1420,7 +1420,7 @@ public: _LIBCPP_INLINE_VISIBILITY bool __invariants() const; _LIBCPP_INLINE_VISIBILITY void __clear_and_shrink() _NOEXCEPT; - + _LIBCPP_INLINE_VISIBILITY bool __is_long() const _NOEXCEPT {return bool(__r_.first().__s.__size_ & __short_mask);} @@ -1682,7 +1682,7 @@ basic_string(basic_string_view<_CharT, _Traits>, _Sz, _Sz, const _Allocator& = _ -> basic_string<_CharT, _Traits, _Allocator>; #endif - + template inline void @@ -4227,21 +4227,17 @@ template const typename basic_string<_CharT, _Traits, _Allocator>::size_type basic_string<_CharT, _Traits, _Allocator>::npos; -template -struct _LIBCPP_TEMPLATE_VIS hash > - : public unary_function, size_t> +template +struct _LIBCPP_TEMPLATE_VIS + hash, _Allocator> > + : public unary_function< + basic_string<_CharT, char_traits<_CharT>, _Allocator>, size_t> { size_t - operator()(const basic_string<_CharT, _Traits, _Allocator>& __val) const _NOEXCEPT; + operator()(const basic_string<_CharT, char_traits<_CharT>, _Allocator>& __val) const _NOEXCEPT + { return __do_string_hash(__val.data(), __val.data() + __val.size()); } }; -template -size_t -hash >::operator()( - const basic_string<_CharT, _Traits, _Allocator>& __val) const _NOEXCEPT -{ - return __do_string_hash(__val.data(), __val.data() + __val.size()); -} template basic_ostream<_CharT, _Traits>& diff --git a/libcxx/test/std/strings/basic.string.hash/char_type_hash.fail.cpp b/libcxx/test/std/strings/basic.string.hash/char_type_hash.fail.cpp new file mode 100644 index 000000000000..542ae05536ea --- /dev/null +++ b/libcxx/test/std/strings/basic.string.hash/char_type_hash.fail.cpp @@ -0,0 +1,66 @@ +//===----------------------------------------------------------------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +// UNSUPPORTED: c++98, c++03 + +// + +// Test that hash specializations for require "char_traits<_CharT>" not just any "_Trait". + +#include + +template +struct trait // copied from <__string> +{ + typedef _CharT char_type; + typedef int int_type; + typedef std::streamoff off_type; + typedef std::streampos pos_type; + typedef std::mbstate_t state_type; + + static inline void assign(char_type& __c1, const char_type& __c2) { + __c1 = __c2; + } + static inline bool eq(char_type __c1, char_type __c2) { return __c1 == __c2; } + static inline bool lt(char_type __c1, char_type __c2) { return __c1 < __c2; } + + static int compare(const char_type* __s1, const char_type* __s2, size_t __n); + static size_t length(const char_type* __s); + static const char_type* find(const char_type* __s, size_t __n, + const char_type& __a); + + static char_type* move(char_type* __s1, const char_type* __s2, size_t __n); + static char_type* copy(char_type* __s1, const char_type* __s2, size_t __n); + static char_type* assign(char_type* __s, size_t __n, char_type __a); + + static inline int_type not_eof(int_type __c) { + return eq_int_type(__c, eof()) ? ~eof() : __c; + } + static inline char_type to_char_type(int_type __c) { return char_type(__c); } + static inline int_type to_int_type(char_type __c) { return int_type(__c); } + static inline bool eq_int_type(int_type __c1, int_type __c2) { + return __c1 == __c2; + } + static inline int_type eof() { return int_type(EOF); } +}; + +template +void test() { + typedef std::basic_string > str_t; + std::hash + h; // call to implicitly-deleted default constructor of 'std::hash' {{*}}}} +} + +int main(int, char**) { + test(); + test(); + test(); + test(); + + return 0; +}