xref: /aosp_15_r20/external/abseil-cpp/absl/strings/match.cc (revision 9356374a3709195abf420251b3e825997ff56c0f)
1*9356374aSAndroid Build Coastguard Worker // Copyright 2017 The Abseil Authors.
2*9356374aSAndroid Build Coastguard Worker //
3*9356374aSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
4*9356374aSAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
5*9356374aSAndroid Build Coastguard Worker // You may obtain a copy of the License at
6*9356374aSAndroid Build Coastguard Worker //
7*9356374aSAndroid Build Coastguard Worker //      https://www.apache.org/licenses/LICENSE-2.0
8*9356374aSAndroid Build Coastguard Worker //
9*9356374aSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*9356374aSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
11*9356374aSAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*9356374aSAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
13*9356374aSAndroid Build Coastguard Worker // limitations under the License.
14*9356374aSAndroid Build Coastguard Worker 
15*9356374aSAndroid Build Coastguard Worker #include "absl/strings/match.h"
16*9356374aSAndroid Build Coastguard Worker 
17*9356374aSAndroid Build Coastguard Worker #include <algorithm>
18*9356374aSAndroid Build Coastguard Worker #include <cstdint>
19*9356374aSAndroid Build Coastguard Worker 
20*9356374aSAndroid Build Coastguard Worker #include "absl/base/config.h"
21*9356374aSAndroid Build Coastguard Worker #include "absl/base/internal/endian.h"
22*9356374aSAndroid Build Coastguard Worker #include "absl/base/optimization.h"
23*9356374aSAndroid Build Coastguard Worker #include "absl/numeric/bits.h"
24*9356374aSAndroid Build Coastguard Worker #include "absl/strings/ascii.h"
25*9356374aSAndroid Build Coastguard Worker #include "absl/strings/internal/memutil.h"
26*9356374aSAndroid Build Coastguard Worker #include "absl/strings/string_view.h"
27*9356374aSAndroid Build Coastguard Worker 
28*9356374aSAndroid Build Coastguard Worker namespace absl {
29*9356374aSAndroid Build Coastguard Worker ABSL_NAMESPACE_BEGIN
30*9356374aSAndroid Build Coastguard Worker 
EqualsIgnoreCase(absl::string_view piece1,absl::string_view piece2)31*9356374aSAndroid Build Coastguard Worker bool EqualsIgnoreCase(absl::string_view piece1,
32*9356374aSAndroid Build Coastguard Worker                       absl::string_view piece2) noexcept {
33*9356374aSAndroid Build Coastguard Worker   return (piece1.size() == piece2.size() &&
34*9356374aSAndroid Build Coastguard Worker           0 == absl::strings_internal::memcasecmp(piece1.data(), piece2.data(),
35*9356374aSAndroid Build Coastguard Worker                                                   piece1.size()));
36*9356374aSAndroid Build Coastguard Worker   // memcasecmp uses absl::ascii_tolower().
37*9356374aSAndroid Build Coastguard Worker }
38*9356374aSAndroid Build Coastguard Worker 
StrContainsIgnoreCase(absl::string_view haystack,absl::string_view needle)39*9356374aSAndroid Build Coastguard Worker bool StrContainsIgnoreCase(absl::string_view haystack,
40*9356374aSAndroid Build Coastguard Worker                            absl::string_view needle) noexcept {
41*9356374aSAndroid Build Coastguard Worker   while (haystack.size() >= needle.size()) {
42*9356374aSAndroid Build Coastguard Worker     if (StartsWithIgnoreCase(haystack, needle)) return true;
43*9356374aSAndroid Build Coastguard Worker     haystack.remove_prefix(1);
44*9356374aSAndroid Build Coastguard Worker   }
45*9356374aSAndroid Build Coastguard Worker   return false;
46*9356374aSAndroid Build Coastguard Worker }
47*9356374aSAndroid Build Coastguard Worker 
StrContainsIgnoreCase(absl::string_view haystack,char needle)48*9356374aSAndroid Build Coastguard Worker bool StrContainsIgnoreCase(absl::string_view haystack,
49*9356374aSAndroid Build Coastguard Worker                            char needle) noexcept {
50*9356374aSAndroid Build Coastguard Worker   char upper_needle = absl::ascii_toupper(static_cast<unsigned char>(needle));
51*9356374aSAndroid Build Coastguard Worker   char lower_needle = absl::ascii_tolower(static_cast<unsigned char>(needle));
52*9356374aSAndroid Build Coastguard Worker   if (upper_needle == lower_needle) {
53*9356374aSAndroid Build Coastguard Worker     return StrContains(haystack, needle);
54*9356374aSAndroid Build Coastguard Worker   } else {
55*9356374aSAndroid Build Coastguard Worker     const char both_cstr[3] = {lower_needle, upper_needle, '\0'};
56*9356374aSAndroid Build Coastguard Worker     return haystack.find_first_of(both_cstr) != absl::string_view::npos;
57*9356374aSAndroid Build Coastguard Worker   }
58*9356374aSAndroid Build Coastguard Worker }
59*9356374aSAndroid Build Coastguard Worker 
StartsWithIgnoreCase(absl::string_view text,absl::string_view prefix)60*9356374aSAndroid Build Coastguard Worker bool StartsWithIgnoreCase(absl::string_view text,
61*9356374aSAndroid Build Coastguard Worker                           absl::string_view prefix) noexcept {
62*9356374aSAndroid Build Coastguard Worker   return (text.size() >= prefix.size()) &&
63*9356374aSAndroid Build Coastguard Worker          EqualsIgnoreCase(text.substr(0, prefix.size()), prefix);
64*9356374aSAndroid Build Coastguard Worker }
65*9356374aSAndroid Build Coastguard Worker 
EndsWithIgnoreCase(absl::string_view text,absl::string_view suffix)66*9356374aSAndroid Build Coastguard Worker bool EndsWithIgnoreCase(absl::string_view text,
67*9356374aSAndroid Build Coastguard Worker                         absl::string_view suffix) noexcept {
68*9356374aSAndroid Build Coastguard Worker   return (text.size() >= suffix.size()) &&
69*9356374aSAndroid Build Coastguard Worker          EqualsIgnoreCase(text.substr(text.size() - suffix.size()), suffix);
70*9356374aSAndroid Build Coastguard Worker }
71*9356374aSAndroid Build Coastguard Worker 
FindLongestCommonPrefix(absl::string_view a,absl::string_view b)72*9356374aSAndroid Build Coastguard Worker absl::string_view FindLongestCommonPrefix(absl::string_view a,
73*9356374aSAndroid Build Coastguard Worker                                           absl::string_view b) {
74*9356374aSAndroid Build Coastguard Worker   const absl::string_view::size_type limit = std::min(a.size(), b.size());
75*9356374aSAndroid Build Coastguard Worker   const char* const pa = a.data();
76*9356374aSAndroid Build Coastguard Worker   const char* const pb = b.data();
77*9356374aSAndroid Build Coastguard Worker   absl::string_view::size_type count = (unsigned) 0;
78*9356374aSAndroid Build Coastguard Worker 
79*9356374aSAndroid Build Coastguard Worker   if (ABSL_PREDICT_FALSE(limit < 8)) {
80*9356374aSAndroid Build Coastguard Worker     while (ABSL_PREDICT_TRUE(count + 2 <= limit)) {
81*9356374aSAndroid Build Coastguard Worker       uint16_t xor_bytes = absl::little_endian::Load16(pa + count) ^
82*9356374aSAndroid Build Coastguard Worker                            absl::little_endian::Load16(pb + count);
83*9356374aSAndroid Build Coastguard Worker       if (ABSL_PREDICT_FALSE(xor_bytes != 0)) {
84*9356374aSAndroid Build Coastguard Worker         if (ABSL_PREDICT_TRUE((xor_bytes & 0xff) == 0)) ++count;
85*9356374aSAndroid Build Coastguard Worker         return absl::string_view(pa, count);
86*9356374aSAndroid Build Coastguard Worker       }
87*9356374aSAndroid Build Coastguard Worker       count += 2;
88*9356374aSAndroid Build Coastguard Worker     }
89*9356374aSAndroid Build Coastguard Worker     if (ABSL_PREDICT_TRUE(count != limit)) {
90*9356374aSAndroid Build Coastguard Worker       if (ABSL_PREDICT_TRUE(pa[count] == pb[count])) ++count;
91*9356374aSAndroid Build Coastguard Worker     }
92*9356374aSAndroid Build Coastguard Worker     return absl::string_view(pa, count);
93*9356374aSAndroid Build Coastguard Worker   }
94*9356374aSAndroid Build Coastguard Worker 
95*9356374aSAndroid Build Coastguard Worker   do {
96*9356374aSAndroid Build Coastguard Worker     uint64_t xor_bytes = absl::little_endian::Load64(pa + count) ^
97*9356374aSAndroid Build Coastguard Worker                          absl::little_endian::Load64(pb + count);
98*9356374aSAndroid Build Coastguard Worker     if (ABSL_PREDICT_FALSE(xor_bytes != 0)) {
99*9356374aSAndroid Build Coastguard Worker       count += static_cast<uint64_t>(absl::countr_zero(xor_bytes) >> 3);
100*9356374aSAndroid Build Coastguard Worker       return absl::string_view(pa, count);
101*9356374aSAndroid Build Coastguard Worker     }
102*9356374aSAndroid Build Coastguard Worker     count += 8;
103*9356374aSAndroid Build Coastguard Worker   } while (ABSL_PREDICT_TRUE(count + 8 < limit));
104*9356374aSAndroid Build Coastguard Worker 
105*9356374aSAndroid Build Coastguard Worker   count = limit - 8;
106*9356374aSAndroid Build Coastguard Worker   uint64_t xor_bytes = absl::little_endian::Load64(pa + count) ^
107*9356374aSAndroid Build Coastguard Worker                        absl::little_endian::Load64(pb + count);
108*9356374aSAndroid Build Coastguard Worker   if (ABSL_PREDICT_TRUE(xor_bytes != 0)) {
109*9356374aSAndroid Build Coastguard Worker     count += static_cast<uint64_t>(absl::countr_zero(xor_bytes) >> 3);
110*9356374aSAndroid Build Coastguard Worker     return absl::string_view(pa, count);
111*9356374aSAndroid Build Coastguard Worker   }
112*9356374aSAndroid Build Coastguard Worker   return absl::string_view(pa, limit);
113*9356374aSAndroid Build Coastguard Worker }
114*9356374aSAndroid Build Coastguard Worker 
FindLongestCommonSuffix(absl::string_view a,absl::string_view b)115*9356374aSAndroid Build Coastguard Worker absl::string_view FindLongestCommonSuffix(absl::string_view a,
116*9356374aSAndroid Build Coastguard Worker                                           absl::string_view b) {
117*9356374aSAndroid Build Coastguard Worker   const absl::string_view::size_type limit = std::min(a.size(), b.size());
118*9356374aSAndroid Build Coastguard Worker   if (limit == 0) return absl::string_view();
119*9356374aSAndroid Build Coastguard Worker 
120*9356374aSAndroid Build Coastguard Worker   const char* pa = a.data() + a.size() - 1;
121*9356374aSAndroid Build Coastguard Worker   const char* pb = b.data() + b.size() - 1;
122*9356374aSAndroid Build Coastguard Worker   absl::string_view::size_type count = (unsigned) 0;
123*9356374aSAndroid Build Coastguard Worker   while (count < limit && *pa == *pb) {
124*9356374aSAndroid Build Coastguard Worker     --pa;
125*9356374aSAndroid Build Coastguard Worker     --pb;
126*9356374aSAndroid Build Coastguard Worker     ++count;
127*9356374aSAndroid Build Coastguard Worker   }
128*9356374aSAndroid Build Coastguard Worker 
129*9356374aSAndroid Build Coastguard Worker   return absl::string_view(++pa, count);
130*9356374aSAndroid Build Coastguard Worker }
131*9356374aSAndroid Build Coastguard Worker 
132*9356374aSAndroid Build Coastguard Worker ABSL_NAMESPACE_END
133*9356374aSAndroid Build Coastguard Worker }  // namespace absl
134