]> git.tdb.fi Git - builder.git/blob - source/package.cpp
e6e3b5099c1814a46ab02586e750f936843e3c27
[builder.git] / source / package.cpp
1 #include <msp/strconv.h>
2 #include <msp/strutils.h>
3 #include "builder.h"
4 #include "misc.h"
5 #include "package.h"
6
7 using namespace std;
8 using namespace Msp;
9
10 PackageRef::PackageRef(Builder &b, const string &n):
11         builder(b),
12         name(n),
13         package(0)
14 { }
15
16 Package *PackageRef::resolve()
17 {
18         if(!package)
19                 package=builder.get_package(name);
20         return package;
21 }
22
23 Package::Package(Builder &b, const string &n, const Path::Path &s):
24         builder(b),
25         name(n),
26         source(s),
27         buildable(true),
28         build_info_ready(false)
29 { }
30
31 Package::Package(Builder &b, const string &n, const vector<string> &info):
32         builder(b),
33         name(n),
34         buildable(false),
35         build_info_ready(true)
36 {
37         for(vector<string>::const_iterator i=info.begin(); i!=info.end(); ++i)
38         {
39                 if(!i->compare(0, 2, "-I"))
40                         export_binfo.incpath.push_back(i->substr(2));
41                 else if(!i->compare(0, 2, "-D"))
42                         export_binfo.defines.push_back(i->substr(2));
43                 else if(!i->compare(0, 2, "-L"))
44                         export_binfo.libpath.push_back(i->substr(2));
45                 else if(!i->compare(0, 2, "-l"))
46                         export_binfo.libs.push_back(i->substr(2));
47         }
48 }
49
50 void Package::resolve_refs()
51 {
52         for(list<PackageRef>::iterator i=requires.begin(); i!=requires.end(); ++i)
53                 i->resolve();
54 }
55
56 void Package::create_build_info()
57 {
58         if(build_info_ready)
59                 return;
60         
61         for(list<PackageRef>::iterator i=requires.begin(); i!=requires.end(); ++i)
62         {
63                 if(!i->get_package())
64                         continue;
65                 i->get_package()->create_build_info();
66                 build_info.add(i->get_package()->get_exported_binfo());
67         }
68         
69         build_info.cflags.push_back("-Wall");
70         build_info.cflags.push_back("-Wshadow");
71         build_info.cflags.push_back("-Wextra");
72         build_info.cflags.push_back("-Wpointer-arith");
73         build_info.cflags.push_back("-Wconversion");
74         build_info.cflags.push_back("-Werror");
75
76         unsigned flags=get_install_flags();
77
78         if(flags&INCLUDE)
79                 export_binfo.incpath.push_back((Path::Path(config.get_option("prefix").value)/"include").str());
80         if(flags&LIB)
81                 export_binfo.libpath.push_back((Path::Path(config.get_option("prefix").value)/"lib").str());
82
83         string optimize=config.get_option("optimize").value;
84         if(strtol(optimize))
85         {
86                 build_info.cflags.push_back("-O"+optimize);
87                 string cpu=config.get_option("cpu").value;
88                 if(cpu!="auto")
89                         build_info.cflags.push_back("-march="+cpu);
90         }
91
92         if(strtobool(config.get_option("debug").value))
93                 build_info.cflags.push_back("-ggdb");
94
95         build_info.unique();
96         export_binfo.unique();
97
98         for(list<Component>::iterator i=components.begin(); i!=components.end(); ++i)
99         {
100                 i->create_build_info();
101                 if(i->get_type()==Component::LIBRARY)
102                         export_binfo.libs.push_back(i->get_name());
103         }
104
105         build_info_ready=true;
106 }
107
108 void Package::process_options(const RawOptionMap &opts)
109 {
110         config.process(opts);
111 }
112
113 Package *Package::create(Builder &b, const string &name)
114 {
115         list<string> argv;
116         argv.push_back("pkg-config");
117         argv.push_back("--cflags");
118         argv.push_back("--libs");
119         argv.push_back(name);
120         vector<string> info=split(run_command(argv));
121         
122         if(info.empty())
123                 return 0;
124         
125         Package *pkg=new Package(b, name, info);
126         return pkg;
127 }
128
129 void Package::init_buildable()
130 {
131         buildable=true;
132
133         config.add_option("tempdir",  "temp",   "Directory for storing temporary files");
134         config.add_option("optimize", "0",      "Apply compiler optimizations");
135         config.add_option("debug",    "0",      "Produce debugging symbols");
136         config.add_option("cpu",      "auto",   "CPU type to optimize for");
137         config.add_option("arch",     "native", "Architecture for cross-compiling");
138
139         const char *home=getenv("HOME");
140         unsigned flags=get_install_flags();
141         if(flags)
142                 config.add_option("prefix",     string(home)+"/local"/*"/usr"*/,            "Installation prefix");
143         /*if(flags&INCLUDE)
144                 config.add_option("includedir", "$prefix/include", "Header installation directory");
145         if(flags&BIN)
146                 config.add_option("includedir", "$prefix/bin",     "Binary installation directory");
147         if(flags&LIB)
148                 config.add_option("includedir", "$prefix/lib",     "Library installation directory");
149         if(flags&DATA)
150                 config.add_option("includedir", "$prefix/share",   "Data installation directory");*/
151 }
152
153 unsigned Package::get_install_flags()
154 {
155         unsigned flags=0;
156         for(ComponentList::iterator i=components.begin(); i!=components.end(); ++i)
157         {
158                 if(i->get_install())
159                 {
160                         if(i->get_type()==Component::PROGRAM)
161                                 flags|=BIN;
162                         else if(i->get_type()==Component::LIBRARY || i->get_type()==Component::MODULE)
163                                 flags|=LIB;
164                 }
165                 if(!i->get_install_headers().empty())
166                         flags|=INCLUDE;
167         }
168
169         return flags;
170 }
171
172 Package::Loader::Loader(Package &p):
173         pkg(p)
174 {
175         add("version",     &Package::version);
176         add("description", &Package::description);
177         add("require",     &Loader::require);
178         add("program",     &Loader::program);
179         add("library",     &Loader::library);
180 }
181
182 Package::Loader::~Loader()
183 {
184         pkg.init_buildable();
185 }
186
187 void Package::Loader::require(const string &n)
188 {
189         pkg.requires.push_back(PackageRef(pkg.builder, n));
190 }
191
192 void Package::Loader::program(const std::string &n)
193 {
194         Component prog(pkg, Component::PROGRAM, n);
195         load_sub(prog);
196         pkg.components.push_back(prog);
197 }
198
199 void Package::Loader::library(const std::string &n)
200 {
201         Component prog(pkg, Component::LIBRARY, n);
202         load_sub(prog);
203         pkg.components.push_back(prog);
204 }