aboutsummaryrefslogtreecommitdiff
path: root/src/Ninja.cpp
blob: 7b3212e62a8a63ba9d5e5a058503afada9f58051 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
#include "../include/ninja/Ninja.hpp"
#include <cassert>

namespace ninja
{
    static std::string joinArgs(const std::vector<NinjaArg> &args)
    {
        std::string result;
        for(const NinjaArg &arg : args)
        {
            if(!result.empty())
                result += ' ';
            result += '\'';
            if(arg.type == NinjaArg::Type::VARIABLE)
                result += "$";
            result += arg.arg;
            result += '\'';
        }
        return result;
    }

    NinjaRule::NinjaRule(NinjaBuildFile *_buildFile, const std::string &_name, const std::string &_command) : 
        buildFile(_buildFile),
        name(_name),
        command(_command)
    {
        assert(buildFile);
    }

    NinjaBuildFile::~NinjaBuildFile()
    {
        for(auto &rule : rules)
        {
            delete rule.second;
        }
    }

    void NinjaRule::build(const std::string &in, const std::string &out, const std::vector<ninja::NinjaArgValue> &additionalArgs)
    {
        buildFile->build(this, in, out, additionalArgs);
    }

    void NinjaBuildFile::defineGlobalVariable(const std::string &name, const std::string &value)
    {
        auto it = globalVariables.find(name);
        if(it != globalVariables.end())
            throw NinjaException(NinjaError::VARIABLE_ALREADY_DEFINED, "Global variable already defined: " + name);
        globalVariables[name] = value;
    }

    NinjaRule* NinjaBuildFile::createRule(const std::string &name, const std::vector<NinjaArg> &commandArgs)
    {
        auto it = rules.find(name);
        if(it != rules.end())
            throw NinjaException(NinjaError::RULE_ALREADY_DEFINED, "Rule already defined: " + name);

        NinjaRule *rule = new NinjaRule(this, name, joinArgs(commandArgs));
        rules[name] = rule;
        return rule;
    }

    void NinjaBuildFile::build(const NinjaRule *rule, const std::string &in, const std::string &out, const std::vector<ninja::NinjaArgValue> &additionalArgs)
    {
        for(const NinjaBuild &build : builds)
        {
            if(build.rule == rule)
                throw NinjaException(NinjaError::RULE_AREADY_BUILT, "Rule already built: " + rule->name);
        }
        builds.push_back({ rule, in, out, additionalArgs });
    }

    std::string NinjaBuildFile::generate() const
    {
        std::string result;
        for(const auto &globalVar : globalVariables)
        {
            result += globalVar.first;
            result += " = ";
            result += globalVar.second;
            result += '\n';
        }

        if(!globalVariables.empty())
            result += '\n';
        
        for(const auto &rule : rules)
        {
            result += "rule ";
            result += rule.first;
            result += "\n  command = ";
            result += rule.second->command;
            if(!rule.second->depFile.empty())
            {
                result += "\n  depfile = ";
                result += rule.second->depFile;
            }
            result += "\n\n";
        }

        // TODO: enclose all names with quotes to escape whitespace?
        for(const NinjaBuild &build : builds)
        {
            result += "build ";
            result += build.out;
            result += ": ";
            result += build.rule->name;
            result += ' ';
            result += build.in;
            for(const NinjaArgValue &additionalArg : build.additionalArgs)
            {
                result += "\n  ";
                result += additionalArg.arg.name;
                result += " = ";
                result += additionalArg.value;

            }
            result += "\n\n";
        }
        return result;
    }
}