• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #ifndef NET_TOOLS_BALSA_STRING_PIECE_UTILS_H_
6 #define NET_TOOLS_BALSA_STRING_PIECE_UTILS_H_
7 
8 #include <ctype.h>
9 
10 #include "base/port.h"
11 #include "base/strings/string_piece.h"
12 
13 namespace net {
14 
15 #if defined(COMPILER_MSVC)
16 struct StringPieceCaseCompare {
17   static const size_t bucket_size = 4;
18 
operatorStringPieceCaseCompare19   size_t operator()(const base::StringPiece& sp) const {
20     // based on __stl_string_hash in http://www.sgi.com/tech/stl/string
21     size_t hash_val = 0;
22     for (base::StringPiece::const_iterator it = sp.begin();
23          it != sp.end(); ++it) {
24       hash_val = 5 * hash_val + tolower(*it);
25     }
26     return hash_val;
27   }
28 
operatorStringPieceCaseCompare29   bool operator()(const base::StringPiece& sp1,
30                   const base::StringPiece& sp2) const {
31     size_t len1 = sp1.length();
32     size_t len2 = sp2.length();
33     bool sp1_shorter = len1 < len2;
34     size_t len = sp1_shorter ? len1 : len2;
35     int rv = _memicmp(sp1.data(), sp2.data(), len);
36     if (rv == 0) {
37       return sp1_shorter;
38     }
39     return rv < 0;
40   }
41 };
42 #else  // COMPILER_MSVC
43 struct StringPieceCaseHash {
44   size_t operator()(const base::StringPiece& sp) const {
45     // based on __stl_string_hash in http://www.sgi.com/tech/stl/string
46     size_t hash_val = 0;
47     for (base::StringPiece::const_iterator it = sp.begin();
48          it != sp.end(); ++it) {
49       hash_val = 5 * hash_val + tolower(*it);
50     }
51     return hash_val;
52   }
53 };
54 #endif  // COMPILER_MSVC
55 
56 struct StringPieceUtils {
EqualIgnoreCaseStringPieceUtils57   static bool EqualIgnoreCase(const base::StringPiece& piece1,
58                               const base::StringPiece& piece2) {
59     base::StringPiece::const_iterator p1i = piece1.begin();
60     base::StringPiece::const_iterator p2i = piece2.begin();
61     if (piece1.empty() && piece2.empty()) {
62       return true;
63     } else if (piece1.size() != piece2.size()) {
64       return false;
65     }
66     while (p1i != piece1.end() && p2i != piece2.end()) {
67       if (tolower(*p1i) != tolower(*p2i))
68         return false;
69       ++p1i;
70       ++p2i;
71     }
72     return true;
73   }
74 
RemoveWhitespaceContextStringPieceUtils75   static void RemoveWhitespaceContext(base::StringPiece* piece1) {
76     base::StringPiece::const_iterator c = piece1->begin();
77     base::StringPiece::const_iterator e = piece1->end();
78     while (c != e && isspace(*c)) {
79       ++c;
80     }
81     if (c == e) {
82       *piece1 = base::StringPiece(c, e-c);
83       return;
84     }
85     --e;
86     while (c != e &&isspace(*e)) {
87       --e;
88     }
89     ++e;
90     *piece1 = base::StringPiece(c, e-c);
91   }
92 
StartsWithIgnoreCaseStringPieceUtils93   static bool StartsWithIgnoreCase(const base::StringPiece& text,
94                                    const base::StringPiece& starts_with) {
95     if (text.size() < starts_with.size())
96       return false;
97     return EqualIgnoreCase(text.substr(0, starts_with.size()), starts_with);
98   }
99 };
100 struct StringPieceCaseEqual {
operatorStringPieceCaseEqual101   bool operator()(const base::StringPiece& piece1,
102                   const base::StringPiece& piece2) const {
103     return StringPieceUtils::EqualIgnoreCase(piece1, piece2);
104   }
105 };
106 
107 
108 
109 }  // namespace net
110 
111 #endif  // NET_TOOLS_BALSA_STRING_PIECE_UTILS_H_
112 
113