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