blob: cc586163c11da4a37c4c2fc9d017edb16e212a21 [file] [log] [blame]
// 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 "command.h"
#include <unordered_map>
#include <unordered_set>
#include "dep.h"
#include "eval.h"
#include "fileutil.h"
#include "flags.h"
#include "log.h"
#include "strutil.h"
#include "var.h"
namespace {
class AutoVar : public Var {
public:
AutoVar() : Var(VarOrigin::AUTOMATIC, nullptr, Loc()) {}
virtual const char* Flavor() const override { return "undefined"; }
virtual void AppendVar(Evaluator*, Value*) override { CHECK(false); }
virtual std::string_view String() const override {
ERROR("$(value %s) is not implemented yet", sym_);
return "";
}
virtual std::string DebugString() const override {
return std::string("AutoVar(") + sym_ + ")";
}
virtual bool IsFunc(Evaluator*) const override { return true; }
protected:
AutoVar(CommandEvaluator* ce, const char* sym) : ce_(ce), sym_(sym) {}
virtual ~AutoVar() = default;
CommandEvaluator* ce_;
const char* sym_;
};
#define DECLARE_AUTO_VAR_CLASS(name) \
class name : public AutoVar { \
public: \
name(CommandEvaluator* ce, const char* sym) : AutoVar(ce, sym) {} \
virtual ~name() = default; \
virtual void Eval(Evaluator* ev, std::string* s) const override; \
}
DECLARE_AUTO_VAR_CLASS(AutoAtVar);
DECLARE_AUTO_VAR_CLASS(AutoLessVar);
DECLARE_AUTO_VAR_CLASS(AutoHatVar);
DECLARE_AUTO_VAR_CLASS(AutoPlusVar);
DECLARE_AUTO_VAR_CLASS(AutoStarVar);
DECLARE_AUTO_VAR_CLASS(AutoQuestionVar);
DECLARE_AUTO_VAR_CLASS(AutoNotImplementedVar);
class AutoSuffixDVar : public AutoVar {
public:
AutoSuffixDVar(CommandEvaluator* ce, const char* sym, Var* wrapped)
: AutoVar(ce, sym), wrapped_(wrapped) {}
virtual ~AutoSuffixDVar() = default;
virtual void Eval(Evaluator* ev, std::string* s) const override;
private:
Var* wrapped_;
};
class AutoSuffixFVar : public AutoVar {
public:
AutoSuffixFVar(CommandEvaluator* ce, const char* sym, Var* wrapped)
: AutoVar(ce, sym), wrapped_(wrapped) {}
virtual ~AutoSuffixFVar() = default;
virtual void Eval(Evaluator* ev, std::string* s) const override;
private:
Var* wrapped_;
};
void AutoAtVar::Eval(Evaluator*, std::string* s) const {
*s += ce_->current_dep_node()->output.str();
}
void AutoLessVar::Eval(Evaluator*, std::string* s) const {
auto& ai = ce_->current_dep_node()->actual_inputs;
if (!ai.empty())
*s += ai[0].str();
}
void AutoHatVar::Eval(Evaluator*, std::string* s) const {
std::unordered_set<std::string_view> seen;
WordWriter ww(s);
for (Symbol ai : ce_->current_dep_node()->actual_inputs) {
if (seen.insert(ai.str()).second)
ww.Write(ai.str());
}
}
void AutoPlusVar::Eval(Evaluator*, std::string* s) const {
WordWriter ww(s);
for (Symbol ai : ce_->current_dep_node()->actual_inputs) {
ww.Write(ai.str());
}
}
void AutoStarVar::Eval(Evaluator*, std::string* s) const {
const DepNode* n = ce_->current_dep_node();
if (!n->output_pattern.IsValid())
return;
Pattern pat(n->output_pattern.str());
s->append(pat.Stem(n->output.str()));
}
void AutoQuestionVar::Eval(Evaluator* ev, std::string* s) const {
std::unordered_set<std::string_view> seen;
if (ev->avoid_io()) {
// Check timestamps using the shell at the start of rule execution
// instead.
*s += "${KATI_NEW_INPUTS}";
if (!ce_->found_new_inputs()) {
std::string def;
WordWriter ww(&def);
ww.Write("KATI_NEW_INPUTS=$(find");
for (Symbol ai : ce_->current_dep_node()->actual_inputs) {
if (seen.insert(ai.str()).second) {
ww.Write(ai.str());
}
}
ww.Write("$(test -e");
ww.Write(ce_->current_dep_node()->output.str());
ww.Write("&& echo -newer");
ww.Write(ce_->current_dep_node()->output.str());
ww.Write(")) && export KATI_NEW_INPUTS");
ev->add_delayed_output_command(def);
ce_->set_found_new_inputs(true);
}
} else {
WordWriter ww(s);
double target_age = GetTimestamp(ce_->current_dep_node()->output.str());
for (Symbol ai : ce_->current_dep_node()->actual_inputs) {
if (seen.insert(ai.str()).second && GetTimestamp(ai.str()) > target_age) {
ww.Write(ai.str());
}
}
}
}
void AutoNotImplementedVar::Eval(Evaluator* ev, std::string*) const {
ev->Error(StringPrintf("Automatic variable `$%s' isn't supported yet", sym_));
}
void AutoSuffixDVar::Eval(Evaluator* ev, std::string* s) const {
std::string buf;
wrapped_->Eval(ev, &buf);
WordWriter ww(s);
for (std::string_view tok : WordScanner(buf)) {
ww.Write(Dirname(tok));
}
}
void AutoSuffixFVar::Eval(Evaluator* ev, std::string* s) const {
std::string buf;
wrapped_->Eval(ev, &buf);
WordWriter ww(s);
for (std::string_view tok : WordScanner(buf)) {
ww.Write(Basename(tok));
}
}
void ParseCommandPrefixes(std::string_view* s, bool* echo, bool* ignore_error) {
*s = TrimLeftSpace(*s);
while (true) {
char c = s->empty() ? 0 : s->front();
if (c == '@') {
*echo = false;
} else if (c == '-') {
*ignore_error = true;
} else if (c == '+') {
// ignore recursion marker
} else {
break;
}
*s = TrimLeftSpace(s->substr(1));
}
}
} // namespace
CommandEvaluator::CommandEvaluator(Evaluator* ev) : ev_(ev) {
#define INSERT_AUTO_VAR(name, sym) \
do { \
Var* v = new name(this, sym); \
Intern(sym).SetGlobalVar(v); \
Intern(sym "D").SetGlobalVar(new AutoSuffixDVar(this, sym "D", v)); \
Intern(sym "F").SetGlobalVar(new AutoSuffixFVar(this, sym "F", v)); \
} while (0)
INSERT_AUTO_VAR(AutoAtVar, "@");
INSERT_AUTO_VAR(AutoLessVar, "<");
INSERT_AUTO_VAR(AutoHatVar, "^");
INSERT_AUTO_VAR(AutoPlusVar, "+");
INSERT_AUTO_VAR(AutoStarVar, "*");
INSERT_AUTO_VAR(AutoQuestionVar, "?");
// TODO: Implement them.
INSERT_AUTO_VAR(AutoNotImplementedVar, "%");
INSERT_AUTO_VAR(AutoNotImplementedVar, "|");
}
std::vector<Command> CommandEvaluator::Eval(const DepNode& n) {
std::vector<Command> result;
ev_->set_loc(n.loc);
ev_->set_current_scope(n.rule_vars);
ev_->SetEvaluatingCommand(true);
current_dep_node_ = &n;
found_new_inputs_ = false;
for (Value* v : n.cmds) {
ev_->set_loc(v->Location());
const std::string&& cmds_buf = v->Eval(ev_);
std::string_view cmds = cmds_buf;
bool global_echo = !g_flags.is_silent_mode;
bool global_ignore_error = false;
ParseCommandPrefixes(&cmds, &global_echo, &global_ignore_error);
if (cmds == "")
continue;
while (true) {
size_t lf_cnt;
size_t index = FindEndOfLine(cmds, 0, &lf_cnt);
if (index == cmds.size())
index = std::string::npos;
std::string_view cmd = TrimLeftSpace(cmds.substr(0, index));
cmds = cmds.substr(index + 1);
bool echo = global_echo;
bool ignore_error = global_ignore_error;
ParseCommandPrefixes(&cmd, &echo, &ignore_error);
if (!cmd.empty()) {
Command& command = result.emplace_back(n.output);
command.cmd = std::string(cmd);
command.echo = echo;
command.ignore_error = ignore_error;
}
if (index == std::string::npos)
break;
}
continue;
}
if (!ev_->delayed_output_commands().empty()) {
std::vector<Command> output_commands;
for (const std::string& cmd : ev_->delayed_output_commands()) {
Command& c = output_commands.emplace_back(n.output);
c.cmd = cmd;
c.echo = false;
c.ignore_error = false;
c.force_no_subshell = true;
}
// Prepend |output_commands|.
result.swap(output_commands);
copy(output_commands.begin(), output_commands.end(), back_inserter(result));
ev_->clear_delayed_output_commands();
}
ev_->set_current_scope(NULL);
ev_->SetEvaluatingCommand(false);
return result;
}