66 mSymbols << sym; |
66 mSymbols << sym; |
67 |
67 |
68 if (mSymbols.IsEmpty()) |
68 if (mSymbols.IsEmpty()) |
69 Error ("Expected expression"); |
69 Error ("Expected expression"); |
70 |
70 |
|
71 AdjustOperators(); |
|
72 |
|
73 for (ExpressionSymbol* sym : mSymbols) |
|
74 { |
|
75 switch (sym->GetType()) |
|
76 { |
|
77 case ExpressionSymbol::eOperatorSymbol: |
|
78 { |
|
79 Print ("\t- Operator: %1\n", static_cast<ExpressionOperator*> (sym)->GetID()); |
|
80 break; |
|
81 } |
|
82 |
|
83 case ExpressionSymbol::eValueSymbol: |
|
84 { |
|
85 ExpressionValue* val = static_cast<ExpressionValue*> (sym); |
|
86 |
|
87 if (val->IsConstexpr()) |
|
88 Print ("\t- Constant expression value: %1\n", val->GetValue()); |
|
89 else |
|
90 { |
|
91 Print ("\t- Databuffer value:\n"); |
|
92 val->GetBuffer()->Dump(); |
|
93 } |
|
94 break; |
|
95 } |
|
96 |
|
97 case ExpressionSymbol::eColonSymbol: |
|
98 { |
|
99 Print ("\t- Colon"); |
|
100 break; |
|
101 } |
|
102 } |
|
103 } |
|
104 |
|
105 exit (0); |
|
106 |
|
107 /* |
71 Verify(); |
108 Verify(); |
72 mResult = Evaluate(); |
109 mResult = Evaluate(); |
|
110 */ |
73 } |
111 } |
74 |
112 |
75 // ============================================================================= |
113 // ============================================================================= |
76 // |
114 // |
77 Expression::~Expression() |
115 Expression::~Expression() |
96 try |
134 try |
97 { |
135 { |
98 ScriptVariable* globalvar; |
136 ScriptVariable* globalvar; |
99 mLexer->MustGetNext(); |
137 mLexer->MustGetNext(); |
100 |
138 |
|
139 Print ("Token type: %1\n", mLexer->DescribeTokenType (mLexer->GetTokenType())); |
|
140 |
101 if (mLexer->GetTokenType() == tkColon) |
141 if (mLexer->GetTokenType() == tkColon) |
102 return new ExpressionColon; |
142 return new ExpressionColon; |
103 |
143 |
104 // Check for operator |
144 // Check for operator |
105 for (const OperatorInfo* op : gOperators) |
145 for (const OperatorInfo& op : gOperators) |
106 if (mLexer->GetTokenType() == op->token) |
146 if (mLexer->GetTokenType() == op.token) |
107 return new ExpressionOperator (op - gOperators); |
147 return new ExpressionOperator ((EOperator) (&op - &gOperators[0])); |
108 |
148 |
109 // Check sub-expression |
149 // Check sub-expression |
110 if (mLexer->GetTokenType() == tkParenStart) |
150 if (mLexer->GetTokenType() == tkParenStart) |
111 { |
151 { |
112 mLexer->MustGetNext(); |
|
113 Expression expr (mParser, mLexer, mType); |
152 Expression expr (mParser, mLexer, mType); |
114 mLexer->MustGetNext (tkParenEnd); |
153 mLexer->MustGetNext (tkParenEnd); |
115 return expr.GetResult(); |
154 return expr.GetResult(); |
116 } |
155 } |
117 |
156 |
118 op = new ExpressionValue; |
157 op = new ExpressionValue (mType); |
119 |
158 |
120 // Check function |
159 // Check function |
121 if (CommandInfo* comm = FindCommandByName (GetTokenString())) |
160 if (CommandInfo* comm = FindCommandByName (GetTokenString())) |
122 { |
161 { |
123 if (mType != EUnknownType && comm->returnvalue != mType) |
162 if (mType != EUnknownType && comm->returnvalue != mType) |
169 // Check for literal |
208 // Check for literal |
170 switch (mType) |
209 switch (mType) |
171 { |
210 { |
172 case EVoidType: |
211 case EVoidType: |
173 case EUnknownType: |
212 case EUnknownType: |
|
213 { |
174 Error ("unknown identifier `%1` (expected keyword, function or variable)", GetTokenString()); |
214 Error ("unknown identifier `%1` (expected keyword, function or variable)", GetTokenString()); |
175 break; |
215 break; |
|
216 } |
176 |
217 |
177 case EBoolType: |
218 case EBoolType: |
|
219 { |
178 if ((tt = mLexer->GetTokenType()) == tkTrue || tt == tkFalse) |
220 if ((tt = mLexer->GetTokenType()) == tkTrue || tt == tkFalse) |
179 { |
221 { |
180 op->SetValue (tt == tkTrue ? 1 : 0); |
222 op->SetValue (tt == tkTrue ? 1 : 0); |
181 return op; |
223 return op; |
182 } |
224 } |
|
225 } |
183 case EIntType: |
226 case EIntType: |
184 if (!mLexer->GetTokenType() != tkNumber) |
227 { |
|
228 if (mLexer->GetTokenType() != tkNumber) |
185 throw failed; |
229 throw failed; |
186 |
230 |
187 op->SetValue (GetTokenString().ToLong()); |
231 op->SetValue (GetTokenString().ToLong()); |
188 return op; |
232 return op; |
|
233 } |
189 |
234 |
190 case EStringType: |
235 case EStringType: |
191 if (!mLexer->GetTokenType() != tkString) |
236 { |
|
237 if (mLexer->GetTokenType() != tkString) |
192 throw failed; |
238 throw failed; |
193 |
239 |
194 op->SetValue (GetStringTableIndex (GetTokenString())); |
240 op->SetValue (GetStringTableIndex (GetTokenString())); |
195 return op; |
241 return op; |
|
242 } |
196 } |
243 } |
197 |
244 |
198 assert (false); |
245 assert (false); |
199 throw failed; |
246 throw failed; |
200 } |
247 } |
201 catch (ELocalException&) |
248 catch (ELocalException&) |
202 { |
249 { |
203 // We use a local enum here since catch(...) would catch Error() calls. |
250 // We use a local enum here since catch(...) would catch Error() calls. |
204 mLexer->SetPosition (pos); |
251 mLexer->SetPosition (pos); |
205 delete op; |
252 delete op; |
206 return false; |
253 return null; |
207 } |
254 } |
208 |
255 |
209 assert (false); |
256 assert (false); |
210 return false; |
257 return null; |
|
258 } |
|
259 |
|
260 // ============================================================================= |
|
261 // |
|
262 // The symbol parsing process only does token-based checking for operators. Thus |
|
263 // ALL minus operators are actually unary minuses simply because both have |
|
264 // tkMinus as their token and the unary minus is prior to the binary minus in |
|
265 // the operator table. Now that we have all symbols present, we can correct |
|
266 // cases like this. |
|
267 // |
|
268 void Expression::AdjustOperators() |
|
269 { |
|
270 for (auto it = mSymbols.begin() + 1; it != mSymbols.end(); ++it) |
|
271 { |
|
272 if ((*it)->GetType() != ExpressionSymbol::eOperatorSymbol) |
|
273 continue; |
|
274 |
|
275 ExpressionOperator* op = static_cast<ExpressionOperator*> (*it); |
|
276 |
|
277 // Unary minus with a value as the previous symbol cannot really be |
|
278 // unary; replace with binary minus. |
|
279 if (op->GetID() == opUnaryMinus && (*(it - 1))->GetType() == ExpressionSymbol::eValueSymbol) |
|
280 { |
|
281 Print ("Changing symbol operator #%1 from %2 to %3\n", |
|
282 it - mSymbols.begin(), op->GetID(), opSubtraction); |
|
283 op->SetID (opSubtraction); |
|
284 } |
|
285 } |
211 } |
286 } |
212 |
287 |
213 // ============================================================================= |
288 // ============================================================================= |
214 // |
289 // |
215 ExpressionValue* Expression::Evaluate() |
290 ExpressionValue* Expression::Evaluate() |
231 return mLexer->GetToken()->text; |
306 return mLexer->GetToken()->text; |
232 } |
307 } |
233 |
308 |
234 // ============================================================================= |
309 // ============================================================================= |
235 // |
310 // |
236 ExpressionOperator::ExpressionOperator (int id) : |
311 ExpressionOperator::ExpressionOperator (EOperator id) : |
237 mID (id), |
312 ExpressionSymbol (eOperatorSymbol), |
238 mType (eOperator) {} |
313 mID (id) {} |
239 |
314 |
240 // ============================================================================= |
315 // ============================================================================= |
241 // |
316 // |
242 ExpressionValue::ExpressionValue(EType valuetype) : |
317 ExpressionValue::ExpressionValue (EType valuetype) : |
|
318 ExpressionSymbol (eValueSymbol), |
243 mBuffer (null), |
319 mBuffer (null), |
244 mType (eOperand), |
|
245 mValueType (valuetype) {} |
320 mValueType (valuetype) {} |
|
321 |
|
322 // ============================================================================= |
|
323 // |
|
324 ExpressionValue::~ExpressionValue() |
|
325 { |
|
326 delete mBuffer; |
|
327 } |
246 |
328 |
247 // ============================================================================= |
329 // ============================================================================= |
248 // |
330 // |
249 void ExpressionValue::ConvertToBuffer() |
331 void ExpressionValue::ConvertToBuffer() |
250 { |
332 { |