root/openpgpsdk/trunk/configure

Revision 251 (checked in by ben, 8 years ago)

Use eq instead of ==, fixes #11.

  • Property svn:executable set to *
Line 
1 #!/usr/bin/perl -w
2
3 use strict;
4
5 use Carp;
6 use File::Temp;
7 use File::Spec;
8 use POSIX qw(:errno_h);
9 use IO::File;
10
11 $|=1;
12
13 our $Log;
14 our $LogFile='/dev/null';
15
16 sub trace {
17     return if !$Log;
18
19     $Log->seek(0,2);
20     print $Log join '',@_;
21 }
22
23 our %Subst=(
24             CRYPTO_LIBS => '-lcrypto',
25             ZLIB => '-lz',
26             INCLUDES => '',
27             CFLAGS => '',
28            );
29
30 my %Args=(
31           '--help' => \&usage,
32           '--use-dmalloc' => sub {
33               $Subst{DM_FLAGS}='-I/usr/local/include -DDMALLOC';
34               $Subst{DM_LIB}='/usr/local/lib/libdmalloc.a';
35           },
36           '--with-openssl=' => sub {
37               my $loc=shift;
38               $Subst{INCLUDES}.=" -I $loc/include";
39               $Subst{CRYPTO_LIBS}="$loc/libcrypto.a";
40           },
41           '--with-zlib=' => sub {
42               my $loc=shift;
43               $Subst{INCLUDES}.=" -I $loc";
44               $Subst{ZLIB}="$loc/libz.a";
45           },
46           '--log=' => sub {
47               $LogFile=shift;
48               $Log=new IO::File(">$LogFile");
49               $Log->autoflush(1);
50           },
51           '--64' => sub {
52               $Subst{CFLAGS}.=' -m64';
53           },
54           '--maintainer' => sub {
55               $Subst{CFLAGS}.=' -fno-builtin';
56           },
57           '--cc=' => sub {
58               my $cc=shift;
59               $Subst{CC}=$cc;
60           }
61          );
62
63 my %Functions=(
64                'socket' => { headers => ['sys/types.h','sys/socket.h'],
65                              call => 'socket(0,0,0)',
66                              libs => [[],['socket','nsl']] },
67                );
68
69 my @Headers=qw(alloca.h);
70 my @Types=qw(time_t);
71 my @RHeaders=qw(openssl/bn.h zlib.h);
72
73 our %Knowledge=(
74                 cc => sub { return $Subst{CC} || chooseBinary('gcc','cc')
75                              || croak 'Can\'t find C compiler'; },
76                 path => sub { return [split /:/,$ENV{PATH}]; },
77                 time_t => sub { return typeInfo('time_t','time.h'); },
78                 base_cflags => sub { return '-Wall -Werror -W -g' if getKnowledge('cc') eq 'gcc'; }
79               );
80
81 while(my $arg=shift) {
82     my $arg2;
83     if($arg =~ /^(.+=)(.+)$/) {
84         $arg=$1;
85         $arg2=$2;
86     }
87     my $code=$Args{$arg};
88     croak "Don't understand: $arg" if !defined $code;
89     &$code($arg2);
90 }
91
92 #my $os=`uname -s`;
93
94 checkHeaders(\@RHeaders);
95
96 $Subst{'CC'}=getKnowledge('cc');
97 $Subst{'CFLAGS'}.=' '.getKnowledge('base_cflags');
98
99 findHeaders(\@Headers);
100 investigateTypes(\@Types);
101 my $libs=findLibraries(\%Functions);
102 $Subst{'LIBS'}=join(' ',map { "-l$_" } @$libs);
103 #exit;
104
105 fixSubst();
106
107 create('src/.depend');
108 create('examples/.depend');
109
110 fileSubst('src/Makefile','#','');
111 fileSubst('examples/Makefile','#','');
112 fileSubst('include/openpgpsdk/configure.h','/*','*/');
113
114 print "make clean\n";
115 system 'make clean' || exit;
116 print "make force_depend\n";
117 if(system('make force_depend') != 0) {
118     print STDERR "Configuration failed\n";
119     exit 1;
120 }
121
122 sub subst {
123     my $line=shift;
124
125     while(my($k,$v)=each %Subst) {
126         $line =~ s/\%$k\%/$v/g;
127     }
128     $line =~ s/\%.+?\%//g;
129     return $line;
130 }
131
132 sub chooseBinary {
133     my $path=getKnowledge('path');
134     while(my $bin=shift) {
135         foreach my $p (@$path) {
136             return $bin if -x "$p/$bin";
137         }
138     }
139     return 'false';
140 }
141
142 sub getKnowledge {
143     my $thing=shift;
144
145     croak "Asked for nonexistent knowledge $thing"
146       if !exists $Knowledge{$thing};
147
148     if(ref $Knowledge{$thing} eq 'CODE') {
149         print "Finding $thing\n";
150         $Knowledge{$thing}=&{$Knowledge{$thing}}();
151     }
152     return $Knowledge{$thing};
153 }
154
155 sub usage {
156     foreach my $k (keys %Args) {
157         print "$k\n";
158     }
159     exit;
160 }
161
162 sub findHeaders {
163     my $list=shift;
164
165     foreach my $h (@$list) {
166         my $n='HAVE_'.uc $h;
167         $n =~ s/[\.\/]/_/;
168         print "Looking for header $h ($n)\n";
169         if(-e "/usr/include/$h") {
170             $Subst{$n}="#define $n 0";
171         } else {
172             $Subst{$n}="#undef $n";
173         }
174     }
175 }
176
177 sub fileSubst {
178     my $file=shift;
179     my $cs=shift;
180     my $ce=shift;
181
182     open(T,"$file.template") || croak "Can't open $file.template: $!";
183     unlink $file;
184     open(M,">$file") || croak "Can't create $file: $!";
185
186     print M "$cs generated by configure from $file.template. Don't edit. $ce\n\n";
187
188     while(my $line=<T>) {
189         print M subst($line);
190     }
191
192     close M;
193     close T;
194
195     chmod 0444,"$file";
196 }
197
198 sub fixSubst {
199     foreach my $k (keys %Subst) {
200         $Subst{$k} =~ s/~/$ENV{HOME}/g;
201     }
202 }
203
204 sub create {
205     my $file=shift;
206
207     print "Creating $file\n";
208     open(D,">$file") || croak "Can't create $file: $!";
209     close D;
210 }
211
212 sub build {
213     my $code=shift;
214     my $link=shift;
215
216     my $cc=getKnowledge('cc');
217     my $fh=new File::Temp(SUFFIX => '.c');
218
219     my($v,$d,$f)=File::Spec->splitpath($fh->filename());
220
221     my $cur=File::Spec->rel2abs(File::Spec->curdir());
222     chdir($d) || croak "chdir($d): $!";
223
224     print $fh $code;
225     $fh->close();
226
227     my $cflag='';
228     $cflag='-c' if !defined $link;
229     my $cmd="$cc $Subst{CFLAGS} $Subst{INCLUDES} $cflag ".$f;
230     $cmd.=" $link" if defined $link;
231     trace("$cmd\n--code--\n$code--------\n");
232     my $ret=system("$cmd >> $LogFile 2>&1");
233
234     my $obj=$fh->filename();
235     $obj =~ s/\.c$/.o/;
236     unlink($obj) || $! == ENOENT || croak "unlink($obj): $!";
237     unlink('a.out') || $! == ENOENT || croak "unlink(a.out): $!";
238
239     chdir($cur) || croak "chdir($cur): $!";
240
241     return $ret == 0;
242 }
243
244 sub typeInfo {
245     my $type=shift;
246     my $header=shift;
247
248     my %info;
249
250     print "Getting info about $type.\n";
251
252     foreach my $fmt (qw(%d %ld)) {
253         my $code="#include <$header>\n";
254         $code .= "#include <stdio.h>\n";
255         $code .= "void f(void) { static $type t; printf(\"$fmt\",t); }\n";
256         if(build($code)) {
257             $info{fmt}=$fmt;
258             print "  $type printf format is $fmt\n";
259         }
260     }
261     croak "Can't determine print format for $type" if !defined $info{fmt};
262
263     return \%info;
264 }
265
266 sub investigateTypes {
267     my $types=shift;
268
269     foreach my $type (@$types) {
270         my $info=getKnowledge($type);
271         $Subst{uc($type)."_FMT"}="\"$info->{fmt}\"";
272     }
273 }
274
275 sub checkHeaders {
276     my $headers=shift;
277
278     foreach my $h (@$headers) {
279         print "Check required header $h\n";
280         build("#include <$h>\n") || croak "Can't find required header $h";
281     }
282 }
283
284 sub findLibraries {
285     my $funcs=shift;
286
287     my @libs;
288   func:
289     foreach my $func (keys %$funcs) {
290         print "Checking libraries for $func()\n";
291         my $code=join("\n",map { "#include <$_>" }
292                       @{$funcs->{$func}->{headers}});
293         $code.="\nint main() { $funcs->{$func}->{call}; return 0; }\n";
294         foreach my $lgroup (@{$funcs->{$func}->{libs}}) {
295             print "  trying ",@$lgroup ? join(', ',@$lgroup) : 'no libraries';
296             if(build($code,join(' ',map { "-l$_" } @$lgroup))) {
297                 push @libs,@$lgroup;
298                 print " ... OK\n";
299                 next func;
300             }
301             print " ... no\n";
302         }
303     }
304     return \@libs;
305 }
306
Note: See TracBrowser for help on using the browser.