• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2009 The RE2 Authors.  All Rights Reserved.
2 // Use of this source code is governed by a BSD-style
3 // license that can be found in the LICENSE file.
4 
5 #include "re2/prefilter_tree.h"
6 
7 #include <stddef.h>
8 #include <algorithm>
9 #include <map>
10 #include <memory>
11 #include <set>
12 #include <string>
13 #include <utility>
14 #include <vector>
15 
16 #include "util/util.h"
17 #include "util/logging.h"
18 #include "util/strutil.h"
19 #include "re2/prefilter.h"
20 #include "re2/re2.h"
21 
22 namespace re2 {
23 
24 static const bool ExtraDebug = false;
25 
PrefilterTree()26 PrefilterTree::PrefilterTree()
27     : compiled_(false),
28       min_atom_len_(3) {
29 }
30 
PrefilterTree(int min_atom_len)31 PrefilterTree::PrefilterTree(int min_atom_len)
32     : compiled_(false),
33       min_atom_len_(min_atom_len) {
34 }
35 
~PrefilterTree()36 PrefilterTree::~PrefilterTree() {
37   for (size_t i = 0; i < prefilter_vec_.size(); i++)
38     delete prefilter_vec_[i];
39 
40   for (size_t i = 0; i < entries_.size(); i++)
41     delete entries_[i].parents;
42 }
43 
Add(Prefilter * prefilter)44 void PrefilterTree::Add(Prefilter* prefilter) {
45   if (compiled_) {
46     LOG(DFATAL) << "Add called after Compile.";
47     return;
48   }
49   if (prefilter != NULL && !KeepNode(prefilter)) {
50     delete prefilter;
51     prefilter = NULL;
52   }
53 
54   prefilter_vec_.push_back(prefilter);
55 }
56 
Compile(std::vector<std::string> * atom_vec)57 void PrefilterTree::Compile(std::vector<std::string>* atom_vec) {
58   if (compiled_) {
59     LOG(DFATAL) << "Compile called already.";
60     return;
61   }
62 
63   // Some legacy users of PrefilterTree call Compile() before
64   // adding any regexps and expect Compile() to have no effect.
65   if (prefilter_vec_.empty())
66     return;
67 
68   compiled_ = true;
69 
70   // TODO(junyer): Use std::unordered_set<Prefilter*> instead?
71   NodeMap nodes;
72   AssignUniqueIds(&nodes, atom_vec);
73 
74   // Identify nodes that are too common among prefilters and are
75   // triggering too many parents. Then get rid of them if possible.
76   // Note that getting rid of a prefilter node simply means they are
77   // no longer necessary for their parent to trigger; that is, we do
78   // not miss out on any regexps triggering by getting rid of a
79   // prefilter node.
80   for (size_t i = 0; i < entries_.size(); i++) {
81     StdIntMap* parents = entries_[i].parents;
82     if (parents->size() > 8) {
83       // This one triggers too many things. If all the parents are AND
84       // nodes and have other things guarding them, then get rid of
85       // this trigger. TODO(vsri): Adjust the threshold appropriately,
86       // make it a function of total number of nodes?
87       bool have_other_guard = true;
88       for (StdIntMap::iterator it = parents->begin();
89            it != parents->end(); ++it) {
90         have_other_guard = have_other_guard &&
91             (entries_[it->first].propagate_up_at_count > 1);
92       }
93 
94       if (have_other_guard) {
95         for (StdIntMap::iterator it = parents->begin();
96              it != parents->end(); ++it)
97           entries_[it->first].propagate_up_at_count -= 1;
98 
99         parents->clear();  // Forget the parents
100       }
101     }
102   }
103 
104   if (ExtraDebug)
105     PrintDebugInfo(&nodes);
106 }
107 
CanonicalNode(NodeMap * nodes,Prefilter * node)108 Prefilter* PrefilterTree::CanonicalNode(NodeMap* nodes, Prefilter* node) {
109   std::string node_string = NodeString(node);
110   std::map<std::string, Prefilter*>::iterator iter = nodes->find(node_string);
111   if (iter == nodes->end())
112     return NULL;
113   return (*iter).second;
114 }
115 
NodeString(Prefilter * node) const116 std::string PrefilterTree::NodeString(Prefilter* node) const {
117   // Adding the operation disambiguates AND/OR/atom nodes.
118   std::string s = StringPrintf("%d", node->op()) + ":";
119   if (node->op() == Prefilter::ATOM) {
120     s += node->atom();
121   } else {
122     for (size_t i = 0; i < node->subs()->size(); i++) {
123       if (i > 0)
124         s += ',';
125       s += StringPrintf("%d", (*node->subs())[i]->unique_id());
126     }
127   }
128   return s;
129 }
130 
KeepNode(Prefilter * node) const131 bool PrefilterTree::KeepNode(Prefilter* node) const {
132   if (node == NULL)
133     return false;
134 
135   switch (node->op()) {
136     default:
137       LOG(DFATAL) << "Unexpected op in KeepNode: " << node->op();
138       return false;
139 
140     case Prefilter::ALL:
141     case Prefilter::NONE:
142       return false;
143 
144     case Prefilter::ATOM:
145       return node->atom().size() >= static_cast<size_t>(min_atom_len_);
146 
147     case Prefilter::AND: {
148       int j = 0;
149       std::vector<Prefilter*>* subs = node->subs();
150       for (size_t i = 0; i < subs->size(); i++)
151         if (KeepNode((*subs)[i]))
152           (*subs)[j++] = (*subs)[i];
153         else
154           delete (*subs)[i];
155 
156       subs->resize(j);
157       return j > 0;
158     }
159 
160     case Prefilter::OR:
161       for (size_t i = 0; i < node->subs()->size(); i++)
162         if (!KeepNode((*node->subs())[i]))
163           return false;
164       return true;
165   }
166 }
167 
AssignUniqueIds(NodeMap * nodes,std::vector<std::string> * atom_vec)168 void PrefilterTree::AssignUniqueIds(NodeMap* nodes,
169                                     std::vector<std::string>* atom_vec) {
170   atom_vec->clear();
171 
172   // Build vector of all filter nodes, sorted topologically
173   // from top to bottom in v.
174   std::vector<Prefilter*> v;
175 
176   // Add the top level nodes of each regexp prefilter.
177   for (size_t i = 0; i < prefilter_vec_.size(); i++) {
178     Prefilter* f = prefilter_vec_[i];
179     if (f == NULL)
180       unfiltered_.push_back(static_cast<int>(i));
181 
182     // We push NULL also on to v, so that we maintain the
183     // mapping of index==regexpid for level=0 prefilter nodes.
184     v.push_back(f);
185   }
186 
187   // Now add all the descendant nodes.
188   for (size_t i = 0; i < v.size(); i++) {
189     Prefilter* f = v[i];
190     if (f == NULL)
191       continue;
192     if (f->op() == Prefilter::AND || f->op() == Prefilter::OR) {
193       const std::vector<Prefilter*>& subs = *f->subs();
194       for (size_t j = 0; j < subs.size(); j++)
195         v.push_back(subs[j]);
196     }
197   }
198 
199   // Identify unique nodes.
200   int unique_id = 0;
201   for (int i = static_cast<int>(v.size()) - 1; i >= 0; i--) {
202     Prefilter *node = v[i];
203     if (node == NULL)
204       continue;
205     node->set_unique_id(-1);
206     Prefilter* canonical = CanonicalNode(nodes, node);
207     if (canonical == NULL) {
208       // Any further nodes that have the same node string
209       // will find this node as the canonical node.
210       nodes->emplace(NodeString(node), node);
211       if (node->op() == Prefilter::ATOM) {
212         atom_vec->push_back(node->atom());
213         atom_index_to_id_.push_back(unique_id);
214       }
215       node->set_unique_id(unique_id++);
216     } else {
217       node->set_unique_id(canonical->unique_id());
218     }
219   }
220   entries_.resize(nodes->size());
221 
222   // Create parent StdIntMap for the entries.
223   for (int i = static_cast<int>(v.size()) - 1; i >= 0; i--) {
224     Prefilter* prefilter = v[i];
225     if (prefilter == NULL)
226       continue;
227 
228     if (CanonicalNode(nodes, prefilter) != prefilter)
229       continue;
230 
231     Entry* entry = &entries_[prefilter->unique_id()];
232     entry->parents = new StdIntMap();
233   }
234 
235   // Fill the entries.
236   for (int i = static_cast<int>(v.size()) - 1; i >= 0; i--) {
237     Prefilter* prefilter = v[i];
238     if (prefilter == NULL)
239       continue;
240 
241     if (CanonicalNode(nodes, prefilter) != prefilter)
242       continue;
243 
244     Entry* entry = &entries_[prefilter->unique_id()];
245 
246     switch (prefilter->op()) {
247       default:
248       case Prefilter::ALL:
249         LOG(DFATAL) << "Unexpected op: " << prefilter->op();
250         return;
251 
252       case Prefilter::ATOM:
253         entry->propagate_up_at_count = 1;
254         break;
255 
256       case Prefilter::OR:
257       case Prefilter::AND: {
258         std::set<int> uniq_child;
259         for (size_t j = 0; j < prefilter->subs()->size(); j++) {
260           Prefilter* child = (*prefilter->subs())[j];
261           Prefilter* canonical = CanonicalNode(nodes, child);
262           if (canonical == NULL) {
263             LOG(DFATAL) << "Null canonical node";
264             return;
265           }
266           int child_id = canonical->unique_id();
267           uniq_child.insert(child_id);
268           // To the child, we want to add to parent indices.
269           Entry* child_entry = &entries_[child_id];
270           if (child_entry->parents->find(prefilter->unique_id()) ==
271               child_entry->parents->end()) {
272             (*child_entry->parents)[prefilter->unique_id()] = 1;
273           }
274         }
275         entry->propagate_up_at_count = prefilter->op() == Prefilter::AND
276                                            ? static_cast<int>(uniq_child.size())
277                                            : 1;
278 
279         break;
280       }
281     }
282   }
283 
284   // For top level nodes, populate regexp id.
285   for (size_t i = 0; i < prefilter_vec_.size(); i++) {
286     if (prefilter_vec_[i] == NULL)
287       continue;
288     int id = CanonicalNode(nodes, prefilter_vec_[i])->unique_id();
289     DCHECK_LE(0, id);
290     Entry* entry = &entries_[id];
291     entry->regexps.push_back(static_cast<int>(i));
292   }
293 }
294 
295 // Functions for triggering during search.
RegexpsGivenStrings(const std::vector<int> & matched_atoms,std::vector<int> * regexps) const296 void PrefilterTree::RegexpsGivenStrings(
297     const std::vector<int>& matched_atoms,
298     std::vector<int>* regexps) const {
299   regexps->clear();
300   if (!compiled_) {
301     // Some legacy users of PrefilterTree call Compile() before
302     // adding any regexps and expect Compile() to have no effect.
303     // This kludge is a counterpart to that kludge.
304     if (prefilter_vec_.empty())
305       return;
306 
307     LOG(ERROR) << "RegexpsGivenStrings called before Compile.";
308     for (size_t i = 0; i < prefilter_vec_.size(); i++)
309       regexps->push_back(static_cast<int>(i));
310   } else {
311     IntMap regexps_map(static_cast<int>(prefilter_vec_.size()));
312     std::vector<int> matched_atom_ids;
313     for (size_t j = 0; j < matched_atoms.size(); j++)
314       matched_atom_ids.push_back(atom_index_to_id_[matched_atoms[j]]);
315     PropagateMatch(matched_atom_ids, &regexps_map);
316     for (IntMap::iterator it = regexps_map.begin();
317          it != regexps_map.end();
318          ++it)
319       regexps->push_back(it->index());
320 
321     regexps->insert(regexps->end(), unfiltered_.begin(), unfiltered_.end());
322   }
323   std::sort(regexps->begin(), regexps->end());
324 }
325 
PropagateMatch(const std::vector<int> & atom_ids,IntMap * regexps) const326 void PrefilterTree::PropagateMatch(const std::vector<int>& atom_ids,
327                                    IntMap* regexps) const {
328   IntMap count(static_cast<int>(entries_.size()));
329   IntMap work(static_cast<int>(entries_.size()));
330   for (size_t i = 0; i < atom_ids.size(); i++)
331     work.set(atom_ids[i], 1);
332   for (IntMap::iterator it = work.begin(); it != work.end(); ++it) {
333     const Entry& entry = entries_[it->index()];
334     // Record regexps triggered.
335     for (size_t i = 0; i < entry.regexps.size(); i++)
336       regexps->set(entry.regexps[i], 1);
337     int c;
338     // Pass trigger up to parents.
339     for (StdIntMap::iterator it = entry.parents->begin();
340          it != entry.parents->end();
341          ++it) {
342       int j = it->first;
343       const Entry& parent = entries_[j];
344       // Delay until all the children have succeeded.
345       if (parent.propagate_up_at_count > 1) {
346         if (count.has_index(j)) {
347           c = count.get_existing(j) + 1;
348           count.set_existing(j, c);
349         } else {
350           c = 1;
351           count.set_new(j, c);
352         }
353         if (c < parent.propagate_up_at_count)
354           continue;
355       }
356       // Trigger the parent.
357       work.set(j, 1);
358     }
359   }
360 }
361 
362 // Debugging help.
PrintPrefilter(int regexpid)363 void PrefilterTree::PrintPrefilter(int regexpid) {
364   LOG(ERROR) << DebugNodeString(prefilter_vec_[regexpid]);
365 }
366 
PrintDebugInfo(NodeMap * nodes)367 void PrefilterTree::PrintDebugInfo(NodeMap* nodes) {
368   LOG(ERROR) << "#Unique Atoms: " << atom_index_to_id_.size();
369   LOG(ERROR) << "#Unique Nodes: " << entries_.size();
370 
371   for (size_t i = 0; i < entries_.size(); i++) {
372     StdIntMap* parents = entries_[i].parents;
373     const std::vector<int>& regexps = entries_[i].regexps;
374     LOG(ERROR) << "EntryId: " << i
375                << " N: " << parents->size() << " R: " << regexps.size();
376     for (StdIntMap::iterator it = parents->begin(); it != parents->end(); ++it)
377       LOG(ERROR) << it->first;
378   }
379   LOG(ERROR) << "Map:";
380   for (std::map<std::string, Prefilter*>::const_iterator iter = nodes->begin();
381        iter != nodes->end(); ++iter)
382     LOG(ERROR) << "NodeId: " << (*iter).second->unique_id()
383                << " Str: " << (*iter).first;
384 }
385 
DebugNodeString(Prefilter * node) const386 std::string PrefilterTree::DebugNodeString(Prefilter* node) const {
387   std::string node_string = "";
388   if (node->op() == Prefilter::ATOM) {
389     DCHECK(!node->atom().empty());
390     node_string += node->atom();
391   } else {
392     // Adding the operation disambiguates AND and OR nodes.
393     node_string +=  node->op() == Prefilter::AND ? "AND" : "OR";
394     node_string += "(";
395     for (size_t i = 0; i < node->subs()->size(); i++) {
396       if (i > 0)
397         node_string += ',';
398       node_string += StringPrintf("%d", (*node->subs())[i]->unique_id());
399       node_string += ":";
400       node_string += DebugNodeString((*node->subs())[i]);
401     }
402     node_string += ")";
403   }
404   return node_string;
405 }
406 
407 }  // namespace re2
408