mirror of https://github.com/procxx/kepka.git
				
				
				
			
		
			
				
	
	
		
			319 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			319 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  *  Copyright 2004 The WebRTC Project Authors. All rights reserved.
 | |
|  *
 | |
|  *  Use of this source code is governed by a BSD-style license
 | |
|  *  that can be found in the LICENSE file in the root of the source
 | |
|  *  tree. An additional intellectual property rights grant can be found
 | |
|  *  in the file PATENTS.  All contributing project authors may
 | |
|  *  be found in the AUTHORS file in the root of the source tree.
 | |
|  */
 | |
| 
 | |
| #ifndef WEBRTC_BASE_STRINGUTILS_H__
 | |
| #define WEBRTC_BASE_STRINGUTILS_H__
 | |
| 
 | |
| #include <ctype.h>
 | |
| #include <stdarg.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| 
 | |
| #if defined(WEBRTC_WIN)
 | |
| #include <malloc.h>
 | |
| #include <wchar.h>
 | |
| #define alloca _alloca
 | |
| #endif  // WEBRTC_WIN 
 | |
| 
 | |
| #if defined(WEBRTC_POSIX)
 | |
| #ifdef BSD
 | |
| #include <stdlib.h>
 | |
| #else  // BSD
 | |
| #include <alloca.h>
 | |
| #endif  // !BSD
 | |
| #endif  // WEBRTC_POSIX
 | |
| 
 | |
| #include <string>
 | |
| 
 | |
| #include "webrtc/base/basictypes.h"
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // Generic string/memory utilities
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| #define STACK_ARRAY(TYPE, LEN) static_cast<TYPE*>(::alloca((LEN)*sizeof(TYPE)))
 | |
| 
 | |
| namespace rtc {
 | |
| 
 | |
| // Complement to memset.  Verifies memory consists of count bytes of value c.
 | |
| bool memory_check(const void* memory, int c, size_t count);
 | |
| 
 | |
| // Determines whether the simple wildcard pattern matches target.
 | |
| // Alpha characters in pattern match case-insensitively.
 | |
| // Asterisks in pattern match 0 or more characters.
 | |
| // Ex: string_match("www.TEST.GOOGLE.COM", "www.*.com") -> true
 | |
| bool string_match(const char* target, const char* pattern);
 | |
| 
 | |
| }  // namespace rtc
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // Rename a bunch of common string functions so they are consistent across
 | |
| // platforms and between char and wchar_t variants.
 | |
| // Here is the full list of functions that are unified:
 | |
| //  strlen, strcmp, stricmp, strncmp, strnicmp
 | |
| //  strchr, vsnprintf, strtoul, tolowercase
 | |
| // tolowercase is like tolower, but not compatible with end-of-file value
 | |
| //
 | |
| // It's not clear if we will ever use wchar_t strings on unix.  In theory,
 | |
| // all strings should be Utf8 all the time, except when interfacing with Win32
 | |
| // APIs that require Utf16.
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| inline char tolowercase(char c) {
 | |
|   return static_cast<char>(tolower(c));
 | |
| }
 | |
| 
 | |
| #if defined(WEBRTC_WIN)
 | |
| 
 | |
| inline size_t strlen(const wchar_t* s) {
 | |
|   return wcslen(s);
 | |
| }
 | |
| inline int strcmp(const wchar_t* s1, const wchar_t* s2) {
 | |
|   return wcscmp(s1, s2);
 | |
| }
 | |
| inline int stricmp(const wchar_t* s1, const wchar_t* s2) {
 | |
|   return _wcsicmp(s1, s2);
 | |
| }
 | |
| inline int strncmp(const wchar_t* s1, const wchar_t* s2, size_t n) {
 | |
|   return wcsncmp(s1, s2, n);
 | |
| }
 | |
| inline int strnicmp(const wchar_t* s1, const wchar_t* s2, size_t n) {
 | |
|   return _wcsnicmp(s1, s2, n);
 | |
| }
 | |
| inline const wchar_t* strchr(const wchar_t* s, wchar_t c) {
 | |
|   return wcschr(s, c);
 | |
| }
 | |
| inline const wchar_t* strstr(const wchar_t* haystack, const wchar_t* needle) {
 | |
|   return wcsstr(haystack, needle);
 | |
| }
 | |
| #ifndef vsnprintf
 | |
| inline int vsnprintf(wchar_t* buf, size_t n, const wchar_t* fmt, va_list args) {
 | |
|   return _vsnwprintf(buf, n, fmt, args);
 | |
| }
 | |
