3 #include <msp/core/maputils.h>
4 #include <msp/fs/utils.h>
5 #include <msp/io/print.h>
6 #include <msp/strings/lexicalcast.h>
7 #include <msp/strings/utils.h>
8 #include "androidapplicationcomponent.h"
9 #include "binarycomponent.h"
10 #include "binarypackage.h"
12 #include "datapackcomponent.h"
14 #include "installcomponent.h"
15 #include "pkgconfigfile.h"
16 #include "sourcearchivecomponent.h"
17 #include "sourcegenerator.h"
18 #include "sourcepackage.h"
20 #include "vcxprojectfile.h"
25 SourcePackage::SourcePackage(Builder &b, const string &n, const FS::Path &f):
27 source_dir(FS::dirname(f)),
34 build_file = builder.get_vfs().get_target(f);
36 build_file = new File(builder, *this, f);
37 source_archive = new SourceArchiveComponent(*this);
38 components.push_back(source_archive);
41 SourcePackage::~SourcePackage()
43 for(ComponentList::iterator i=components.begin(); i!=components.end(); ++i)
47 FS::Path SourcePackage::get_temp_directory() const
49 string subdir = builder.get_current_arch().get_name();
53 subdir += build_type->get_name();
56 const FS::Path &temp = builder.get_temp_directory();
57 if(temp.is_absolute())
58 return temp/name/subdir;
60 return source_dir/temp/subdir;
63 FS::Path SourcePackage::get_output_directory() const
65 const Architecture &arch = builder.get_current_arch();
69 return source_dir/arch.get_name();
72 const Component &SourcePackage::get_component(const string &n) const
74 for(ComponentList::const_iterator i=components.begin(); i!=components.end(); ++i)
75 if((*i)->get_name()==n)
80 bool SourcePackage::match_feature(const string &feat, const string *comp) const
82 string value = config.get_option("with_"+feat).value;
86 return lexical_cast<bool>(value);
89 void SourcePackage::set_build_type(const BuildType &t)
94 void SourcePackage::do_prepare()
96 BuildInfo final_build_info;
99 final_build_info.update_from(build_type->get_build_info());
101 final_build_info.update_from(build_info);
102 build_info = final_build_info;
104 build_info.incpath.push_back((builder.get_prefix()/"include").str());
105 build_info.libpath.push_back((builder.get_prefix()/"lib").str());
107 for(FeatureList::iterator i=features.begin(); i!=features.end(); ++i)
109 string ident = "WITH_"+toupper(i->name);
110 string value = config.get_option("with_"+i->name).value;
112 if(i->choices.empty())
114 if(!lexical_cast<bool>(value))
119 build_info.defines[ident] = value;
121 export_binfo.defines[ident] = value;
124 for(ComponentList::iterator i=components.begin(); i!=components.end(); ++i)
127 (*i)->create_build_info();
129 (*i)->update_exported_build_info(export_binfo);
134 for(ComponentList::iterator i=components.begin(); i!=components.end(); ++i)
135 (*i)->create_targets();
137 const Architecture &arch = builder.get_native_arch();
138 if(!export_binfo.libs.empty())
140 export_binfo.incpath.push_back((builder.get_prefix()/"include").str());
141 export_binfo.libpath.push_back((builder.get_prefix()/"lib").str());
143 if(arch.get_system()=="linux")
145 PkgConfigFile *pc = new PkgConfigFile(builder, *this);
146 builder.get_build_graph().get_target("install")->add_dependency(*builder.get_toolchain().get_tool("CP").create_target(*pc));
150 if(arch.get_system()=="windows")
151 new VcxProjectFile(builder, *this);
154 void SourcePackage::save_caches()
161 SourcePackage::Loader::Loader(SourcePackage &p):
162 DataFile::DerivedObjectLoader<SourcePackage, Package::Loader>(p),
163 FeatureConditional(p, p.name)
168 SourcePackage::Loader::Loader(SourcePackage &p, const Config::InputOptions &o):
169 DataFile::DerivedObjectLoader<SourcePackage, Package::Loader>(p),
170 FeatureConditional(p, p.name)
175 void SourcePackage::Loader::init(const Config::InputOptions *o)
178 add("android_application", &Loader::component<AndroidApplicationComponent>);
179 add("build_info", &Loader::build_info);
180 add("datapack", &Loader::component<DataPackComponent>);
181 add("description", &SourcePackage::description);
182 add("feature", &Loader::feature);
183 add("generate", &Loader::generate);
184 add("install", &Loader::component<InstallComponent>);
185 add("interface_version", &Loader::interface_version);
186 add("library", &Loader::component_arg<BinaryComponent, BinaryComponent::Type, BinaryComponent::LIBRARY>);
187 add("module", &Loader::component_arg<BinaryComponent, BinaryComponent::Type, BinaryComponent::MODULE>);
188 add("program", &Loader::component_arg<BinaryComponent, BinaryComponent::Type, BinaryComponent::PROGRAM>);
189 add("source_archive", &Loader::source_archive);
190 add("source_tarball", &Loader::source_archive);
191 add("tarball", &Loader::tarball);
192 add("version", &Loader::version);
195 void SourcePackage::Loader::finish()
197 /* Make sure the source tarball is last in the list so targets from all
198 other components wil be created first */
199 ComponentList::iterator i = find(obj.components.begin(), obj.components.end(), obj.source_archive);
200 if(i!=obj.components.end())
201 obj.components.splice(obj.components.end(), obj.components, i);
204 void SourcePackage::Loader::feature(const string &n, const string &d)
207 feat.description = d;
209 obj.features.push_back(feat);
211 const Config::Option &opt = obj.config.add_option(feat);
214 Config::InputOptions::const_iterator i = options->find(opt.name);
215 if(i!=options->end())
216 obj.config.set_option(opt.name, i->second);
221 void SourcePackage::Loader::component(const string &n)
223 C *comp = new C(obj, n);
225 obj.components.push_back(comp);
228 template<typename C, typename A, A a>
229 void SourcePackage::Loader::component_arg(const string &n)
231 C *comp = new C(obj, n, a);
233 obj.components.push_back(comp);
236 void SourcePackage::Loader::build_info()
238 load_sub(obj.build_info);
241 void SourcePackage::Loader::generate(const string &tag)
243 SourceGenerator *gen = new SourceGenerator(obj.builder, obj, tag);
245 obj.local_tools.add_tool(gen);
248 void SourcePackage::Loader::interface_version(const string &v)
250 obj.interface_version = v;
251 if(obj.version.empty())
255 void SourcePackage::Loader::source_archive()
257 load_sub(*obj.source_archive);
260 void SourcePackage::Loader::tarball(const string &)
262 IO::print("%s: Deprecated tarball component ignored\n", get_source());
265 void SourcePackage::Loader::version(const string &v)
269 string::size_type i = 0;
270 for(unsigned dots=0; i<obj.version.size(); ++i)
271 if(obj.version[i]=='.' && ++dots>=2)
273 obj.interface_version = obj.version.substr(0, i);