#include <msp/fs/utils.h>
+#include <msp/strings/format.h>
#include "builder.h"
#include "executable.h"
#include "externaltask.h"
SourceGenerator::SourceGenerator(Builder &b, const SourcePackage &p, const string &t):
Tool(b, t),
package(p)
-{ }
+{
+ set_run(&_run);
+}
Target *SourceGenerator::create_source(const Component &comp, const FS::Path &path) const
{
return new TemplateFile(builder, comp, path);
}
-Target *SourceGenerator::create_target(const list<Target *> &sources, const string &)
+Target *SourceGenerator::create_target(const vector<Target *> &sources, const string &)
{
- if(sources.size()!=1)
+ if(sources.empty())
throw invalid_argument("SourceGenerator::create_target");
+ if(out_suffixes.empty())
+ throw logic_error("No output suffixes");
TemplateFile &tmpl = dynamic_cast<TemplateFile &>(*sources.front());
const Component *comp = tmpl.get_component();
const SourcePackage *pkg = tmpl.get_package();
- string base = FS::basepart(FS::basename(tmpl.get_path()));
+ FS::Path subdir;
+ string base;
+ if(processing_unit==COMPONENT)
+ base = comp->get_name();
+ else
+ {
+ subdir = FS::dirname(FS::relative(tmpl.get_path(), pkg->get_source_directory()));
+ if(processing_unit==ONE_FILE)
+ base = FS::basepart(FS::basename(tmpl.get_path()));
+ else if(processing_unit==DIRECTORY)
+ {
+ base = FS::basename(subdir);
+ subdir = FS::dirname(subdir);
+ }
+ }
Target *primary = 0;
- for(list<string>::const_iterator i=out_suffixes.begin(); i!=out_suffixes.end(); ++i)
+ for(const string &s: out_suffixes)
{
- Tool *tool = builder.get_toolchain().get_tool_for_suffix(*i, true);
+ Tool *tool = builder.get_toolchain().get_tool_for_suffix(s, true);
if(tool)
{
- FS::Path fn = pkg->get_temp_directory()/comp->get_name()/(base+*i);
+ FS::Path fn = pkg->get_temp_directory()/"generated"/subdir/(base+s);
Target *target = tool->create_source(*comp, fn);
target->set_tool(*this);
- target->add_dependency(tmpl);
+ for(Target *t: sources)
+ target->add_dependency(*t);
if(primary)
primary->add_side_effect(*target);
else
primary = target;
}
+ else
+ throw runtime_error("No tool found for suffix "+s);
}
return primary;
}
-Task *SourceGenerator::run(const Target &target) const
+Task *SourceGenerator::_run(const SourceFile &out_src)
{
- const SourceFile &out_src = dynamic_cast<const SourceFile &>(target);
const FS::Path &work_dir = out_src.get_package()->get_source_directory();
+ const SourceGenerator &tool = dynamic_cast<const SourceGenerator &>(*out_src.get_tool());
vector<string> args;
- args.push_back(executable->get_path().str());
+ args.push_back(tool.get_executable()->get_path().str());
+ args.insert(args.end(), tool.arguments.begin(), tool.arguments.end());
- const Target::Dependencies &deps = target.get_dependencies();
- for(Target::Dependencies::const_iterator i=deps.begin(); i!=deps.end(); ++i)
- if(const TemplateFile *tmpl = dynamic_cast<const TemplateFile *>(*i))
+ for(const Target *d: out_src.get_dependencies())
+ if(const TemplateFile *tmpl = dynamic_cast<const TemplateFile *>(d))
args.push_back(FS::relative(tmpl->get_path(), work_dir).str());
+ if(!tool.out_argument.empty())
+ args.push_back(tool.out_argument);
args.push_back(FS::relative(out_src.get_path(), work_dir).str());
return new ExternalTask(args, work_dir);
SourceGenerator::Loader::Loader(SourceGenerator &sg):
- DataFile::ObjectLoader<SourceGenerator>(sg)
+ DataFile::ObjectLoader<SourceGenerator>(sg),
+ ConditionalLoader(sg.package, format("%s/%s", sg.package.get_name(), sg.tag))
{
+ add("argument", &Loader::argument);
+ add("arguments", &Loader::arguments);
add("command", &Loader::command);
add("in_suffix", &Loader::in_suffix);
+ add("out_argument", &SourceGenerator::out_argument);
add("out_suffix", &Loader::out_suffix);
+ add("processing_unit", static_cast<ProcessingUnit SourceGenerator::*>(&SourceGenerator::processing_unit));
+}
+
+void SourceGenerator::Loader::argument(const string &a)
+{
+ obj.arguments.push_back(a);
+}
+
+void SourceGenerator::Loader::arguments(const vector<string> &a)
+{
+ obj.arguments.insert(obj.arguments.end(), a.begin(), a.end());
}
void SourceGenerator::Loader::command(const string &c)
{
- obj.set_command((obj.package.get_source_directory()/c).str());
+ if(c.find('/')!=string::npos)
+ obj.set_command((obj.package.get_source_directory()/c).str());
+ else
+ obj.set_command(c);
}
void SourceGenerator::Loader::in_suffix(const string &s)