1 //===- subzero/src/IceRangeSpec.cpp - Include/exclude specification -------===//
2 //
3 // The Subzero Code Generator
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 ///
10 /// \file
11 /// \brief Implements a class for specifying sets of names and number ranges to
12 /// match against. This is specified as a comma-separated list of clauses.
13 /// Each clause optionally starts with '-' to indicate exclusion instead of
14 /// inclusion. A clause can be a name, or a numeric range X:Y, or a single
15 /// number X. The X:Y form indicates a range of numbers greater than or equal
16 /// to X and strictly less than Y. A missing "X" is taken to be 0, and a
17 /// missing "Y" is taken to be infinite. E.g., "0:" and ":" specify the entire
18 /// set.
19 ///
20 /// This is essentially the same implementation as in szbuild.py, except that
21 /// regular expressions are not used for the names.
22 ///
23 //===----------------------------------------------------------------------===//
24
25 #include "IceRangeSpec.h"
26 #include "IceStringPool.h"
27
28 #include <cctype>
29 #include <string>
30 #include <unordered_set>
31 #include <vector>
32
33 namespace Ice {
34
35 bool RangeSpec::HasNames = false;
36
37 namespace {
38
39 /// Helper function to parse "X" or "X:Y" into First and Last.
40 /// - "X" is treated as "X:X+1".
41 /// - ":Y" is treated as "0:Y".
42 /// - "X:" is treated as "X:inf"
43 ///
44 /// Behavior is undefined if "X" or "Y" is not a proper number (since std::stoul
45 /// throws an exception).
46 ///
47 /// If the string doesn't contain 1 or 2 ':' delimiters, or X>=Y,
48 /// report_fatal_error is called.
getRange(const std::string & Token,uint32_t * First,uint32_t * Last)49 void getRange(const std::string &Token, uint32_t *First, uint32_t *Last) {
50 bool Error = false;
51 auto Tokens = RangeSpec::tokenize(Token, RangeSpec::DELIM_RANGE);
52 if (Tokens.size() == 1) {
53 *First = std::stoul(Tokens[0]);
54 *Last = *First + 1;
55 } else if (Tokens.size() == 2) {
56 *First = Tokens[0].empty() ? 0 : std::stoul(Tokens[0]);
57 *Last = Tokens[1].empty() ? RangeSpec::RangeMax : std::stoul(Tokens[1]);
58 } else {
59 Error = true;
60 }
61 if (*First >= *Last) {
62 Error = true;
63 }
64 if (Error) {
65 llvm::report_fatal_error("Invalid range " + Token);
66 }
67 }
68
69 /// Helper function to add one token to the include or exclude set. The token
70 /// is examined and then treated as either a numeric range or a single name.
record(const std::string & Token,RangeSpec::Desc * D)71 void record(const std::string &Token, RangeSpec::Desc *D) {
72 if (Token.empty())
73 return;
74 // Mark that an include or exclude was explicitly given. This affects the
75 // default decision when matching a value that wasn't explicitly provided in
76 // the include or exclude list.
77 D->IsExplicit = true;
78 // A range is identified by starting with a digit or a ':'.
79 if (Token[0] == RangeSpec::DELIM_RANGE || std::isdigit(Token[0])) {
80 uint32_t First, Last;
81 getRange(Token, &First, &Last);
82 if (Last == RangeSpec::RangeMax) {
83 D->AllFrom = std::min(D->AllFrom, First);
84 } else {
85 if (Last >= D->Numbers.size())
86 D->Numbers.resize(Last + 1);
87 D->Numbers.set(First, Last);
88 }
89 } else {
90 // Otherwise treat it as a single name.
91 D->Names.insert(Token);
92 }
93 }
94
95 } // end of anonymous namespace
96
tokenize(const std::string & Spec,char Delimiter)97 std::vector<std::string> RangeSpec::tokenize(const std::string &Spec,
98 char Delimiter) {
99 std::vector<std::string> Tokens;
100 if (!Spec.empty()) {
101 std::string::size_type StartPos = 0;
102 std::string::size_type DelimPos = 0;
103 while (DelimPos != std::string::npos) {
104 DelimPos = Spec.find(Delimiter, StartPos);
105 Tokens.emplace_back(Spec.substr(StartPos, DelimPos - StartPos));
106 StartPos = DelimPos + 1;
107 }
108 }
109 return Tokens;
110 }
111
112 /// Initialize the RangeSpec with the given string. Calling init multiple times
113 /// (e.g. init("A");init("B");) is equivalent to init("A,B"); .
init(const std::string & Spec)114 void RangeSpec::init(const std::string &Spec) {
115 auto Tokens = tokenize(Spec, DELIM_LIST);
116 for (const auto &Token : Tokens) {
117 if (Token[0] == '-') {
118 exclude(Token.substr(1));
119 } else {
120 include(Token);
121 }
122 }
123 if (!Includes.Names.empty() || !Excludes.Names.empty())
124 HasNames = true;
125 }
126
127 /// Determine whether the given Name/Number combo match the specification given
128 /// to the init() method. Explicit excludes take precedence over explicit
129 /// includes. If the combo doesn't match any explicit include or exclude:
130 /// - false if the init() string is empty (no explicit includes or excludes)
131 /// - true if there is at least one explicit exclude and no explicit includes
132 /// - false otherwise (at least one explicit include)
match(const std::string & Name,uint32_t Number) const133 bool RangeSpec::match(const std::string &Name, uint32_t Number) const {
134 // No match if it is explicitly excluded by name or number.
135 if (Excludes.Names.find(Name) != Excludes.Names.end())
136 return false;
137 if (Number >= Excludes.AllFrom)
138 return false;
139 if (Number < Excludes.Numbers.size() && Excludes.Numbers[Number])
140 return false;
141
142 // Positive match if it is explicitly included by name or number.
143 if (Includes.Names.find(Name) != Includes.Names.end())
144 return true;
145 if (Number >= Includes.AllFrom)
146 return true;
147 if (Number < Includes.Numbers.size() && Includes.Numbers[Number])
148 return true;
149
150 // Otherwise use the default decision.
151 return Excludes.IsExplicit && !Includes.IsExplicit;
152 }
153
include(const std::string & Token)154 void RangeSpec::include(const std::string &Token) { record(Token, &Includes); }
155
exclude(const std::string & Token)156 void RangeSpec::exclude(const std::string &Token) { record(Token, &Excludes); }
157
158 } // end of namespace Ice
159