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