python: Build shared python modules for generic use.
[ira/wip.git] / source / build / smb_build / input.pm
1 # Samba Build System
2 # - the input checking functions
3 #
4 #  Copyright (C) Stefan (metze) Metzmacher 2004
5 #  Copyright (C) Jelmer Vernooij 2004
6 #  Released under the GNU GPL
7
8 use smb_build::config;
9 use strict;
10 package smb_build::input;
11 use File::Basename;
12
13 my $srcdir = $config::config{srcdir};
14
15 sub strtrim($)
16 {
17         $_ = shift;
18         s/^[\t\n ]*//g;
19         s/[\t\n ]*$//g;
20         return $_;
21 }
22
23 sub str2array($)
24 {
25         $_ = shift;
26         s/^[\t\n ]*//g;
27         s/[\t\n ]*$//g;
28         s/([\t\n ]+)/ /g;
29
30         return () if (length($_)==0);
31         return split /[ \t\n]/;
32 }
33
34 sub add_libreplace($)
35 {
36         my ($part) = @_;
37
38         return if ($part->{NAME} eq "LIBREPLACE");
39         return if ($part->{NAME} eq "LIBREPLACE_HOSTCC");
40         return if ($part->{NAME} eq "REPLACE_READDIR");
41
42         foreach my $n (@{$part->{PRIVATE_DEPENDENCIES}}) {
43                 return if ($n eq "LIBREPLACE");
44                 return if ($n eq "LIBREPLACE_HOSTCC");
45         }
46         foreach my $n (@{$part->{PUBLIC_DEPENDENCIES}}) {
47                 return if ($n eq "LIBREPLACE");
48                 return if ($n eq "LIBREPLACE_HOSTCC");
49         }
50
51         if (defined($part->{USE_HOSTCC}) && $part->{USE_HOSTCC} eq "YES") {
52                 unshift (@{$part->{PRIVATE_DEPENDENCIES}}, "LIBREPLACE_HOSTCC");
53         } else {
54                 unshift (@{$part->{PRIVATE_DEPENDENCIES}}, "LIBREPLACE");
55         }
56 }
57
58 sub check_subsystem($$$)
59 {
60         my ($INPUT, $subsys, $default_ot) = @_;
61         return if ($subsys->{ENABLE} ne "YES");
62         
63         unless(defined($subsys->{OUTPUT_TYPE})) {
64                 $subsys->{OUTPUT_TYPE} = $default_ot;
65         }
66
67         unless (defined($subsys->{INIT_FUNCTION_TYPE})) { $subsys->{INIT_FUNCTION_TYPE} = "NTSTATUS (*) (void)"; }
68         unless (defined($subsys->{INIT_FUNCTION_SENTINEL})) { $subsys->{INIT_FUNCTION_SENTINEL} = "NULL"; }
69
70         add_libreplace($subsys);
71 }
72
73 sub check_module($$$)
74 {
75         my ($INPUT, $mod, $default_ot) = @_;
76
77         die("Module $mod->{NAME} does not have a SUBSYSTEM set") if not defined($mod->{SUBSYSTEM});
78
79         my $use_default = 0;
80         
81         if (not exists($INPUT->{$mod->{SUBSYSTEM}}{INIT_FUNCTIONS})) {
82                 $INPUT->{$mod->{SUBSYSTEM}}{INIT_FUNCTIONS} = [];
83         }
84
85         if (!(defined($INPUT->{$mod->{SUBSYSTEM}}))) {
86                 $mod->{ENABLE} = "NO";
87                 return;
88         }
89
90         if ($INPUT->{$mod->{SUBSYSTEM}} eq "NO") {
91                 $mod->{ENABLE} = "NO";
92                 return;
93         }
94
95         return if ($mod->{ENABLE} ne "YES");
96
97         if (exists($INPUT->{$mod->{SUBSYSTEM}}{INIT_FUNCTION_TYPE})) {
98                 $mod->{INIT_FUNCTION_TYPE} = $INPUT->{$mod->{SUBSYSTEM}}{INIT_FUNCTION_TYPE};
99         } else {
100                 $mod->{INIT_FUNCTION_TYPE} = "NTSTATUS (*) (void)";
101         }
102
103         if (not defined($mod->{OUTPUT_TYPE})) {
104                 if ($INPUT->{$mod->{SUBSYSTEM}}->{TYPE} eq "EXT_LIB") {
105                         $mod->{OUTPUT_TYPE} = ["SHARED_LIBRARY"];
106                 } else {
107                         $mod->{OUTPUT_TYPE} = $default_ot;
108                 }
109         }
110
111         if (grep(/SHARED_LIBRARY/, @{$mod->{OUTPUT_TYPE}})) {
112                 my $sane_subsystem = lc($mod->{SUBSYSTEM});
113                 $sane_subsystem =~ s/^lib//;
114                 $mod->{INSTALLDIR} = "MODULESDIR/$sane_subsystem";
115                 push (@{$mod->{PUBLIC_DEPENDENCIES}}, $mod->{SUBSYSTEM});
116         } 
117         if (grep(/INTEGRATED/, @{$mod->{OUTPUT_TYPE}})) {
118                 push (@{$INPUT->{$mod->{SUBSYSTEM}}{INIT_FUNCTIONS}}, $mod->{INIT_FUNCTION}) if defined($mod->{INIT_FUNCTION});
119         }
120         add_libreplace($mod);
121 }
122
123 sub check_library($$$)
124 {
125         my ($INPUT, $lib, $default_ot) = @_;
126
127         return if ($lib->{ENABLE} ne "YES");
128
129         unless (defined($lib->{OUTPUT_TYPE})) {
130                 $lib->{OUTPUT_TYPE} = $default_ot;
131         }
132
133         if (defined($lib->{VERSION}) and not defined($lib->{SO_VERSION})) {
134                 print "$lib->{NAME}: Please specify SO_VERSION when specifying VERSION\n";
135                 return;
136         }
137
138         if (defined($lib->{SO_VERSION}) and not defined($lib->{VERSION})) {
139                 print "$lib->{NAME}: Please specify VERSION when specifying SO_VERSION\n";
140                 return;
141         }
142
143         unless (defined($lib->{INIT_FUNCTION_TYPE})) { $lib->{INIT_FUNCTION_TYPE} = "NTSTATUS (*) (void)"; }
144
145         unless (defined($lib->{INIT_FUNCTION_SENTINEL})) { $lib->{INIT_FUNCTION_SENTINEL} = "NULL"; }
146
147         unless(defined($lib->{INSTALLDIR})) {
148                 $lib->{INSTALLDIR} = "LIBDIR";
149         }
150
151         add_libreplace($lib);
152 }
153
154 sub check_python($$$)
155 {
156         my ($INPUT, $python, $default_ot) = @_;
157
158         return if ($INPUT->{LIBPYTHON}{ENABLE} ne "YES");
159
160         $python->{INSTALLDIR} = "PYTHONDIR";
161         unless (defined($python->{CFLAGS})) { $python->{CFLAGS} = []; }
162         if (defined($python->{SWIG_FILE})) {
163                 my $dirname = dirname($python->{SWIG_FILE});
164                 my $basename = basename($python->{SWIG_FILE}, ".i");
165
166                 $dirname .= "/" unless $dirname =~ /\/$/;
167                 $dirname = "" if $dirname eq "./";
168
169                 $python->{OBJ_FILES} = ["$dirname$basename\_wrap.o"];
170                 $python->{LIBRARY_REALNAME} = "_$basename.\$(SHLIBEXT)";
171                 $python->{PYTHON_FILES} = ["$dirname$basename.py"];
172                 push (@{$python->{CFLAGS}}, $config::config{CFLAG_NO_UNUSED_MACROS});
173                 push (@{$python->{CFLAGS}}, $config::config{CFLAG_NO_CAST_QUAL});
174                 $python->{INIT_FUNCTION} = "{ (char *)\"_$basename\", init_$basename }";
175         } else {
176                 my $basename = $python->{NAME};
177                 $basename =~ s/^python_//g;
178                 $python->{LIBRARY_REALNAME} = "$basename.\$(SHLIBEXT)";
179                 $python->{INIT_FUNCTION} = "{ (char *)\"$basename\", init$basename }";
180         }
181         push (@{$python->{CFLAGS}}, @{$INPUT->{EXT_LIB_PYTHON}->{CFLAGS}});
182
183         $python->{SUBSYSTEM} = "LIBPYTHON";
184
185         check_module($INPUT, $python, ["SHARED_LIBRARY", "STATIC_LIBRARY"]);
186 }
187
188 sub check_binary($$)
189 {
190         my ($INPUT, $bin) = @_;
191
192         return if ($bin->{ENABLE} ne "YES");
193
194         ($bin->{BINARY} = (lc $bin->{NAME})) if not defined($bin->{BINARY});
195
196         $bin->{OUTPUT_TYPE} = ["BINARY"];
197         add_libreplace($bin);
198 }
199
200 sub import_integrated($$)
201 {
202         my ($lib, $depend) = @_;
203
204         foreach my $mod (values %$depend) {
205                 next if(not defined($mod->{OUTPUT_TYPE}));
206                 next if(not grep(/INTEGRATED/, @{$mod->{OUTPUT_TYPE}}));
207                 next if(not defined($mod->{SUBSYSTEM}));
208                 next if($mod->{SUBSYSTEM} ne $lib->{NAME});
209                 next if($mod->{ENABLE} ne "YES");
210
211                 push (@{$lib->{FULL_OBJ_LIST}}, "\$($mod->{TYPE}_$mod->{NAME}_FULL_OBJ_LIST)");
212                 push (@{$lib->{LINK_FLAGS}}, "\$($mod->{TYPE}_$mod->{NAME}_LINK_FLAGS)");
213                 push (@{$lib->{CFLAGS}}, @{$mod->{CFLAGS}}) if defined($mod->{CFLAGS});
214                 push (@{$lib->{PUBLIC_DEPENDENCIES}}, @{$mod->{PUBLIC_DEPENDENCIES}}) if defined($mod->{PUBLIC_DEPENDENCIES});
215                 push (@{$lib->{PRIVATE_DEPENDENCIES}}, @{$mod->{PRIVATE_DEPENDENCIES}}) if defined($mod->{PRIVATE_DEPENDENCIES});
216
217                 $mod->{ENABLE} = "NO";
218         }
219 }
220
221 sub calc_unique_deps($$$$$$$$)
222 {
223         sub calc_unique_deps($$$$$$$$);
224         my ($name, $INPUT, $deps, $udeps, $withlibs, $forward, $pubonly, $busy) = @_;
225
226         foreach my $n (@$deps) {
227                 die("Dependency unknown: $n (for $name)") unless (defined($INPUT->{$n}));
228                 die("Recursive dependency: $n, list: " . join(',', @$busy)) if (grep (/^$n$/, @$busy));
229                 next if (grep /^$n$/, @$udeps);
230                 my $dep = $INPUT->{$n};
231
232                 push (@{$udeps}, $dep->{NAME}) if $forward;
233
234                 if (defined ($dep->{OUTPUT_TYPE}) && 
235                         ($withlibs or 
236                         (@{$dep->{OUTPUT_TYPE}}[0] eq "INTEGRATED") or 
237                         (@{$dep->{OUTPUT_TYPE}}[0] eq "STATIC_LIBRARY"))) {
238                                 push (@$busy, $dep->{NAME});
239                                 calc_unique_deps($dep->{NAME}, $INPUT, $dep->{PUBLIC_DEPENDENCIES}, $udeps, $withlibs, $forward, $pubonly, $busy);
240                                 calc_unique_deps($dep->{NAME}, $INPUT, $dep->{PRIVATE_DEPENDENCIES}, $udeps, $withlibs, $forward, $pubonly, $busy) unless $pubonly;
241                                 pop (@$busy);
242                 }
243
244                 unshift (@{$udeps}, $dep->{NAME}) unless $forward;
245         }
246 }
247
248 sub check($$$$$)
249 {
250         my ($INPUT, $enabled, $subsys_ot, $lib_ot, $module_ot) = @_;
251
252         foreach my $part (values %$INPUT) {
253                 unless (defined($part->{STANDARD_VISIBILITY})) {
254                         if ($part->{TYPE} eq "MODULE" or $part->{TYPE} eq "BINARY") {
255                                 $part->{STANDARD_VISIBILITY} = "hidden";
256                         } else {
257                                 $part->{STANDARD_VISIBILITY} = "default";
258                         }
259                 }
260
261                 unless (defined($part->{PUBLIC_HEADERS})) {
262                         $part->{PUBLIC_HEADERS} = [];
263                 }
264                 
265                 if (defined($part->{PUBLIC_PROTO_HEADER})) {
266                         push (@{$part->{PUBLIC_HEADERS}}, $part->{PUBLIC_PROTO_HEADER});
267                 }
268
269                 if (defined($enabled->{$part->{NAME}})) { 
270                         $part->{ENABLE} = $enabled->{$part->{NAME}};
271                         next;
272                 }
273                 
274                 unless(defined($part->{ENABLE})) {
275                         if ($part->{TYPE} eq "EXT_LIB") {
276                                 $part->{ENABLE} = "NO";
277                         } else {
278                                 $part->{ENABLE} = "YES";
279                         }
280                 }
281         }
282
283         foreach my $k (keys %$INPUT) {
284                 my $part = $INPUT->{$k};
285
286                 $part->{LINK_FLAGS} = [];
287                 $part->{FULL_OBJ_LIST} = ["\$($part->{TYPE}_$part->{NAME}_OBJ_LIST)"];
288
289                 if ($part->{TYPE} eq "SUBSYSTEM") { 
290                         check_subsystem($INPUT, $part, $subsys_ot);
291                 } elsif ($part->{TYPE} eq "MODULE") {
292                         check_module($INPUT, $part, $module_ot);
293                 } elsif ($part->{TYPE} eq "LIBRARY") {
294                         check_library($INPUT, $part, $lib_ot);
295                 } elsif ($part->{TYPE} eq "BINARY") {
296                         check_binary($INPUT, $part);
297                 } elsif ($part->{TYPE} eq "PYTHON") {
298                         check_python($INPUT, $part, $module_ot);
299                 } elsif ($part->{TYPE} eq "EXT_LIB") {
300                 } else {
301                         die("Unknown type $part->{TYPE}");
302                 }
303         }
304
305         foreach my $part (values %$INPUT) {
306                 if (defined($part->{INIT_FUNCTIONS})) {
307                         push (@{$part->{LINK_FLAGS}}, "\$(DYNEXP)");
308                 }
309                 import_integrated($part, $INPUT);
310         }
311
312         foreach my $part (values %$INPUT) {
313                 $part->{UNIQUE_DEPENDENCIES_LINK} = [];
314                 calc_unique_deps($part->{NAME}, $INPUT, $part->{PUBLIC_DEPENDENCIES}, $part->{UNIQUE_DEPENDENCIES_LINK}, 0, 0, 0, []);
315                 calc_unique_deps($part->{NAME}, $INPUT, $part->{PRIVATE_DEPENDENCIES}, $part->{UNIQUE_DEPENDENCIES_LINK}, 0, 0, 0, []);
316         }
317
318         foreach my $part (values %$INPUT) {
319                 $part->{UNIQUE_DEPENDENCIES_COMPILE} = [];
320                 calc_unique_deps($part->{NAME}, $INPUT, $part->{PUBLIC_DEPENDENCIES}, $part->{UNIQUE_DEPENDENCIES_COMPILE}, 1, 1, 1, []);
321                 calc_unique_deps($part->{NAME}, $INPUT, $part->{PRIVATE_DEPENDENCIES}, $part->{UNIQUE_DEPENDENCIES_COMPILE}, 1, 1, 1, []);
322         }
323
324         foreach my $part (values %$INPUT) {
325                 $part->{UNIQUE_DEPENDENCIES_ALL} = [];
326                 calc_unique_deps($part->{NAME}, $INPUT, $part->{PUBLIC_DEPENDENCIES}, $part->{UNIQUE_DEPENDENCIES_ALL}, 1, 0, 0, []);
327                 calc_unique_deps($part->{NAME}, $INPUT, $part->{PRIVATE_DEPENDENCIES}, $part->{UNIQUE_DEPENDENCIES_ALL}, 1, 0, 0, []);
328         }
329
330         return $INPUT;
331 }
332
333 1;