git-subtree-dir: third_party/abseil_cpp git-subtree-mainline:ffb2ae54begit-subtree-split:768eb2ca28
		
			
				
	
	
		
			148 lines
		
	
	
	
		
			5.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			148 lines
		
	
	
	
		
			5.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
//
 | 
						|
// 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.
 | 
						|
//
 | 
						|
 | 
						|
// These routines provide mem versions of standard C string routines,
 | 
						|
// such as strpbrk.  They function exactly the same as the str versions,
 | 
						|
// so if you wonder what they are, replace the word "mem" by
 | 
						|
// "str" and check out the man page.  I could return void*, as the
 | 
						|
// strutil.h mem*() routines tend to do, but I return char* instead
 | 
						|
// since this is by far the most common way these functions are called.
 | 
						|
//
 | 
						|
// The difference between the mem and str versions is the mem version
 | 
						|
// takes a pointer and a length, rather than a '\0'-terminated string.
 | 
						|
// The memcase* routines defined here assume the locale is "C"
 | 
						|
// (they use absl::ascii_tolower instead of tolower).
 | 
						|
//
 | 
						|
// These routines are based on the BSD library.
 | 
						|
//
 | 
						|
// Here's a list of routines from string.h, and their mem analogues.
 | 
						|
// Functions in lowercase are defined in string.h; those in UPPERCASE
 | 
						|
// are defined here:
 | 
						|
//
 | 
						|
// strlen                  --
 | 
						|
// strcat strncat          MEMCAT
 | 
						|
// strcpy strncpy          memcpy
 | 
						|
// --                      memccpy   (very cool function, btw)
 | 
						|
// --                      memmove
 | 
						|
// --                      memset
 | 
						|
// strcmp strncmp          memcmp
 | 
						|
// strcasecmp strncasecmp  MEMCASECMP
 | 
						|
// strchr                  memchr
 | 
						|
// strcoll                 --
 | 
						|
// strxfrm                 --
 | 
						|
// strdup strndup          MEMDUP
 | 
						|
// strrchr                 MEMRCHR
 | 
						|
// strspn                  MEMSPN
 | 
						|
// strcspn                 MEMCSPN
 | 
						|
// strpbrk                 MEMPBRK
 | 
						|
// strstr                  MEMSTR MEMMEM
 | 
						|
// (g)strcasestr           MEMCASESTR MEMCASEMEM
 | 
						|
// strtok                  --
 | 
						|
// strprefix               MEMPREFIX      (strprefix is from strutil.h)
 | 
						|
// strcaseprefix           MEMCASEPREFIX  (strcaseprefix is from strutil.h)
 | 
						|
// strsuffix               MEMSUFFIX      (strsuffix is from strutil.h)
 | 
						|
// strcasesuffix           MEMCASESUFFIX  (strcasesuffix is from strutil.h)
 | 
						|
// --                      MEMIS
 | 
						|
// --                      MEMCASEIS
 | 
						|
// strcount                MEMCOUNT       (strcount is from strutil.h)
 | 
						|
 | 
						|
#ifndef ABSL_STRINGS_INTERNAL_MEMUTIL_H_
 | 
						|
#define ABSL_STRINGS_INTERNAL_MEMUTIL_H_
 | 
						|
 | 
						|
#include <cstddef>
 | 
						|
#include <cstring>
 | 
						|
 | 
						|
#include "absl/base/port.h"  // disable some warnings on Windows
 | 
						|
#include "absl/strings/ascii.h"  // for absl::ascii_tolower
 | 
						|
 | 
						|
namespace absl {
 | 
						|
ABSL_NAMESPACE_BEGIN
 | 
						|
namespace strings_internal {
 | 
						|
 | 
						|
inline char* memcat(char* dest, size_t destlen, const char* src,
 | 
						|
                    size_t srclen) {
 | 
						|
  return reinterpret_cast<char*>(memcpy(dest + destlen, src, srclen));
 | 
						|
}
 | 
						|
 | 
						|
int memcasecmp(const char* s1, const char* s2, size_t len);
 | 
						|
char* memdup(const char* s, size_t slen);
 | 
						|
char* memrchr(const char* s, int c, size_t slen);
 | 
						|
size_t memspn(const char* s, size_t slen, const char* accept);
 | 
						|
size_t memcspn(const char* s, size_t slen, const char* reject);
 | 
						|
char* mempbrk(const char* s, size_t slen, const char* accept);
 | 
						|
 | 
						|
// This is for internal use only.  Don't call this directly
 | 
						|
template <bool case_sensitive>
 | 
						|
const char* int_memmatch(const char* haystack, size_t haylen,
 | 
						|
                         const char* needle, size_t neelen) {
 | 
						|
  if (0 == neelen) {
 | 
						|
    return haystack;  // even if haylen is 0
 | 
						|
  }
 | 
						|
  const char* hayend = haystack + haylen;
 | 
						|
  const char* needlestart = needle;
 | 
						|
  const char* needleend = needlestart + neelen;
 | 
						|
 | 
						|
  for (; haystack < hayend; ++haystack) {
 | 
						|
    char hay = case_sensitive
 | 
						|
                   ? *haystack
 | 
						|
                   : absl::ascii_tolower(static_cast<unsigned char>(*haystack));
 | 
						|
    char nee = case_sensitive
 | 
						|
                   ? *needle
 | 
						|
                   : absl::ascii_tolower(static_cast<unsigned char>(*needle));
 | 
						|
    if (hay == nee) {
 | 
						|
      if (++needle == needleend) {
 | 
						|
        return haystack + 1 - neelen;
 | 
						|
      }
 | 
						|
    } else if (needle != needlestart) {
 | 
						|
      // must back up haystack in case a prefix matched (find "aab" in "aaab")
 | 
						|
      haystack -= needle - needlestart;  // for loop will advance one more
 | 
						|
      needle = needlestart;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return nullptr;
 | 
						|
}
 | 
						|
 | 
						|
// These are the guys you can call directly
 | 
						|
inline const char* memstr(const char* phaystack, size_t haylen,
 | 
						|
                          const char* pneedle) {
 | 
						|
  return int_memmatch<true>(phaystack, haylen, pneedle, strlen(pneedle));
 | 
						|
}
 | 
						|
 | 
						|
inline const char* memcasestr(const char* phaystack, size_t haylen,
 | 
						|
                              const char* pneedle) {
 | 
						|
  return int_memmatch<false>(phaystack, haylen, pneedle, strlen(pneedle));
 | 
						|
}
 | 
						|
 | 
						|
inline const char* memmem(const char* phaystack, size_t haylen,
 | 
						|
                          const char* pneedle, size_t needlelen) {
 | 
						|
  return int_memmatch<true>(phaystack, haylen, pneedle, needlelen);
 | 
						|
}
 | 
						|
 | 
						|
inline const char* memcasemem(const char* phaystack, size_t haylen,
 | 
						|
                              const char* pneedle, size_t needlelen) {
 | 
						|
  return int_memmatch<false>(phaystack, haylen, pneedle, needlelen);
 | 
						|
}
 | 
						|
 | 
						|
// This is significantly faster for case-sensitive matches with very
 | 
						|
// few possible matches.  See unit test for benchmarks.
 | 
						|
const char* memmatch(const char* phaystack, size_t haylen, const char* pneedle,
 | 
						|
                     size_t neelen);
 | 
						|
 | 
						|
}  // namespace strings_internal
 | 
						|
ABSL_NAMESPACE_END
 | 
						|
}  // namespace absl
 | 
						|
 | 
						|
#endif  // ABSL_STRINGS_INTERNAL_MEMUTIL_H_
 |