]> git.ipfire.org Git - thirdparty/openssl.git/blob - Configure
Make apps/progs.pl more flexible
[thirdparty/openssl.git] / Configure
1 #! /usr/bin/env perl
2 # -*- mode: perl; -*-
3
4 ## Configure -- OpenSSL source tree configuration script
5
6 require 5.10.0;
7 use strict;
8 use File::Basename;
9 use File::Spec::Functions qw/:DEFAULT abs2rel rel2abs/;
10 use File::Path qw/mkpath/;
11
12 # see INSTALL for instructions.
13
14 my $usage="Usage: Configure [no-<cipher> ...] [enable-<cipher> ...] [-Dxxx] [-lxxx] [-Lxxx] [-fxxx] [-Kxxx] [no-hw-xxx|no-hw] [[no-]threads] [[no-]shared] [[no-]zlib|zlib-dynamic] [no-asm] [no-dso] [no-egd] [sctp] [386] [--prefix=DIR] [--openssldir=OPENSSLDIR] [--with-xxx[=vvv]] [--config=FILE] os/compiler[:flags]\n";
15
16 # Options:
17 #
18 # --config add the given configuration file, which will be read after
19 # any "Configurations*" files that are found in the same
20 # directory as this script.
21 # --prefix prefix for the OpenSSL installation, which includes the
22 # directories bin, lib, include, share/man, share/doc/openssl
23 # This becomes the value of INSTALLTOP in Makefile
24 # (Default: /usr/local)
25 # --openssldir OpenSSL data area, such as openssl.cnf, certificates and keys.
26 # If it's a relative directory, it will be added on the directory
27 # given with --prefix.
28 # This becomes the value of OPENSSLDIR in Makefile and in C.
29 # (Default: PREFIX/ssl)
30 #
31 # --cross-compile-prefix Add specified prefix to binutils components.
32 #
33 # --api One of 0.9.8, 1.0.0 or 1.1.0. Do not compile support for
34 # interfaces deprecated as of the specified OpenSSL version.
35 #
36 # no-hw-xxx do not compile support for specific crypto hardware.
37 # Generic OpenSSL-style methods relating to this support
38 # are always compiled but return NULL if the hardware
39 # support isn't compiled.
40 # no-hw do not compile support for any crypto hardware.
41 # [no-]threads [don't] try to create a library that is suitable for
42 # multithreaded applications (default is "threads" if we
43 # know how to do it)
44 # [no-]shared [don't] try to create shared libraries when supported.
45 # [no-]pic [don't] try to build position independent code when supported.
46 # If disabled, it also disables shared and dynamic-engine.
47 # no-asm do not use assembler
48 # no-dso do not compile in any native shared-library methods. This
49 # will ensure that all methods just return NULL.
50 # no-egd do not compile support for the entropy-gathering daemon APIs
51 # [no-]zlib [don't] compile support for zlib compression.
52 # zlib-dynamic Like "zlib", but the zlib library is expected to be a shared
53 # library and will be loaded in run-time by the OpenSSL library.
54 # sctp include SCTP support
55 # 386 generate 80386 code
56 # enable-weak-ssl-ciphers
57 # Enable weak ciphers that are disabled by default. This currently
58 # only includes RC4 based ciphers.
59 # no-sse2 disables IA-32 SSE2 code, above option implies no-sse2
60 # no-<cipher> build without specified algorithm (rsa, idea, rc5, ...)
61 # -<xxx> +<xxx> compiler options are passed through
62 #
63 # DEBUG_SAFESTACK use type-safe stacks to enforce type-safety on stack items
64 # provided to stack calls. Generates unique stack functions for
65 # each possible stack type.
66 # BN_LLONG use the type 'long long' in crypto/bn/bn.h
67 # RC4_CHAR use 'char' instead of 'int' for RC4_INT in crypto/rc4/rc4.h
68 # Following are set automatically by this script
69 #
70 # MD5_ASM use some extra md5 assember,
71 # SHA1_ASM use some extra sha1 assember, must define L_ENDIAN for x86
72 # RMD160_ASM use some extra ripemd160 assember,
73 # SHA256_ASM sha256_block is implemented in assembler
74 # SHA512_ASM sha512_block is implemented in assembler
75 # AES_ASM ASE_[en|de]crypt is implemented in assembler
76
77 # Minimum warning options... any contributions to OpenSSL should at least get
78 # past these.
79
80 my $gcc_devteam_warn = "-DPEDANTIC -DREF_DEBUG -DDEBUG_UNUSED -DBIO_DEBUG"
81 . " -pedantic"
82 . " -Wall"
83 . " -Wno-long-long"
84 . " -Wsign-compare"
85 . " -Wmissing-prototypes"
86 . " -Wshadow"
87 . " -Wformat"
88 . " -Wtype-limits"
89 . " -Werror"
90 ;
91
92 # These are used in addition to $gcc_devteam_warn when the compiler is clang.
93 # TODO(openssl-team): fix problems and investigate if (at least) the
94 # following warnings can also be enabled:
95 # -Wswitch-enum
96 # -Wcast-align
97 # -Wunreachable-code
98 # -Wlanguage-extension-token -- no, we use asm()
99 # -Wunused-macros -- no, too tricky for BN and _XOPEN_SOURCE etc
100 # -Wextended-offsetof -- no, needed in CMS ASN1 code
101 my $clang_devteam_warn = ""
102 . " -Qunused-arguments"
103 . " -Wextra"
104 . " -Wno-unused-parameter"
105 . " -Wno-missing-field-initializers"
106 . " -Wno-language-extension-token"
107 . " -Wno-extended-offsetof"
108 . " -Wconditional-uninitialized"
109 . " -Wincompatible-pointer-types-discards-qualifiers"
110 . " -Wmissing-variable-declarations"
111 ;
112
113 # This adds backtrace information to the memory leak info. Is only used
114 # when crypto-mdebug-backtrace is enabled.
115 my $memleak_devteam_backtrace = "-rdynamic";
116
117 my $strict_warnings = 0;
118
119 # As for $BSDthreads. Idea is to maintain "collective" set of flags,
120 # which would cover all BSD flavors. -pthread applies to them all,
121 # but is treated differently. OpenBSD expands is as -D_POSIX_THREAD
122 # -lc_r, which is sufficient. FreeBSD 4.x expands it as -lc_r,
123 # which has to be accompanied by explicit -D_THREAD_SAFE and
124 # sometimes -D_REENTRANT. FreeBSD 5.x expands it as -lc_r, which
125 # seems to be sufficient?
126 our $BSDthreads="-pthread -D_THREAD_SAFE -D_REENTRANT";
127
128 #
129 # API compability name to version number mapping.
130 #
131 my $maxapi = "1.1.0"; # API for "no-deprecated" builds
132 my $apitable = {
133 "1.1.0" => "0x10100000L",
134 "1.0.0" => "0x10000000L",
135 "0.9.8" => "0x00908000L",
136 };
137
138 our %table = ();
139 our %config = ();
140 our %withargs = ();
141
142 # Forward declarations ###############################################
143
144 # read_config(filename)
145 #
146 # Reads a configuration file and populates %table with the contents
147 # (which the configuration file places in %targets).
148 sub read_config;
149
150 # resolve_config(target)
151 #
152 # Resolves all the late evalutations, inheritances and so on for the
153 # chosen target and any target it inherits from.
154 sub resolve_config;
155
156
157 # Information collection #############################################
158
159 # Unified build supports separate build dir
160 my $srcdir = catdir(absolutedir(dirname($0))); # catdir ensures local syntax
161 my $blddir = catdir(absolutedir(".")); # catdir ensures local syntax
162 my $dofile = abs2rel(catfile($srcdir, "util/dofile.pl"));
163
164 $config{sourcedir} = abs2rel($srcdir);
165 $config{builddir} = abs2rel($blddir);
166
167 # Collect version numbers
168 $config{version} = "unknown";
169 $config{version_num} = "unknown";
170 $config{shlib_version_number} = "unknown";
171 $config{shlib_version_history} = "unknown";
172
173 collect_information(
174 collect_from_file(catfile($srcdir,'include/openssl/opensslv.h')),
175 qr/OPENSSL.VERSION.TEXT.*OpenSSL (\S+) / => sub { $config{version} = $1; },
176 qr/OPENSSL.VERSION.NUMBER.*(0x\S+)/ => sub { $config{version_num}=$1 },
177 qr/SHLIB_VERSION_NUMBER *"([^"]+)"/ => sub { $config{shlib_version_number}=$1 },
178 qr/SHLIB_VERSION_HISTORY *"([^"]*)"/ => sub { $config{shlib_version_history}=$1 }
179 );
180 if ($config{shlib_version_history} ne "") { $config{shlib_version_history} .= ":"; }
181
182 ($config{major}, $config{minor})
183 = ($config{version} =~ /^([0-9]+)\.([0-9\.]+)/);
184 ($config{shlib_major}, $config{shlib_minor})
185 = ($config{shlib_version_number} =~ /^([0-9]+)\.([0-9\.]+)/);
186 die "erroneous version information in opensslv.h: ",
187 "$config{major}, $config{minor}, $config{shlib_major}, $config{shlib_minor}\n"
188 if ($config{major} eq "" || $config{minor} eq ""
189 || $config{shlib_major} eq "" || $config{shlib_minor} eq "");
190
191 # Collect target configurations
192
193 my $pattern = catfile(dirname($0), "Configurations", "*.conf");
194 foreach (sort glob($pattern) ) {
195 &read_config($_);
196 }
197
198
199 print "Configuring OpenSSL version $config{version} (0x$config{version_num})\n";
200
201 $config{prefix}="";
202 $config{openssldir}="";
203 $config{processor}="";
204 $config{libdir}="";
205 $config{cross_compile_prefix}="";
206 $config{fipslibdir}="/usr/local/ssl/fips-2.0/lib/";
207 my $nofipscanistercheck=0;
208 $config{baseaddr}="0xFB00000";
209 my $auto_threads=1; # enable threads automatically? true by default
210 my $default_ranlib;
211 $config{fips}=0;
212
213 # Top level directories to build
214 $config{dirs} = [ "crypto", "ssl", "engines", "apps", "test", "tools" ];
215 # crypto/ subdirectories to build
216 $config{sdirs} = [
217 "objects",
218 "md2", "md4", "md5", "sha", "mdc2", "hmac", "ripemd", "whrlpool", "poly1305", "blake2",
219 "des", "aes", "rc2", "rc4", "rc5", "idea", "bf", "cast", "camellia", "seed", "chacha", "modes",
220 "bn", "ec", "rsa", "dsa", "dh", "dso", "engine",
221 "buffer", "bio", "stack", "lhash", "rand", "err",
222 "evp", "asn1", "pem", "x509", "x509v3", "conf", "txt_db", "pkcs7", "pkcs12", "comp", "ocsp", "ui",
223 "cms", "ts", "srp", "cmac", "ct", "async", "kdf"
224 ];
225
226 # Known TLS and DTLS protocols
227 my @tls = qw(ssl3 tls1 tls1_1 tls1_2);
228 my @dtls = qw(dtls1 dtls1_2);
229
230 # Explicitelly known options that are possible to disable. They can
231 # be regexps, and will be used like this: /^no-${option}$/
232 # For developers: keep it sorted alphabetically
233
234 my @disablables = (
235 "aes",
236 "afalgeng",
237 "asm",
238 "async",
239 "autoalginit",
240 "autoerrinit",
241 "bf",
242 "blake2",
243 "camellia",
244 "capieng",
245 "cast",
246 "chacha",
247 "cmac",
248 "cms",
249 "comp",
250 "crypto-mdebug",
251 "crypto-mdebug-backtrace",
252 "ct",
253 "deprecated",
254 "des",
255 "dgram",
256 "dh",
257 "dsa",
258 "dso",
259 "dtls",
260 "dynamic-engine",
261 "ec",
262 "ec2m",
263 "ecdh",
264 "ecdsa",
265 "ec_nistp_64_gcc_128",
266 "egd",
267 "engine",
268 "err",
269 "filenames",
270 "heartbeats",
271 "hmac",
272 "hw(-.+)?",
273 "idea",
274 "makedepend",
275 "md2",
276 "md4",
277 "md5",
278 "mdc2",
279 "md[-_]ghost94",
280 "multiblock",
281 "nextprotoneg",
282 "ocb",
283 "ocsp",
284 "pic",
285 "poly1305",
286 "posix-io",
287 "psk",
288 "rc2",
289 "rc4",
290 "rc5",
291 "rdrand",
292 "rfc3779",
293 "rijndael", # Old AES name
294 "ripemd",
295 "rmd160",
296 "rsa",
297 "scrypt",
298 "sct",
299 "sctp",
300 "seed",
301 "sha",
302 "shared",
303 "sock",
304 "srp",
305 "srtp",
306 "sse2",
307 "ssl",
308 "ssl-trace",
309 "static-engine",
310 "stdio",
311 "threads",
312 "tls",
313 "ts",
314 "ui",
315 "unit-test",
316 "whirlpool",
317 "weak-ssl-ciphers",
318 "zlib",
319 "zlib-dynamic",
320 );
321 foreach my $proto ((@tls, @dtls))
322 {
323 push(@disablables, $proto);
324 push(@disablables, "$proto-method");
325 }
326
327 my @deprecated_disablables = (
328 "ssl2",
329 );
330
331 # All of the following is disabled by default (RC5 was enabled before 0.9.8):
332
333 our %disabled = ( # "what" => "comment"
334 "ec_nistp_64_gcc_128" => "default",
335 "egd" => "default",
336 "md2" => "default",
337 "rc5" => "default",
338 "sctp" => "default",
339 "shared" => "default",
340 "ssl-trace" => "default",
341 "ssl3" => "default",
342 "ssl3-method" => "default",
343 "unit-test" => "default",
344 "weak-ssl-ciphers" => "default",
345 "zlib" => "default",
346 "zlib-dynamic" => "default",
347 "crypto-mdebug" => "default",
348 "heartbeats" => "default",
349 );
350
351 # Note: => pair form used for aesthetics, not to truly make a hash table
352 my @disable_cascades = (
353 # "what" => [ "cascade", ... ]
354 sub { $config{processor} eq "386" }
355 => [ "sse2" ],
356 "ssl" => [ "ssl3" ],
357 "ssl3-method" => [ "ssl3" ],
358 "zlib" => [ "zlib-dynamic" ],
359 "rijndael" => [ "aes" ],
360 "des" => [ "mdc2" ],
361 "ec" => [ "ecdsa", "ecdh" ],
362
363 "dgram" => [ "dtls" ],
364 "dtls" => [ @dtls ],
365
366 # SSL 3.0, (D)TLS 1.0 and TLS 1.1 require MD5 and SHA
367 "md5" => [ "ssl", "tls1", "tls1_1", "dtls1" ],
368 "sha" => [ "ssl", "tls1", "tls1_1", "dtls1" ],
369
370 # Additionally, SSL 3.0 requires either RSA or DSA+DH
371 sub { $disabled{rsa}
372 && ($disabled{dsa} || $disabled{dh}); }
373 => [ "ssl" ],
374
375 # (D)TLS 1.0 and TLS 1.1 also require either RSA or DSA+DH
376 # or ECDSA + ECDH. (D)TLS 1.2 has this requirement as well.
377 # (XXX: We don't support PSK-only builds).
378 sub { $disabled{rsa}
379 && ($disabled{dsa} || $disabled{dh})
380 && ($disabled{ecdsa} || $disabled{ecdh}); }
381 => [ "tls1", "tls1_1", "tls1_2",
382 "dtls1", "dtls1_2" ],
383
384 "tls" => [ @tls ],
385
386 # SRP and HEARTBEATS require TLSEXT
387 "tlsext" => [ "srp", "heartbeats" ],
388
389 "crypto-mdebug" => [ "crypto-mdebug-backtrace" ],
390
391 # Without DSO, we can't load dynamic engines, so don't build them dynamic
392 "dso" => [ "dynamic-engine" ],
393
394 # Without position independent code, there can be no shared libraries or DSOs
395 "pic" => [ "shared" ],
396 "shared" => [ "dynamic-engine" ],
397 "engine" => [ "afalgeng" ],
398 );
399
400 # Avoid protocol support holes. Also disable all versions below N, if version
401 # N is disabled while N+1 is enabled.
402 #
403 my @list = (reverse @tls);
404 while ((my $first, my $second) = (shift @list, shift @list)) {
405 last unless @list;
406 push @disable_cascades, ( sub { !$disabled{$first} && $disabled{$second} }
407 => [ @list ] );
408 unshift @list, $second;
409 }
410 my @list = (reverse @dtls);
411 while ((my $first, my $second) = (shift @list, shift @list)) {
412 last unless @list;
413 push @disable_cascades, ( sub { !$disabled{$first} && $disabled{$second} }
414 => [ @list ] );
415 unshift @list, $second;
416 }
417
418 # Explicit "no-..." options will be collected in %disabled along with the defaults.
419 # To remove something from %disabled, use "enable-foo".
420 # For symmetry, "disable-foo" is a synonym for "no-foo".
421
422 my @generated_headers = (
423 "include/openssl/opensslconf.h",
424 "crypto/include/internal/bn_conf.h",
425 "crypto/include/internal/dso_conf.h"
426 );
427
428 my @generated_by_make_headers = (
429 "crypto/buildinf.h"
430 );
431
432
433 my $no_sse2=0;
434
435 &usage if ($#ARGV < 0);
436
437 my $user_cflags="";
438 my @user_defines=();
439 $config{openssl_api_defines}=[];
440 $config{openssl_algorithm_defines}=[];
441 $config{openssl_thread_defines}=[];
442 $config{openssl_sys_defines}=[];
443 $config{openssl_other_defines}=[];
444 my $libs="";
445 my $target="";
446 $config{options}="";
447 $config{build_type} = "release";
448
449 my $classic = 0;
450
451 my @argvcopy=@ARGV;
452
453 if (grep /^reconf(igure)?$/, @argvcopy) {
454 if (-f "./configdata.pm") {
455 my $file = "./configdata.pm";
456 unless (my $return = do $file) {
457 die "couldn't parse $file: $@" if $@;
458 die "couldn't do $file: $!" unless defined $return;
459 die "couldn't run $file" unless $return;
460 }
461
462 @argvcopy = defined($configdata::config{perlargv}) ?
463 @{$configdata::config{perlargv}} : ();
464 die "Incorrect data to reconfigure, please do a normal configuration\n"
465 if (grep(/^reconf/,@argvcopy));
466 $ENV{CROSS_COMPILE} = $configdata::config{cross_compile_prefix}
467 if defined($configdata::config{cross_compile_prefix});
468 $ENV{CROSS_COMPILE} = $configdata::config{cc}
469 if defined($configdata::config{cc});
470
471 print "Reconfiguring with: ", join(" ",@argvcopy), "\n";
472 print " CROSS_COMPILE = ",$ENV{CROSS_COMPILE},"\n"
473 if $ENV{CROSS_COMPILE};
474 print " CC = ",$ENV{CC},"\n" if $ENV{CC};
475 } elsif (open IN, "<Makefile") {
476 #
477 # THIS SECTION IS TEMPORARY, it helps transitioning from Makefile
478 # centered information gathering the reading configdata.pm
479 #
480 while (<IN>) {
481 s|\R$||;
482 if (/^CONFIGURE_ARGS=\s*(.*)\s*/) {
483 # Older form, we split the string and hope for the best
484 @argvcopy = split /\s+/, $_;
485 die "Incorrect data to reconfigure, please do a normal configuration\n"
486 if (grep(/^reconf/,@argvcopy));
487 } elsif (/^CROSS_COMPILE=\s*(.*)/) {
488 $ENV{CROSS_COMPILE}=$1;
489 } elsif (/^CC=\s*(?:\$\(CROSS_COMPILE\))?(.*?)$/) {
490 $ENV{CC}=$1;
491 }
492 }
493 #
494 # END OF TEMPORARY SECTION
495 #
496 } else {
497 die "Insufficient data to reconfigure, please do a normal configuration\n";
498 }
499 }
500
501 $config{perlargv} = [ @argvcopy ];
502
503 my %unsupported_options = ();
504 my %deprecated_options = ();
505 foreach (@argvcopy)
506 {
507 # VMS is a case insensitive environment, and depending on settings
508 # out of our control, we may receive options uppercased. Let's
509 # downcase at least the part before any equal sign.
510 if ($^O eq "VMS")
511 {
512 s/^([^=]*)/lc($1)/e;
513 }
514 s /^-no-/no-/; # some people just can't read the instructions
515
516 # rewrite some options in "enable-..." form
517 s /^-?-?shared$/enable-shared/;
518 s /^sctp$/enable-sctp/;
519 s /^threads$/enable-threads/;
520 s /^zlib$/enable-zlib/;
521 s /^zlib-dynamic$/enable-zlib-dynamic/;
522
523 if (/^(no|disable|enable)-(.+)$/)
524 {
525 my $word = $2;
526 if (grep { $word =~ /^${_}$/ } @deprecated_disablables)
527 {
528 $deprecated_options{$_} = 1;
529 next;
530 }
531 elsif (!grep { $word =~ /^${_}$/ } @disablables)
532 {
533 $unsupported_options{$_} = 1;
534 next;
535 }
536 }
537 if (/^no-(.+)$/ || /^disable-(.+)$/)
538 {
539 foreach my $proto ((@tls, @dtls))
540 {
541 if ($1 eq "$proto-method")
542 {
543 $disabled{"$proto"} = "option($proto-method)";
544 last;
545 }
546 }
547 if ($1 eq "dtls")
548 {
549 foreach my $proto (@dtls)
550 {
551 $disabled{$proto} = "option(dtls)";
552 }
553 }
554 elsif ($1 eq "ssl")
555 {
556 # Last one of its kind
557 $disabled{"ssl3"} = "option(ssl)";
558 }
559 elsif ($1 eq "tls")
560 {
561 # XXX: Tests will fail if all SSL/TLS
562 # protocols are disabled.
563 foreach my $proto (@tls)
564 {
565 $disabled{$proto} = "option(tls)";
566 }
567 }
568 elsif ($1 eq "static-engine")
569 {
570 delete $disabled{"dynamic-engine"};
571 }
572 elsif ($1 eq "dynamic-engine")
573 {
574 $disabled{"dynamic-engine"} = "option";
575 }
576 else
577 {
578 $disabled{$1} = "option";
579 }
580 # No longer an automatic choice
581 $auto_threads = 0 if ($1 eq "threads");
582 }
583 elsif (/^enable-(.+)$/)
584 {
585 if ($1 eq "static-engine")
586 {
587 $disabled{"dynamic-engine"} = "option";
588 }
589 elsif ($1 eq "dynamic-engine")
590 {
591 delete $disabled{"dynamic-engine"};
592 }
593 elsif ($1 eq "zlib-dynamic")
594 {
595 delete $disabled{"zlib"};
596 }
597 my $algo = $1;
598 delete $disabled{$algo};
599
600 # No longer an automatic choice
601 $auto_threads = 0 if ($1 eq "threads");
602 }
603 elsif (/^--strict-warnings$/)
604 {
605 $strict_warnings = 1;
606 }
607 elsif (/^--debug$/)
608 {
609 $config{build_type} = "debug";
610 }
611 elsif (/^--release$/)
612 {
613 $config{build_type} = "release";
614 }
615 elsif (/^386$/)
616 { $config{processor}=386; }
617 elsif (/^fips$/)
618 {
619 $config{fips}=1;
620 }
621 elsif (/^rsaref$/)
622 {
623 # No RSAref support any more since it's not needed.
624 # The check for the option is there so scripts aren't
625 # broken
626 }
627 elsif (/^nofipscanistercheck$/)
628 {
629 $config{fips} = 1;
630 $nofipscanistercheck = 1;
631 }
632 elsif (/^[-+]/)
633 {
634 if (/^--classic$/)
635 {
636 $classic=1;
637 }
638 elsif (/^--prefix=(.*)$/)
639 {
640 $config{prefix}=$1;
641 die "Directory given with --prefix MUST be absolute\n"
642 unless file_name_is_absolute($config{prefix});
643 }
644 elsif (/^--api=(.*)$/)
645 {
646 $config{api}=$1;
647 }
648 elsif (/^--libdir=(.*)$/)
649 {
650 $config{libdir}=$1;
651 }
652 elsif (/^--openssldir=(.*)$/)
653 {
654 $config{openssldir}=$1;
655 }
656 elsif (/^--with-zlib-lib=(.*)$/)
657 {
658 $withargs{zlib_lib}=$1;
659 }
660 elsif (/^--with-zlib-include=(.*)$/)
661 {
662 $withargs{zlib_include}="-I$1";
663 }
664 elsif (/^--with-fipslibdir=(.*)$/)
665 {
666 $config{fipslibdir}="$1/";
667 }
668 elsif (/^--with-baseaddr=(.*)$/)
669 {
670 $config{baseaddr}="$1";
671 }
672 elsif (/^--cross-compile-prefix=(.*)$/)
673 {
674 $config{cross_compile_prefix}=$1;
675 }
676 elsif (/^--config=(.*)$/)
677 {
678 read_config $1;
679 }
680 elsif (/^-[lL](.*)$/ or /^-Wl,/)
681 {
682 $libs.=$_." ";
683 }
684 elsif (/^-D(.*)$/)
685 {
686 push @user_defines, $1;
687 }
688 else # common if (/^[-+]/), just pass down...
689 {
690 $_ =~ s/%([0-9a-f]{1,2})/chr(hex($1))/gei;
691 $user_cflags.=" ".$_;
692 }
693 }
694 else
695 {
696 die "target already defined - $target (offending arg: $_)\n" if ($target ne "");
697 $target=$_;
698 }
699 unless ($_ eq $target || /^no-/ || /^disable-/)
700 {
701 # "no-..." follows later after implied disactivations
702 # have been derived. (Don't take this too seroiusly,
703 # we really only write OPTIONS to the Makefile out of
704 # nostalgia.)
705
706 if ($config{options} eq "")
707 { $config{options} = $_; }
708 else
709 { $config{options} .= " ".$_; }
710 }
711
712 if (defined($config{api}) && !exists $apitable->{$config{api}}) {
713 die "***** Unsupported api compatibility level: $config{api}\n",
714 }
715
716 if (keys %deprecated_options)
717 {
718 warn "***** Deprecated options: ",
719 join(", ", keys %deprecated_options), "\n";
720 }
721 if (keys %unsupported_options)
722 {
723 die "***** Unsupported options: ",
724 join(", ", keys %unsupported_options), "\n";
725 }
726 }
727
728 if ($config{fips})
729 {
730 delete $disabled{"shared"} if ($disabled{"shared"} =~ /^default/);
731 }
732 else
733 {
734 @{$config{dirs}} = grep !/^fips$/, @{$config{dirs}};
735 }
736
737 my @tocheckfor = (keys %disabled);
738 while (@tocheckfor) {
739 my %new_tocheckfor = ();
740 my @cascade_copy = (@disable_cascades);
741 while (@cascade_copy) {
742 my ($test, $descendents) = (shift @cascade_copy, shift @cascade_copy);
743 if (ref($test) eq "CODE" ? $test->() : defined($disabled{$test})) {
744 map {
745 $new_tocheckfor{$_} = 1; $disabled{$_} = "forced";
746 } grep { !defined($disabled{$_}) } @$descendents;
747 }
748 }
749 @tocheckfor = (keys %new_tocheckfor);
750 }
751
752 if ($target eq "TABLE") {
753 foreach (sort keys %table) {
754 print_table_entry($_, "TABLE");
755 }
756 exit 0;
757 }
758
759 if ($target eq "LIST") {
760 foreach (sort keys %table) {
761 print $_,"\n" unless $table{$_}->{template};
762 }
763 exit 0;
764 }
765
766 if ($target eq "HASH") {
767 print "%table = (\n";
768 foreach (sort keys %table) {
769 print_table_entry($_, "HASH");
770 }
771 exit 0;
772 }
773
774 # Backward compatibility?
775 if ($target =~ m/^CygWin32(-.*)$/) {
776 $target = "Cygwin".$1;
777 }
778
779 foreach (sort (keys %disabled))
780 {
781 $config{options} .= " no-$_";
782
783 printf " no-%-12s %-10s", $_, "[$disabled{$_}]";
784
785 if (/^dso$/)
786 { }
787 elsif (/^threads$/)
788 { }
789 elsif (/^shared$/)
790 { }
791 elsif (/^pic$/)
792 { }
793 elsif (/^zlib$/)
794 { }
795 elsif (/^dynamic-engine$/)
796 { }
797 elsif (/^makedepend$/)
798 { }
799 elsif (/^zlib-dynamic$/)
800 { }
801 elsif (/^sse2$/)
802 { $no_sse2 = 1; }
803 elsif (/^engine$/)
804 {
805 @{$config{dirs}} = grep !/^engines$/, @{$config{dirs}};
806 @{$config{sdirs}} = grep !/^engine$/, @{$config{sdirs}};
807 push @{$config{openssl_other_defines}}, "OPENSSL_NO_ENGINE";
808 }
809 else
810 {
811 my ($ALGO, $algo);
812 ($ALGO = $algo = $_) =~ tr/[\-a-z]/[_A-Z]/;
813
814 if (/^asm$/ || /^err$/ || /^hw$/ || /^hw-/ || /^async$/
815 || /^autoalginit/ || /^autoerrinit/)
816 {
817 push @{$config{openssl_other_defines}}, "OPENSSL_NO_$ALGO";
818 print " OPENSSL_NO_$ALGO";
819
820 if (/^err$/) { push @user_defines, "OPENSSL_NO_ERR"; }
821 }
822 else
823 {
824 ($ALGO,$algo) = ("RMD160","rmd160") if ($algo eq "ripemd");
825
826 push @{$config{openssl_algorithm_defines}}, "OPENSSL_NO_$ALGO";
827 print " OPENSSL_NO_$ALGO";
828
829 # fix-up crypto/directory name(s)
830 $algo="whrlpool" if $algo eq "whirlpool";
831 $algo="ripemd" if $algo eq "rmd160";
832 @{$config{sdirs}} = grep { $_ ne $algo} @{$config{sdirs}};
833
834 print " (skip dir)";
835 }
836 }
837
838 print "\n";
839 }
840
841 print "Configuring for $target\n";
842
843 # Support for legacy targets having a name starting with 'debug-'
844 my ($d, $t) = $target =~ m/^(debug-)?(.*)$/;
845 if ($d) {
846 $config{build_type} = "debug";
847
848 # If we do not find debug-foo in the table, the target is set to foo.
849 if (!$table{$target}) {
850 $target = $t;
851 }
852 }
853 $config{target} = $target;
854 my %target = resolve_config($target);
855
856 &usage if (!%target || $target{template});
857
858 %target = ( %{$table{DEFAULTS}}, %target );
859
860 $target{exe_extension}="";
861 $target{exe_extension}=".exe" if ($config{target} eq "DJGPP"
862 || $config{target} =~ /^(?:Cygwin|mingw)/);
863 $target{exe_extension}=".pm" if ($config{target} =~ /vos/);
864
865 ($target{shared_extension_simple}=$target{shared_extension})
866 =~ s|\.\$\(SHLIB_MAJOR\)\.\$\(SHLIB_MINOR\)||;
867 $target{dso_extension}=$target{shared_extension_simple};
868 ($target{shared_import_extension}=$target{shared_extension_simple}.".a")
869 if ($config{target} =~ /^(?:Cygwin|mingw)/);
870
871
872 $config{cross_compile_prefix} = $ENV{'CROSS_COMPILE'}
873 if $config{cross_compile_prefix} eq "";
874
875 # Allow overriding the names of some tools. USE WITH CARE
876 $config{perl} = $ENV{'PERL'} || which("perl5") || which("perl") || "perl";
877 $target{cc} = $ENV{'CC'} || $target{cc} || "cc";
878 $target{ranlib} = $ENV{'RANLIB'} || $target{ranlib} || which("ranlib") || "true";
879 $target{ar} = $ENV{'AR'} || $target{ar} || "ar";
880 $target{nm} = $ENV{'NM'} || $target{nm} || "nm";
881
882 # For cflags, lflags, plib_lflags, ex_libs and defines, add the debug_
883 # or release_ attributes.
884 # Do it in such a way that no spurious space is appended (hence the grep).
885 $config{defines} = [];
886 $config{cflags} = "";
887 $config{ex_libs} = "";
888 $config{shared_ldflag} = "";
889
890 # Make sure build_scheme is consistent.
891 $target{build_scheme} = [ $target{build_scheme} ]
892 if ref($target{build_scheme}) ne "ARRAY";
893
894 ###### TO BE REMOVED WHEN CLASSIC BUILD IS REMOVED
895 ######
896 ###### If the user has chosen --classic, we give it to them.
897 ###### If they try that with an out-of-source config, we complain.
898 if ($target{build_scheme}->[0] eq "unified" && $classic) {
899 die "Can't perform a classic build out of source tree\n"
900 if $srcdir ne $blddir;
901
902 $target{build_scheme} = { unix => [ "unixmake" ],
903 windows => undef,
904 VMS => undef } -> {$target{build_scheme}->[1]};
905
906 die "Classic mode unavailable on this platform\n"
907 unless defined($target{build_scheme});
908 }
909
910 my ($builder, $builder_platform, @builder_opts) =
911 @{$target{build_scheme}};
912
913 if ($target =~ /^mingw/ && `$target{cc} --target-help 2>&1` =~ m/-mno-cygwin/m)
914 {
915 $config{cflags} .= " -mno-cygwin";
916 $config{shared_ldflag} .= " -mno-cygwin";
917 }
918
919 if ($target =~ /linux.*-mips/ && !$disabled{asm} && $user_cflags !~ /-m(ips|arch=)/) {
920 # minimally required architecture flags for assembly modules
921 $config{cflags}="-mips2 $config{cflags}" if ($target =~ /mips32/);
922 $config{cflags}="-mips3 $config{cflags}" if ($target =~ /mips64/);
923 }
924
925 my $no_shared_warn=0;
926 my $no_user_cflags=0;
927 my $no_user_defines=0;
928
929 # The DSO code currently always implements all functions so that no
930 # applications will have to worry about that from a compilation point
931 # of view. However, the "method"s may return zero unless that platform
932 # has support compiled in for them. Currently each method is enabled
933 # by a define "DSO_<name>" ... we translate the "dso_scheme" config
934 # string entry into using the following logic;
935 if (!$disabled{dso} && $target{dso_scheme} ne "")
936 {
937 $target{dso_scheme} =~ tr/[a-z]/[A-Z]/;
938 if ($target{dso_scheme} eq "DLFCN")
939 {
940 unshift @{$config{defines}}, "DSO_DLFCN", "HAVE_DLFCN_H";
941 }
942 elsif ($target{dso_scheme} eq "DLFCN_NO_H")
943 {
944 unshift @{$config{defines}}, "DSO_DLFCN";
945 }
946 else
947 {
948 unshift @{$config{defines}}, "DSO_$target{dso_scheme}";
949 }
950 }
951
952 $config{ex_libs}="$libs$config{ex_libs}" if ($libs ne "");
953
954 if ($disabled{asm})
955 {
956 if ($config{fips})
957 {
958 @{$config{defines}} = grep !/^[BL]_ENDIAN$/, @{$config{defines}};
959 @{$target{defines}} = grep !/^[BL]_ENDIAN$/, @{$target{defines}};
960 }
961 }
962
963 # If threads aren't disabled, check how possible they are
964 unless ($disabled{threads}) {
965 if ($auto_threads) {
966 # Enabled by default, disable it forcibly if unavailable
967 if ($target{thread_scheme} eq "(unknown)") {
968 $disabled{threads} = "unavailable";
969 }
970 } else {
971 # The user chose to enable threads explicitely, let's see
972 # if there's a chance that's possible
973 if ($target{thread_scheme} eq "(unknown)") {
974 # If the user asked for "threads" and we don't have internal
975 # knowledge how to do it, [s]he is expected to provide any
976 # system-dependent compiler options that are necessary. We
977 # can't truly check that the given options are correct, but
978 # we expect the user to know what [s]He is doing.
979 if ($no_user_cflags && $no_user_defines) {
980 die "You asked for multi-threading support, but didn't\n"
981 ,"provide any system-specific compiler options\n";
982 }
983 }
984 }
985 }
986
987 # If threads still aren't disabled, add a C macro to ensure the source
988 # code knows about it. Any other flag is taken care of by the configs.
989 unless($disabled{threads}) {
990 foreach (("defines", "openssl_thread_defines")) {
991 push @{$config{$_}}, "OPENSSL_THREADS";
992 }
993 }
994
995 # With "deprecated" disable all deprecated features.
996 if (defined($disabled{"deprecated"})) {
997 $config{api} = $maxapi;
998 }
999
1000 if ($target{shared_target} eq "")
1001 {
1002 $no_shared_warn = 1
1003 if ((!$disabled{shared} || !$disabled{"dynamic-engine"})
1004 && !$config{fips});
1005 $disabled{shared} = "no-shared-target";
1006 $disabled{pic} = $disabled{shared} = $disabled{"dynamic-engine"} =
1007 "no-shared-target";
1008 }
1009
1010 if ($disabled{"dynamic-engine"}) {
1011 push @{$config{defines}}, "OPENSSL_NO_DYNAMIC_ENGINE";
1012 $config{dynamic_engines} = 0;
1013 } else {
1014 push @{$config{defines}}, "OPENSSL_NO_STATIC_ENGINE";
1015 $config{dynamic_engines} = 1;
1016 }
1017
1018 #
1019 # Platform fix-ups
1020 #
1021
1022 # This saves the build files from having to check
1023 if ($disabled{pic})
1024 {
1025 $target{shared_cflag} = $target{shared_ldflag} =
1026 $target{shared_rcflag} = "";
1027 }
1028 else
1029 {
1030 push @{$config{defines}}, "OPENSSL_PIC";
1031 }
1032
1033 if ($target{sys_id} ne "")
1034 {
1035 push @{$config{openssl_sys_defines}}, "OPENSSL_SYS_$target{sys_id}";
1036 }
1037
1038 unless ($disabled{asm}) {
1039 $target{cpuid_asm_src}=$table{DEFAULTS}->{cpuid_asm_src} if ($config{processor} eq "386");
1040 $target{bn_asm_src} =~ s/\w+-gf2m.c// if (defined($disabled{ec2m}));
1041
1042 # bn-586 is the only one implementing bn_*_part_words
1043 push @{$config{defines}}, "OPENSSL_BN_ASM_PART_WORDS" if ($target{bn_asm_src} =~ /bn-586/);
1044 push @{$config{defines}}, "OPENSSL_IA32_SSE2" if (!$no_sse2 && $target{bn_asm_src} =~ /86/);
1045
1046 push @{$config{defines}}, "OPENSSL_BN_ASM_MONT" if ($target{bn_asm_src} =~ /-mont/);
1047 push @{$config{defines}}, "OPENSSL_BN_ASM_MONT5" if ($target{bn_asm_src} =~ /-mont5/);
1048 push @{$config{defines}}, "OPENSSL_BN_ASM_GF2m" if ($target{bn_asm_src} =~ /-gf2m/);
1049
1050 if ($config{fips}) {
1051 push @{$config{openssl_other_defines}}, "OPENSSL_FIPS";
1052 }
1053
1054 if ($target{sha1_asm_src}) {
1055 push @{$config{defines}}, "SHA1_ASM" if ($target{sha1_asm_src} =~ /sx86/ || $target{sha1_asm_src} =~ /sha1/);
1056 push @{$config{defines}}, "SHA256_ASM" if ($target{sha1_asm_src} =~ /sha256/);
1057 push @{$config{defines}}, "SHA512_ASM" if ($target{sha1_asm_src} =~ /sha512/);
1058 }
1059 if ($target{md5_asm_src}) {
1060 push @{$config{defines}}, "MD5_ASM";
1061 }
1062 $target{cast_asm_src}=$table{DEFAULTS}->{cast_asm_src} unless $disabled{pic}; # CAST assembler is not PIC
1063 if ($target{rmd160_asm_src}) {
1064 push @{$config{defines}}, "RMD160_ASM";
1065 }
1066 if ($target{aes_asm_src}) {
1067 push @{$config{defines}}, "AES_ASM" if ($target{aes_asm_src} =~ m/\baes-/);;
1068 # aes-ctr.fake is not a real file, only indication that assembler
1069 # module implements AES_ctr32_encrypt...
1070 push @{$config{defines}}, "AES_CTR_ASM" if ($target{aes_asm_src} =~ s/\s*aes-ctr\.fake//);
1071 # aes-xts.fake indicates presence of AES_xts_[en|de]crypt...
1072 push @{$config{defines}}, "AES_XTS_ASM" if ($target{aes_asm_src} =~ s/\s*aes-xts\.fake//);
1073 $target{aes_asm_src} =~ s/\s*(vpaes|aesni)-x86\.s//g if ($no_sse2);
1074 push @{$config{defines}}, "VPAES_ASM" if ($target{aes_asm_src} =~ m/vpaes/);
1075 push @{$config{defines}}, "BSAES_ASM" if ($target{aes_asm_src} =~ m/bsaes/);
1076 }
1077 if ($target{wp_asm_src} =~ /mmx/) {
1078 if ($config{processor} eq "386") {
1079 $target{wp_asm_src}=$table{DEFAULTS}->{wp_asm_src};
1080 } elsif (!$disabled{"whirlpool"}) {
1081 push @{$config{defines}}, "WHIRLPOOL_ASM";
1082 }
1083 }
1084 if ($target{modes_asm_src} =~ /ghash-/) {
1085 push @{$config{defines}}, "GHASH_ASM";
1086 }
1087 if ($target{ec_asm_src} =~ /ecp_nistz256/) {
1088 push @{$config{defines}}, "ECP_NISTZ256_ASM";
1089 }
1090 if ($target{poly1305_asm_src} ne "") {
1091 push @{$config{defines}}, "POLY1305_ASM";
1092 }
1093 }
1094
1095 my $ecc = $target{cc};
1096 if ($^O ne "VMS" && !$disabled{makedepend}) {
1097 # Is the compiler gcc or clang? $ecc is used below to see if
1098 # error-checking can be turned on.
1099 my $ccpcc = "$config{cross_compile_prefix}$target{cc}";
1100 open(PIPE, "$ccpcc --version 2>&1 |");
1101 my $lines = 2;
1102 while ( <PIPE> ) {
1103 # Find the version number and save the major.
1104 m|(?:.*)\b(\d+)\.\d+\.\d+\b(?:.*)|;
1105 my $compiler_major = $1;
1106 # We know that GNU C version 3 and up as well as all clang
1107 # versions support dependency generation
1108 $config{makedepprog} = $ccpcc
1109 if (/clang/ || (/gcc/ && $compiler_major > 3));
1110 $ecc = "clang" if /clang/;
1111 $ecc = "gcc" if /gcc/;
1112 last if ($config{makedepprog} || !$lines--);
1113 }
1114 close(PIPE);
1115
1116 $config{makedepprog} = which('makedepend') unless $config{makedepprog};
1117 $disabled{makedepend} = "unavailable" unless $config{makedepprog};
1118 }
1119
1120
1121
1122 # Deal with bn_ops ###################################################
1123
1124 $config{bn_ll} =0;
1125 $config{export_var_as_fn} =0;
1126 my $def_int="unsigned int";
1127 $config{rc4_int} =$def_int;
1128 ($config{b64l},$config{b64},$config{b32})=(0,0,1);
1129
1130 my $count = 0;
1131 foreach (sort split(/\s+/,$target{bn_ops})) {
1132 $count++ if /SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT/;
1133 $config{export_var_as_fn}=1 if $_ eq 'EXPORT_VAR_AS_FN';
1134 $config{bn_ll}=1 if $_ eq 'BN_LLONG';
1135 $config{rc4_int}="unsigned char" if $_ eq 'RC4_CHAR';
1136 ($config{b64l},$config{b64},$config{b32})
1137 =(0,1,0) if $_ eq 'SIXTY_FOUR_BIT';
1138 ($config{b64l},$config{b64},$config{b32})
1139 =(1,0,0) if $_ eq 'SIXTY_FOUR_BIT_LONG';
1140 ($config{b64l},$config{b64},$config{b32})
1141 =(0,0,1) if $_ eq 'THIRTY_TWO_BIT';
1142 }
1143 die "Exactly one of SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT can be set in bn_ops\n"
1144 if $count > 1;
1145
1146
1147 # Hack cflags for better warnings (dev option) #######################
1148
1149 # "Stringify" the C flags string. This permits it to be made part of a string
1150 # and works as well on command lines.
1151 $config{cflags} =~ s/([\\\"])/\\$1/g;
1152
1153 if (defined($config{api})) {
1154 $config{openssl_api_defines} = [ "OPENSSL_MIN_API=".$apitable->{$config{api}} ];
1155 my $apiflag = sprintf("OPENSSL_API_COMPAT=%s", $apitable->{$config{api}});
1156 push @{$config{defines}}, $apiflag;
1157 }
1158
1159 if ($strict_warnings)
1160 {
1161 my $wopt;
1162 die "ERROR --strict-warnings requires gcc or clang"
1163 unless $ecc eq 'gcc' || $ecc eq 'clang';
1164 foreach $wopt (split /\s+/, $gcc_devteam_warn)
1165 {
1166 $config{cflags} .= " $wopt" unless ($config{cflags} =~ /(?:^|\s)$wopt(?:\s|$)/)
1167 }
1168 if ($ecc eq "clang")
1169 {
1170 foreach $wopt (split /\s+/, $clang_devteam_warn)
1171 {
1172 $config{cflags} .= " $wopt" unless ($config{cflags} =~ /(?:^|\s)$wopt(?:\s|$)/)
1173 }
1174 }
1175 }
1176
1177 unless ($disabled{"crypto-mdebug-backtrace"})
1178 {
1179 foreach my $wopt (split /\s+/, $memleak_devteam_backtrace)
1180 {
1181 $config{cflags} .= " $wopt" unless ($config{cflags} =~ /(?:^|\s)$wopt(?:\s|$)/)
1182 }
1183 if ($target =~ /^BSD-/)
1184 {
1185 $config{ex_libs} .= " -lexecinfo";
1186 }
1187 }
1188
1189 if ($user_cflags ne "") { $config{cflags}="$config{cflags}$user_cflags"; }
1190 else { $no_user_cflags=1; }
1191 if (@user_defines) { $config{defines}=[ @{$config{defines}}, @user_defines ]; }
1192 else { $no_user_defines=1; }
1193
1194 # ALL MODIFICATIONS TO %config and %target MUST BE DONE FROM HERE ON
1195
1196 unless ($disabled{afalgeng}) {
1197 $config{afalgeng}="";
1198 if ($target =~ m/^linux/) {
1199 my $minver = 4*10000 + 1*100 + 0;
1200 if ($config{cross_compile_prefix} eq "") {
1201 my $verstr = `uname -r`;
1202 my ($ma, $mi1, $mi2) = split("\\.", $verstr);
1203 ($mi2) = $mi2 =~ /(\d+)/;
1204 my $ver = $ma*10000 + $mi1*100 + $mi2;
1205 if ($ver < $minver) {
1206 $disabled{afalgeng} = "too-old-kernel";
1207 } else {
1208 push @{$config{engdirs}}, "afalg";
1209 }
1210 } else {
1211 $disabled{afalgeng} = "cross-compiling";
1212 }
1213 } else {
1214 $disabled{afalgeng} = "not-linux";
1215 }
1216 }
1217
1218 push @{$config{openssl_other_defines}}, "OPENSSL_NO_AFALGENG" if ($disabled{afalgeng});
1219
1220 # If we use the unified build, collect information from build.info files
1221 my %unified_info = ();
1222
1223 my $buildinfo_debug = defined($ENV{CONFIGURE_DEBUG_BUILDINFO});
1224 if ($builder eq "unified") {
1225 # Store the name of the template file we will build the build file from
1226 # in %config. This may be useful for the build file itself.
1227 my $build_file_template =
1228 catfile($srcdir, "Configurations",
1229 $builder_platform."-".$target{build_file}.".tmpl");
1230 $build_file_template =
1231 catfile($srcdir, "Configurations", $target{build_file}.".tmpl")
1232 if (! -f $build_file_template);
1233 $config{build_file_template} = $build_file_template;
1234
1235 use lib catdir(dirname(__FILE__),"util");
1236 use with_fallback qw(Text::Template);
1237
1238 sub cleandir {
1239 my $base = shift;
1240 my $dir = shift;
1241 my $relativeto = shift || ".";
1242
1243 $dir = catdir($base,$dir) unless isabsolute($dir);
1244
1245 # Make sure the directories we're building in exists
1246 mkpath($dir);
1247
1248 my $res = abs2rel(absolutedir($dir), rel2abs($relativeto));
1249 #print STDERR "DEBUG[cleandir]: $dir , $base => $res\n";
1250 return $res;
1251 }
1252
1253 sub cleanfile {
1254 my $base = shift;
1255 my $file = shift;
1256 my $relativeto = shift || ".";
1257
1258 $file = catfile($base,$file) unless isabsolute($file);
1259
1260 my $d = dirname($file);
1261 my $f = basename($file);
1262
1263 # Make sure the directories we're building in exists
1264 mkpath($d);
1265
1266 my $res = abs2rel(catfile(absolutedir($d), $f), rel2abs($relativeto));
1267 #print STDERR "DEBUG[cleanfile]: $d , $f => $res\n";
1268 return $res;
1269 }
1270
1271 my @build_infos = ( [ ".", "build.info" ] );
1272 foreach (@{$config{dirs}}) {
1273 push @build_infos, [ $_, "build.info" ]
1274 if (-f catfile($srcdir, $_, "build.info"));
1275 }
1276 foreach (@{$config{sdirs}}) {
1277 push @build_infos, [ catdir("crypto", $_), "build.info" ]
1278 if (-f catfile($srcdir, "crypto", $_, "build.info"));
1279 }
1280 foreach (@{$config{engdirs}}) {
1281 push @build_infos, [ catdir("engines", $_), "build.info" ]
1282 if (-f catfile($srcdir, "engines", $_, "build.info"));
1283 }
1284
1285 $config{build_infos} = [ ];
1286
1287 foreach (@build_infos) {
1288 my $sourced = catdir($srcdir, $_->[0]);
1289 my $buildd = catdir($blddir, $_->[0]);
1290
1291 mkpath($buildd);
1292
1293 my $f = $_->[1];
1294 # The basic things we're trying to build
1295 my @programs = ();
1296 my @libraries = ();
1297 my @engines = ();
1298 my @scripts = ();
1299 my @extra = ();
1300 my @overrides = ();
1301 my @intermediates = ();
1302 my @rawlines = ();
1303
1304 my %ordinals = ();
1305 my %sources = ();
1306 my %includes = ();
1307 my %depends = ();
1308 my %renames = ();
1309 my %sharednames = ();
1310 my %generate = ();
1311
1312 push @{$config{build_infos}}, catfile(abs2rel($sourced, $blddir), $f);
1313 my $template = Text::Template->new(TYPE => 'FILE',
1314 SOURCE => catfile($sourced, $f));
1315 die "Something went wrong with $sourced/$f: $!\n" unless $template;
1316 my @text =
1317 split /^/m,
1318 $template->fill_in(HASH => { config => \%config,
1319 target => \%target,
1320 disabled => \%disabled,
1321 builddir => abs2rel($buildd, $blddir),
1322 sourcedir => abs2rel($sourced, $blddir),
1323 buildtop => abs2rel($blddir, $blddir),
1324 sourcetop => abs2rel($srcdir, $blddir) },
1325 DELIMITERS => [ "{-", "-}" ]);
1326
1327 # The top item of this stack has the following values
1328 # -2 positive already run and we found ELSE (following ELSIF should fail)
1329 # -1 positive already run (skip until ENDIF)
1330 # 0 negatives so far (if we're at a condition, check it)
1331 # 1 last was positive (don't skip lines until next ELSE, ELSIF or ENDIF)
1332 # 2 positive ELSE (following ELSIF should fail)
1333 my @skip = ();
1334 collect_information(
1335 collect_from_array([ @text ],
1336 qr/\\$/ => sub { my $l1 = shift; my $l2 = shift;
1337 $l1 =~ s/\\$//; $l1.$l2 }),
1338 # Info we're looking for
1339 qr/^\s*IF\[((?:\\.|[^\\\]])*)\]\s*$/
1340 => sub {
1341 if (! @skip || $skip[$#skip] > 0) {
1342 push @skip, !! $1;
1343 } else {
1344 push @skip, -1;
1345 }
1346 },
1347 qr/^\s*ELSIF\[((?:\\.|[^\\\]])*)\]\s*$/
1348 => sub { die "ELSIF out of scope" if ! @skip;
1349 die "ELSIF following ELSE" if abs($skip[$#skip]) == 2;
1350 $skip[$#skip] = -1 if $skip[$#skip] != 0;
1351 $skip[$#skip] = !! $1
1352 if $skip[$#skip] == 0; },
1353 qr/^\s*ELSE\s*$/
1354 => sub { die "ELSE out of scope" if ! @skip;
1355 $skip[$#skip] = -2 if $skip[$#skip] != 0;
1356 $skip[$#skip] = 2 if $skip[$#skip] == 0; },
1357 qr/^\s*ENDIF\s*$/
1358 => sub { die "ENDIF out of scope" if ! @skip;
1359 pop @skip; },
1360 qr/^\s*PROGRAMS\s*=\s*(.*)\s*$/
1361 => sub { push @programs, split(/\s+/, $1)
1362 if !@skip || $skip[$#skip] > 0 },
1363 qr/^\s*LIBS\s*=\s*(.*)\s*$/
1364 => sub { push @libraries, split(/\s+/, $1)
1365 if !@skip || $skip[$#skip] > 0 },
1366 qr/^\s*ENGINES\s*=\s*(.*)\s*$/
1367 => sub { push @engines, split(/\s+/, $1)
1368 if !@skip || $skip[$#skip] > 0 },
1369 qr/^\s*SCRIPTS\s*=\s*(.*)\s*$/
1370 => sub { push @scripts, split(/\s+/, $1)
1371 if !@skip || $skip[$#skip] > 0 },
1372 qr/^\s*EXTRA\s*=\s*(.*)\s*$/
1373 => sub { push @extra, split(/\s+/, $1)
1374 if !@skip || $skip[$#skip] > 0 },
1375 qr/^\s*OVERRIDES\s*=\s*(.*)\s*$/
1376 => sub { push @overrides, split(/\s+/, $1)
1377 if !@skip || $skip[$#skip] > 0 },
1378
1379 qr/^\s*ORDINALS\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/,
1380 => sub { push @{$ordinals{$1}}, split(/\s+/, $2)
1381 if !@skip || $skip[$#skip] > 0 },
1382 qr/^\s*SOURCE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1383 => sub { push @{$sources{$1}}, split(/\s+/, $2)
1384 if !@skip || $skip[$#skip] > 0 },
1385 qr/^\s*INCLUDE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1386 => sub { push @{$includes{$1}}, split(/\s+/, $2)
1387 if !@skip || $skip[$#skip] > 0 },
1388 qr/^\s*DEPEND\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1389 => sub { push @{$depends{$1}}, split(/\s+/, $2)
1390 if !@skip || $skip[$#skip] > 0 },
1391 qr/^\s*GENERATE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1392 => sub { push @{$generate{$1}}, $2
1393 if !@skip || $skip[$#skip] > 0 },
1394 qr/^\s*RENAME\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1395 => sub { push @{$renames{$1}}, split(/\s+/, $2)
1396 if !@skip || $skip[$#skip] > 0 },
1397 qr/^\s*SHARED_NAME\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1398 => sub { push @{$sharednames{$1}}, split(/\s+/, $2)
1399 if !@skip || $skip[$#skip] > 0 },
1400 qr/^\s*BEGINRAW\[((?:\\.|[^\\\]])+)\]\s*$/
1401 => sub {
1402 my $lineiterator = shift;
1403 my $target_kind = $1;
1404 while (defined $lineiterator->()) {
1405 s|\R$||;
1406 if (/^\s*ENDRAW\[((?:\\.|[^\\\]])+)\]\s*$/) {
1407 die "ENDRAW doesn't match BEGINRAW"
1408 if $1 ne $target_kind;
1409 last;
1410 }
1411 next if @skip && $skip[$#skip] <= 0;
1412 push @rawlines, $_
1413 if ($target_kind eq $target{build_file}
1414 || $target_kind eq $target{build_file}."(".$builder_platform.")");
1415 }
1416 },
1417 qr/^(?:#.*|\s*)$/ => sub { },
1418 "OTHERWISE" => sub { die "Something wrong with this line:\n$_\nat $sourced/$f" },
1419 "BEFORE" => sub {
1420 if ($buildinfo_debug) {
1421 print STDERR "DEBUG: Parsing ",join(" ", @_),"\n";
1422 print STDERR "DEBUG: ... before parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
1423 }
1424 },
1425 "AFTER" => sub {
1426 if ($buildinfo_debug) {
1427 print STDERR "DEBUG: .... after parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
1428 }
1429 },
1430 );
1431 die "runaway IF?" if (@skip);
1432
1433 foreach (keys %renames) {
1434 die "$_ renamed to more than one thing: "
1435 ,join(" ", @{$renames{$_}}),"\n"
1436 if scalar @{$renames{$_}} > 1;
1437 my $dest = cleanfile($buildd, $_, $blddir);
1438 my $to = cleanfile($buildd, $renames{$_}->[0], $blddir);
1439 die "$dest renamed to more than one thing: "
1440 ,$unified_info{rename}->{$dest}, $to
1441 unless !defined($unified_info{rename}->{$dest})
1442 or $unified_info{rename}->{$dest} eq $to;
1443 $unified_info{rename}->{$dest} = $to;
1444 }
1445
1446 foreach (@programs) {
1447 my $program = cleanfile($buildd, $_, $blddir);
1448 if ($unified_info{rename}->{$program}) {
1449 $program = $unified_info{rename}->{$program};
1450 }
1451 $unified_info{programs}->{$program} = 1;
1452 }
1453
1454 foreach (@libraries) {
1455 my $library = cleanfile($buildd, $_, $blddir);
1456 if ($unified_info{rename}->{$library}) {
1457 $library = $unified_info{rename}->{$library};
1458 }
1459 $unified_info{libraries}->{$library} = 1;
1460 }
1461
1462 die <<"EOF" if scalar @engines and !$config{dynamic_engines};
1463 ENGINES can only be used if configured with 'dynamic-engine'.
1464 This is usually a fault in a build.info file.
1465 EOF
1466 foreach (@engines) {
1467 my $library = cleanfile($buildd, $_, $blddir);
1468 if ($unified_info{rename}->{$library}) {
1469 $library = $unified_info{rename}->{$library};
1470 }
1471 $unified_info{engines}->{$library} = 1;
1472 }
1473
1474 foreach (@scripts) {
1475 my $script = cleanfile($buildd, $_, $blddir);
1476 if ($unified_info{rename}->{$script}) {
1477 $script = $unified_info{rename}->{$script};
1478 }
1479 $unified_info{scripts}->{$script} = 1;
1480 }
1481
1482 foreach (@extra) {
1483 my $extra = cleanfile($buildd, $_, $blddir);
1484 $unified_info{extra}->{$extra} = 1;
1485 }
1486
1487 foreach (@overrides) {
1488 my $override = cleanfile($buildd, $_, $blddir);
1489 $unified_info{overrides}->{$override} = 1;
1490 }
1491
1492 push @{$unified_info{rawlines}}, @rawlines;
1493
1494 unless ($disabled{shared}) {
1495 # Check sharednames.
1496 foreach (keys %sharednames) {
1497 my $dest = cleanfile($buildd, $_, $blddir);
1498 if ($unified_info{rename}->{$dest}) {
1499 $dest = $unified_info{rename}->{$dest};
1500 }
1501 die "shared_name for $dest with multiple values: "
1502 ,join(" ", @{$sharednames{$_}}),"\n"
1503 if scalar @{$sharednames{$_}} > 1;
1504 my $to = cleanfile($buildd, $sharednames{$_}->[0], $blddir);
1505 die "shared_name found for a library $dest that isn't defined\n"
1506 unless $unified_info{libraries}->{$dest};
1507 die "shared_name for $dest with multiple values: "
1508 ,$unified_info{sharednames}->{$dest}, ", ", $to
1509 unless !defined($unified_info{sharednames}->{$dest})
1510 or $unified_info{sharednames}->{$dest} eq $to;
1511 $unified_info{sharednames}->{$dest} = $to;
1512 }
1513
1514 # Additionally, we set up sharednames for libraries that don't
1515 # have any, as themselves.
1516 foreach (keys %{$unified_info{libraries}}) {
1517 if (!defined $unified_info{sharednames}->{$_}) {
1518 $unified_info{sharednames}->{$_} = $_
1519 }
1520 }
1521 }
1522
1523 foreach (keys %ordinals) {
1524 my $dest = $_;
1525 my $ddest = cleanfile($buildd, $_, $blddir);
1526 if ($unified_info{rename}->{$ddest}) {
1527 $ddest = $unified_info{rename}->{$ddest};
1528 }
1529 foreach (@{$ordinals{$dest}}) {
1530 my %known_ordinals =
1531 (
1532 crypto =>
1533 cleanfile($sourced, catfile("util", "libcrypto.num"), $blddir),
1534 ssl =>
1535 cleanfile($sourced, catfile("util", "libssl.num"), $blddir)
1536 );
1537 my $o = $known_ordinals{$_};
1538 die "Ordinals for $ddest defined more than once\n"
1539 if $unified_info{ordinals}->{$ddest};
1540 $unified_info{ordinals}->{$ddest} = [ $_, $o ];
1541 }
1542 }
1543
1544 foreach (keys %sources) {
1545 my $dest = $_;
1546 my $ddest = cleanfile($buildd, $_, $blddir);
1547 if ($unified_info{rename}->{$ddest}) {
1548 $ddest = $unified_info{rename}->{$ddest};
1549 }
1550 foreach (@{$sources{$dest}}) {
1551 my $s = cleanfile($sourced, $_, $blddir);
1552
1553 # If it isn't in the source tree, we assume it's generated
1554 # in the build tree
1555 if (! -f $s) {
1556 $s = cleanfile($buildd, $_, $blddir);
1557 }
1558 # We recognise C and asm files
1559 if ($s =~ /\.[csS]\b$/) {
1560 (my $o = $_) =~ s/\.[csS]\b$/.o/;
1561 $o = cleanfile($buildd, $o, $blddir);
1562 $unified_info{sources}->{$ddest}->{$o} = 1;
1563 $unified_info{sources}->{$o}->{$s} = 1;
1564 } else {
1565 $unified_info{sources}->{$ddest}->{$s} = 1;
1566 }
1567 }
1568 }
1569
1570 foreach (keys %generate) {
1571 my $dest = $_;
1572 my $ddest = cleanfile($buildd, $_, $blddir);
1573 if ($unified_info{rename}->{$ddest}) {
1574 $ddest = $unified_info{rename}->{$ddest};
1575 }
1576 die "more than one generator for $dest: "
1577 ,join(" ", @{$generate{$_}}),"\n"
1578 if scalar @{$generate{$_}} > 1;
1579 my @generator = split /\s+/, $generate{$dest}->[0];
1580 $generator[0] = cleanfile($sourced, $generator[0], $blddir),
1581 $unified_info{generate}->{$ddest} = [ @generator ];
1582 }
1583
1584 foreach (keys %depends) {
1585 my $dest = $_;
1586 my $ddest = cleanfile($buildd, $_, $blddir);
1587 if ($unified_info{rename}->{$ddest}) {
1588 $ddest = $unified_info{rename}->{$ddest};
1589 }
1590 foreach (@{$depends{$dest}}) {
1591 my $d = cleanfile($sourced, $_, $blddir);
1592
1593 # If we know it's generated, or assume it is because we can't
1594 # find it in the source tree, we set file we depend on to be
1595 # in the build tree rather than the source tree, and assume
1596 # and that there are lines to build it in a BEGINRAW..ENDRAW
1597 # section or in the Makefile template.
1598 if (! -f $d
1599 || (grep { $d eq $_ }
1600 map { cleanfile($srcdir, $_, $blddir) }
1601 (@generated_headers, @generated_by_make_headers))) {
1602 $d = cleanfile($buildd, $_, $blddir);
1603 }
1604 # Take note if the file to depend on is being renamed
1605 if ($unified_info{rename}->{$d}) {
1606 $d = $unified_info{rename}->{$d};
1607 }
1608 $unified_info{depends}->{$ddest}->{$d} = 1;
1609 # If we depend on a header file, let's make sure it
1610 # can get included
1611 if ($d =~ /\.h$/) {
1612 my $i = dirname($d);
1613 push @{$unified_info{includes}->{$ddest}}, $i
1614 unless grep { $_ eq $i } @{$unified_info{includes}->{$ddest}};
1615 }
1616 }
1617 }
1618
1619 foreach (keys %includes) {
1620 my $dest = $_;
1621 my $ddest = cleanfile($buildd, $_, $blddir);
1622 if ($unified_info{rename}->{$ddest}) {
1623 $ddest = $unified_info{rename}->{$ddest};
1624 }
1625 foreach (@{$includes{$dest}}) {
1626 my $i = cleandir($sourced, $_, $blddir);
1627 push @{$unified_info{includes}->{$ddest}}, $i
1628 unless grep { $_ eq $i } @{$unified_info{includes}->{$ddest}};
1629 }
1630 }
1631 }
1632
1633 ### Make unified_info a bit more efficient
1634 # One level structures
1635 foreach (("programs", "libraries", "engines", "scripts", "extra", "overrides")) {
1636 $unified_info{$_} = [ sort keys %{$unified_info{$_}} ];
1637 }
1638 # Two level structures
1639 foreach my $l1 (("sources", "ldadd", "depends")) {
1640 foreach my $l2 (sort keys %{$unified_info{$l1}}) {
1641 $unified_info{$l1}->{$l2} =
1642 [ sort keys %{$unified_info{$l1}->{$l2}} ];
1643 }
1644 }
1645 }
1646
1647 # For the schemes that need it, we provide the old *_obj configs
1648 # from the *_asm_obj ones
1649 foreach (grep /_(asm|aux)_src$/, keys %target) {
1650 my $src = $_;
1651 (my $obj = $_) =~ s/_(asm|aux)_src$/_obj/;
1652 ($target{$obj} = $target{$src}) =~ s/\.[csS]\b/.o/g;
1653 }
1654
1655 # Write down our configuration where it fits #########################
1656
1657 open(OUT,">configdata.pm") || die "unable to create configdata.pm: $!\n";
1658 print OUT <<"EOF";
1659 package configdata;
1660
1661 use strict;
1662 use warnings;
1663
1664 use Exporter;
1665 #use vars qw(\@ISA \@EXPORT);
1666 our \@ISA = qw(Exporter);
1667 our \@EXPORT = qw(\%config \%target \%disabled \%withargs \%unified_info \@disablables);
1668
1669 EOF
1670 print OUT "our %config = (\n";
1671 foreach (sort keys %config) {
1672 if (ref($config{$_}) eq "ARRAY") {
1673 print OUT " ", $_, " => [ ", join(", ",
1674 map { quotify("perl", $_) }
1675 @{$config{$_}}), " ],\n";
1676 } else {
1677 print OUT " ", $_, " => ", quotify("perl", $config{$_}), ",\n"
1678 }
1679 }
1680 print OUT <<"EOF";
1681 );
1682
1683 EOF
1684 print OUT "our %target = (\n";
1685 foreach (sort keys %target) {
1686 if (ref($target{$_}) eq "ARRAY") {
1687 print OUT " ", $_, " => [ ", join(", ",
1688 map { quotify("perl", $_) }
1689 @{$target{$_}}), " ],\n";
1690 } else {
1691 print OUT " ", $_, " => ", quotify("perl", $target{$_}), ",\n"
1692 }
1693 }
1694 print OUT <<"EOF";
1695 );
1696
1697 EOF
1698 print OUT "our \%available_protocols = (\n";
1699 print OUT " tls => [ ", join(", ", map { quotify("perl", $_) } @tls), " ],\n";
1700 print OUT " dtls => [ ", join(", ", map { quotify("perl", $_) } @dtls), " ],\n";
1701 print OUT <<"EOF";
1702 );
1703
1704 EOF
1705 print OUT "our \@disablables = (\n";
1706 foreach (@disablables) {
1707 print OUT " ", quotify("perl", $_), ",\n";
1708 }
1709 print OUT <<"EOF";
1710 );
1711
1712 EOF
1713 print OUT "our \%disabled = (\n";
1714 foreach (sort keys %disabled) {
1715 print OUT " ", quotify("perl", $_), " => ", quotify("perl", $disabled{$_}), ",\n";
1716 }
1717 print OUT <<"EOF";
1718 );
1719
1720 EOF
1721 print OUT "our %withargs = (\n";
1722 foreach (sort keys %withargs) {
1723 if (ref($withargs{$_}) eq "ARRAY") {
1724 print OUT " ", $_, " => [ ", join(", ",
1725 map { quotify("perl", $_) }
1726 @{$withargs{$_}}), " ],\n";
1727 } else {
1728 print OUT " ", $_, " => ", quotify("perl", $withargs{$_}), ",\n"
1729 }
1730 }
1731 print OUT <<"EOF";
1732 );
1733
1734 EOF
1735 if ($builder eq "unified") {
1736 my $recurse;
1737 $recurse = sub {
1738 my $indent = shift;
1739 foreach (@_) {
1740 if (ref $_ eq "ARRAY") {
1741 print OUT " "x$indent, "[\n";
1742 foreach (@$_) {
1743 $recurse->($indent + 4, $_);
1744 }
1745 print OUT " "x$indent, "],\n";
1746 } elsif (ref $_ eq "HASH") {
1747 my %h = %$_;
1748 print OUT " "x$indent, "{\n";
1749 foreach (sort keys %h) {
1750 if (ref $h{$_} eq "") {
1751 print OUT " "x($indent + 4), quotify("perl", $_), " => ", quotify("perl", $h{$_}), ",\n";
1752 } else {
1753 print OUT " "x($indent + 4), quotify("perl", $_), " =>\n";
1754 $recurse->($indent + 8, $h{$_});
1755 }
1756 }
1757 print OUT " "x$indent, "},\n";
1758 } else {
1759 print OUT " "x$indent, quotify("perl", $_), ",\n";
1760 }
1761 }
1762 };
1763 print OUT "our %unified_info = (\n";
1764 foreach (sort keys %unified_info) {
1765 if (ref $unified_info{$_} eq "") {
1766 print OUT " "x4, quotify("perl", $_), " => ", quotify("perl", $unified_info{$_}), ",\n";
1767 } else {
1768 print OUT " "x4, quotify("perl", $_), " =>\n";
1769 $recurse->(8, $unified_info{$_});
1770 }
1771 }
1772 print OUT <<"EOF";
1773 );
1774
1775 EOF
1776 }
1777 print OUT "1;\n";
1778 close(OUT);
1779
1780
1781 print "IsMK1MF =", ($builder eq "mk1mf" ? "yes" : "no"), "\n";
1782 print "CC =$target{cc}\n";
1783 print "CFLAG =$target{cflags} $config{cflags}\n";
1784 print "SHARED_CFLAG =$target{shared_cflag}\n";
1785 print "DEFINES =",join(" ", @{$target{defines}}, @{$config{defines}}),"\n";
1786 print "LFLAG =$target{lflags}\n";
1787 print "PLIB_LFLAG =$target{plib_lflags}\n";
1788 print "EX_LIBS =$target{ex_libs} $config{ex_libs}\n";
1789 print "APPS_OBJ =$target{apps_obj}\n";
1790 print "CPUID_OBJ =$target{cpuid_obj}\n";
1791 print "UPLINK_OBJ =$target{uplink_obj}\n";
1792 print "BN_ASM =$target{bn_obj}\n";
1793 print "EC_ASM =$target{ec_obj}\n";
1794 print "DES_ENC =$target{des_obj}\n";
1795 print "AES_ENC =$target{aes_obj}\n";
1796 print "BF_ENC =$target{bf_obj}\n";
1797 print "CAST_ENC =$target{cast_obj}\n";
1798 print "RC4_ENC =$target{rc4_obj}\n";
1799 print "RC5_ENC =$target{rc5_obj}\n";
1800 print "MD5_OBJ_ASM =$target{md5_obj}\n";
1801 print "SHA1_OBJ_ASM =$target{sha1_obj}\n";
1802 print "RMD160_OBJ_ASM=$target{rmd160_obj}\n";
1803 print "CMLL_ENC =$target{cmll_obj}\n";
1804 print "MODES_OBJ =$target{modes_obj}\n";
1805 print "PADLOCK_OBJ =$target{padlock_obj}\n";
1806 print "CHACHA_ENC =$target{chacha_obj}\n";
1807 print "POLY1305_OBJ =$target{poly1305_obj}\n";
1808 print "BLAKE2_OBJ =$target{blake2_obj}\n";
1809 print "PROCESSOR =$config{processor}\n";
1810 print "RANLIB =$target{ranlib}\n";
1811 print "ARFLAGS =$target{arflags}\n";
1812 print "PERL =$config{perl}\n";
1813 print "\n";
1814 print "SIXTY_FOUR_BIT_LONG mode\n" if $config{b64l};
1815 print "SIXTY_FOUR_BIT mode\n" if $config{b64};
1816 print "THIRTY_TWO_BIT mode\n" if $config{b32};
1817 print "BN_LLONG mode\n" if $config{bn_ll};
1818 print "RC4 uses $config{rc4_int}\n" if $config{rc4_int} ne $def_int;
1819
1820 for (@generated_headers) {
1821 mkpath(catdir($blddir, dirname($_)));
1822 run_dofile(catfile($blddir, $_),
1823 catfile($srcdir, $_.".in"));
1824 }
1825
1826 ###
1827 ### When the old "unixmake" scheme goes away, so does this function
1828 ###
1829 sub build_Makefile {
1830 run_dofile("Makefile","Makefile.in");
1831
1832 # Copy all Makefile.in to Makefile (except top-level)
1833 use File::Find;
1834 use IO::File;
1835 find(
1836 {
1837 preprocess => sub {
1838 grep(!/^\./, @_);
1839 },
1840 wanted => sub {
1841 return if ($_ ne "Makefile.in" || $File::Find::dir eq ".");
1842 my $in = IO::File->new($_, "r") or
1843 die sprintf "Error reading Makefile.in in %s: !$\n",
1844 $File::Find::dir;
1845 my $out = IO::File->new("Makefile", "w") or
1846 die sprintf "Error writing Makefile in %s: !$\n",
1847 $File::Find::dir;
1848 print $out "# Generated from $_, do not edit\n";
1849 while (my $line = <$in>) { print $out $line }
1850 $in->close() or
1851 die sprintf "Error reading Makefile.in in %s: !$\n",
1852 $File::Find::dir;
1853 $out->close() or
1854 die sprintf "Error writing Makefile in %s: !$\n",
1855 $File::Find::dir;
1856 },
1857 },
1858 ".");
1859 }
1860
1861 my %builders = (
1862 unified => sub {
1863 run_dofile(catfile($blddir, $target{build_file}),
1864 $config{build_file_template},
1865 catfile($srcdir, "Configurations", "common.tmpl"));
1866 },
1867 unixmake => sub {
1868 build_Makefile();
1869
1870 run_dofile("util/domd", "util/domd.in");
1871 chmod 0755, "util/domd";
1872 },
1873 mk1mf => sub {
1874 my $platform = shift;
1875 # The only reason we do this is to have something to build MINFO from
1876 build_Makefile();
1877
1878 # create the ms/version32.rc file if needed
1879 my ($v1, $v2, $v3, $v4);
1880 if ($config{version_num} =~ /^0x([0-9a-f]{1})([0-9a-f]{2})([0-9a-f]{2})([0-9a-f]{2})([0-9a-f]{1})L$/i) {
1881 $v1=hex $1;
1882 $v2=hex $2;
1883 $v3=hex $3;
1884 $v4=hex $4;
1885 }
1886 open (OUT,">ms/version32.rc") || die "Can't open ms/version32.rc";
1887 print OUT <<"EOF";
1888 #include <winver.h>
1889
1890 LANGUAGE 0x09,0x01
1891
1892 1 VERSIONINFO
1893 FILEVERSION $v1,$v2,$v3,$v4
1894 PRODUCTVERSION $v1,$v2,$v3,$v4
1895 FILEFLAGSMASK 0x3fL
1896 #ifdef _DEBUG
1897 FILEFLAGS 0x01L
1898 #else
1899 FILEFLAGS 0x00L
1900 #endif
1901 FILEOS VOS__WINDOWS32
1902 FILETYPE VFT_DLL
1903 FILESUBTYPE 0x0L
1904 BEGIN
1905 BLOCK "StringFileInfo"
1906 BEGIN
1907 BLOCK "040904b0"
1908 BEGIN
1909 // Required:
1910 VALUE "CompanyName", "The OpenSSL Project, http://www.openssl.org/\\0"
1911 VALUE "FileDescription", "OpenSSL Shared Library\\0"
1912 VALUE "FileVersion", "$config{version}\\0"
1913 #if defined(CRYPTO)
1914 VALUE "InternalName", "libcrypto32\\0"
1915 VALUE "OriginalFilename", "libcrypto32.dll\\0"
1916 #elif defined(SSL)
1917 VALUE "InternalName", "libssl32\\0"
1918 VALUE "OriginalFilename", "libssl32.dll\\0"
1919 #endif
1920 VALUE "ProductName", "The OpenSSL Toolkit\\0"
1921 VALUE "ProductVersion", "$config{version}\\0"
1922 // Optional:
1923 //VALUE "Comments", "\\0"
1924 VALUE "LegalCopyright", "Copyright © 1998-2015 The OpenSSL Project. Copyright © 1995-1998 Eric A. Young, Tim J. Hudson. All rights reserved.\\0"
1925 //VALUE "LegalTrademarks", "\\0"
1926 //VALUE "PrivateBuild", "\\0"
1927 //VALUE "SpecialBuild", "\\0"
1928 END
1929 END
1930 BLOCK "VarFileInfo"
1931 BEGIN
1932 VALUE "Translation", 0x409, 0x4b0
1933 END
1934 END
1935 EOF
1936 close(OUT);
1937 },
1938 );
1939
1940 $builders{$builder}->($builder_platform, @builder_opts);
1941
1942 print <<"EOF";
1943
1944 Configured for $target.
1945 EOF
1946
1947 print <<"EOF" if ($disabled{threads} eq "unavailable");
1948
1949 The library could not be configured for supporting multi-threaded
1950 applications as the compiler options required on this system are not known.
1951 See file INSTALL for details if you need multi-threading.
1952 EOF
1953
1954 print <<"EOF" if ($no_shared_warn);
1955
1956 The options 'shared', 'pic' and 'dynamic-engine' aren't supported on this
1957 platform, so we will pretend you gave the option 'no-pic', which also disables
1958 'shared' and 'dynamic-engine'. If you know how to implement shared libraries
1959 or position independent code, please let us know (but please first make sure
1960 you have tried with a current version of OpenSSL).
1961 EOF
1962
1963 exit(0);
1964
1965 ######################################################################
1966 #
1967 # Helpers and utility functions
1968 #
1969
1970 # Configuration file reading #########################################
1971
1972 # Note: All of the helper functions are for lazy evaluation. They all
1973 # return a CODE ref, which will return the intended value when evaluated.
1974 # Thus, whenever there's mention of a returned value, it's about that
1975 # intended value.
1976
1977 # Helper function to implement conditional inheritance depending on the
1978 # value of $disabled{asm}. Used in inherit_from values as follows:
1979 #
1980 # inherit_from => [ "template", asm("asm_tmpl") ]
1981 #
1982 sub asm {
1983 my @x = @_;
1984 sub {
1985 $disabled{asm} ? () : @x;
1986 }
1987 }
1988
1989 # Helper function to implement conditional value variants, with a default
1990 # plus additional values based on the value of $config{build_type}.
1991 # Arguments are given in hash table form:
1992 #
1993 # picker(default => "Basic string: ",
1994 # debug => "debug",
1995 # release => "release")
1996 #
1997 # When configuring with --debug, the resulting string will be
1998 # "Basic string: debug", and when not, it will be "Basic string: release"
1999 #
2000 # This can be used to create variants of sets of flags according to the
2001 # build type:
2002 #
2003 # cflags => picker(default => "-Wall",
2004 # debug => "-g -O0",
2005 # release => "-O3")
2006 #
2007 sub picker {
2008 my %opts = @_;
2009 return sub { add($opts{default} || (),
2010 $opts{$config{build_type}} || ())->(); }
2011 }
2012
2013 # Helper function to combine several values of different types into one.
2014 # This is useful if you want to combine a string with the result of a
2015 # lazy function, such as:
2016 #
2017 # cflags => combine("-Wall", sub { $disabled{zlib} ? () : "-DZLIB" })
2018 #
2019 sub combine {
2020 my @stuff = @_;
2021 return sub { add(@stuff)->(); }
2022 }
2023
2024 # Helper function to implement conditional values depending on the value
2025 # of $disabled{threads}. Can be used as follows:
2026 #
2027 # cflags => combine("-Wall", threads("-pthread"))
2028 #
2029 sub threads {
2030 my @flags = @_;
2031 return sub { add($disabled{threads} ? () : @flags)->(); }
2032 }
2033
2034
2035
2036 our $add_called = 0;
2037 # Helper function to implement adding values to already existing configuration
2038 # values. It handles elements that are ARRAYs, CODEs and scalars
2039 sub _add {
2040 my $separator = shift;
2041
2042 # If there's any ARRAY in the collection of values OR the separator
2043 # is undef, we will return an ARRAY of combined values, otherwise a
2044 # string of joined values with $separator as the separator.
2045 my $found_array = !defined($separator);
2046
2047 my @values =
2048 map {
2049 my $res = $_;
2050 while (ref($res) eq "CODE") {
2051 $res = $res->();
2052 }
2053 if (defined($res)) {
2054 if (ref($res) eq "ARRAY") {
2055 $found_array = 1;
2056 @$res;
2057 } else {
2058 $res;
2059 }
2060 } else {
2061 ();
2062 }
2063 } (@_);
2064
2065 $add_called = 1;
2066
2067 if ($found_array) {
2068 [ @values ];
2069 } else {
2070 join($separator, grep { defined($_) && $_ ne "" } @values);
2071 }
2072 }
2073 sub add_before {
2074 my $separator = " ";
2075 if (ref($_[$#_]) eq "HASH") {
2076 my $opts = pop;
2077 $separator = $opts->{separator};
2078 }
2079 my @x = @_;
2080 sub { _add($separator, @x, @_) };
2081 }
2082 sub add {
2083 my $separator = " ";
2084 if (ref($_[$#_]) eq "HASH") {
2085 my $opts = pop;
2086 $separator = $opts->{separator};
2087 }
2088 my @x = @_;
2089 sub { _add($separator, @_, @x) };
2090 }
2091
2092 # configuration reader, evaluates the input file as a perl script and expects
2093 # it to fill %targets with target configurations. Those are then added to
2094 # %table.
2095 sub read_config {
2096 my $fname = shift;
2097 open(CONFFILE, "< $fname")
2098 or die "Can't open configuration file '$fname'!\n";
2099 my $x = $/;
2100 undef $/;
2101 my $content = <CONFFILE>;
2102 $/ = $x;
2103 close(CONFFILE);
2104 my %targets = ();
2105 {
2106 local %table = %::table; # Protect %table from tampering
2107
2108 eval $content;
2109 warn $@ if $@;
2110 }
2111
2112 # For each target, check that it's configured with a hash table.
2113 foreach (keys %targets) {
2114 if (ref($targets{$_}) ne "HASH") {
2115 if (ref($targets{$_}) eq "") {
2116 warn "Deprecated target configuration for $_, ignoring...\n";
2117 } else {
2118 warn "Misconfigured target configuration for $_ (should be a hash table), ignoring...\n";
2119 }
2120 delete $targets{$_};
2121 }
2122 }
2123
2124 %table = (%table, %targets);
2125
2126 }
2127
2128 # configuration resolver. Will only resolve all the lazy evalutation
2129 # codeblocks for the chozen target and all those it inherits from,
2130 # recursively
2131 sub resolve_config {
2132 my $target = shift;
2133 my @breadcrumbs = @_;
2134
2135 # my $extra_checks = defined($ENV{CONFIGURE_EXTRA_CHECKS});
2136
2137 if (grep { $_ eq $target } @breadcrumbs) {
2138 die "inherit_from loop! target backtrace:\n "
2139 ,$target,"\n ",join("\n ", @breadcrumbs),"\n";
2140 }
2141
2142 if (!defined($table{$target})) {
2143 warn "Warning! target $target doesn't exist!\n";
2144 return ();
2145 }
2146 # Recurse through all inheritances. They will be resolved on the
2147 # fly, so when this operation is done, they will all just be a
2148 # bunch of attributes with string values.
2149 # What we get here, though, are keys with references to lists of
2150 # the combined values of them all. We will deal with lists after
2151 # this stage is done.
2152 my %combined_inheritance = ();
2153 if ($table{$target}->{inherit_from}) {
2154 my @inherit_from =
2155 map { ref($_) eq "CODE" ? $_->() : $_ } @{$table{$target}->{inherit_from}};
2156 foreach (@inherit_from) {
2157 my %inherited_config = resolve_config($_, $target, @breadcrumbs);
2158
2159 # 'template' is a marker that's considered private to
2160 # the config that had it.
2161 delete $inherited_config{template};
2162
2163 map {
2164 if (!$combined_inheritance{$_}) {
2165 $combined_inheritance{$_} = [];
2166 }
2167 push @{$combined_inheritance{$_}}, $inherited_config{$_};
2168 } keys %inherited_config;
2169 }
2170 }
2171
2172 # We won't need inherit_from in this target any more, since we've
2173 # resolved all the inheritances that lead to this
2174 delete $table{$target}->{inherit_from};
2175
2176 # Now is the time to deal with those lists. Here's the place to
2177 # decide what shall be done with those lists, all based on the
2178 # values of the target we're currently dealing with.
2179 # - If a value is a coderef, it will be executed with the list of
2180 # inherited values as arguments.
2181 # - If the corresponding key doesn't have a value at all or is the
2182 # emoty string, the inherited value list will be run through the
2183 # default combiner (below), and the result becomes this target's
2184 # value.
2185 # - Otherwise, this target's value is assumed to be a string that
2186 # will simply override the inherited list of values.
2187 my $default_combiner = add();
2188
2189 my %all_keys =
2190 map { $_ => 1 } (keys %combined_inheritance,
2191 keys %{$table{$target}});
2192
2193 sub process_values {
2194 my $object = shift;
2195 my $inherited = shift; # Always a [ list ]
2196 my $target = shift;
2197 my $entry = shift;
2198
2199 $add_called = 0;
2200
2201 while(ref($object) eq "CODE") {
2202 $object = $object->(@$inherited);
2203 }
2204 if (!defined($object)) {
2205 return ();
2206 }
2207 elsif (ref($object) eq "ARRAY") {
2208 local $add_called; # To make sure recursive calls don't affect it
2209 return [ map { process_values($_, $inherited, $target, $entry) }
2210 @$object ];
2211 } elsif (ref($object) eq "") {
2212 return $object;
2213 } else {
2214 die "cannot handle reference type ",ref($object)
2215 ," found in target ",$target," -> ",$entry,"\n";
2216 }
2217 }
2218
2219 foreach (sort keys %all_keys) {
2220 my $previous = $combined_inheritance{$_};
2221
2222 # Current target doesn't have a value for the current key?
2223 # Assign it the default combiner, the rest of this loop body
2224 # will handle it just like any other coderef.
2225 if (!exists $table{$target}->{$_}) {
2226 $table{$target}->{$_} = $default_combiner;
2227 }
2228
2229 $table{$target}->{$_} = process_values($table{$target}->{$_},
2230 $combined_inheritance{$_},
2231 $target, $_);
2232 unless(defined($table{$target}->{$_})) {
2233 delete $table{$target}->{$_};
2234 }
2235 # if ($extra_checks &&
2236 # $previous && !($add_called || $previous ~~ $table{$target}->{$_})) {
2237 # warn "$_ got replaced in $target\n";
2238 # }
2239 }
2240
2241 # Finally done, return the result.
2242 return %{$table{$target}};
2243 }
2244
2245 sub usage
2246 {
2247 print STDERR $usage;
2248 print STDERR "\npick os/compiler from:\n";
2249 my $j=0;
2250 my $i;
2251 my $k=0;
2252 foreach $i (sort keys %table)
2253 {
2254 next if $table{$i}->{template};
2255 next if $i =~ /^debug/;
2256 $k += length($i) + 1;
2257 if ($k > 78)
2258 {
2259 print STDERR "\n";
2260 $k=length($i);
2261 }
2262 print STDERR $i . " ";
2263 }
2264 foreach $i (sort keys %table)
2265 {
2266 next if $table{$i}->{template};
2267 next if $i !~ /^debug/;
2268 $k += length($i) + 1;
2269 if ($k > 78)
2270 {
2271 print STDERR "\n";
2272 $k=length($i);
2273 }
2274 print STDERR $i . " ";
2275 }
2276 print STDERR "\n\nNOTE: If in doubt, on Unix-ish systems use './config'.\n";
2277 exit(1);
2278 }
2279
2280 sub run_dofile
2281 {
2282 my $out = shift;
2283 my @templates = @_;
2284
2285 unlink $out || warn "Can't remove $out, $!"
2286 if -f $out;
2287 foreach (@templates) {
2288 die "Can't open $_, $!" unless -f $_;
2289 }
2290 my $cmd = "$config{perl} \"-I.\" \"-Mconfigdata\" $dofile -o\"Configure\" \"".join("\" \"",@templates)."\" > \"$out.new\"";
2291 #print STDERR "DEBUG[run_dofile]: \$cmd = $cmd\n";
2292 system($cmd);
2293 exit 1 if $? != 0;
2294 rename("$out.new", $out) || die "Can't rename $out.new, $!";
2295 }
2296
2297 # Configuration printer ##############################################
2298
2299 sub print_table_entry
2300 {
2301 my $target = shift;
2302 my %target = resolve_config($target);
2303 my $type = shift;
2304
2305 # Don't print the templates
2306 return if $target{template};
2307
2308 my @sequence = (
2309 "sys_id",
2310 "cc",
2311 "cflags",
2312 "defines",
2313 "unistd",
2314 "ld",
2315 "lflags",
2316 "plib_lflags",
2317 "ex_libs",
2318 "bn_ops",
2319 "cpuid_obj",
2320 "bn_obj",
2321 "ec_obj",
2322 "des_obj",
2323 "aes_obj",
2324 "bf_obj",
2325 "md5_obj",
2326 "sha1_obj",
2327 "cast_obj",
2328 "rc4_obj",
2329 "rmd160_obj",
2330 "rc5_obj",
2331 "wp_obj",
2332 "cmll_obj",
2333 "modes_obj",
2334 "padlock_obj",
2335 "thread_scheme",
2336 "perlasm_scheme",
2337 "dso_scheme",
2338 "shared_target",
2339 "shared_cflag",
2340 "shared_ldflag",
2341 "shared_rcflag",
2342 "shared_extension",
2343 "shared_extension_simple",
2344 "shared_import_extension",
2345 "dso_extension",
2346 "obj_extension",
2347 "exe_extension",
2348 "ranlib",
2349 "ar",
2350 "arflags",
2351 "multilib",
2352 "build_scheme",
2353 );
2354
2355 if ($type eq "TABLE") {
2356 print "\n";
2357 print "*** $target\n";
2358 foreach (@sequence) {
2359 if (ref($target{$_}) eq "ARRAY") {
2360 printf "\$%-12s = %s\n", $_, join(" ", @{$target{$_}});
2361 } else {
2362 printf "\$%-12s = %s\n", $_, $target{$_};
2363 }
2364 }
2365 } elsif ($type eq "HASH") {
2366 my $largest =
2367 length((sort { length($a) <=> length($b) } @sequence)[-1]);
2368 print " '$target' => {\n";
2369 foreach (@sequence) {
2370 if ($target{$_}) {
2371 if (ref($target{$_}) eq "ARRAY") {
2372 print " '",$_,"'"," " x ($largest - length($_))," => [ ",join(", ", map { "'$_'" } @{$target{$_}})," ],\n";
2373 } else {
2374 print " '",$_,"'"," " x ($largest - length($_))," => '",$target{$_},"',\n";
2375 }
2376 }
2377 }
2378 print " },\n";
2379 }
2380 }
2381
2382 # Utility routines ###################################################
2383
2384 # On VMS, if the given file is a logical name, File::Spec::Functions
2385 # will consider it an absolute path. There are cases when we want a
2386 # purely syntactic check without checking the environment.
2387 sub isabsolute {
2388 my $file = shift;
2389
2390 # On non-platforms, we just use file_name_is_absolute().
2391 return file_name_is_absolute($file) unless $^O eq "VMS";
2392
2393 # If the file spec includes a device or a directpry spec,
2394 # file_name_is_absolute() is perfectly safe.
2395 return file_name_is_absolute($file) if $file =~ m|[:\[]|;
2396
2397 # Here, we know the given file spec isn't absolute
2398 return 0;
2399 }
2400
2401 # Makes a directory absolute and cleans out /../ in paths like foo/../bar
2402 # On some platforms, this uses rel2abs(), while on others, realpath() is used.
2403 # realpath() requires that at least all path components except the last is an
2404 # existing directory. On VMS, the last component of the directory spec must
2405 # exist.
2406 sub absolutedir {
2407 my $dir = shift;
2408
2409 # realpath() is quite buggy on VMS. It uses LIB$FID_TO_NAME, which
2410 # will return the volume name for the device, no matter what. Also,
2411 # it will return an incorrect directory spec if the argument is a
2412 # directory that doesn't exist.
2413 if ($^O eq "VMS") {
2414 return rel2abs($dir);
2415 }
2416
2417 # We use realpath() on Unix, since no other will properly clean out
2418 # a directory spec.
2419 use Cwd qw/realpath/;
2420
2421 return realpath($dir);
2422 }
2423
2424 sub which
2425 {
2426 my($name)=@_;
2427 my $path;
2428 foreach $path (split /:/, $ENV{PATH})
2429 {
2430 my $fullpath = "$path/$name$target{exe_extension}";
2431 if (-f $fullpath and -x $fullpath)
2432 {
2433 return $fullpath
2434 unless ($name eq "perl" and
2435 system("$fullpath -e " . '\'exit($]<5.0);\''));
2436 }
2437 }
2438 }
2439
2440 sub quotify {
2441 my %processors = (
2442 perl => sub { my $x = shift;
2443 $x =~ s/([\\\$\@"])/\\$1/g;
2444 return '"'.$x.'"'; },
2445 );
2446 my $for = shift;
2447 my $processor =
2448 defined($processors{$for}) ? $processors{$for} : sub { shift; };
2449
2450 map { $processor->($_); } @_;
2451 }
2452
2453 # collect_from_file($filename, $line_concat_cond_re, $line_concat)
2454 # $filename is a file name to read from
2455 # $line_concat_cond_re is a regexp detecting a line continuation ending
2456 # $line_concat is a CODEref that takes care of concatenating two lines
2457 sub collect_from_file {
2458 my $filename = shift;
2459 my $line_concat_cond_re = shift;
2460 my $line_concat = shift;
2461
2462 open my $fh, $filename || die "unable to read $filename: $!\n";
2463 return sub {
2464 my $saved_line = "";
2465 $_ = "";
2466 while (<$fh>) {
2467 s|\R$||;
2468 if (defined $line_concat) {
2469 $_ = $line_concat->($saved_line, $_);
2470 $saved_line = "";
2471 }
2472 if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
2473 $saved_line = $_;
2474 next;
2475 }
2476 return $_;
2477 }
2478 die "$filename ending with continuation line\n" if $_;
2479 close $fh;
2480 return undef;
2481 }
2482 }
2483
2484 # collect_from_array($array, $line_concat_cond_re, $line_concat)
2485 # $array is an ARRAYref of lines
2486 # $line_concat_cond_re is a regexp detecting a line continuation ending
2487 # $line_concat is a CODEref that takes care of concatenating two lines
2488 sub collect_from_array {
2489 my $array = shift;
2490 my $line_concat_cond_re = shift;
2491 my $line_concat = shift;
2492 my @array = (@$array);
2493
2494 return sub {
2495 my $saved_line = "";
2496 $_ = "";
2497 while (defined($_ = shift @array)) {
2498 s|\R$||;
2499 if (defined $line_concat) {
2500 $_ = $line_concat->($saved_line, $_);
2501 $saved_line = "";
2502 }
2503 if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
2504 $saved_line = $_;
2505 next;
2506 }
2507 return $_;
2508 }
2509 die "input text ending with continuation line\n" if $_;
2510 return undef;
2511 }
2512 }
2513
2514 # collect_information($lineiterator, $line_continue, $regexp => $CODEref, ...)
2515 # $lineiterator is a CODEref that delivers one line at a time.
2516 # All following arguments are regex/CODEref pairs, where the regexp detects a
2517 # line and the CODEref does something with the result of the regexp.
2518 sub collect_information {
2519 my $lineiterator = shift;
2520 my %collectors = @_;
2521
2522 while(defined($_ = $lineiterator->())) {
2523 s|\R$||;
2524 my $found = 0;
2525 if ($collectors{"BEFORE"}) {
2526 $collectors{"BEFORE"}->($_);
2527 }
2528 foreach my $re (keys %collectors) {
2529 if ($re !~ /^OTHERWISE|BEFORE|AFTER$/ && /$re/) {
2530 $collectors{$re}->($lineiterator);
2531 $found = 1;
2532 };
2533 }
2534 if ($collectors{"OTHERWISE"}) {
2535 $collectors{"OTHERWISE"}->($lineiterator, $_)
2536 unless $found || !defined $collectors{"OTHERWISE"};
2537 }
2538 if ($collectors{"AFTER"}) {
2539 $collectors{"AFTER"}->($_);
2540 }
2541 }
2542 }