123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133 |
- // Copyright 2017 The Abseil Authors.
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // https://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- #include "y_absl/strings/match.h"
- #include <algorithm>
- #include <cstdint>
- #include "y_absl/base/config.h"
- #include "y_absl/base/internal/endian.h"
- #include "y_absl/base/optimization.h"
- #include "y_absl/numeric/bits.h"
- #include "y_absl/strings/ascii.h"
- #include "y_absl/strings/internal/memutil.h"
- #include "y_absl/strings/string_view.h"
- namespace y_absl {
- Y_ABSL_NAMESPACE_BEGIN
- bool EqualsIgnoreCase(y_absl::string_view piece1,
- y_absl::string_view piece2) noexcept {
- return (piece1.size() == piece2.size() &&
- 0 == y_absl::strings_internal::memcasecmp(piece1.data(), piece2.data(),
- piece1.size()));
- // memcasecmp uses y_absl::ascii_tolower().
- }
- bool StrContainsIgnoreCase(y_absl::string_view haystack,
- y_absl::string_view needle) noexcept {
- while (haystack.size() >= needle.size()) {
- if (StartsWithIgnoreCase(haystack, needle)) return true;
- haystack.remove_prefix(1);
- }
- return false;
- }
- bool StrContainsIgnoreCase(y_absl::string_view haystack,
- char needle) noexcept {
- char upper_needle = y_absl::ascii_toupper(static_cast<unsigned char>(needle));
- char lower_needle = y_absl::ascii_tolower(static_cast<unsigned char>(needle));
- if (upper_needle == lower_needle) {
- return StrContains(haystack, needle);
- } else {
- const char both_cstr[3] = {lower_needle, upper_needle, '\0'};
- return haystack.find_first_of(both_cstr) != y_absl::string_view::npos;
- }
- }
- bool StartsWithIgnoreCase(y_absl::string_view text,
- y_absl::string_view prefix) noexcept {
- return (text.size() >= prefix.size()) &&
- EqualsIgnoreCase(text.substr(0, prefix.size()), prefix);
- }
- bool EndsWithIgnoreCase(y_absl::string_view text,
- y_absl::string_view suffix) noexcept {
- return (text.size() >= suffix.size()) &&
- EqualsIgnoreCase(text.substr(text.size() - suffix.size()), suffix);
- }
- y_absl::string_view FindLongestCommonPrefix(y_absl::string_view a,
- y_absl::string_view b) {
- const y_absl::string_view::size_type limit = std::min(a.size(), b.size());
- const char* const pa = a.data();
- const char* const pb = b.data();
- y_absl::string_view::size_type count = (unsigned) 0;
- if (Y_ABSL_PREDICT_FALSE(limit < 8)) {
- while (Y_ABSL_PREDICT_TRUE(count + 2 <= limit)) {
- uint16_t xor_bytes = y_absl::little_endian::Load16(pa + count) ^
- y_absl::little_endian::Load16(pb + count);
- if (Y_ABSL_PREDICT_FALSE(xor_bytes != 0)) {
- if (Y_ABSL_PREDICT_TRUE((xor_bytes & 0xff) == 0)) ++count;
- return y_absl::string_view(pa, count);
- }
- count += 2;
- }
- if (Y_ABSL_PREDICT_TRUE(count != limit)) {
- if (Y_ABSL_PREDICT_TRUE(pa[count] == pb[count])) ++count;
- }
- return y_absl::string_view(pa, count);
- }
- do {
- uint64_t xor_bytes = y_absl::little_endian::Load64(pa + count) ^
- y_absl::little_endian::Load64(pb + count);
- if (Y_ABSL_PREDICT_FALSE(xor_bytes != 0)) {
- count += static_cast<uint64_t>(y_absl::countr_zero(xor_bytes) >> 3);
- return y_absl::string_view(pa, count);
- }
- count += 8;
- } while (Y_ABSL_PREDICT_TRUE(count + 8 < limit));
- count = limit - 8;
- uint64_t xor_bytes = y_absl::little_endian::Load64(pa + count) ^
- y_absl::little_endian::Load64(pb + count);
- if (Y_ABSL_PREDICT_TRUE(xor_bytes != 0)) {
- count += static_cast<uint64_t>(y_absl::countr_zero(xor_bytes) >> 3);
- return y_absl::string_view(pa, count);
- }
- return y_absl::string_view(pa, limit);
- }
- y_absl::string_view FindLongestCommonSuffix(y_absl::string_view a,
- y_absl::string_view b) {
- const y_absl::string_view::size_type limit = std::min(a.size(), b.size());
- if (limit == 0) return y_absl::string_view();
- const char* pa = a.data() + a.size() - 1;
- const char* pb = b.data() + b.size() - 1;
- y_absl::string_view::size_type count = (unsigned) 0;
- while (count < limit && *pa == *pb) {
- --pa;
- --pb;
- ++count;
- }
- return y_absl::string_view(++pa, count);
- }
- Y_ABSL_NAMESPACE_END
- } // namespace y_absl
|