// Copyright 2015 Google Inc. All rights reserved
|
//
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
// you may not use this file except in compliance with the License.
|
// You may obtain a copy of the License at
|
//
|
// http://www.apache.org/licenses/LICENSE-2.0
|
//
|
// Unless required by applicable law or agreed to in writing, software
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
// See the License for the specific language governing permissions and
|
// limitations under the License.
|
|
// +build ignore
|
|
#include "rule.h"
|
|
#include "expr.h"
|
#include "log.h"
|
#include "parser.h"
|
#include "stringprintf.h"
|
#include "strutil.h"
|
#include "symtab.h"
|
|
Rule::Rule() : is_double_colon(false), is_suffix_rule(false), cmd_lineno(0) {}
|
|
void Rule::ParseInputs(const StringPiece& inputs_str) {
|
bool is_order_only = false;
|
for (auto const& input : WordScanner(inputs_str)) {
|
if (input == "|") {
|
is_order_only = true;
|
continue;
|
}
|
Symbol input_sym = Intern(TrimLeadingCurdir(input));
|
(is_order_only ? order_only_inputs : inputs).push_back(input_sym);
|
}
|
}
|
|
void Rule::ParsePrerequisites(const StringPiece& line,
|
size_t separator_pos,
|
const RuleStmt* rule_stmt) {
|
// line is either
|
// prerequisites [ ; command ]
|
// or
|
// target-prerequisites : prereq-patterns [ ; command ]
|
// First, separate command. At this point separator_pos should point to ';'
|
// unless null.
|
StringPiece prereq_string = line;
|
if (separator_pos != string::npos &&
|
rule_stmt->sep != RuleStmt::SEP_SEMICOLON) {
|
CHECK(line[separator_pos] == ';');
|
// TODO: Maybe better to avoid Intern here?
|
cmds.push_back(Value::NewLiteral(
|
Intern(TrimLeftSpace(line.substr(separator_pos + 1))).str()));
|
prereq_string = line.substr(0, separator_pos);
|
}
|
|
if ((separator_pos = prereq_string.find(':')) == string::npos) {
|
// Simple prerequisites
|
ParseInputs(prereq_string);
|
return;
|
}
|
|
// Static pattern rule.
|
if (!output_patterns.empty()) {
|
ERROR_LOC(loc, "*** mixed implicit and normal rules: deprecated syntax");
|
}
|
|
// Empty static patterns should not produce rules, but need to eat the
|
// commands So return a rule with no outputs nor output_patterns
|
if (outputs.empty()) {
|
return;
|
}
|
|
StringPiece target_prereq = prereq_string.substr(0, separator_pos);
|
StringPiece prereq_patterns = prereq_string.substr(separator_pos + 1);
|
|
for (StringPiece target_pattern : WordScanner(target_prereq)) {
|
target_pattern = TrimLeadingCurdir(target_pattern);
|
for (Symbol target : outputs) {
|
if (!Pattern(target_pattern).Match(target.str())) {
|
WARN_LOC(loc, "target `%s' doesn't match the target pattern",
|
target.c_str());
|
}
|
}
|
output_patterns.push_back(Intern(target_pattern));
|
}
|
|
if (output_patterns.empty()) {
|
ERROR_LOC(loc, "*** missing target pattern.");
|
}
|
if (output_patterns.size() > 1) {
|
ERROR_LOC(loc, "*** multiple target patterns.");
|
}
|
if (!IsPatternRule(output_patterns[0].str())) {
|
ERROR_LOC(loc, "*** target pattern contains no '%%'.");
|
}
|
ParseInputs(prereq_patterns);
|
}
|
|
string Rule::DebugString() const {
|
vector<string> v;
|
v.push_back(StringPrintf("outputs=[%s]", JoinSymbols(outputs, ",").c_str()));
|
v.push_back(StringPrintf("inputs=[%s]", JoinSymbols(inputs, ",").c_str()));
|
if (!order_only_inputs.empty()) {
|
v.push_back(StringPrintf("order_only_inputs=[%s]",
|
JoinSymbols(order_only_inputs, ",").c_str()));
|
}
|
if (!output_patterns.empty()) {
|
v.push_back(StringPrintf("output_patterns=[%s]",
|
JoinSymbols(output_patterns, ",").c_str()));
|
}
|
if (is_double_colon)
|
v.push_back("is_double_colon");
|
if (is_suffix_rule)
|
v.push_back("is_suffix_rule");
|
if (!cmds.empty()) {
|
v.push_back(StringPrintf("cmds=[%s]", JoinValues(cmds, ",").c_str()));
|
}
|
return JoinStrings(v, " ");
|
}
|