1 /*
2  * Copyright (C) 2014 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include <ctype.h>
18 
19 #include <android-base/stringprintf.h>
20 #include <cutils/properties.h>
21 
22 #include "LogWhiteBlackList.h"
23 
24 // White and Black list
25 
Prune(uid_t uid,pid_t pid)26 Prune::Prune(uid_t uid, pid_t pid) : mUid(uid), mPid(pid) {
27 }
28 
cmp(uid_t uid,pid_t pid) const29 int Prune::cmp(uid_t uid, pid_t pid) const {
30     if ((mUid == uid_all) || (mUid == uid)) {
31         if (mPid == pid_all) {
32             return 0;
33         }
34         return pid - mPid;
35     }
36     return uid - mUid;
37 }
38 
format()39 std::string Prune::format() {
40     if (mUid != uid_all) {
41         if (mPid != pid_all) {
42             return android::base::StringPrintf("%u/%u", mUid, mPid);
43         }
44         return android::base::StringPrintf("%u", mUid);
45     }
46     if (mPid != pid_all) {
47         return android::base::StringPrintf("/%u", mPid);
48     }
49     // NB: mPid == pid_all can not happen if mUid == uid_all
50     return std::string("/");
51 }
52 
PruneList()53 PruneList::PruneList() {
54     init(NULL);
55 }
56 
~PruneList()57 PruneList::~PruneList() {
58     PruneCollection::iterator it;
59     for (it = mNice.begin(); it != mNice.end();) {
60         it = mNice.erase(it);
61     }
62     for (it = mNaughty.begin(); it != mNaughty.end();) {
63         it = mNaughty.erase(it);
64     }
65 }
66 
init(const char * str)67 int PruneList::init(const char *str) {
68     mWorstUidEnabled = true;
69     mWorstPidOfSystemEnabled = true;
70     PruneCollection::iterator it;
71     for (it = mNice.begin(); it != mNice.end();) {
72         it = mNice.erase(it);
73     }
74     for (it = mNaughty.begin(); it != mNaughty.end();) {
75         it = mNaughty.erase(it);
76     }
77 
78     static const char _default[] = "default";
79     // default here means take ro.logd.filter, persist.logd.filter then
80     // internal default in that order.
81     if (str && !strcmp(str, _default)) {
82         str = NULL;
83     }
84     static const char _disable[] = "disable";
85     if (str && !strcmp(str, _disable)) {
86         str = "";
87     }
88 
89     std::string filter;
90 
91     if (str) {
92         filter = str;
93     } else {
94         char property[PROPERTY_VALUE_MAX];
95         property_get("ro.logd.filter", property, _default);
96         filter = property;
97         property_get("persist.logd.filter", property, filter.c_str());
98         // default here means take ro.logd.filter
99         if (strcmp(property, _default)) {
100             filter = property;
101         }
102     }
103 
104     // default here means take internal default.
105     if (filter == _default) {
106         // See README.property for description of filter format
107         filter = "~! ~1000/!";
108     }
109     if (filter == _disable) {
110         filter = "";
111     }
112 
113     mWorstUidEnabled = false;
114     mWorstPidOfSystemEnabled = false;
115 
116     for(str = filter.c_str(); *str; ++str) {
117         if (isspace(*str)) {
118             continue;
119         }
120 
121         PruneCollection *list;
122         if ((*str == '~') || (*str == '!')) { // ~ supported, ! undocumented
123             ++str;
124             // special case, translates to worst UID at priority in blacklist
125             if (*str == '!') {
126                 mWorstUidEnabled = true;
127                 ++str;
128                 if (!*str) {
129                     break;
130                 }
131                 if (!isspace(*str)) {
132                     return 1;
133                 }
134                 continue;
135             }
136             // special case, translated to worst PID of System at priority
137             static const char worstPid[] = "1000/!";
138             if (!strncmp(str, worstPid, sizeof(worstPid) - 1)) {
139                 mWorstPidOfSystemEnabled = true;
140                 str += sizeof(worstPid) - 1;
141                 if (!*str) {
142                     break;
143                 }
144                 if (!isspace(*str)) {
145                     return 1;
146                 }
147                 continue;
148             }
149             if (!*str) {
150                 return 1;
151             }
152             list = &mNaughty;
153         } else {
154             list = &mNice;
155         }
156 
157         uid_t uid = Prune::uid_all;
158         if (isdigit(*str)) {
159             uid = 0;
160             do {
161                 uid = uid * 10 + *str++ - '0';
162             } while (isdigit(*str));
163         }
164 
165         pid_t pid = Prune::pid_all;
166         if (*str == '/') {
167             ++str;
168             if (isdigit(*str)) {
169                 pid = 0;
170                 do {
171                     pid = pid * 10 + *str++ - '0';
172                 } while (isdigit(*str));
173             }
174         }
175 
176         if ((uid == Prune::uid_all) && (pid == Prune::pid_all)) {
177             return 1;
178         }
179 
180         if (*str && !isspace(*str)) {
181             return 1;
182         }
183 
184         // insert sequentially into list
185         PruneCollection::iterator it = list->begin();
186         while (it != list->end()) {
187             Prune &p = *it;
188             int m = uid - p.mUid;
189             if (m == 0) {
190                 if (p.mPid == p.pid_all) {
191                     break;
192                 }
193                 if ((pid == p.pid_all) && (p.mPid != p.pid_all)) {
194                     it = list->erase(it);
195                     continue;
196                 }
197                 m = pid - p.mPid;
198             }
199             if (m <= 0) {
200                 if (m < 0) {
201                     list->insert(it, Prune(uid,pid));
202                 }
203                 break;
204             }
205             ++it;
206         }
207         if (it == list->end()) {
208             list->push_back(Prune(uid,pid));
209         }
210         if (!*str) {
211             break;
212         }
213     }
214 
215     return 0;
216 }
217 
format()218 std::string PruneList::format() {
219     static const char nice_format[] = " %s";
220     const char *fmt = nice_format + 1;
221 
222     std::string string;
223 
224     if (mWorstUidEnabled) {
225         string = "~!";
226         fmt = nice_format;
227         if (mWorstPidOfSystemEnabled) {
228             string += " ~1000/!";
229         }
230     }
231 
232     PruneCollection::iterator it;
233 
234     for (it = mNice.begin(); it != mNice.end(); ++it) {
235         string += android::base::StringPrintf(fmt, (*it).format().c_str());
236         fmt = nice_format;
237     }
238 
239     static const char naughty_format[] = " ~%s";
240     fmt = naughty_format + (*fmt != ' ');
241     for (it = mNaughty.begin(); it != mNaughty.end(); ++it) {
242         string += android::base::StringPrintf(fmt, (*it).format().c_str());
243         fmt = naughty_format;
244     }
245 
246     return string;
247 }
248 
249 // ToDo: Lists are in sorted order, Prune->cmp() returns + or -
250 // If there is scaling issues, resort to a better algorithm than linear
251 // based on these assumptions.
252 
naughty(LogBufferElement * element)253 bool PruneList::naughty(LogBufferElement *element) {
254     PruneCollection::iterator it;
255     for (it = mNaughty.begin(); it != mNaughty.end(); ++it) {
256         if (!(*it).cmp(element)) {
257             return true;
258         }
259     }
260     return false;
261 }
262 
nice(LogBufferElement * element)263 bool PruneList::nice(LogBufferElement *element) {
264     PruneCollection::iterator it;
265     for (it = mNice.begin(); it != mNice.end(); ++it) {
266         if (!(*it).cmp(element)) {
267             return true;
268         }
269     }
270     return false;
271 }
272