]> git.tdb.fi Git - builder.git/blob - source/package.cpp
Add command line options (not all of them work yet)
[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("--silence-errors");
118         argv.push_back("--cflags");
119         argv.push_back("--libs");
120         argv.push_back(name);
121         vector<string> info=split(run_command(argv));
122         
123         if(info.empty())
124         {
125                 if(name=="opengl")
126                         info.push_back("-lGL");
127                 else if(name=="pthread")
128                         info.push_back("-lpthread");
129                 else
130                         return 0;
131         }
132         
133         Package *pkg=new Package(b, name, info);
134         return pkg;
135 }
136
137 void Package::init_buildable()
138 {
139         buildable=true;
140
141         config.add_option("tempdir",  "temp",   "Directory for storing temporary files");
142         config.add_option("optimize", "0",      "Apply compiler optimizations");
143         config.add_option("debug",    "0",      "Produce debugging symbols");
144         config.add_option("cpu",      "auto",   "CPU type to optimize for");
145         config.add_option("arch",     "native", "Architecture for cross-compiling");
146
147         const char *home=getenv("HOME");
148         unsigned flags=get_install_flags();
149         if(flags)
150                 config.add_option("prefix",     string(home)+"/local"/*"/usr"*/,            "Installation prefix");
151         /*if(flags&INCLUDE)
152                 config.add_option("includedir", "$prefix/include", "Header installation directory");
153         if(flags&BIN)
154                 config.add_option("includedir", "$prefix/bin",     "Binary installation directory");
155         if(flags&LIB)
156                 config.add_option("includedir", "$prefix/lib",     "Library installation directory");
157         if(flags&DATA)
158                 config.add_option("includedir", "$prefix/share",   "Data installation directory");*/
159 }
160
161 unsigned Package::get_install_flags()
162 {
163         unsigned flags=0;
164         for(ComponentList::iterator i=components.begin(); i!=components.end(); ++i)
165         {
166                 if(i->get_install())
167                 {
168                         if(i->get_type()==Component::PROGRAM)
169                                 flags|=BIN;
170                         else if(i->get_type()==Component::LIBRARY || i->get_type()==Component::MODULE)
171                                 flags|=LIB;
172                 }
173                 if(!i->get_install_headers().empty())
174                         flags|=INCLUDE;
175         }
176
177         return flags;
178 }
179
180 Package::Loader::Loader(Package &p):
181         pkg(p)
182 {
183         add("version",     &Package::version);
184         add("description", &Package::description);
185         add("require",     &Loader::require);
186         add("program",     &Loader::program);
187         add("library",     &Loader::library);
188         add("headers",     &Loader::headers);
189 }
190
191 Package::Loader::~Loader()
192 {
193         pkg.init_buildable();
194 }
195
196 void Package::Loader::require(const string &n)
197 {
198         pkg.requires.push_back(PackageRef(pkg.builder, n));
199 }
200
201 void Package::Loader::program(const string &n)
202 {
203         Component prog(pkg, Component::PROGRAM, n);
204         load_sub(prog);
205         pkg.components.push_back(prog);
206 }
207
208 void Package::Loader::library(const string &n)
209 {
210         Component prog(pkg, Component::LIBRARY, n);
211         load_sub(prog);
212         pkg.components.push_back(prog);
213 }
214
215 void Package::Loader::headers(const string &n)
216 {
217         Component prog(pkg, Component::HEADERS, n);
218         load_sub(prog);
219         pkg.components.push_back(prog);
220 }