1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "tools/gn/parse_tree.h"
9 #include "base/stl_util.h"
10 #include "base/strings/string_number_conversions.h"
11 #include "tools/gn/functions.h"
12 #include "tools/gn/operators.h"
13 #include "tools/gn/scope.h"
14 #include "tools/gn/string_utils.h"
18 std::string
IndentFor(int value
) {
19 return std::string(value
, ' ');
24 Comments::Comments() {
27 Comments::~Comments() {
30 void Comments::ReverseSuffix() {
31 for (int i
= 0, j
= static_cast<int>(suffix_
.size() - 1); i
< j
; ++i
, --j
)
32 std::swap(suffix_
[i
], suffix_
[j
]);
35 ParseNode::ParseNode() {
38 ParseNode::~ParseNode() {
41 const AccessorNode
* ParseNode::AsAccessor() const { return NULL
; }
42 const BinaryOpNode
* ParseNode::AsBinaryOp() const { return NULL
; }
43 const BlockCommentNode
* ParseNode::AsBlockComment() const { return NULL
; }
44 const BlockNode
* ParseNode::AsBlock() const { return NULL
; }
45 const ConditionNode
* ParseNode::AsConditionNode() const { return NULL
; }
46 const EndNode
* ParseNode::AsEnd() const { return NULL
; }
47 const FunctionCallNode
* ParseNode::AsFunctionCall() const { return NULL
; }
48 const IdentifierNode
* ParseNode::AsIdentifier() const { return NULL
; }
49 const ListNode
* ParseNode::AsList() const { return NULL
; }
50 const LiteralNode
* ParseNode::AsLiteral() const { return NULL
; }
51 const UnaryOpNode
* ParseNode::AsUnaryOp() const { return NULL
; }
53 Comments
* ParseNode::comments_mutable() {
55 comments_
.reset(new Comments
);
56 return comments_
.get();
59 void ParseNode::PrintComments(std::ostream
& out
, int indent
) const {
61 std::string ind
= IndentFor(indent
+ 1);
62 for (const auto& token
: comments_
->before())
63 out
<< ind
<< "+BEFORE_COMMENT(\"" << token
.value() << "\")\n";
64 for (const auto& token
: comments_
->suffix())
65 out
<< ind
<< "+SUFFIX_COMMENT(\"" << token
.value() << "\")\n";
66 for (const auto& token
: comments_
->after())
67 out
<< ind
<< "+AFTER_COMMENT(\"" << token
.value() << "\")\n";
71 // AccessorNode ---------------------------------------------------------------
73 AccessorNode::AccessorNode() {
76 AccessorNode::~AccessorNode() {
79 const AccessorNode
* AccessorNode::AsAccessor() const {
83 Value
AccessorNode::Execute(Scope
* scope
, Err
* err
) const {
85 return ExecuteArrayAccess(scope
, err
);
87 return ExecuteScopeAccess(scope
, err
);
92 LocationRange
AccessorNode::GetRange() const {
94 return LocationRange(base_
.location(), index_
->GetRange().end());
96 return LocationRange(base_
.location(), member_
->GetRange().end());
98 return LocationRange();
101 Err
AccessorNode::MakeErrorDescribing(const std::string
& msg
,
102 const std::string
& help
) const {
103 return Err(GetRange(), msg
, help
);
106 void AccessorNode::Print(std::ostream
& out
, int indent
) const {
107 out
<< IndentFor(indent
) << "ACCESSOR\n";
108 PrintComments(out
, indent
);
109 out
<< IndentFor(indent
+ 1) << base_
.value() << "\n";
111 index_
->Print(out
, indent
+ 1);
113 member_
->Print(out
, indent
+ 1);
116 Value
AccessorNode::ExecuteArrayAccess(Scope
* scope
, Err
* err
) const {
117 Value index_value
= index_
->Execute(scope
, err
);
118 if (err
->has_error())
120 if (!index_value
.VerifyTypeIs(Value::INTEGER
, err
))
123 const Value
* base_value
= scope
->GetValue(base_
.value(), true);
125 *err
= MakeErrorDescribing("Undefined identifier.");
128 if (!base_value
->VerifyTypeIs(Value::LIST
, err
))
131 int64 index_int
= index_value
.int_value();
133 *err
= Err(index_
->GetRange(), "Negative array subscript.",
134 "You gave me " + base::Int64ToString(index_int
) + ".");
137 size_t index_sizet
= static_cast<size_t>(index_int
);
138 if (index_sizet
>= base_value
->list_value().size()) {
139 *err
= Err(index_
->GetRange(), "Array subscript out of range.",
140 "You gave me " + base::Int64ToString(index_int
) +
141 " but I was expecting something from 0 to " +
143 static_cast<int64
>(base_value
->list_value().size()) - 1) +
148 // Doing this assumes that there's no way in the language to do anything
149 // between the time the reference is created and the time that the reference
150 // is used. If there is, this will crash! Currently, this is just used for
151 // array accesses where this "shouldn't" happen.
152 return base_value
->list_value()[index_sizet
];
155 Value
AccessorNode::ExecuteScopeAccess(Scope
* scope
, Err
* err
) const {
156 // We jump through some hoops here since ideally a.b will count "b" as
157 // accessed in the given scope. The value "a" might be in some normal nested
158 // scope and we can modify it, but it might also be inherited from the
159 // readonly root scope and we can't do used variable tracking on it. (It's
160 // not legal to const cast it away since the root scope will be in readonly
161 // mode and being accessed from multiple threads without locking.) So this
162 // code handles both cases.
163 const Value
* result
= NULL
;
165 // Look up the value in the scope named by "base_".
166 Value
* mutable_base_value
= scope
->GetMutableValue(base_
.value(), true);
167 if (mutable_base_value
) {
168 // Common case: base value is mutable so we can track variable accesses
169 // for unused value warnings.
170 if (!mutable_base_value
->VerifyTypeIs(Value::SCOPE
, err
))
172 result
= mutable_base_value
->scope_value()->GetValue(
173 member_
->value().value(), true);
175 // Fall back to see if the value is on a read-only scope.
176 const Value
* const_base_value
= scope
->GetValue(base_
.value(), true);
177 if (const_base_value
) {
178 // Read only value, don't try to mark the value access as a "used" one.
179 if (!const_base_value
->VerifyTypeIs(Value::SCOPE
, err
))
182 const_base_value
->scope_value()->GetValue(member_
->value().value());
184 *err
= Err(base_
, "Undefined identifier.");
190 *err
= Err(member_
.get(), "No value named \"" +
191 member_
->value().value() + "\" in scope \"" + base_
.value() + "\"");
197 // BinaryOpNode ---------------------------------------------------------------
199 BinaryOpNode::BinaryOpNode() {
202 BinaryOpNode::~BinaryOpNode() {
205 const BinaryOpNode
* BinaryOpNode::AsBinaryOp() const {
209 Value
BinaryOpNode::Execute(Scope
* scope
, Err
* err
) const {
210 return ExecuteBinaryOperator(scope
, this, left_
.get(), right_
.get(), err
);
213 LocationRange
BinaryOpNode::GetRange() const {
214 return left_
->GetRange().Union(right_
->GetRange());
217 Err
BinaryOpNode::MakeErrorDescribing(const std::string
& msg
,
218 const std::string
& help
) const {
219 return Err(op_
, msg
, help
);
222 void BinaryOpNode::Print(std::ostream
& out
, int indent
) const {
223 out
<< IndentFor(indent
) << "BINARY(" << op_
.value() << ")\n";
224 PrintComments(out
, indent
);
225 left_
->Print(out
, indent
+ 1);
226 right_
->Print(out
, indent
+ 1);
229 // BlockNode ------------------------------------------------------------------
231 BlockNode::BlockNode(bool has_scope
) : has_scope_(has_scope
) {
234 BlockNode::~BlockNode() {
235 STLDeleteContainerPointers(statements_
.begin(), statements_
.end());
238 const BlockNode
* BlockNode::AsBlock() const {
242 Value
BlockNode::Execute(Scope
* containing_scope
, Err
* err
) const {
244 Scope
our_scope(containing_scope
);
245 Value ret
= ExecuteBlockInScope(&our_scope
, err
);
246 if (err
->has_error())
249 // Check for unused vars in the scope.
250 our_scope
.CheckForUnusedVars(err
);
253 return ExecuteBlockInScope(containing_scope
, err
);
256 LocationRange
BlockNode::GetRange() const {
257 if (begin_token_
.type() != Token::INVALID
&&
258 end_
->value().type() != Token::INVALID
) {
259 return begin_token_
.range().Union(end_
->value().range());
260 } else if (!statements_
.empty()) {
261 return statements_
[0]->GetRange().Union(
262 statements_
[statements_
.size() - 1]->GetRange());
264 return LocationRange();
267 Err
BlockNode::MakeErrorDescribing(const std::string
& msg
,
268 const std::string
& help
) const {
269 return Err(GetRange(), msg
, help
);
272 void BlockNode::Print(std::ostream
& out
, int indent
) const {
273 out
<< IndentFor(indent
) << "BLOCK\n";
274 PrintComments(out
, indent
);
275 for (const auto& statement
: statements_
)
276 statement
->Print(out
, indent
+ 1);
277 if (end_
&& end_
->comments())
278 end_
->Print(out
, indent
+ 1);
281 Value
BlockNode::ExecuteBlockInScope(Scope
* our_scope
, Err
* err
) const {
282 for (size_t i
= 0; i
< statements_
.size() && !err
->has_error(); i
++) {
283 // Check for trying to execute things with no side effects in a block.
284 const ParseNode
* cur
= statements_
[i
];
285 if (cur
->AsList() || cur
->AsLiteral() || cur
->AsUnaryOp() ||
286 cur
->AsIdentifier()) {
287 *err
= cur
->MakeErrorDescribing(
288 "This statement has no effect.",
289 "Either delete it or do something with the result.");
292 cur
->Execute(our_scope
, err
);
297 // ConditionNode --------------------------------------------------------------
299 ConditionNode::ConditionNode() {
302 ConditionNode::~ConditionNode() {
305 const ConditionNode
* ConditionNode::AsConditionNode() const {
309 Value
ConditionNode::Execute(Scope
* scope
, Err
* err
) const {
310 Value condition_result
= condition_
->Execute(scope
, err
);
311 if (err
->has_error())
313 if (condition_result
.type() != Value::BOOLEAN
) {
314 *err
= condition_
->MakeErrorDescribing(
315 "Condition does not evaluate to a boolean value.",
316 std::string("This is a value of type \"") +
317 Value::DescribeType(condition_result
.type()) +
319 err
->AppendRange(if_token_
.range());
323 if (condition_result
.boolean_value()) {
324 if_true_
->ExecuteBlockInScope(scope
, err
);
325 } else if (if_false_
) {
326 // The else block is optional. It's either another condition (for an
327 // "else if" and we can just Execute it and the condition will handle
328 // the scoping) or it's a block indicating an "else" in which ase we
329 // need to be sure it inherits our scope.
330 const BlockNode
* if_false_block
= if_false_
->AsBlock();
332 if_false_block
->ExecuteBlockInScope(scope
, err
);
334 if_false_
->Execute(scope
, err
);
340 LocationRange
ConditionNode::GetRange() const {
342 return if_token_
.range().Union(if_false_
->GetRange());
343 return if_token_
.range().Union(if_true_
->GetRange());
346 Err
ConditionNode::MakeErrorDescribing(const std::string
& msg
,
347 const std::string
& help
) const {
348 return Err(if_token_
, msg
, help
);
351 void ConditionNode::Print(std::ostream
& out
, int indent
) const {
352 out
<< IndentFor(indent
) << "CONDITION\n";
353 PrintComments(out
, indent
);
354 condition_
->Print(out
, indent
+ 1);
355 if_true_
->Print(out
, indent
+ 1);
357 if_false_
->Print(out
, indent
+ 1);
360 // FunctionCallNode -----------------------------------------------------------
362 FunctionCallNode::FunctionCallNode() {
365 FunctionCallNode::~FunctionCallNode() {
368 const FunctionCallNode
* FunctionCallNode::AsFunctionCall() const {
372 Value
FunctionCallNode::Execute(Scope
* scope
, Err
* err
) const {
373 return functions::RunFunction(scope
, this, args_
.get(), block_
.get(), err
);
376 LocationRange
FunctionCallNode::GetRange() const {
377 if (function_
.type() == Token::INVALID
)
378 return LocationRange(); // This will be null in some tests.
380 return function_
.range().Union(block_
->GetRange());
381 return function_
.range().Union(args_
->GetRange());
384 Err
FunctionCallNode::MakeErrorDescribing(const std::string
& msg
,
385 const std::string
& help
) const {
386 return Err(function_
, msg
, help
);
389 void FunctionCallNode::Print(std::ostream
& out
, int indent
) const {
390 out
<< IndentFor(indent
) << "FUNCTION(" << function_
.value() << ")\n";
391 PrintComments(out
, indent
);
392 args_
->Print(out
, indent
+ 1);
394 block_
->Print(out
, indent
+ 1);
397 // IdentifierNode --------------------------------------------------------------
399 IdentifierNode::IdentifierNode() {
402 IdentifierNode::IdentifierNode(const Token
& token
) : value_(token
) {
405 IdentifierNode::~IdentifierNode() {
408 const IdentifierNode
* IdentifierNode::AsIdentifier() const {
412 Value
IdentifierNode::Execute(Scope
* scope
, Err
* err
) const {
413 const Value
* value
= scope
->GetValue(value_
.value(), true);
416 *err
= MakeErrorDescribing("Undefined identifier");
421 result
.set_origin(this);
425 LocationRange
IdentifierNode::GetRange() const {
426 return value_
.range();
429 Err
IdentifierNode::MakeErrorDescribing(const std::string
& msg
,
430 const std::string
& help
) const {
431 return Err(value_
, msg
, help
);
434 void IdentifierNode::Print(std::ostream
& out
, int indent
) const {
435 out
<< IndentFor(indent
) << "IDENTIFIER(" << value_
.value() << ")\n";
436 PrintComments(out
, indent
);
439 // ListNode -------------------------------------------------------------------
441 ListNode::ListNode() {
444 ListNode::~ListNode() {
445 STLDeleteContainerPointers(contents_
.begin(), contents_
.end());
448 const ListNode
* ListNode::AsList() const {
452 Value
ListNode::Execute(Scope
* scope
, Err
* err
) const {
453 Value
result_value(this, Value::LIST
);
454 std::vector
<Value
>& results
= result_value
.list_value();
455 results
.reserve(contents_
.size());
457 for (const auto& cur
: contents_
) {
458 if (cur
->AsBlockComment())
460 results
.push_back(cur
->Execute(scope
, err
));
461 if (err
->has_error())
463 if (results
.back().type() == Value::NONE
) {
464 *err
= cur
->MakeErrorDescribing(
465 "This does not evaluate to a value.",
466 "I can't do something with nothing.");
473 LocationRange
ListNode::GetRange() const {
474 return LocationRange(begin_token_
.location(),
475 end_
->value().location());
478 Err
ListNode::MakeErrorDescribing(const std::string
& msg
,
479 const std::string
& help
) const {
480 return Err(begin_token_
, msg
, help
);
483 void ListNode::Print(std::ostream
& out
, int indent
) const {
484 out
<< IndentFor(indent
) << "LIST\n";
485 PrintComments(out
, indent
);
486 for (const auto& cur
: contents_
)
487 cur
->Print(out
, indent
+ 1);
488 if (end_
&& end_
->comments())
489 end_
->Print(out
, indent
+ 1);
492 // LiteralNode -----------------------------------------------------------------
494 LiteralNode::LiteralNode() {
497 LiteralNode::LiteralNode(const Token
& token
) : value_(token
) {
500 LiteralNode::~LiteralNode() {
503 const LiteralNode
* LiteralNode::AsLiteral() const {
507 Value
LiteralNode::Execute(Scope
* scope
, Err
* err
) const {
508 switch (value_
.type()) {
509 case Token::TRUE_TOKEN
:
510 return Value(this, true);
511 case Token::FALSE_TOKEN
:
512 return Value(this, false);
513 case Token::INTEGER
: {
515 if (!base::StringToInt64(value_
.value(), &result_int
)) {
516 *err
= MakeErrorDescribing("This does not look like an integer");
519 return Value(this, result_int
);
521 case Token::STRING
: {
522 Value
v(this, Value::STRING
);
523 ExpandStringLiteral(scope
, value_
, &v
, err
);
532 LocationRange
LiteralNode::GetRange() const {
533 return value_
.range();
536 Err
LiteralNode::MakeErrorDescribing(const std::string
& msg
,
537 const std::string
& help
) const {
538 return Err(value_
, msg
, help
);
541 void LiteralNode::Print(std::ostream
& out
, int indent
) const {
542 out
<< IndentFor(indent
) << "LITERAL(" << value_
.value() << ")\n";
543 PrintComments(out
, indent
);
546 // UnaryOpNode ----------------------------------------------------------------
548 UnaryOpNode::UnaryOpNode() {
551 UnaryOpNode::~UnaryOpNode() {
554 const UnaryOpNode
* UnaryOpNode::AsUnaryOp() const {
558 Value
UnaryOpNode::Execute(Scope
* scope
, Err
* err
) const {
559 Value operand_value
= operand_
->Execute(scope
, err
);
560 if (err
->has_error())
562 return ExecuteUnaryOperator(scope
, this, operand_value
, err
);
565 LocationRange
UnaryOpNode::GetRange() const {
566 return op_
.range().Union(operand_
->GetRange());
569 Err
UnaryOpNode::MakeErrorDescribing(const std::string
& msg
,
570 const std::string
& help
) const {
571 return Err(op_
, msg
, help
);
574 void UnaryOpNode::Print(std::ostream
& out
, int indent
) const {
575 out
<< IndentFor(indent
) << "UNARY(" << op_
.value() << ")\n";
576 PrintComments(out
, indent
);
577 operand_
->Print(out
, indent
+ 1);
580 // BlockCommentNode ------------------------------------------------------------
582 BlockCommentNode::BlockCommentNode() {
585 BlockCommentNode::~BlockCommentNode() {
588 const BlockCommentNode
* BlockCommentNode::AsBlockComment() const {
592 Value
BlockCommentNode::Execute(Scope
* scope
, Err
* err
) const {
596 LocationRange
BlockCommentNode::GetRange() const {
597 return comment_
.range();
600 Err
BlockCommentNode::MakeErrorDescribing(const std::string
& msg
,
601 const std::string
& help
) const {
602 return Err(comment_
, msg
, help
);
605 void BlockCommentNode::Print(std::ostream
& out
, int indent
) const {
606 out
<< IndentFor(indent
) << "BLOCK_COMMENT(" << comment_
.value() << ")\n";
607 PrintComments(out
, indent
);
611 // EndNode ---------------------------------------------------------------------
613 EndNode::EndNode(const Token
& token
) : value_(token
) {
616 EndNode::~EndNode() {
619 const EndNode
* EndNode::AsEnd() const {
623 Value
EndNode::Execute(Scope
* scope
, Err
* err
) const {
627 LocationRange
EndNode::GetRange() const {
628 return value_
.range();
631 Err
EndNode::MakeErrorDescribing(const std::string
& msg
,
632 const std::string
& help
) const {
633 return Err(value_
, msg
, help
);
636 void EndNode::Print(std::ostream
& out
, int indent
) const {
637 out
<< IndentFor(indent
) << "END(" << value_
.value() << ")\n";
638 PrintComments(out
, indent
);