| #endif // !vsnprintf
 | |
| inline unsigned long strtoul(const wchar_t* snum, wchar_t** end, int base) {
 | |
|   return wcstoul(snum, end, base);
 | |
| }
 | |
| inline wchar_t tolowercase(wchar_t c) {
 | |
|   return static_cast<wchar_t>(towlower(c));
 | |
| }
 | |
| 
 | |
| #endif  // WEBRTC_WIN 
 | |
| 
 | |
| #if defined(WEBRTC_POSIX)
 | |
| 
 | |
| inline int _stricmp(const char* s1, const char* s2) {
 | |
|   return strcasecmp(s1, s2);
 | |
| }
 | |
| inline int _strnicmp(const char* s1, const char* s2, size_t n) {
 | |
|   return strncasecmp(s1, s2, n);
 | |
| }
 | |
| 
 | |
| #endif // WEBRTC_POSIX
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // Traits simplifies porting string functions to be CTYPE-agnostic
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| namespace rtc {
 | |
| 
 | |
| const size_t SIZE_UNKNOWN = static_cast<size_t>(-1);
 | |
| 
 | |
| template<class CTYPE>
 | |
| struct Traits {
 | |
|   // STL string type
 | |
|   //typedef XXX string;
 | |
|   // Null-terminated string
 | |
|   //inline static const CTYPE* empty_str();
 | |
| };
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // String utilities which work with char or wchar_t
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| template<class CTYPE>
 | |
| inline const CTYPE* nonnull(const CTYPE* str, const CTYPE* def_str = NULL) {
 | |
|   return str ? str : (def_str ? def_str : Traits<CTYPE>::empty_str());
 | |
| }
 | |
| 
 | |
| template<class CTYPE>
 | |
| const CTYPE* strchr(const CTYPE* str, const CTYPE* chs) {
 | |
|   for (size_t i=0; str[i]; ++i) {
 | |
|     for (size_t j=0; chs[j]; ++j) {
 | |
|       if (str[i] == chs[j]) {
 | |
|         return str + i;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| template<class CTYPE>
 | |
| const CTYPE* strchrn(const CTYPE* str, size_t slen, CTYPE ch) {
 | |
|   for (size_t i=0; i<slen && str[i]; ++i) {
 | |
|     if (str[i] == ch) {
 | |
|       return str + i;
 | |
|     }
 | |
|   }
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| template<class CTYPE>
 | |
| size_t strlenn(const CTYPE* buffer, size_t buflen) {
 | |
|   size_t bufpos = 0;
 | |
|   while (buffer[bufpos] && (bufpos < buflen)) {
 | |
|     ++bufpos;
 | |
|   }
 | |
|   return bufpos;
 | |
| }
 | |
| 
 | |
| // Safe versions of strncpy, strncat, snprintf and vsnprintf that always
 | |
| // null-terminate.
 | |
| 
 | |
| template<class CTYPE>
 | |
| size_t strcpyn(CTYPE* buffer, size_t buflen,
 | |
|                const CTYPE* source, size_t srclen = SIZE_UNKNOWN) {
 | |
|   if (buflen <= 0)
 | |
|     return 0;
 | |
| 
 | |
|   if (srclen == SIZE_UNKNOWN) {
 | |
|     srclen = strlenn(source, buflen - 1);
 | |
|   } else if (srclen >= buflen) {
 | |
|     srclen = buflen - 1;
 | |
|   }
 | |
|   memcpy(buffer, source, srclen * sizeof(CTYPE));
 | |
|   buffer[srclen] = 0;
 | |
|   return srclen;
 | |
| }
 | |
| 
 | |
| template<class CTYPE>
 | |
| size_t strcatn(CTYPE* buffer, size_t buflen,
 | |
|                const CTYPE* source, size_t srclen = SIZE_UNKNOWN) {
 | |
|   if (buflen <= 0)
 | |
|     return 0;
 | |
| 
 | |
|   size_t bufpos = strlenn(buffer, buflen - 1);
 | |
|   return bufpos + strcpyn(buffer + bufpos, buflen - bufpos, source, srclen);
 | |
| }
 | |
| 
 | |
| // Some compilers (clang specifically) require vsprintfn be defined before
 | |
| // sprintfn.
 | |
| template<class CTYPE>
 | |
| size_t vsprintfn(CTYPE* buffer, size_t buflen, const CTYPE* format,
 | |
|                  va_list args) {
 | |
|   int len = vsnprintf(buffer, buflen, format, args);
 | |
|   if ((len < 0) || (static_cast<size_t>(len) >= buflen)) {
 | |
|     len = static_cast<int>(buflen - 1);
 | |
|     buffer[len] = 0;
 | |
|   }
 | |
|   return len;
 | |
| }
 | |
| 
 | |
| template<class CTYPE>
 | |
| size_t sprintfn(CTYPE* buffer, size_t buflen, const CTYPE* format, ...);
 | |
| template<class CTYPE>
 | |
| size_t sprintfn(CTYPE* buffer, size_t buflen, const CTYPE* format, ...) {
 | |
|   va_list args;
 | |
|   va_start(args, format);
 | |
|   size_t len = vsprintfn(buffer, buflen, format, args);
 | |
|   va_end(args);
 | |
|   return len;
 | |
| }
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // Allow safe comparing and copying ascii (not UTF-8) with both wide and
 | |
| // non-wide character strings.
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| inline int asccmp(const char* s1, const char* s2) {
 | |
|   return strcmp(s1, s2);
 | |
| }
 | |
| inline int ascicmp(const char* s1, const char* s2) {
 | |
|   return strcmp(s1, s2);
 | |
| }
 | |
| inline int ascncmp(const char* s1, const char* s2, size_t n) {
 | |
|   return strncmp(s1, s2, n);
 | |
| }
 | |
| inline int ascnicmp(const char* s1, const char* s2, size_t n) {
 | |
|   return strncmp(s1, s2, n);
 | |
| }
 | |
| inline size_t asccpyn(char* buffer, size_t buflen,
 | |
|                       const char* source, size_t srclen = SIZE_UNKNOWN) {
 | |
|   return strcpyn(buffer, buflen, source, srclen);
 | |
| }
 | |
| 
 | |
| #if defined(WEBRTC_WIN)
 | |
| 
 | |
| typedef wchar_t(*CharacterTransformation)(wchar_t);
 | |
| inline wchar_t identity(wchar_t c) { return c; }
 | |
| int ascii_string_compare(const wchar_t* s1, const char* s2, size_t n,
 | |
|                          CharacterTransformation transformation);
 | |
| 
 | |
| inline int asccmp(const wchar_t* s1, const char* s2) {
 | |
|   return ascii_string_compare(s1, s2, static_cast<size_t>(-1), identity);
 | |
| }
 | |
| inline int ascicmp(const wchar_t* s1, const char* s2) {
 | |
|   return ascii_string_compare(s1, s2, static_cast<size_t>(-1), tolowercase);
 | |
| }
 | |
| inline int ascncmp(const wchar_t* s1, const char* s2, size_t n) {
 | |
|   return ascii_string_compare(s1, s2, n, identity);
 | |
| }
 | |
| inline int ascnicmp(const wchar_t* s1, const char* s2, size_t n) {
 | |
|   return ascii_string_compare(s1, s2, n, tolowercase);
 | |
| }
 | |
| size_t asccpyn(wchar_t* buffer, size_t buflen,
 | |
|                const char* source, size_t srclen = SIZE_UNKNOWN);
 | |
| 
 | |
| #endif  // WEBRTC_WIN 
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // Traits<char> specializations
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| template<>
 | |
| struct Traits<char> {
 | |
|   typedef std::string string;
 | |
|   inline static const char* empty_str() { return ""; }
 | |
| };
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // Traits<wchar_t> specializations (Windows only, currently)
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| #if defined(WEBRTC_WIN)
 | |
| 
 | |
| template<>
 | |
| struct Traits<wchar_t> {
 | |
|   typedef std::wstring string;
 | |
|   inline static const wchar_t* empty_str() { return L""; }
 | |
| };
 | |
| 
 | |
| #endif  // WEBRTC_WIN 
 | |
| 
 | |
| // Replaces all occurrences of "search" with "replace".
 | |
| void replace_substrs(const char *search,
 | |
|                      size_t search_len,
 | |
|                      const char *replace,
 | |
|                      size_t replace_len,
 | |
|                      std::string *s);
 | |
| 
 | |
| // True iff s1 starts with s2.
 | |
| bool starts_with(const char *s1, const char *s2);
 | |
| 
 | |
| // True iff s1 ends with s2.
 | |
| bool ends_with(const char *s1, const char *s2);
 | |
| 
 | |
| // Remove leading and trailing whitespaces.
 | |
| std::string string_trim(const std::string& s);
 | |
| 
 | |
| }  // namespace rtc
 | |
| 
 | |
| #endif // WEBRTC_BASE_STRINGUTILS_H__
 |