1*ec779b8eSAndroid Build Coastguard Worker /*
2*ec779b8eSAndroid Build Coastguard Worker * Copyright 2014 The Android Open Source Project
3*ec779b8eSAndroid Build Coastguard Worker *
4*ec779b8eSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*ec779b8eSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*ec779b8eSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*ec779b8eSAndroid Build Coastguard Worker *
8*ec779b8eSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*ec779b8eSAndroid Build Coastguard Worker *
10*ec779b8eSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*ec779b8eSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*ec779b8eSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*ec779b8eSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*ec779b8eSAndroid Build Coastguard Worker * limitations under the License.
15*ec779b8eSAndroid Build Coastguard Worker */
16*ec779b8eSAndroid Build Coastguard Worker
17*ec779b8eSAndroid Build Coastguard Worker #include <string.h>
18*ec779b8eSAndroid Build Coastguard Worker #include <AStringUtils.h>
19*ec779b8eSAndroid Build Coastguard Worker
20*ec779b8eSAndroid Build Coastguard Worker namespace android {
21*ec779b8eSAndroid Build Coastguard Worker
22*ec779b8eSAndroid Build Coastguard Worker // static
Compare(const char * a,const char * b,size_t len,bool ignoreCase)23*ec779b8eSAndroid Build Coastguard Worker int AStringUtils::Compare(const char *a, const char *b, size_t len, bool ignoreCase) {
24*ec779b8eSAndroid Build Coastguard Worker // this method relies on a trailing '\0' if a or b are shorter than len
25*ec779b8eSAndroid Build Coastguard Worker return ignoreCase ? strncasecmp(a, b, len) : strncmp(a, b, len);
26*ec779b8eSAndroid Build Coastguard Worker }
27*ec779b8eSAndroid Build Coastguard Worker
28*ec779b8eSAndroid Build Coastguard Worker // static
MatchesGlob(const char * glob,size_t globLen,const char * str,size_t strLen,bool ignoreCase)29*ec779b8eSAndroid Build Coastguard Worker bool AStringUtils::MatchesGlob(
30*ec779b8eSAndroid Build Coastguard Worker const char *glob, size_t globLen, const char *str, size_t strLen, bool ignoreCase) {
31*ec779b8eSAndroid Build Coastguard Worker // this method does not assume a trailing '\0'
32*ec779b8eSAndroid Build Coastguard Worker size_t ix = 0, globIx = 0;
33*ec779b8eSAndroid Build Coastguard Worker
34*ec779b8eSAndroid Build Coastguard Worker // pattern must match until first '*'
35*ec779b8eSAndroid Build Coastguard Worker while (globIx < globLen && glob[globIx] != '*') {
36*ec779b8eSAndroid Build Coastguard Worker ++globIx;
37*ec779b8eSAndroid Build Coastguard Worker }
38*ec779b8eSAndroid Build Coastguard Worker if (strLen < globIx || Compare(str, glob, globIx /* len */, ignoreCase)) {
39*ec779b8eSAndroid Build Coastguard Worker return false;
40*ec779b8eSAndroid Build Coastguard Worker }
41*ec779b8eSAndroid Build Coastguard Worker ix = globIx;
42*ec779b8eSAndroid Build Coastguard Worker
43*ec779b8eSAndroid Build Coastguard Worker // process by * separated sections
44*ec779b8eSAndroid Build Coastguard Worker while (globIx < globLen) {
45*ec779b8eSAndroid Build Coastguard Worker ++globIx;
46*ec779b8eSAndroid Build Coastguard Worker size_t start = globIx;
47*ec779b8eSAndroid Build Coastguard Worker while (globIx < globLen && glob[globIx] != '*') {
48*ec779b8eSAndroid Build Coastguard Worker ++globIx;
49*ec779b8eSAndroid Build Coastguard Worker }
50*ec779b8eSAndroid Build Coastguard Worker size_t len = globIx - start;
51*ec779b8eSAndroid Build Coastguard Worker const char *pattern = glob + start;
52*ec779b8eSAndroid Build Coastguard Worker
53*ec779b8eSAndroid Build Coastguard Worker if (globIx == globLen) {
54*ec779b8eSAndroid Build Coastguard Worker // last pattern must match tail
55*ec779b8eSAndroid Build Coastguard Worker if (ix + len > strLen) {
56*ec779b8eSAndroid Build Coastguard Worker return false;
57*ec779b8eSAndroid Build Coastguard Worker }
58*ec779b8eSAndroid Build Coastguard Worker const char *tail = str + strLen - len;
59*ec779b8eSAndroid Build Coastguard Worker return !Compare(tail, pattern, len, ignoreCase);
60*ec779b8eSAndroid Build Coastguard Worker }
61*ec779b8eSAndroid Build Coastguard Worker // progress after first occurrence of pattern
62*ec779b8eSAndroid Build Coastguard Worker while (ix + len <= strLen && Compare(str + ix, pattern, len, ignoreCase)) {
63*ec779b8eSAndroid Build Coastguard Worker ++ix;
64*ec779b8eSAndroid Build Coastguard Worker }
65*ec779b8eSAndroid Build Coastguard Worker if (ix + len > strLen) {
66*ec779b8eSAndroid Build Coastguard Worker return false;
67*ec779b8eSAndroid Build Coastguard Worker }
68*ec779b8eSAndroid Build Coastguard Worker ix += len;
69*ec779b8eSAndroid Build Coastguard Worker // we will loop around as globIx < globLen
70*ec779b8eSAndroid Build Coastguard Worker }
71*ec779b8eSAndroid Build Coastguard Worker
72*ec779b8eSAndroid Build Coastguard Worker // we only get here if there were no * in the pattern
73*ec779b8eSAndroid Build Coastguard Worker return ix == strLen;
74*ec779b8eSAndroid Build Coastguard Worker }
75*ec779b8eSAndroid Build Coastguard Worker
76*ec779b8eSAndroid Build Coastguard Worker } // namespace android
77*ec779b8eSAndroid Build Coastguard Worker
78