]> git.ipfire.org Git - thirdparty/openssl.git/blob - Configure
Create provider errors and use them
[thirdparty/openssl.git] / Configure
1 #! /usr/bin/env perl
2 # -*- mode: perl; -*-
3 # Copyright 2016-2018 The OpenSSL Project Authors. All Rights Reserved.
4 #
5 # Licensed under the Apache License 2.0 (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 use 5.10.0;
13 use strict;
14 use Config;
15 use FindBin;
16 use lib "$FindBin::Bin/util/perl";
17 use File::Basename;
18 use File::Spec::Functions qw/:DEFAULT abs2rel rel2abs splitdir/;
19 use File::Path qw/mkpath/;
20 use OpenSSL::Glob;
21
22 # see INSTALL for instructions.
23
24 my $orig_death_handler = $SIG{__DIE__};
25 $SIG{__DIE__} = \&death_handler;
26
27 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-egd] [sctp] [386] [--prefix=DIR] [--openssldir=OPENSSLDIR] [--with-xxx[=vvv]] [--config=FILE] os/compiler[:flags]\n";
28
29 # Options:
30 #
31 # --config add the given configuration file, which will be read after
32 # any "Configurations*" files that are found in the same
33 # directory as this script.
34 # --prefix prefix for the OpenSSL installation, which includes the
35 # directories bin, lib, include, share/man, share/doc/openssl
36 # This becomes the value of INSTALLTOP in Makefile
37 # (Default: /usr/local)
38 # --openssldir OpenSSL data area, such as openssl.cnf, certificates and keys.
39 # If it's a relative directory, it will be added on the directory
40 # given with --prefix.
41 # This becomes the value of OPENSSLDIR in Makefile and in C.
42 # (Default: PREFIX/ssl)
43 #
44 # --cross-compile-prefix Add specified prefix to binutils components.
45 #
46 # --api One of 0.9.8, 1.0.0, 1.0.1, 1.0.2, 1.1.0, 1.1.1, or 3.0.0 / 3.
47 # Do not compile support for interfaces deprecated as of the
48 # specified OpenSSL version.
49 #
50 # no-hw-xxx do not compile support for specific crypto hardware.
51 # Generic OpenSSL-style methods relating to this support
52 # are always compiled but return NULL if the hardware
53 # support isn't compiled.
54 # no-hw do not compile support for any crypto hardware.
55 # [no-]threads [don't] try to create a library that is suitable for
56 # multithreaded applications (default is "threads" if we
57 # know how to do it)
58 # [no-]shared [don't] try to create shared libraries when supported.
59 # [no-]pic [don't] try to build position independent code when supported.
60 # If disabled, it also disables shared and dynamic-engine.
61 # no-asm do not use assembler
62 # no-egd do not compile support for the entropy-gathering daemon APIs
63 # [no-]zlib [don't] compile support for zlib compression.
64 # zlib-dynamic Like "zlib", but the zlib library is expected to be a shared
65 # library and will be loaded in run-time by the OpenSSL library.
66 # sctp include SCTP support
67 # enable-weak-ssl-ciphers
68 # Enable weak ciphers that are disabled by default.
69 # 386 generate 80386 code in assembly modules
70 # no-sse2 disables IA-32 SSE2 code in assembly modules, the above
71 # mentioned '386' option implies this one
72 # no-<cipher> build without specified algorithm (rsa, idea, rc5, ...)
73 # -<xxx> +<xxx> compiler options are passed through
74 # -static while -static is also a pass-through compiler option (and
75 # as such is limited to environments where it's actually
76 # meaningful), it triggers a number configuration options,
77 # namely no-pic, no-shared and no-threads. It is
78 # argued that the only reason to produce statically linked
79 # binaries (and in context it means executables linked with
80 # -static flag, and not just executables linked with static
81 # libcrypto.a) is to eliminate dependency on specific run-time,
82 # a.k.a. libc version. The mentioned config options are meant
83 # to achieve just that. Unfortunately on Linux it's impossible
84 # to eliminate the dependency completely for openssl executable
85 # because of getaddrinfo and gethostbyname calls, which can
86 # invoke dynamically loadable library facility anyway to meet
87 # the lookup requests. For this reason on Linux statically
88 # linked openssl executable has rather debugging value than
89 # production quality.
90 #
91 # DEBUG_SAFESTACK use type-safe stacks to enforce type-safety on stack items
92 # provided to stack calls. Generates unique stack functions for
93 # each possible stack type.
94 # BN_LLONG use the type 'long long' in crypto/bn/bn.h
95 # RC4_CHAR use 'char' instead of 'int' for RC4_INT in crypto/rc4/rc4.h
96 # Following are set automatically by this script
97 #
98 # MD5_ASM use some extra md5 assembler,
99 # SHA1_ASM use some extra sha1 assembler, must define L_ENDIAN for x86
100 # RMD160_ASM use some extra ripemd160 assembler,
101 # SHA256_ASM sha256_block is implemented in assembler
102 # SHA512_ASM sha512_block is implemented in assembler
103 # AES_ASM AES_[en|de]crypt is implemented in assembler
104
105 # Minimum warning options... any contributions to OpenSSL should at least get
106 # past these.
107
108 # DEBUG_UNUSED enables __owur (warn unused result) checks.
109 # -DPEDANTIC complements -pedantic and is meant to mask code that
110 # is not strictly standard-compliant and/or implementation-specific,
111 # e.g. inline assembly, disregards to alignment requirements, such
112 # that -pedantic would complain about. Incidentally -DPEDANTIC has
113 # to be used even in sanitized builds, because sanitizer too is
114 # supposed to and does take notice of non-standard behaviour. Then
115 # -pedantic with pre-C9x compiler would also complain about 'long
116 # long' not being supported. As 64-bit algorithms are common now,
117 # it grew impossible to resolve this without sizeable additional
118 # code, so we just tell compiler to be pedantic about everything
119 # but 'long long' type.
120
121 my %gcc_devteam_warn = ();
122 {
123 my @common = qw( -DDEBUG_UNUSED
124 -DPEDANTIC -pedantic -Wno-long-long
125 -Wall
126 -Wextra
127 -Wno-unused-parameter
128 -Wno-missing-field-initializers
129 -Wswitch
130 -Wsign-compare
131 -Wshadow
132 -Wformat
133 -Wtype-limits
134 -Wundef
135 -Werror );
136 %gcc_devteam_warn = (
137 CFLAGS => [ @common, qw( -Wmissing-prototypes
138 -Wstrict-prototypes ) ],
139 CXXFLAGS => [ @common ]
140 );
141 }
142
143 # These are used in addition to $gcc_devteam_warn when the compiler is clang.
144 # TODO(openssl-team): fix problems and investigate if (at least) the
145 # following warnings can also be enabled:
146 # -Wcast-align
147 # -Wunreachable-code -- no, too ugly/compiler-specific
148 # -Wlanguage-extension-token -- no, we use asm()
149 # -Wunused-macros -- no, too tricky for BN and _XOPEN_SOURCE etc
150 # -Wextended-offsetof -- no, needed in CMS ASN1 code
151 my %clang_devteam_warn = ();
152 {
153 my @common = qw( -Wswitch-default
154 -Wno-parentheses-equality
155 -Wno-language-extension-token
156 -Wno-extended-offsetof
157 -Wconditional-uninitialized
158 -Wincompatible-pointer-types-discards-qualifiers
159 -Wno-unknown-warning-option );
160 %clang_devteam_warn = (
161 CFLAGS => [ @common, qw( -Wmissing-variable-declarations ) ],
162 CXXFLAGS => [ @common ]
163 );
164 }
165
166 # This adds backtrace information to the memory leak info. Is only used
167 # when crypto-mdebug-backtrace is enabled.
168 my $memleak_devteam_backtrace = "-rdynamic";
169
170 my $strict_warnings = 0;
171
172 # As for $BSDthreads. Idea is to maintain "collective" set of flags,
173 # which would cover all BSD flavors. -pthread applies to them all,
174 # but is treated differently. OpenBSD expands is as -D_POSIX_THREAD
175 # -lc_r, which is sufficient. FreeBSD 4.x expands it as -lc_r,
176 # which has to be accompanied by explicit -D_THREAD_SAFE and
177 # sometimes -D_REENTRANT. FreeBSD 5.x expands it as -lc_r, which
178 # seems to be sufficient?
179 our $BSDthreads="-pthread -D_THREAD_SAFE -D_REENTRANT";
180
181 #
182 # API compatibility name to version number mapping.
183 #
184 my $maxapi = "3.0.0"; # API for "no-deprecated" builds
185 my $apitable = {
186 "3.0.0" => 3,
187 "1.1.1" => 2,
188 "1.1.0" => 2,
189 "1.0.2" => 1,
190 "1.0.1" => 1,
191 "1.0.0" => 1,
192 "0.9.8" => 0,
193 };
194
195 our %table = ();
196 our %config = ();
197 our %withargs = ();
198 our $now_printing; # set to current entry's name in print_table_entry
199 # (todo: right thing would be to encapsulate name
200 # into %target [class] and make print_table_entry
201 # a method)
202
203 # Forward declarations ###############################################
204
205 # read_config(filename)
206 #
207 # Reads a configuration file and populates %table with the contents
208 # (which the configuration file places in %targets).
209 sub read_config;
210
211 # resolve_config(target)
212 #
213 # Resolves all the late evaluations, inheritances and so on for the
214 # chosen target and any target it inherits from.
215 sub resolve_config;
216
217
218 # Information collection #############################################
219
220 # Unified build supports separate build dir
221 my $srcdir = catdir(absolutedir(dirname($0))); # catdir ensures local syntax
222 my $blddir = catdir(absolutedir(".")); # catdir ensures local syntax
223 my $dofile = abs2rel(catfile($srcdir, "util/dofile.pl"));
224
225 my $local_config_envname = 'OPENSSL_LOCAL_CONFIG_DIR';
226
227 $config{sourcedir} = abs2rel($srcdir);
228 $config{builddir} = abs2rel($blddir);
229
230 # Collect reconfiguration information if needed
231 my @argvcopy=@ARGV;
232
233 if (grep /^reconf(igure)?$/, @argvcopy) {
234 die "reconfiguring with other arguments present isn't supported"
235 if scalar @argvcopy > 1;
236 if (-f "./configdata.pm") {
237 my $file = "./configdata.pm";
238 unless (my $return = do $file) {
239 die "couldn't parse $file: $@" if $@;
240 die "couldn't do $file: $!" unless defined $return;
241 die "couldn't run $file" unless $return;
242 }
243
244 @argvcopy = defined($configdata::config{perlargv}) ?
245 @{$configdata::config{perlargv}} : ();
246 die "Incorrect data to reconfigure, please do a normal configuration\n"
247 if (grep(/^reconf/,@argvcopy));
248 $config{perlenv} = $configdata::config{perlenv} // {};
249 } else {
250 die "Insufficient data to reconfigure, please do a normal configuration\n";
251 }
252 }
253
254 $config{perlargv} = [ @argvcopy ];
255
256 # Collect version numbers
257 $config{major} = "unknown";
258 $config{minor} = "unknown";
259 $config{patch} = "unknown";
260 $config{prerelease} = "";
261 $config{build_metadata} = "";
262 $config{shlib_version} = "unknown";
263
264 collect_information(
265 collect_from_file(catfile($srcdir,'include/openssl/opensslv.h')),
266 qr/#\s+define\s+OPENSSL_VERSION_MAJOR\s+(\d+)/ =>
267 sub { $config{major} = $1; },
268 qr/#\s+define\s+OPENSSL_VERSION_MINOR\s+(\d+)/ =>
269 sub { $config{minor} = $1; },
270 qr/#\s+define\s+OPENSSL_VERSION_PATCH\s+(\d+)/ =>
271 sub { $config{patch} = $1; },
272 qr/#\s+define\s+OPENSSL_VERSION_PRE_RELEASE\s+"((?:\\.|[^"])*)"/ =>
273 sub { $config{prerelease} = $1; },
274 qr/#\s+define\s+OPENSSL_VERSION_BUILD_METADATA\s+"((?:\\.|[^"])*)"/ =>
275 sub { $config{build_metadata} = $1; },
276 qr/#\s+define\s+OPENSSL_SHLIB_VERSION\s+([\d\.]+)/ =>
277 sub { $config{shlib_version} = $1; },
278 );
279 die "erroneous version information in opensslv.h: ",
280 "$config{major}.$config{minor}.$config{patch}, $config{shlib_version}\n"
281 if ($config{major} eq "unknown"
282 || $config{minor} eq "unknown"
283 || $config{patch} eq "unknown"
284 || $config{shlib_version} eq "unknown");
285
286 $config{version} = "$config{major}.$config{minor}.$config{patch}";
287 $config{full_version} = "$config{version}$config{prerelease}$config{build_metadata}";
288
289 # Collect target configurations
290
291 my $pattern = catfile(dirname($0), "Configurations", "*.conf");
292 foreach (sort glob($pattern)) {
293 &read_config($_);
294 }
295
296 if (defined env($local_config_envname)) {
297 if ($^O eq 'VMS') {
298 # VMS environment variables are logical names,
299 # which can be used as is
300 $pattern = $local_config_envname . ':' . '*.conf';
301 } else {
302 $pattern = catfile(env($local_config_envname), '*.conf');
303 }
304
305 foreach (sort glob($pattern)) {
306 &read_config($_);
307 }
308 }
309
310 # Save away perl command information
311 $config{perl_cmd} = $^X;
312 $config{perl_version} = $Config{version};
313 $config{perl_archname} = $Config{archname};
314
315 $config{prefix}="";
316 $config{openssldir}="";
317 $config{processor}="";
318 $config{libdir}="";
319 my $auto_threads=1; # enable threads automatically? true by default
320 my $default_ranlib;
321
322 # Known TLS and DTLS protocols
323 my @tls = qw(ssl3 tls1 tls1_1 tls1_2 tls1_3);
324 my @dtls = qw(dtls1 dtls1_2);
325
326 # Explicitly known options that are possible to disable. They can
327 # be regexps, and will be used like this: /^no-${option}$/
328 # For developers: keep it sorted alphabetically
329
330 my @disablables = (
331 "ktls",
332 "afalgeng",
333 "aria",
334 "asan",
335 "asm",
336 "async",
337 "autoalginit",
338 "autoerrinit",
339 "autoload-config",
340 "bf",
341 "blake2",
342 "buildtest-c++",
343 "camellia",
344 "capieng",
345 "cast",
346 "chacha",
347 "cmac",
348 "cms",
349 "comp",
350 "crmf",
351 "crypto-mdebug",
352 "crypto-mdebug-backtrace",
353 "ct",
354 "deprecated",
355 "des",
356 "devcryptoeng",
357 "dgram",
358 "dh",
359 "dsa",
360 "dtls",
361 "dynamic-engine",
362 "ec",
363 "ec2m",
364 "ecdh",
365 "ecdsa",
366 "ec_nistp_64_gcc_128",
367 "egd",
368 "engine",
369 "err",
370 "external-tests",
371 "filenames",
372 "fips",
373 "fuzz-libfuzzer",
374 "fuzz-afl",
375 "gost",
376 "idea",
377 "legacy",
378 "makedepend",
379 "md2",
380 "md4",
381 "mdc2",
382 "module",
383 "msan",
384 "multiblock",
385 "nextprotoneg",
386 "pinshared",
387 "ocb",
388 "ocsp",
389 "padlockeng",
390 "pic",
391 "poly1305",
392 "posix-io",
393 "psk",
394 "rc2",
395 "rc4",
396 "rc5",
397 "rdrand",
398 "rfc3779",
399 "rmd160",
400 "scrypt",
401 "sctp",
402 "seed",
403 "shared",
404 "siphash",
405 "siv",
406 "sm2",
407 "sm3",
408 "sm4",
409 "sock",
410 "srp",
411 "srtp",
412 "sse2",
413 "ssl",
414 "ssl-trace",
415 "static-engine",
416 "stdio",
417 "tests",
418 "threads",
419 "tls",
420 "trace",
421 "ts",
422 "ubsan",
423 "ui-console",
424 "unit-test",
425 "whirlpool",
426 "weak-ssl-ciphers",
427 "zlib",
428 "zlib-dynamic",
429 );
430 foreach my $proto ((@tls, @dtls))
431 {
432 push(@disablables, $proto);
433 push(@disablables, "$proto-method") unless $proto eq "tls1_3";
434 }
435
436 my %deprecated_disablables = (
437 "ssl2" => undef,
438 "buf-freelists" => undef,
439 "hw" => "hw", # causes cascade, but no macro
440 "hw-padlock" => "padlockeng",
441 "ripemd" => "rmd160",
442 "ui" => "ui-console",
443 "dso" => undef,
444 "heartbeats" => undef,
445 );
446
447 # All of the following are disabled by default:
448
449 our %disabled = ( # "what" => "comment"
450 "asan" => "default",
451 "buildtest-c++" => "default",
452 "crypto-mdebug" => "default",
453 "crypto-mdebug-backtrace" => "default",
454 "devcryptoeng" => "default",
455 "ec_nistp_64_gcc_128" => "default",
456 "egd" => "default",
457 "external-tests" => "default",
458 "fuzz-libfuzzer" => "default",
459 "fuzz-afl" => "default",
460 "md2" => "default",
461 "msan" => "default",
462 "rc5" => "default",
463 "sctp" => "default",
464 "ssl-trace" => "default",
465 "ssl3" => "default",
466 "ssl3-method" => "default",
467 "trace" => "default",
468 "ubsan" => "default",
469 "unit-test" => "default",
470 "weak-ssl-ciphers" => "default",
471 "zlib" => "default",
472 "zlib-dynamic" => "default",
473 "ktls" => "default",
474 );
475
476 # Note: => pair form used for aesthetics, not to truly make a hash table
477 my @disable_cascades = (
478 # "what" => [ "cascade", ... ]
479 sub { $config{processor} eq "386" }
480 => [ "sse2" ],
481 "ssl" => [ "ssl3" ],
482 "ssl3-method" => [ "ssl3" ],
483 "zlib" => [ "zlib-dynamic" ],
484 "des" => [ "mdc2" ],
485 "ec" => [ "ecdsa", "ecdh" ],
486
487 "dgram" => [ "dtls", "sctp" ],
488 "sock" => [ "dgram" ],
489 "dtls" => [ @dtls ],
490 sub { 0 == scalar grep { !$disabled{$_} } @dtls }
491 => [ "dtls" ],
492
493 "tls" => [ @tls ],
494 sub { 0 == scalar grep { !$disabled{$_} } @tls }
495 => [ "tls" ],
496
497 "crypto-mdebug" => [ "crypto-mdebug-backtrace" ],
498
499 # If no modules, then no dynamic engines either
500 "module" => [ "dynamic-engine" ],
501
502 # Without shared libraries, dynamic engines aren't possible.
503 # This is due to them having to link with libcrypto and register features
504 # using the ENGINE functionality, and since that relies on global tables,
505 # those *have* to be exacty the same as the ones accessed from the app,
506 # which cannot be guaranteed if shared libraries aren't present.
507 # (note that even with shared libraries, both the app and dynamic engines
508 # must be linked with the same library)
509 "shared" => [ "dynamic-engine" ],
510 # Other modules don't necessarily have to link with libcrypto, so shared
511 # libraries do not have to be a condition to produce those.
512
513 # Without position independent code, there can be no shared libraries
514 # or modules.
515 "pic" => [ "shared", "module" ],
516
517 "module" => [ "fips", "legacy" ],
518
519 "engine" => [ grep /eng$/, @disablables ],
520 "hw" => [ "padlockeng" ],
521
522 # no-autoalginit is only useful when building non-shared
523 "autoalginit" => [ "shared", "apps" ],
524
525 "stdio" => [ "apps", "capieng", "egd" ],
526 "apps" => [ "tests" ],
527 "tests" => [ "external-tests" ],
528 "comp" => [ "zlib" ],
529 "ec" => [ "tls1_3", "sm2" ],
530 "sm3" => [ "sm2" ],
531 sub { !$disabled{"unit-test"} } => [ "heartbeats" ],
532
533 sub { !$disabled{"msan"} } => [ "asm" ],
534
535 sub { $disabled{cmac}; } => [ "siv" ],
536 "legacy" => [ "md2" ],
537 );
538
539 # Avoid protocol support holes. Also disable all versions below N, if version
540 # N is disabled while N+1 is enabled.
541 #
542 my @list = (reverse @tls);
543 while ((my $first, my $second) = (shift @list, shift @list)) {
544 last unless @list;
545 push @disable_cascades, ( sub { !$disabled{$first} && $disabled{$second} }
546 => [ @list ] );
547 unshift @list, $second;
548 }
549 my @list = (reverse @dtls);
550 while ((my $first, my $second) = (shift @list, shift @list)) {
551 last unless @list;
552 push @disable_cascades, ( sub { !$disabled{$first} && $disabled{$second} }
553 => [ @list ] );
554 unshift @list, $second;
555 }
556
557 # Explicit "no-..." options will be collected in %disabled along with the defaults.
558 # To remove something from %disabled, use "enable-foo".
559 # For symmetry, "disable-foo" is a synonym for "no-foo".
560
561 &usage if ($#ARGV < 0);
562
563 # For the "make variables" CINCLUDES and CDEFINES, we support lists with
564 # platform specific list separators. Users from those platforms should
565 # recognise those separators from how you set up the PATH to find executables.
566 # The default is the Unix like separator, :, but as an exception, we also
567 # support the space as separator.
568 my $list_separator_re =
569 { VMS => qr/(?<!\^),/,
570 MSWin32 => qr/(?<!\\);/ } -> {$^O} // qr/(?<!\\)[:\s]/;
571 # All the "make variables" we support
572 # Some get pre-populated for the sake of backward compatibility
573 # (we supported those before the change to "make variable" support.
574 my %user = (
575 AR => env('AR'),
576 ARFLAGS => [],
577 AS => undef,
578 ASFLAGS => [],
579 CC => env('CC'),
580 CFLAGS => [ env('CFLAGS') || () ],
581 CXX => env('CXX'),
582 CXXFLAGS => [ env('CXXFLAGS') || () ],
583 CPP => undef,
584 CPPFLAGS => [ env('CPPFLAGS') || () ], # -D, -I, -Wp,
585 CPPDEFINES => [], # Alternative for -D
586 CPPINCLUDES => [], # Alternative for -I
587 CROSS_COMPILE => env('CROSS_COMPILE'),
588 HASHBANGPERL=> env('HASHBANGPERL') || env('PERL'),
589 LD => undef,
590 LDFLAGS => [ env('LDFLAGS') || () ], # -L, -Wl,
591 LDLIBS => [ env('LDLIBS') || () ], # -l
592 MT => undef,
593 MTFLAGS => [],
594 PERL => env('PERL') || ($^O ne "VMS" ? $^X : "perl"),
595 RANLIB => env('RANLIB'),
596 RC => env('RC') || env('WINDRES'),
597 RCFLAGS => [],
598 RM => undef,
599 );
600 # Info about what "make variables" may be prefixed with the cross compiler
601 # prefix. This should NEVER mention any such variable with a list for value.
602 my @user_crossable = qw ( AR AS CC CXX CPP LD MT RANLIB RC );
603 # The same but for flags given as Configure options. These are *additional*
604 # input, as opposed to the VAR=string option that override the corresponding
605 # config target attributes
606 my %useradd = (
607 CPPDEFINES => [],
608 CPPINCLUDES => [],
609 CPPFLAGS => [],
610 CFLAGS => [],
611 CXXFLAGS => [],
612 LDFLAGS => [],
613 LDLIBS => [],
614 );
615
616 my %user_synonyms = (
617 HASHBANGPERL=> 'PERL',
618 RC => 'WINDRES',
619 );
620
621 # Some target attributes have been renamed, this is the translation table
622 my %target_attr_translate =(
623 ar => 'AR',
624 as => 'AS',
625 cc => 'CC',
626 cxx => 'CXX',
627 cpp => 'CPP',
628 hashbangperl => 'HASHBANGPERL',
629 ld => 'LD',
630 mt => 'MT',
631 ranlib => 'RANLIB',
632 rc => 'RC',
633 rm => 'RM',
634 );
635
636 # Initialisers coming from 'config' scripts
637 $config{defines} = [ split(/$list_separator_re/, env('__CNF_CPPDEFINES')) ];
638 $config{includes} = [ split(/$list_separator_re/, env('__CNF_CPPINCLUDES')) ];
639 $config{cppflags} = [ env('__CNF_CPPFLAGS') || () ];
640 $config{cflags} = [ env('__CNF_CFLAGS') || () ];
641 $config{cxxflags} = [ env('__CNF_CXXFLAGS') || () ];
642 $config{lflags} = [ env('__CNF_LDFLAGS') || () ];
643 $config{ex_libs} = [ env('__CNF_LDLIBS') || () ];
644
645 $config{openssl_api_defines}=[];
646 $config{openssl_sys_defines}=[];
647 $config{openssl_feature_defines}=[];
648 $config{options}="";
649 $config{build_type} = "release";
650 my $target="";
651
652 my %cmdvars = (); # Stores FOO='blah' type arguments
653 my %unsupported_options = ();
654 my %deprecated_options = ();
655 # If you change this, update apps/version.c
656 my @known_seed_sources = qw(getrandom devrandom os egd none rdcpu librandom);
657 my @seed_sources = ();
658 while (@argvcopy)
659 {
660 $_ = shift @argvcopy;
661
662 # Support env variable assignments among the options
663 if (m|^(\w+)=(.+)?$|)
664 {
665 $cmdvars{$1} = $2;
666 # Every time a variable is given as a configuration argument,
667 # it acts as a reset if the variable.
668 if (exists $user{$1})
669 {
670 $user{$1} = ref $user{$1} eq "ARRAY" ? [] : undef;
671 }
672 #if (exists $useradd{$1})
673 # {
674 # $useradd{$1} = [];
675 # }
676 next;
677 }
678
679 # VMS is a case insensitive environment, and depending on settings
680 # out of our control, we may receive options uppercased. Let's
681 # downcase at least the part before any equal sign.
682 if ($^O eq "VMS")
683 {
684 s/^([^=]*)/lc($1)/e;
685 }
686
687 # some people just can't read the instructions, clang people have to...
688 s/^-no-(?!integrated-as)/no-/;
689
690 # rewrite some options in "enable-..." form
691 s /^-?-?shared$/enable-shared/;
692 s /^sctp$/enable-sctp/;
693 s /^threads$/enable-threads/;
694 s /^zlib$/enable-zlib/;
695 s /^zlib-dynamic$/enable-zlib-dynamic/;
696
697 if (/^(no|disable|enable)-(.+)$/)
698 {
699 my $word = $2;
700 if ($word !~ m|hw(?:-.+)| # special treatment for hw regexp opt
701 && !exists $deprecated_disablables{$word}
702 && !grep { $word eq $_ } @disablables)
703 {
704 $unsupported_options{$_} = 1;
705 next;
706 }
707 }
708 if (/^no-(.+)$/ || /^disable-(.+)$/)
709 {
710 foreach my $proto ((@tls, @dtls))
711 {
712 if ($1 eq "$proto-method")
713 {
714 $disabled{"$proto"} = "option($proto-method)";
715 last;
716 }
717 }
718 if ($1 eq "dtls")
719 {
720 foreach my $proto (@dtls)
721 {
722 $disabled{$proto} = "option(dtls)";
723 }
724 $disabled{"dtls"} = "option(dtls)";
725 }
726 elsif ($1 eq "ssl")
727 {
728 # Last one of its kind
729 $disabled{"ssl3"} = "option(ssl)";
730 }
731 elsif ($1 eq "tls")
732 {
733 # XXX: Tests will fail if all SSL/TLS
734 # protocols are disabled.
735 foreach my $proto (@tls)
736 {
737 $disabled{$proto} = "option(tls)";
738 }
739 }
740 elsif ($1 eq "static-engine")
741 {
742 delete $disabled{"dynamic-engine"};
743 }
744 elsif ($1 eq "dynamic-engine")
745 {
746 $disabled{"dynamic-engine"} = "option";
747 }
748 elsif (exists $deprecated_disablables{$1})
749 {
750 $deprecated_options{$_} = 1;
751 if (defined $deprecated_disablables{$1})
752 {
753 $disabled{$deprecated_disablables{$1}} = "option";
754 }
755 }
756 elsif ($1 =~ m|hw(?:-.+)|) # deprecate hw options in regexp form
757 {
758 $deprecated_options{$_} = 1;
759 }
760 else
761 {
762 $disabled{$1} = "option";
763 }
764 # No longer an automatic choice
765 $auto_threads = 0 if ($1 eq "threads");
766 }
767 elsif (/^enable-(.+)$/)
768 {
769 if ($1 eq "static-engine")
770 {
771 $disabled{"dynamic-engine"} = "option";
772 }
773 elsif ($1 eq "dynamic-engine")
774 {
775 delete $disabled{"dynamic-engine"};
776 }
777 elsif ($1 eq "zlib-dynamic")
778 {
779 delete $disabled{"zlib"};
780 }
781 my $algo = $1;
782 delete $disabled{$algo};
783
784 # No longer an automatic choice
785 $auto_threads = 0 if ($1 eq "threads");
786 }
787 elsif (/^--strict-warnings$/)
788 {
789 # Pretend that our strict flags is a C flag, and replace it
790 # with the proper flags later on
791 push @{$useradd{CFLAGS}}, '--ossl-strict-warnings';
792 push @{$useradd{CXXFLAGS}}, '--ossl-strict-warnings';
793 $strict_warnings=1;
794 }
795 elsif (/^--debug$/)
796 {
797 $config{build_type} = "debug";
798 }
799 elsif (/^--release$/)
800 {
801 $config{build_type} = "release";
802 }
803 elsif (/^386$/)
804 { $config{processor}=386; }
805 elsif (/^fips$/)
806 {
807 die "FIPS mode not supported\n";
808 }
809 elsif (/^rsaref$/)
810 {
811 # No RSAref support any more since it's not needed.
812 # The check for the option is there so scripts aren't
813 # broken
814 }
815 elsif (/^nofipscanistercheck$/)
816 {
817 die "FIPS mode not supported\n";
818 }
819 elsif (/^[-+]/)
820 {
821 if (/^--prefix=(.*)$/)
822 {
823 $config{prefix}=$1;
824 die "Directory given with --prefix MUST be absolute\n"
825 unless file_name_is_absolute($config{prefix});
826 }
827 elsif (/^--api=(.*)$/)
828 {
829 $config{api}=$1;
830 }
831 elsif (/^--libdir=(.*)$/)
832 {
833 $config{libdir}=$1;
834 }
835 elsif (/^--openssldir=(.*)$/)
836 {
837 $config{openssldir}=$1;
838 }
839 elsif (/^--with-zlib-lib=(.*)$/)
840 {
841 $withargs{zlib_lib}=$1;
842 }
843 elsif (/^--with-zlib-include=(.*)$/)
844 {
845 $withargs{zlib_include}=$1;
846 }
847 elsif (/^--with-fuzzer-lib=(.*)$/)
848 {
849 $withargs{fuzzer_lib}=$1;
850 }
851 elsif (/^--with-fuzzer-include=(.*)$/)
852 {
853 $withargs{fuzzer_include}=$1;
854 }
855 elsif (/^--with-rand-seed=(.*)$/)
856 {
857 foreach my $x (split(m|,|, $1))
858 {
859 die "Unknown --with-rand-seed choice $x\n"
860 if ! grep { $x eq $_ } @known_seed_sources;
861 push @seed_sources, $x;
862 }
863 }
864 elsif (/^--cross-compile-prefix=(.*)$/)
865 {
866 $user{CROSS_COMPILE}=$1;
867 }
868 elsif (/^--config=(.*)$/)
869 {
870 read_config $1;
871 }
872 elsif (/^-l(.*)$/)
873 {
874 push @{$useradd{LDLIBS}}, $_;
875 }
876 elsif (/^-framework$/)
877 {
878 push @{$useradd{LDLIBS}}, $_, shift(@argvcopy);
879 }
880 elsif (/^-L(.*)$/ or /^-Wl,/)
881 {
882 push @{$useradd{LDFLAGS}}, $_;
883 }
884 elsif (/^-rpath$/ or /^-R$/)
885 # -rpath is the OSF1 rpath flag
886 # -R is the old Solaris rpath flag
887 {
888 my $rpath = shift(@argvcopy) || "";
889 $rpath .= " " if $rpath ne "";
890 push @{$useradd{LDFLAGS}}, $_, $rpath;
891 }
892 elsif (/^-static$/)
893 {
894 push @{$useradd{LDFLAGS}}, $_;
895 $disabled{"pic"} = "forced";
896 $disabled{"shared"} = "forced";
897 $disabled{"threads"} = "forced";
898 }
899 elsif (/^-D(.*)$/)
900 {
901 push @{$useradd{CPPDEFINES}}, $1;
902 }
903 elsif (/^-I(.*)$/)
904 {
905 push @{$useradd{CPPINCLUDES}}, $1;
906 }
907 elsif (/^-Wp,$/)
908 {
909 push @{$useradd{CPPFLAGS}}, $1;
910 }
911 else # common if (/^[-+]/), just pass down...
912 {
913 $_ =~ s/%([0-9a-f]{1,2})/chr(hex($1))/gei;
914 push @{$useradd{CFLAGS}}, $_;
915 push @{$useradd{CXXFLAGS}}, $_;
916 }
917 }
918 else
919 {
920 die "target already defined - $target (offending arg: $_)\n" if ($target ne "");
921 $target=$_;
922 }
923 unless ($_ eq $target || /^no-/ || /^disable-/)
924 {
925 # "no-..." follows later after implied deactivations
926 # have been derived. (Don't take this too seriously,
927 # we really only write OPTIONS to the Makefile out of
928 # nostalgia.)
929
930 if ($config{options} eq "")
931 { $config{options} = $_; }
932 else
933 { $config{options} .= " ".$_; }
934 }
935 }
936
937 if (defined($config{api}) && !exists $apitable->{$config{api}}) {
938 die "***** Unsupported api compatibility level: $config{api}\n",
939 }
940
941 if (keys %deprecated_options)
942 {
943 warn "***** Deprecated options: ",
944 join(", ", keys %deprecated_options), "\n";
945 }
946 if (keys %unsupported_options)
947 {
948 die "***** Unsupported options: ",
949 join(", ", keys %unsupported_options), "\n";
950 }
951
952 # If any %useradd entry has been set, we must check that the "make
953 # variables" haven't been set. We start by checking of any %useradd entry
954 # is set.
955 if (grep { scalar @$_ > 0 } values %useradd) {
956 # Hash of env / make variables names. The possible values are:
957 # 1 - "make vars"
958 # 2 - %useradd entry set
959 # 3 - both set
960 my %detected_vars =
961 map { my $v = 0;
962 $v += 1 if $cmdvars{$_};
963 $v += 2 if @{$useradd{$_}};
964 $_ => $v }
965 keys %useradd;
966
967 # If any of the corresponding "make variables" is set, we error
968 if (grep { $_ & 1 } values %detected_vars) {
969 my $names = join(', ', grep { $detected_vars{$_} > 0 }
970 sort keys %detected_vars);
971 die <<"_____";
972 ***** Mixing make variables and additional compiler/linker flags as
973 ***** configure command line option is not permitted.
974 ***** Affected make variables: $names
975 _____
976 }
977 }
978
979 # Check through all supported command line variables to see if any of them
980 # were set, and canonicalise the values we got. If no compiler or linker
981 # flag or anything else that affects %useradd was set, we also check the
982 # environment for values.
983 my $anyuseradd =
984 grep { defined $_ && (ref $_ ne 'ARRAY' || @$_) } values %useradd;
985 foreach (keys %user) {
986 my $value = $cmdvars{$_};
987 $value //= env($_) unless $anyuseradd;
988 $value //=
989 defined $user_synonyms{$_} ? $cmdvars{$user_synonyms{$_}} : undef;
990 $value //= defined $user_synonyms{$_} ? env($user_synonyms{$_}) : undef
991 unless $anyuseradd;
992
993 if (defined $value) {
994 if (ref $user{$_} eq 'ARRAY') {
995 $user{$_} = [ split /$list_separator_re/, $value ];
996 } elsif (!defined $user{$_}) {
997 $user{$_} = $value;
998 }
999 }
1000 }
1001
1002 if (grep { /-rpath\b/ } ($user{LDFLAGS} ? @{$user{LDFLAGS}} : ())
1003 && !$disabled{shared}
1004 && !($disabled{asan} && $disabled{msan} && $disabled{ubsan})) {
1005 die "***** Cannot simultaneously use -rpath, shared libraries, and\n",
1006 "***** any of asan, msan or ubsan\n";
1007 }
1008
1009 my @tocheckfor = (keys %disabled);
1010 while (@tocheckfor) {
1011 my %new_tocheckfor = ();
1012 my @cascade_copy = (@disable_cascades);
1013 while (@cascade_copy) {
1014 my ($test, $descendents) = (shift @cascade_copy, shift @cascade_copy);
1015 if (ref($test) eq "CODE" ? $test->() : defined($disabled{$test})) {
1016 foreach(grep { !defined($disabled{$_}) } @$descendents) {
1017 $new_tocheckfor{$_} = 1; $disabled{$_} = "forced";
1018 }
1019 }
1020 }
1021 @tocheckfor = (keys %new_tocheckfor);
1022 }
1023
1024 our $die = sub { die @_; };
1025 if ($target eq "TABLE") {
1026 local $die = sub { warn @_; };
1027 foreach (sort keys %table) {
1028 print_table_entry($_, "TABLE");
1029 }
1030 exit 0;
1031 }
1032
1033 if ($target eq "LIST") {
1034 foreach (sort keys %table) {
1035 print $_,"\n" unless $table{$_}->{template};
1036 }
1037 exit 0;
1038 }
1039
1040 if ($target eq "HASH") {
1041 local $die = sub { warn @_; };
1042 print "%table = (\n";
1043 foreach (sort keys %table) {
1044 print_table_entry($_, "HASH");
1045 }
1046 exit 0;
1047 }
1048
1049 print "Configuring OpenSSL version $config{full_version} ";
1050 print "for target $target\n";
1051
1052 if (scalar(@seed_sources) == 0) {
1053 print "Using os-specific seed configuration\n";
1054 push @seed_sources, 'os';
1055 }
1056 if (scalar(grep { $_ eq 'none' } @seed_sources) > 0) {
1057 die "Cannot seed with none and anything else" if scalar(@seed_sources) > 1;
1058 warn <<_____ if scalar(@seed_sources) == 1;
1059
1060 ============================== WARNING ===============================
1061 You have selected the --with-rand-seed=none option, which effectively
1062 disables automatic reseeding of the OpenSSL random generator.
1063 All operations depending on the random generator such as creating keys
1064 will not work unless the random generator is seeded manually by the
1065 application.
1066
1067 Please read the 'Note on random number generation' section in the
1068 INSTALL instructions and the RAND_DRBG(7) manual page for more details.
1069 ============================== WARNING ===============================
1070
1071 _____
1072 }
1073 push @{$config{openssl_feature_defines}},
1074 map { (my $x = $_) =~ tr|[\-a-z]|[_A-Z]|; "OPENSSL_RAND_SEED_$x" }
1075 @seed_sources;
1076
1077 # Backward compatibility?
1078 if ($target =~ m/^CygWin32(-.*)$/) {
1079 $target = "Cygwin".$1;
1080 }
1081
1082 # Support for legacy targets having a name starting with 'debug-'
1083 my ($d, $t) = $target =~ m/^(debug-)?(.*)$/;
1084 if ($d) {
1085 $config{build_type} = "debug";
1086
1087 # If we do not find debug-foo in the table, the target is set to foo.
1088 if (!$table{$target}) {
1089 $target = $t;
1090 }
1091 }
1092
1093 &usage if !$table{$target} || $table{$target}->{template};
1094
1095 $config{target} = $target;
1096 my %target = resolve_config($target);
1097
1098 foreach (keys %target_attr_translate) {
1099 $target{$target_attr_translate{$_}} = $target{$_}
1100 if $target{$_};
1101 delete $target{$_};
1102 }
1103
1104 %target = ( %{$table{DEFAULTS}}, %target );
1105
1106 # Make the flags to build DSOs the same as for shared libraries unless they
1107 # are already defined
1108 $target{module_cflags} = $target{shared_cflag} unless defined $target{module_cflags};
1109 $target{module_cxxflags} = $target{shared_cxxflag} unless defined $target{module_cxxflags};
1110 $target{module_ldflags} = $target{shared_ldflag} unless defined $target{module_ldflags};
1111 {
1112 my $shared_info_pl =
1113 catfile(dirname($0), "Configurations", "shared-info.pl");
1114 my %shared_info = read_eval_file($shared_info_pl);
1115 push @{$target{_conf_fname_int}}, $shared_info_pl;
1116 my $si = $target{shared_target};
1117 while (ref $si ne "HASH") {
1118 last if ! defined $si;
1119 if (ref $si eq "CODE") {
1120 $si = $si->();
1121 } else {
1122 $si = $shared_info{$si};
1123 }
1124 }
1125
1126 # Some of the 'shared_target' values don't have any entried in
1127 # %shared_info. That's perfectly fine, AS LONG AS the build file
1128 # template knows how to handle this. That is currently the case for
1129 # Windows and VMS.
1130 if (defined $si) {
1131 # Just as above, copy certain shared_* attributes to the corresponding
1132 # module_ attribute unless the latter is already defined
1133 $si->{module_cflags} = $si->{shared_cflag} unless defined $si->{module_cflags};
1134 $si->{module_cxxflags} = $si->{shared_cxxflag} unless defined $si->{module_cxxflags};
1135 $si->{module_ldflags} = $si->{shared_ldflag} unless defined $si->{module_ldflags};
1136 foreach (sort keys %$si) {
1137 $target{$_} = defined $target{$_}
1138 ? add($si->{$_})->($target{$_})
1139 : $si->{$_};
1140 }
1141 }
1142 }
1143
1144 my %conf_files = map { $_ => 1 } (@{$target{_conf_fname_int}});
1145 $config{conf_files} = [ sort keys %conf_files ];
1146
1147 foreach my $feature (@{$target{disable}}) {
1148 if (exists $deprecated_disablables{$feature}) {
1149 warn "***** config $target disables deprecated feature $feature\n";
1150 } elsif (!grep { $feature eq $_ } @disablables) {
1151 die "***** config $target disables unknown feature $feature\n";
1152 }
1153 $disabled{$feature} = 'config';
1154 }
1155 foreach my $feature (@{$target{enable}}) {
1156 if ("default" eq ($disabled{$feature} // "")) {
1157 if (exists $deprecated_disablables{$feature}) {
1158 warn "***** config $target enables deprecated feature $feature\n";
1159 } elsif (!grep { $feature eq $_ } @disablables) {
1160 die "***** config $target enables unknown feature $feature\n";
1161 }
1162 delete $disabled{$feature};
1163 }
1164 }
1165
1166 $target{CXXFLAGS}//=$target{CFLAGS} if $target{CXX};
1167 $target{cxxflags}//=$target{cflags} if $target{CXX};
1168 $target{exe_extension}=".exe" if ($config{target} eq "DJGPP");
1169 $target{exe_extension}=".pm" if ($config{target} =~ /vos/);
1170
1171 # Fill %config with values from %user, and in case those are undefined or
1172 # empty, use values from %target (acting as a default).
1173 foreach (keys %user) {
1174 my $ref_type = ref $user{$_};
1175
1176 # Temporary function. Takes an intended ref type (empty string or "ARRAY")
1177 # and a value that's to be coerced into that type.
1178 my $mkvalue = sub {
1179 my $type = shift;
1180 my $value = shift;
1181 my $undef_p = shift;
1182
1183 die "Too many arguments for \$mkvalue" if @_;
1184
1185 while (ref $value eq 'CODE') {
1186 $value = $value->();
1187 }
1188
1189 if ($type eq 'ARRAY') {
1190 return undef unless defined $value;
1191 return undef if ref $value ne 'ARRAY' && !$value;
1192 return undef if ref $value eq 'ARRAY' && !@$value;
1193 return [ $value ] unless ref $value eq 'ARRAY';
1194 }
1195 return undef unless $value;
1196 return $value;
1197 };
1198
1199 $config{$_} =
1200 $mkvalue->($ref_type, $user{$_})
1201 || $mkvalue->($ref_type, $target{$_});
1202 delete $config{$_} unless defined $config{$_};
1203 }
1204
1205 # Allow overriding the build file name
1206 $config{build_file} = env('BUILDFILE') || $target{build_file} || "Makefile";
1207
1208 ######################################################################
1209 # Build up information for skipping certain directories depending on disabled
1210 # features, as well as setting up macros for disabled features.
1211
1212 # This is a tentative database of directories to skip. Some entries may not
1213 # correspond to anything real, but that's ok, they will simply be ignored.
1214 # The actual processing of these entries is done in the build.info lookup
1215 # loop further down.
1216 #
1217 # The key is a Unix formated path in the source tree, the value is an index
1218 # into %disabled_info, so any existing path gets added to a corresponding
1219 # 'skipped' entry in there with the list of skipped directories.
1220 my %skipdir = ();
1221 my %disabled_info = (); # For configdata.pm
1222 foreach my $what (sort keys %disabled) {
1223 # There are deprecated disablables that translate to themselves.
1224 # They cause disabling cascades, but should otherwise not regiter.
1225 next if $deprecated_disablables{$what};
1226
1227 $config{options} .= " no-$what";
1228
1229 if (!grep { $what eq $_ } ( 'buildtest-c++', 'fips', 'threads', 'shared',
1230 'module', 'pic', 'dynamic-engine', 'makedepend',
1231 'zlib-dynamic', 'zlib', 'sse2', 'legacy' )) {
1232 (my $WHAT = uc $what) =~ s|-|_|g;
1233 my $skipdir = $what;
1234
1235 # fix-up crypto/directory name(s)
1236 $skipdir = "ripemd" if $what eq "rmd160";
1237 $skipdir = "whrlpool" if $what eq "whirlpool";
1238
1239 my $macro = $disabled_info{$what}->{macro} = "OPENSSL_NO_$WHAT";
1240 push @{$config{openssl_feature_defines}}, $macro;
1241
1242 $skipdir{engines} = $what if $what eq 'engine';
1243 $skipdir{"crypto/$skipdir"} = $what
1244 unless $what eq 'async' || $what eq 'err';
1245 }
1246 }
1247
1248 # Make sure build_scheme is consistent.
1249 $target{build_scheme} = [ $target{build_scheme} ]
1250 if ref($target{build_scheme}) ne "ARRAY";
1251
1252 my ($builder, $builder_platform, @builder_opts) =
1253 @{$target{build_scheme}};
1254
1255 foreach my $checker (($builder_platform."-".$target{build_file}."-checker.pm",
1256 $builder_platform."-checker.pm")) {
1257 my $checker_path = catfile($srcdir, "Configurations", $checker);
1258 if (-f $checker_path) {
1259 my $fn = $ENV{CONFIGURE_CHECKER_WARN}
1260 ? sub { warn $@; } : sub { die $@; };
1261 if (! do $checker_path) {
1262 if ($@) {
1263 $fn->($@);
1264 } elsif ($!) {
1265 $fn->($!);
1266 } else {
1267 $fn->("The detected tools didn't match the platform\n");
1268 }
1269 }
1270 last;
1271 }
1272 }
1273
1274 push @{$config{defines}}, "NDEBUG" if $config{build_type} eq "release";
1275
1276 if ($target =~ /^mingw/ && `$config{CC} --target-help 2>&1` =~ m/-mno-cygwin/m)
1277 {
1278 push @{$config{cflags}}, "-mno-cygwin";
1279 push @{$config{cxxflags}}, "-mno-cygwin" if $config{CXX};
1280 push @{$config{shared_ldflag}}, "-mno-cygwin";
1281 }
1282
1283 if ($target =~ /linux.*-mips/ && !$disabled{asm}
1284 && !grep { $_ !~ /-m(ips|arch=)/ } (@{$user{CFLAGS}},
1285 @{$useradd{CFLAGS}})) {
1286 # minimally required architecture flags for assembly modules
1287 my $value;
1288 $value = '-mips2' if ($target =~ /mips32/);
1289 $value = '-mips3' if ($target =~ /mips64/);
1290 unshift @{$config{cflags}}, $value;
1291 unshift @{$config{cxxflags}}, $value if $config{CXX};
1292 }
1293
1294 # If threads aren't disabled, check how possible they are
1295 unless ($disabled{threads}) {
1296 if ($auto_threads) {
1297 # Enabled by default, disable it forcibly if unavailable
1298 if ($target{thread_scheme} eq "(unknown)") {
1299 $disabled{threads} = "unavailable";
1300 }
1301 } else {
1302 # The user chose to enable threads explicitly, let's see
1303 # if there's a chance that's possible
1304 if ($target{thread_scheme} eq "(unknown)") {
1305 # If the user asked for "threads" and we don't have internal
1306 # knowledge how to do it, [s]he is expected to provide any
1307 # system-dependent compiler options that are necessary. We
1308 # can't truly check that the given options are correct, but
1309 # we expect the user to know what [s]He is doing.
1310 if (!@{$user{CFLAGS}} && !@{$useradd{CFLAGS}}
1311 && !@{$user{CPPDEFINES}} && !@{$useradd{CPPDEFINES}}) {
1312 die "You asked for multi-threading support, but didn't\n"
1313 ,"provide any system-specific compiler options\n";
1314 }
1315 }
1316 }
1317 }
1318
1319 # If threads still aren't disabled, add a C macro to ensure the source
1320 # code knows about it. Any other flag is taken care of by the configs.
1321 unless($disabled{threads}) {
1322 push @{$config{openssl_feature_defines}}, "OPENSSL_THREADS";
1323 }
1324
1325 # With "deprecated" disable all deprecated features.
1326 if (defined($disabled{"deprecated"})) {
1327 $config{api} = $maxapi;
1328 }
1329
1330 my $no_shared_warn=0;
1331 if ($target{shared_target} eq "")
1332 {
1333 $no_shared_warn = 1
1334 if (!$disabled{shared} || !$disabled{"dynamic-engine"});
1335 $disabled{pic} = $disabled{shared} = $disabled{"dynamic-engine"} =
1336 $disabled{module} = "no-shared-target";
1337 }
1338
1339 if ($disabled{"dynamic-engine"}) {
1340 push @{$config{openssl_feature_defines}}, "OPENSSL_NO_DYNAMIC_ENGINE";
1341 $config{dynamic_engines} = 0;
1342 } else {
1343 push @{$config{openssl_feature_defines}}, "OPENSSL_NO_STATIC_ENGINE";
1344 $config{dynamic_engines} = 1;
1345 }
1346
1347 unless ($disabled{asan}) {
1348 push @{$config{cflags}}, "-fsanitize=address";
1349 push @{$config{cxxflags}}, "-fsanitize=address" if $config{CXX};
1350 }
1351
1352 unless ($disabled{ubsan}) {
1353 # -DPEDANTIC or -fnosanitize=alignment may also be required on some
1354 # platforms.
1355 push @{$config{cflags}}, "-fsanitize=undefined", "-fno-sanitize-recover=all";
1356 push @{$config{cxxflags}}, "-fsanitize=undefined", "-fno-sanitize-recover=all"
1357 if $config{CXX};
1358 }
1359
1360 unless ($disabled{msan}) {
1361 push @{$config{cflags}}, "-fsanitize=memory";
1362 push @{$config{cxxflags}}, "-fsanitize=memory" if $config{CXX};
1363 }
1364
1365 unless ($disabled{"fuzz-libfuzzer"} && $disabled{"fuzz-afl"}
1366 && $disabled{asan} && $disabled{ubsan} && $disabled{msan}) {
1367 push @{$config{cflags}}, "-fno-omit-frame-pointer", "-g";
1368 push @{$config{cxxflags}}, "-fno-omit-frame-pointer", "-g" if $config{CXX};
1369 }
1370 #
1371 # Platform fix-ups
1372 #
1373
1374 # This saves the build files from having to check
1375 if ($disabled{pic})
1376 {
1377 foreach (qw(shared_cflag shared_cxxflag shared_cppflag
1378 shared_defines shared_includes shared_ldflag
1379 module_cflags module_cxxflags module_cppflags
1380 module_defines module_includes module_lflags))
1381 {
1382 delete $config{$_};
1383 $target{$_} = "";
1384 }
1385 }
1386 else
1387 {
1388 push @{$config{lib_defines}}, "OPENSSL_PIC";
1389 }
1390
1391 if ($target{sys_id} ne "")
1392 {
1393 push @{$config{openssl_sys_defines}}, "OPENSSL_SYS_$target{sys_id}";
1394 }
1395
1396 unless ($disabled{asm}) {
1397 $target{cpuid_asm_src}=$table{DEFAULTS}->{cpuid_asm_src} if ($config{processor} eq "386");
1398 push @{$config{lib_defines}}, "OPENSSL_CPUID_OBJ" if ($target{cpuid_asm_src} ne "mem_clr.c");
1399
1400 $target{bn_asm_src} =~ s/\w+-gf2m.c// if (defined($disabled{ec2m}));
1401
1402 # bn-586 is the only one implementing bn_*_part_words
1403 push @{$config{lib_defines}}, "OPENSSL_BN_ASM_PART_WORDS" if ($target{bn_asm_src} =~ /bn-586/);
1404 push @{$config{lib_defines}}, "OPENSSL_IA32_SSE2" if (!$disabled{sse2} && $target{bn_asm_src} =~ /86/);
1405
1406 push @{$config{lib_defines}}, "OPENSSL_BN_ASM_MONT" if ($target{bn_asm_src} =~ /-mont/);
1407 push @{$config{lib_defines}}, "OPENSSL_BN_ASM_MONT5" if ($target{bn_asm_src} =~ /-mont5/);
1408 push @{$config{lib_defines}}, "OPENSSL_BN_ASM_GF2m" if ($target{bn_asm_src} =~ /-gf2m/);
1409 push @{$config{lib_defines}}, "BN_DIV3W" if ($target{bn_asm_src} =~ /-div3w/);
1410
1411 if ($target{sha1_asm_src}) {
1412 push @{$config{lib_defines}}, "SHA1_ASM" if ($target{sha1_asm_src} =~ /sx86/ || $target{sha1_asm_src} =~ /sha1/);
1413 push @{$config{lib_defines}}, "SHA256_ASM" if ($target{sha1_asm_src} =~ /sha256/);
1414 push @{$config{lib_defines}}, "SHA512_ASM" if ($target{sha1_asm_src} =~ /sha512/);
1415 }
1416 if ($target{keccak1600_asm_src} ne $table{DEFAULTS}->{keccak1600_asm_src}) {
1417 push @{$config{lib_defines}}, "KECCAK1600_ASM";
1418 }
1419 if ($target{rc4_asm_src} ne $table{DEFAULTS}->{rc4_asm_src}) {
1420 push @{$config{lib_defines}}, "RC4_ASM";
1421 }
1422 if ($target{md5_asm_src}) {
1423 push @{$config{lib_defines}}, "MD5_ASM";
1424 }
1425 $target{cast_asm_src}=$table{DEFAULTS}->{cast_asm_src} unless $disabled{pic}; # CAST assembler is not PIC
1426 if ($target{rmd160_asm_src}) {
1427 push @{$config{lib_defines}}, "RMD160_ASM";
1428 }
1429 if ($target{aes_asm_src}) {
1430 push @{$config{lib_defines}}, "AES_ASM" if ($target{aes_asm_src} =~ m/\baes-/);;
1431 # aes-ctr.fake is not a real file, only indication that assembler
1432 # module implements AES_ctr32_encrypt...
1433 push @{$config{lib_defines}}, "AES_CTR_ASM" if ($target{aes_asm_src} =~ s/\s*aes-ctr\.fake//);
1434 # aes-xts.fake indicates presence of AES_xts_[en|de]crypt...
1435 push @{$config{lib_defines}}, "AES_XTS_ASM" if ($target{aes_asm_src} =~ s/\s*aes-xts\.fake//);
1436 $target{aes_asm_src} =~ s/\s*(vpaes|aesni)-x86\.s//g if ($disabled{sse2});
1437 push @{$config{lib_defines}}, "VPAES_ASM" if ($target{aes_asm_src} =~ m/vpaes/);
1438 push @{$config{lib_defines}}, "BSAES_ASM" if ($target{aes_asm_src} =~ m/bsaes/);
1439 }
1440 if ($target{wp_asm_src} =~ /mmx/) {
1441 if ($config{processor} eq "386") {
1442 $target{wp_asm_src}=$table{DEFAULTS}->{wp_asm_src};
1443 } elsif (!$disabled{"whirlpool"}) {
1444 push @{$config{lib_defines}}, "WHIRLPOOL_ASM";
1445 }
1446 }
1447 if ($target{modes_asm_src} =~ /ghash-/) {
1448 push @{$config{lib_defines}}, "GHASH_ASM";
1449 }
1450 if ($target{ec_asm_src} =~ /ecp_nistz256/) {
1451 push @{$config{lib_defines}}, "ECP_NISTZ256_ASM";
1452 }
1453 if ($target{ec_asm_src} =~ /x25519/) {
1454 push @{$config{lib_defines}}, "X25519_ASM";
1455 }
1456 if ($target{padlock_asm_src} ne $table{DEFAULTS}->{padlock_asm_src}) {
1457 push @{$config{dso_defines}}, "PADLOCK_ASM";
1458 }
1459 if ($target{poly1305_asm_src} ne "") {
1460 push @{$config{lib_defines}}, "POLY1305_ASM";
1461 }
1462 }
1463
1464 my %predefined_C = compiler_predefined($config{CROSS_COMPILE}.$config{CC});
1465 my %predefined_CXX = $config{CXX}
1466 ? compiler_predefined($config{CROSS_COMPILE}.$config{CXX})
1467 : ();
1468
1469 # Check for makedepend capabilities.
1470 if (!$disabled{makedepend}) {
1471 if ($config{target} =~ /^(VC|vms)-/) {
1472 # For VC- and vms- targets, there's nothing more to do here. The
1473 # functionality is hard coded in the corresponding build files for
1474 # cl (Windows) and CC/DECC (VMS).
1475 } elsif (($predefined_C{__GNUC__} // -1) >= 3
1476 && !($predefined_C{__APPLE_CC__} && !$predefined_C{__clang__})) {
1477 # We know that GNU C version 3 and up as well as all clang
1478 # versions support dependency generation, but Xcode did not
1479 # handle $cc -M before clang support (but claims __GNUC__ = 3)
1480 $config{makedepprog} = "\$(CROSS_COMPILE)$config{CC}";
1481 } else {
1482 # In all other cases, we look for 'makedepend', and disable the
1483 # capability if not found.
1484 $config{makedepprog} = which('makedepend');
1485 $disabled{makedepend} = "unavailable" unless $config{makedepprog};
1486 }
1487 }
1488
1489 if (!$disabled{asm} && !$predefined_C{__MACH__} && $^O ne 'VMS') {
1490 # probe for -Wa,--noexecstack option...
1491 if ($predefined_C{__clang__}) {
1492 # clang has builtin assembler, which doesn't recognize --help,
1493 # but it apparently recognizes the option in question on all
1494 # supported platforms even when it's meaningless. In other words
1495 # probe would fail, but probed option always accepted...
1496 push @{$config{cflags}}, "-Wa,--noexecstack", "-Qunused-arguments";
1497 } else {
1498 my $cc = $config{CROSS_COMPILE}.$config{CC};
1499 open(PIPE, "$cc -Wa,--help -c -o null.$$.o -x assembler /dev/null 2>&1 |");
1500 while(<PIPE>) {
1501 if (m/--noexecstack/) {
1502 push @{$config{cflags}}, "-Wa,--noexecstack";
1503 last;
1504 }
1505 }
1506 close(PIPE);
1507 unlink("null.$$.o");
1508 }
1509 }
1510
1511 # Deal with bn_ops ###################################################
1512
1513 $config{bn_ll} =0;
1514 $config{export_var_as_fn} =0;
1515 my $def_int="unsigned int";
1516 $config{rc4_int} =$def_int;
1517 ($config{b64l},$config{b64},$config{b32})=(0,0,1);
1518
1519 my $count = 0;
1520 foreach (sort split(/\s+/,$target{bn_ops})) {
1521 $count++ if /SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT/;
1522 $config{export_var_as_fn}=1 if $_ eq 'EXPORT_VAR_AS_FN';
1523 $config{bn_ll}=1 if $_ eq 'BN_LLONG';
1524 $config{rc4_int}="unsigned char" if $_ eq 'RC4_CHAR';
1525 ($config{b64l},$config{b64},$config{b32})
1526 =(0,1,0) if $_ eq 'SIXTY_FOUR_BIT';
1527 ($config{b64l},$config{b64},$config{b32})
1528 =(1,0,0) if $_ eq 'SIXTY_FOUR_BIT_LONG';
1529 ($config{b64l},$config{b64},$config{b32})
1530 =(0,0,1) if $_ eq 'THIRTY_TWO_BIT';
1531 }
1532 die "Exactly one of SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT can be set in bn_ops\n"
1533 if $count > 1;
1534
1535
1536 # Hack cflags for better warnings (dev option) #######################
1537
1538 # "Stringify" the C and C++ flags string. This permits it to be made part of
1539 # a string and works as well on command lines.
1540 $config{cflags} = [ map { (my $x = $_) =~ s/([\\\"])/\\$1/g; $x }
1541 @{$config{cflags}} ];
1542 $config{cxxflags} = [ map { (my $x = $_) =~ s/([\\\"])/\\$1/g; $x }
1543 @{$config{cxxflags}} ] if $config{CXX};
1544
1545 $config{openssl_api_defines} = [
1546 "OPENSSL_MIN_API=".($apitable->{$config{api} // ""} // -1)
1547 ];
1548
1549 my %strict_warnings_collection=( CFLAGS => [], CXXFLAGS => []);
1550 if ($strict_warnings)
1551 {
1552 my $wopt;
1553 my $gccver = $predefined_C{__GNUC__} // -1;
1554 my $gxxver = $predefined_CXX{__GNUC__} // -1;
1555
1556 warn "WARNING --strict-warnings requires gcc[>=4] or gcc-alike"
1557 unless $gccver >= 4;
1558 warn "WARNING --strict-warnings requires g++[>=4] or g++-alike"
1559 unless $gxxver >= 4;
1560 foreach (qw(CFLAGS CXXFLAGS))
1561 {
1562 push @{$strict_warnings_collection{$_}},
1563 @{$gcc_devteam_warn{$_}};
1564 }
1565 push @{$strict_warnings_collection{CFLAGS}},
1566 @{$clang_devteam_warn{CFLAGS}}
1567 if (defined($predefined_C{__clang__}));
1568 push @{$strict_warnings_collection{CXXFLAGS}},
1569 @{$clang_devteam_warn{CXXFLAGS}}
1570 if (defined($predefined_CXX{__clang__}));
1571 }
1572 foreach my $idx (qw(CFLAGS CXXFLAGS))
1573 {
1574 $useradd{$idx} = [ map { $_ eq '--ossl-strict-warnings'
1575 ? @{$strict_warnings_collection{$idx}}
1576 : ( $_ ) }
1577 @{$useradd{$idx}} ];
1578 }
1579
1580 unless ($disabled{"crypto-mdebug-backtrace"})
1581 {
1582 foreach my $wopt (split /\s+/, $memleak_devteam_backtrace)
1583 {
1584 push @{$config{cflags}}, $wopt
1585 unless grep { $_ eq $wopt } @{$config{cflags}};
1586 push @{$config{cxxflags}}, $wopt
1587 if ($config{CXX}
1588 && !grep { $_ eq $wopt } @{$config{cxxflags}});
1589 }
1590 if ($target =~ /^BSD-/)
1591 {
1592 push @{$config{ex_libs}}, "-lexecinfo";
1593 }
1594 }
1595
1596 unless ($disabled{afalgeng}) {
1597 $config{afalgeng}="";
1598 if (grep { $_ eq 'afalgeng' } @{$target{enable}}) {
1599 my $minver = 4*10000 + 1*100 + 0;
1600 if ($config{CROSS_COMPILE} eq "") {
1601 my $verstr = `uname -r`;
1602 my ($ma, $mi1, $mi2) = split("\\.", $verstr);
1603 ($mi2) = $mi2 =~ /(\d+)/;
1604 my $ver = $ma*10000 + $mi1*100 + $mi2;
1605 if ($ver < $minver) {
1606 $disabled{afalgeng} = "too-old-kernel";
1607 } else {
1608 push @{$config{engdirs}}, "afalg";
1609 }
1610 } else {
1611 $disabled{afalgeng} = "cross-compiling";
1612 }
1613 } else {
1614 $disabled{afalgeng} = "not-linux";
1615 }
1616 }
1617
1618 push @{$config{openssl_feature_defines}}, "OPENSSL_NO_AFALGENG" if ($disabled{afalgeng});
1619
1620 unless ($disabled{ktls}) {
1621 $config{ktls}="";
1622 if ($target =~ m/^linux/) {
1623 my $usr = "/usr/$config{cross_compile_prefix}";
1624 chop($usr);
1625 if ($config{cross_compile_prefix} eq "") {
1626 $usr = "/usr";
1627 }
1628 my $minver = (4 << 16) + (13 << 8) + 0;
1629 my @verstr = split(" ",`cat $usr/include/linux/version.h | grep LINUX_VERSION_CODE`);
1630
1631 if ($verstr[2] < $minver) {
1632 $disabled{ktls} = "too-old-kernel";
1633 }
1634 } else {
1635 $disabled{ktls} = "not-linux";
1636 }
1637 }
1638
1639 push @{$config{openssl_other_defines}}, "OPENSSL_NO_KTLS" if ($disabled{ktls});
1640
1641 # Finish up %config by appending things the user gave us on the command line
1642 # apart from "make variables"
1643 foreach (keys %useradd) {
1644 # The must all be lists, so we assert that here
1645 die "internal error: \$useradd{$_} isn't an ARRAY\n"
1646 unless ref $useradd{$_} eq 'ARRAY';
1647
1648 if (defined $config{$_}) {
1649 push @{$config{$_}}, @{$useradd{$_}};
1650 } else {
1651 $config{$_} = [ @{$useradd{$_}} ];
1652 }
1653 }
1654
1655 # ALL MODIFICATIONS TO %config and %target MUST BE DONE FROM HERE ON
1656
1657 # If we use the unified build, collect information from build.info files
1658 my %unified_info = ();
1659
1660 my $buildinfo_debug = defined($ENV{CONFIGURE_DEBUG_BUILDINFO});
1661 if ($builder eq "unified") {
1662 use with_fallback qw(Text::Template);
1663
1664 sub cleandir {
1665 my $base = shift;
1666 my $dir = shift;
1667 my $relativeto = shift || ".";
1668
1669 $dir = catdir($base,$dir) unless isabsolute($dir);
1670
1671 # Make sure the directories we're building in exists
1672 mkpath($dir);
1673
1674 my $res = abs2rel(absolutedir($dir), rel2abs($relativeto));
1675 #print STDERR "DEBUG[cleandir]: $dir , $base => $res\n";
1676 return $res;
1677 }
1678
1679 sub cleanfile {
1680 my $base = shift;
1681 my $file = shift;
1682 my $relativeto = shift || ".";
1683
1684 $file = catfile($base,$file) unless isabsolute($file);
1685
1686 my $d = dirname($file);
1687 my $f = basename($file);
1688
1689 # Make sure the directories we're building in exists
1690 mkpath($d);
1691
1692 my $res = abs2rel(catfile(absolutedir($d), $f), rel2abs($relativeto));
1693 #print STDERR "DEBUG[cleanfile]: $d , $f => $res\n";
1694 return $res;
1695 }
1696
1697 # Store the name of the template file we will build the build file from
1698 # in %config. This may be useful for the build file itself.
1699 my @build_file_template_names =
1700 ( $builder_platform."-".$target{build_file}.".tmpl",
1701 $target{build_file}.".tmpl" );
1702 my @build_file_templates = ();
1703
1704 # First, look in the user provided directory, if given
1705 if (defined env($local_config_envname)) {
1706 @build_file_templates =
1707 map {
1708 if ($^O eq 'VMS') {
1709 # VMS environment variables are logical names,
1710 # which can be used as is
1711 $local_config_envname . ':' . $_;
1712 } else {
1713 catfile(env($local_config_envname), $_);
1714 }
1715 }
1716 @build_file_template_names;
1717 }
1718 # Then, look in our standard directory
1719 push @build_file_templates,
1720 ( map { cleanfile($srcdir, catfile("Configurations", $_), $blddir) }
1721 @build_file_template_names );
1722
1723 my $build_file_template;
1724 for $_ (@build_file_templates) {
1725 $build_file_template = $_;
1726 last if -f $build_file_template;
1727
1728 $build_file_template = undef;
1729 }
1730 if (!defined $build_file_template) {
1731 die "*** Couldn't find any of:\n", join("\n", @build_file_templates), "\n";
1732 }
1733 $config{build_file_templates}
1734 = [ cleanfile($srcdir, catfile("Configurations", "common0.tmpl"),
1735 $blddir),
1736 $build_file_template,
1737 cleanfile($srcdir, catfile("Configurations", "common.tmpl"),
1738 $blddir) ];
1739
1740 my @build_dirs = ( [ ] ); # current directory
1741
1742 $config{build_infos} = [ ];
1743
1744 my %ordinals = ();
1745 while (@build_dirs) {
1746 my @curd = @{shift @build_dirs};
1747 my $sourced = catdir($srcdir, @curd);
1748 my $buildd = catdir($blddir, @curd);
1749
1750 my $unixdir = join('/', @curd);
1751 if (exists $skipdir{$unixdir}) {
1752 my $what = $skipdir{$unixdir};
1753 push @{$disabled_info{$what}->{skipped}}, catdir(@curd);
1754 next;
1755 }
1756
1757 mkpath($buildd);
1758
1759 my $f = 'build.info';
1760 # The basic things we're trying to build
1761 my @programs = ();
1762 my @libraries = ();
1763 my @modules = ();
1764 my @scripts = ();
1765
1766 my %attributes = ();
1767 my %sources = ();
1768 my %shared_sources = ();
1769 my %includes = ();
1770 my %defines = ();
1771 my %depends = ();
1772 my %generate = ();
1773
1774 # We want to detect configdata.pm in the source tree, so we
1775 # don't use it if the build tree is different.
1776 my $src_configdata = cleanfile($srcdir, "configdata.pm", $blddir);
1777
1778 push @{$config{build_infos}}, catfile(abs2rel($sourced, $blddir), $f);
1779 my $template =
1780 Text::Template->new(TYPE => 'FILE',
1781 SOURCE => catfile($sourced, $f),
1782 PREPEND => qq{use lib "$FindBin::Bin/util/perl";});
1783 die "Something went wrong with $sourced/$f: $!\n" unless $template;
1784 my @text =
1785 split /^/m,
1786 $template->fill_in(HASH => { config => \%config,
1787 target => \%target,
1788 disabled => \%disabled,
1789 withargs => \%withargs,
1790 builddir => abs2rel($buildd, $blddir),
1791 sourcedir => abs2rel($sourced, $blddir),
1792 buildtop => abs2rel($blddir, $blddir),
1793 sourcetop => abs2rel($srcdir, $blddir) },
1794 DELIMITERS => [ "{-", "-}" ]);
1795
1796 # The top item of this stack has the following values
1797 # -2 positive already run and we found ELSE (following ELSIF should fail)
1798 # -1 positive already run (skip until ENDIF)
1799 # 0 negatives so far (if we're at a condition, check it)
1800 # 1 last was positive (don't skip lines until next ELSE, ELSIF or ENDIF)
1801 # 2 positive ELSE (following ELSIF should fail)
1802 my @skip = ();
1803 collect_information(
1804 collect_from_array([ @text ],
1805 qr/\\$/ => sub { my $l1 = shift; my $l2 = shift;
1806 $l1 =~ s/\\$//; $l1.$l2 }),
1807 # Info we're looking for
1808 qr/^\s*IF\[((?:\\.|[^\\\]])*)\]\s*$/
1809 => sub {
1810 if (! @skip || $skip[$#skip] > 0) {
1811 push @skip, !! $1;
1812 } else {
1813 push @skip, -1;
1814 }
1815 },
1816 qr/^\s*ELSIF\[((?:\\.|[^\\\]])*)\]\s*$/
1817 => sub { die "ELSIF out of scope" if ! @skip;
1818 die "ELSIF following ELSE" if abs($skip[$#skip]) == 2;
1819 $skip[$#skip] = -1 if $skip[$#skip] != 0;
1820 $skip[$#skip] = !! $1
1821 if $skip[$#skip] == 0; },
1822 qr/^\s*ELSE\s*$/
1823 => sub { die "ELSE out of scope" if ! @skip;
1824 $skip[$#skip] = -2 if $skip[$#skip] != 0;
1825 $skip[$#skip] = 2 if $skip[$#skip] == 0; },
1826 qr/^\s*ENDIF\s*$/
1827 => sub { die "ENDIF out of scope" if ! @skip;
1828 pop @skip; },
1829 qr/^\s*SUBDIRS\s*=\s*(.*)\s*$/
1830 => sub {
1831 if (!@skip || $skip[$#skip] > 0) {
1832 foreach (tokenize($1)) {
1833 push @build_dirs, [ @curd, splitdir($_, 1) ];
1834 }
1835 }
1836 },
1837 qr/^\s*PROGRAMS(?:{([\w=]+(?:\s*,\s*[\w=]+)*)})?\s*=\s*(.*)\s*$/
1838 => sub {
1839 if (!@skip || $skip[$#skip] > 0) {
1840 my @a = tokenize($1, qr|\s*,\s*|);
1841 my @p = tokenize($2);
1842 push @programs, @p;
1843 foreach my $a (@a) {
1844 my $ak = $a;
1845 my $av = 1;
1846 if ($a =~ m|^(.*?)\s*=\s*(.*?)$|) {
1847 $ak = $1;
1848 $av = $2;
1849 }
1850 foreach my $p (@p) {
1851 $attributes{$p}->{$ak} = $av;
1852 }
1853 }
1854 }
1855 },
1856 qr/^\s*LIBS(?:{([\w=]+(?:\s*,\s*[\w=]+)*)})?\s*=\s*(.*)\s*$/
1857 => sub {
1858 if (!@skip || $skip[$#skip] > 0) {
1859 my @a = tokenize($1, qr|\s*,\s*|);
1860 my @l = tokenize($2);
1861 push @libraries, @l;
1862 foreach my $a (@a) {
1863 my $ak = $a;
1864 my $av = 1;
1865 if ($a =~ m|^(.*?)\s*=\s*(.*?)$|) {
1866 $ak = $1;
1867 $av = $2;
1868 }
1869 foreach my $l (@l) {
1870 $attributes{$l}->{$ak} = $av;
1871 }
1872 }
1873 }
1874 },
1875 qr/^\s*MODULES(?:{([\w=]+(?:\s*,\s*[\w=]+)*)})?\s*=\s*(.*)\s*$/
1876 => sub {
1877 if (!@skip || $skip[$#skip] > 0) {
1878 my @a = tokenize($1, qr|\s*,\s*|);
1879 my @m = tokenize($2);
1880 push @modules, @m;
1881 foreach my $a (@a) {
1882 my $ak = $a;
1883 my $av = 1;
1884 if ($a =~ m|^(.*?)\s*=\s*(.*?)$|) {
1885 $ak = $1;
1886 $av = $2;
1887 }
1888 foreach my $m (@m) {
1889 $attributes{$m}->{$ak} = $av;
1890 }
1891 }
1892 }
1893 },
1894 qr/^\s*SCRIPTS(?:{([\w=]+(?:\s*,\s*[\w=]+)*)})?\s*=\s*(.*)\s*$/
1895 => sub {
1896 if (!@skip || $skip[$#skip] > 0) {
1897 my @a = tokenize($1, qr|\s*,\s*|);
1898 my @s = tokenize($2);
1899 push @scripts, @s;
1900 foreach my $a (@a) {
1901 my $ak = $a;
1902 my $av = 1;
1903 if ($a =~ m|^(.*?)\s*=\s*(.*?)$|) {
1904 $ak = $1;
1905 $av = $2;
1906 }
1907 foreach my $s (@s) {
1908 $attributes{$s}->{$ak} = $av;
1909 }
1910 }
1911 }
1912 },
1913
1914 qr/^\s*ORDINALS\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/,
1915 => sub { push @{$ordinals{$1}}, tokenize($2)
1916 if !@skip || $skip[$#skip] > 0 },
1917 qr/^\s*SOURCE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1918 => sub { push @{$sources{$1}}, tokenize($2)
1919 if !@skip || $skip[$#skip] > 0 },
1920 qr/^\s*SHARED_SOURCE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1921 => sub { push @{$shared_sources{$1}}, tokenize($2)
1922 if !@skip || $skip[$#skip] > 0 },
1923 qr/^\s*INCLUDE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1924 => sub { push @{$includes{$1}}, tokenize($2)
1925 if !@skip || $skip[$#skip] > 0 },
1926 qr/^\s*DEFINE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1927 => sub { push @{$defines{$1}}, tokenize($2)
1928 if !@skip || $skip[$#skip] > 0 },
1929 qr/^\s*DEPEND\[((?:\\.|[^\\\]])*)\]\s*=\s*(.*)\s*$/
1930 => sub { push @{$depends{$1}}, tokenize($2)
1931 if !@skip || $skip[$#skip] > 0 },
1932 qr/^\s*GENERATE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1933 => sub { push @{$generate{$1}}, $2
1934 if !@skip || $skip[$#skip] > 0 },
1935 qr/^\s*(?:#.*)?$/ => sub { },
1936 "OTHERWISE" => sub { die "Something wrong with this line:\n$_\nat $sourced/$f" },
1937 "BEFORE" => sub {
1938 if ($buildinfo_debug) {
1939 print STDERR "DEBUG: Parsing ",join(" ", @_),"\n";
1940 print STDERR "DEBUG: ... before parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
1941 }
1942 },
1943 "AFTER" => sub {
1944 if ($buildinfo_debug) {
1945 print STDERR "DEBUG: .... after parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
1946 }
1947 },
1948 );
1949 die "runaway IF?" if (@skip);
1950
1951 if (grep { defined $attributes{$_}->{engine} } keys %attributes
1952 and !$config{dynamic_engines}) {
1953 die <<"EOF"
1954 ENGINES can only be used if configured with 'dynamic-engine'.
1955 This is usually a fault in a build.info file.
1956 EOF
1957 }
1958
1959 foreach (keys %attributes) {
1960 my $dest = $_;
1961 my $ddest = cleanfile($buildd, $_, $blddir);
1962 foreach (keys %{$attributes{$dest} // {}}) {
1963 $unified_info{attributes}->{$ddest}->{$_} =
1964 $attributes{$dest}->{$_};
1965 }
1966 }
1967
1968 {
1969 my %infos = ( programs => [ @programs ],
1970 libraries => [ @libraries ],
1971 modules => [ @modules ],
1972 scripts => [ @scripts ] );
1973 foreach my $k (keys %infos) {
1974 foreach (@{$infos{$k}}) {
1975 my $item = cleanfile($buildd, $_, $blddir);
1976 $unified_info{$k}->{$item} = 1;
1977 }
1978 }
1979 }
1980
1981 # Check that we haven't defined any library as both shared and
1982 # explicitly static. That is forbidden.
1983 my @doubles = ();
1984 foreach (grep /\.a$/, keys %{$unified_info{libraries}}) {
1985 (my $l = $_) =~ s/\.a$//;
1986 push @doubles, $l if defined $unified_info{libraries}->{$l};
1987 }
1988 die "these libraries are both explicitly static and shared:\n ",
1989 join(" ", @doubles), "\n"
1990 if @doubles;
1991
1992 foreach (keys %sources) {
1993 my $dest = $_;
1994 my $ddest = cleanfile($buildd, $_, $blddir);
1995 foreach (@{$sources{$dest}}) {
1996 my $s = cleanfile($sourced, $_, $blddir);
1997
1998 # If it isn't in the source tree, we assume it's generated
1999 # in the build tree
2000 if ($s eq $src_configdata || ! -f $s || $generate{$_}) {
2001 $s = cleanfile($buildd, $_, $blddir);
2002 }
2003 # We recognise C++, C and asm files
2004 if ($s =~ /\.(cc|cpp|c|s|S)$/) {
2005 my $o = $_;
2006 $o =~ s/\.[csS]$/.o/; # C and assembler
2007 $o =~ s/\.(cc|cpp)$/_cc.o/; # C++
2008 $o = cleanfile($buildd, $o, $blddir);
2009 $unified_info{sources}->{$ddest}->{$o} = -1;
2010 $unified_info{sources}->{$o}->{$s} = -1;
2011 } elsif ($s =~ /\.rc$/) {
2012 # We also recognise resource files
2013 my $o = $_;
2014 $o =~ s/\.rc$/.res/; # Resource configuration
2015 my $o = cleanfile($buildd, $o, $blddir);
2016 $unified_info{sources}->{$ddest}->{$o} = -1;
2017 $unified_info{sources}->{$o}->{$s} = -1;
2018 } else {
2019 $unified_info{sources}->{$ddest}->{$s} = 1;
2020 }
2021 }
2022 }
2023
2024 foreach (keys %shared_sources) {
2025 my $dest = $_;
2026 my $ddest = cleanfile($buildd, $_, $blddir);
2027 foreach (@{$shared_sources{$dest}}) {
2028 my $s = cleanfile($sourced, $_, $blddir);
2029
2030 # If it isn't in the source tree, we assume it's generated
2031 # in the build tree
2032 if ($s eq $src_configdata || ! -f $s || $generate{$_}) {
2033 $s = cleanfile($buildd, $_, $blddir);
2034 }
2035
2036 if ($s =~ /\.(cc|cpp|c|s|S)$/) {
2037 # We recognise C++, C and asm files
2038 my $o = $_;
2039 $o =~ s/\.[csS]$/.o/; # C and assembler
2040 $o =~ s/\.(cc|cpp)$/_cc.o/; # C++
2041 $o = cleanfile($buildd, $o, $blddir);
2042 $unified_info{shared_sources}->{$ddest}->{$o} = -1;
2043 $unified_info{sources}->{$o}->{$s} = -1;
2044 } elsif ($s =~ /\.rc$/) {
2045 # We also recognise resource files
2046 my $o = $_;
2047 $o =~ s/\.rc$/.res/; # Resource configuration
2048 my $o = cleanfile($buildd, $o, $blddir);
2049 $unified_info{shared_sources}->{$ddest}->{$o} = -1;
2050 $unified_info{sources}->{$o}->{$s} = -1;
2051 } elsif ($s =~ /\.ld$/) {
2052 # We also recognise linker scripts (or corresponding)
2053 # We know they are generated files
2054 my $ld = cleanfile($buildd, $_, $blddir);
2055 $unified_info{shared_sources}->{$ddest}->{$ld} = 1;
2056 } else {
2057 die "unrecognised source file type for shared library: $s\n";
2058 }
2059 }
2060 }
2061
2062 foreach (keys %generate) {
2063 my $dest = $_;
2064 my $ddest = cleanfile($buildd, $_, $blddir);
2065 die "more than one generator for $dest: "
2066 ,join(" ", @{$generate{$_}}),"\n"
2067 if scalar @{$generate{$_}} > 1;
2068 my @generator = split /\s+/, $generate{$dest}->[0];
2069 $generator[0] = cleanfile($sourced, $generator[0], $blddir),
2070 $unified_info{generate}->{$ddest} = [ @generator ];
2071 }
2072
2073 foreach (keys %depends) {
2074 my $dest = $_;
2075 my $ddest = $dest eq "" ? "" : cleanfile($sourced, $_, $blddir);
2076
2077 # If the destination doesn't exist in source, it can only be
2078 # a generated file in the build tree.
2079 if ($ddest ne "" && ($ddest eq $src_configdata || ! -f $ddest)) {
2080 $ddest = cleanfile($buildd, $_, $blddir);
2081 }
2082 foreach (@{$depends{$dest}}) {
2083 my $d = cleanfile($sourced, $_, $blddir);
2084
2085 # If we know it's generated, or assume it is because we can't
2086 # find it in the source tree, we set file we depend on to be
2087 # in the build tree rather than the source tree, and assume
2088 # and that there are lines to build it in a BEGINRAW..ENDRAW
2089 # section or in the Makefile template.
2090 if ($d eq $src_configdata
2091 || ! -f $d
2092 || (grep { $d eq $_ }
2093 map { cleanfile($srcdir, $_, $blddir) }
2094 grep { /\.h$/ } keys %{$unified_info{generate}})) {
2095 $d = cleanfile($buildd, $_, $blddir);
2096 }
2097 # Take note if the file to depend on is being renamed
2098 # Take extra care with files ending with .a, they should
2099 # be treated without that extension, and the extension
2100 # should be added back after treatment.
2101 $d =~ /(\.a)?$/;
2102 my $e = $1 // "";
2103 $d = $`.$e;
2104 $unified_info{depends}->{$ddest}->{$d} = 1;
2105 }
2106 }
2107
2108 foreach (keys %includes) {
2109 my $dest = $_;
2110 my $ddest = cleanfile($sourced, $_, $blddir);
2111
2112 # If the destination doesn't exist in source, it can only be
2113 # a generated file in the build tree.
2114 if ($ddest eq $src_configdata || ! -f $ddest) {
2115 $ddest = cleanfile($buildd, $_, $blddir);
2116 }
2117 foreach (@{$includes{$dest}}) {
2118 my $is = cleandir($sourced, $_, $blddir);
2119 my $ib = cleandir($buildd, $_, $blddir);
2120 push @{$unified_info{includes}->{$ddest}->{source}}, $is
2121 unless grep { $_ eq $is } @{$unified_info{includes}->{$ddest}->{source}};
2122 push @{$unified_info{includes}->{$ddest}->{build}}, $ib
2123 unless grep { $_ eq $ib } @{$unified_info{includes}->{$ddest}->{build}};
2124 }
2125 }
2126
2127 foreach (keys %defines) {
2128 my $dest = $_;
2129 my $ddest = cleanfile($sourced, $_, $blddir);
2130
2131 # If the destination doesn't exist in source, it can only be
2132 # a generated file in the build tree.
2133 if (! -f $ddest) {
2134 $ddest = cleanfile($buildd, $_, $blddir);
2135 if ($unified_info{rename}->{$ddest}) {
2136 $ddest = $unified_info{rename}->{$ddest};
2137 }
2138 }
2139 foreach (@{$defines{$dest}}) {
2140 m|^([^=]*)(=.*)?$|;
2141 die "0 length macro name not permitted\n" if $1 eq "";
2142 die "$1 defined more than once\n"
2143 if defined $unified_info{defines}->{$ddest}->{$1};
2144 $unified_info{defines}->{$ddest}->{$1} = $2;
2145 }
2146 }
2147 }
2148
2149 my $ordinals_text = join(', ', sort keys %ordinals);
2150 warn <<"EOF" if $ordinals_text;
2151
2152 WARNING: ORDINALS were specified for $ordinals_text
2153 They are ignored and should be replaced with a combination of GENERATE,
2154 DEPEND and SHARED_SOURCE.
2155 EOF
2156
2157 # Massage the result
2158
2159 # If we depend on a header file or a perl module, add an inclusion of
2160 # its directory to allow smoothe inclusion
2161 foreach my $dest (keys %{$unified_info{depends}}) {
2162 next if $dest eq "";
2163 foreach my $d (keys %{$unified_info{depends}->{$dest}}) {
2164 next unless $d =~ /\.(h|pm)$/;
2165 my $i = dirname($d);
2166 my $spot =
2167 $d eq "configdata.pm" || defined($unified_info{generate}->{$d})
2168 ? 'build' : 'source';
2169 push @{$unified_info{includes}->{$dest}->{$spot}}, $i
2170 unless grep { $_ eq $i } @{$unified_info{includes}->{$dest}->{$spot}};
2171 }
2172 }
2173
2174 # Go through all intermediary files and change their names to something that
2175 # reflects what they will be built for. Note that for some source files,
2176 # this leads to duplicate object files because they are used multiple times.
2177 # the goal is to rename all object files according to this scheme:
2178 # {productname}-{midfix}-{origobjname}.[o|res]
2179 # the {midfix} is a keyword indicating the type of product, which is mostly
2180 # valuable for libraries since they come in two forms.
2181 #
2182 # This also reorganises the {sources} and {shared_sources} so that the
2183 # former only contains ALL object files that are supposed to end up in
2184 # static libraries and programs, while the latter contains ALL object files
2185 # that are supposed to end up in shared libraries and DSOs.
2186 # The main reason for having two different source structures is to allow
2187 # the same name to be used for the static and the shared variants of a
2188 # library.
2189 {
2190 # Take copies so we don't get interference from added stuff
2191 my %unified_copy = ();
2192 foreach (('sources', 'shared_sources')) {
2193 $unified_copy{$_} = { %{$unified_info{$_}} }
2194 if defined($unified_info{$_});
2195 delete $unified_info{$_};
2196 }
2197 foreach my $prodtype (('programs', 'libraries', 'modules', 'scripts')) {
2198 # $intent serves multi purposes:
2199 # - give a prefix for the new object files names
2200 # - in the case of libraries, rearrange the object files so static
2201 # libraries use the 'sources' structure exclusively, while shared
2202 # libraries use the 'shared_sources' structure exclusively.
2203 my $intent = {
2204 programs => { bin => { src => [ 'sources' ],
2205 dst => 'sources' } },
2206 libraries => { lib => { src => [ 'sources' ],
2207 dst => 'sources' },
2208 shlib => { prodselect =>
2209 sub { grep !/\.a$/, @_ },
2210 src => [ 'sources',
2211 'shared_sources' ],
2212 dst => 'shared_sources' } },
2213 modules => { dso => { src => [ 'sources' ],
2214 dst => 'sources' } },
2215 scripts => { script => { src => [ 'sources' ],
2216 dst => 'sources' } }
2217 } -> {$prodtype};
2218 foreach my $kind (keys %$intent) {
2219 next if ($intent->{$kind}->{dst} eq 'shared_sources'
2220 && $disabled{shared});
2221
2222 my @src = @{$intent->{$kind}->{src}};
2223 my $dst = $intent->{$kind}->{dst};
2224 my $prodselect = $intent->{$kind}->{prodselect} // sub { @_ };
2225 foreach my $prod ($prodselect->(keys %{$unified_info{$prodtype}})) {
2226 # %prod_sources has all applicable objects as keys, and
2227 # their corresponding sources as values
2228 my %prod_sources =
2229 map { $_ => [ keys %{$unified_copy{sources}->{$_}} ] }
2230 map { keys %{$unified_copy{$_}->{$prod}} }
2231 @src;
2232 foreach (keys %prod_sources) {
2233 # Only affect object files and resource files,
2234 # the others simply get a new value
2235 # (+1 instead of -1)
2236 if ($_ =~ /\.(o|res)$/) {
2237 (my $prodname = $prod) =~ s|\.a$||;
2238 my $newobj =
2239 catfile(dirname($_),
2240 basename($prodname)
2241 . '-' . $kind
2242 . '-' . basename($_));
2243 $unified_info{$dst}->{$prod}->{$newobj} = 1;
2244 foreach my $src (@{$prod_sources{$_}}) {
2245 $unified_info{sources}->{$newobj}->{$src} = 1;
2246 }
2247 # Adjust dependencies
2248 foreach my $deps (keys %{$unified_info{depends}->{$_}}) {
2249 $unified_info{depends}->{$_}->{$deps} = -1;
2250 $unified_info{depends}->{$newobj}->{$deps} = 1;
2251 }
2252 # Adjust includes
2253 foreach my $k (('source', 'build')) {
2254 next unless
2255 defined($unified_info{includes}->{$_}->{$k});
2256 my @incs = @{$unified_info{includes}->{$_}->{$k}};
2257 $unified_info{includes}->{$newobj}->{$k} = [ @incs ];
2258 }
2259 } else {
2260 $unified_info{$dst}->{$prod}->{$_} = 1;
2261 }
2262 }
2263 }
2264 }
2265 }
2266 }
2267 # At this point, we have a number of sources with the value -1. They
2268 # aren't part of the local build and are probably meant for a different
2269 # platform, and can therefore be cleaned away. That happens when making
2270 # %unified_info more efficient below.
2271
2272 ### Make unified_info a bit more efficient
2273 # One level structures
2274 foreach (("programs", "libraries", "modules", "scripts")) {
2275 $unified_info{$_} = [ sort keys %{$unified_info{$_}} ];
2276 }
2277 # Two level structures
2278 foreach my $l1 (("sources", "shared_sources", "ldadd", "depends")) {
2279 foreach my $l2 (sort keys %{$unified_info{$l1}}) {
2280 my @items =
2281 sort
2282 grep { $unified_info{$l1}->{$l2}->{$_} > 0 }
2283 keys %{$unified_info{$l1}->{$l2}};
2284 if (@items) {
2285 $unified_info{$l1}->{$l2} = [ @items ];
2286 } else {
2287 delete $unified_info{$l1}->{$l2};
2288 }
2289 }
2290 }
2291 # Defines
2292 foreach my $dest (sort keys %{$unified_info{defines}}) {
2293 $unified_info{defines}->{$dest}
2294 = [ map { $_.$unified_info{defines}->{$dest}->{$_} }
2295 sort keys %{$unified_info{defines}->{$dest}} ];
2296 }
2297 # Includes
2298 foreach my $dest (sort keys %{$unified_info{includes}}) {
2299 if (defined($unified_info{includes}->{$dest}->{build})) {
2300 my @source_includes = ();
2301 @source_includes = ( @{$unified_info{includes}->{$dest}->{source}} )
2302 if defined($unified_info{includes}->{$dest}->{source});
2303 $unified_info{includes}->{$dest} =
2304 [ @{$unified_info{includes}->{$dest}->{build}} ];
2305 foreach my $inc (@source_includes) {
2306 push @{$unified_info{includes}->{$dest}}, $inc
2307 unless grep { $_ eq $inc } @{$unified_info{includes}->{$dest}};
2308 }
2309 } elsif (defined($unified_info{includes}->{$dest}->{source})) {
2310 $unified_info{includes}->{$dest} =
2311 [ @{$unified_info{includes}->{$dest}->{source}} ];
2312 } else {
2313 delete $unified_info{includes}->{$dest};
2314 }
2315 }
2316
2317 # For convenience collect information regarding directories where
2318 # files are generated, those generated files and the end product
2319 # they end up in where applicable. Then, add build rules for those
2320 # directories
2321 my %loopinfo = ( "lib" => [ @{$unified_info{libraries}} ],
2322 "dso" => [ @{$unified_info{modules}} ],
2323 "bin" => [ @{$unified_info{programs}} ],
2324 "script" => [ @{$unified_info{scripts}} ] );
2325 foreach my $type (keys %loopinfo) {
2326 foreach my $product (@{$loopinfo{$type}}) {
2327 my %dirs = ();
2328 my $pd = dirname($product);
2329
2330 foreach (@{$unified_info{sources}->{$product} // []},
2331 @{$unified_info{shared_sources}->{$product} // []}) {
2332 my $d = dirname($_);
2333
2334 # We don't want to create targets for source directories
2335 # when building out of source
2336 next if ($config{sourcedir} ne $config{builddir}
2337 && $d =~ m|^\Q$config{sourcedir}\E|);
2338 # We already have a "test" target, and the current directory
2339 # is just silly to make a target for
2340 next if $d eq "test" || $d eq ".";
2341
2342 $dirs{$d} = 1;
2343 push @{$unified_info{dirinfo}->{$d}->{deps}}, $_
2344 if $d ne $pd;
2345 }
2346 foreach (keys %dirs) {
2347 push @{$unified_info{dirinfo}->{$_}->{products}->{$type}},
2348 $product;
2349 }
2350 }
2351 }
2352 }
2353
2354 # For the schemes that need it, we provide the old *_obj configs
2355 # from the *_asm_obj ones
2356 foreach (grep /_(asm|aux)_src$/, keys %target) {
2357 my $src = $_;
2358 (my $obj = $_) =~ s/_(asm|aux)_src$/_obj/;
2359 $target{$obj} = $target{$src};
2360 $target{$obj} =~ s/\.[csS]\b/.o/g; # C and assembler
2361 $target{$obj} =~ s/\.(cc|cpp)\b/_cc.o/g; # C++
2362 }
2363
2364 # Write down our configuration where it fits #########################
2365
2366 print "Creating configdata.pm\n";
2367 open(OUT,">configdata.pm") || die "unable to create configdata.pm: $!\n";
2368 print OUT <<"EOF";
2369 #! $config{HASHBANGPERL}
2370
2371 package configdata;
2372
2373 use strict;
2374 use warnings;
2375
2376 use Exporter;
2377 #use vars qw(\@ISA \@EXPORT);
2378 our \@ISA = qw(Exporter);
2379 our \@EXPORT = qw(\%config \%target \%disabled \%withargs \%unified_info \@disablables);
2380
2381 EOF
2382 print OUT "our %config = (\n";
2383 foreach (sort keys %config) {
2384 if (ref($config{$_}) eq "ARRAY") {
2385 print OUT " ", $_, " => [ ", join(", ",
2386 map { quotify("perl", $_) }
2387 @{$config{$_}}), " ],\n";
2388 } elsif (ref($config{$_}) eq "HASH") {
2389 print OUT " ", $_, " => {";
2390 if (scalar keys %{$config{$_}} > 0) {
2391 print OUT "\n";
2392 foreach my $key (sort keys %{$config{$_}}) {
2393 print OUT " ",
2394 join(" => ",
2395 quotify("perl", $key),
2396 defined $config{$_}->{$key}
2397 ? quotify("perl", $config{$_}->{$key})
2398 : "undef");
2399 print OUT ",\n";
2400 }
2401 print OUT " ";
2402 }
2403 print OUT "},\n";
2404 } else {
2405 print OUT " ", $_, " => ", quotify("perl", $config{$_}), ",\n"
2406 }
2407 }
2408 print OUT <<"EOF";
2409 );
2410
2411 EOF
2412 print OUT "our %target = (\n";
2413 foreach (sort keys %target) {
2414 if (ref($target{$_}) eq "ARRAY") {
2415 print OUT " ", $_, " => [ ", join(", ",
2416 map { quotify("perl", $_) }
2417 @{$target{$_}}), " ],\n";
2418 } else {
2419 print OUT " ", $_, " => ", quotify("perl", $target{$_}), ",\n"
2420 }
2421 }
2422 print OUT <<"EOF";
2423 );
2424
2425 EOF
2426 print OUT "our \%available_protocols = (\n";
2427 print OUT " tls => [ ", join(", ", map { quotify("perl", $_) } @tls), " ],\n";
2428 print OUT " dtls => [ ", join(", ", map { quotify("perl", $_) } @dtls), " ],\n";
2429 print OUT <<"EOF";
2430 );
2431
2432 EOF
2433 print OUT "our \@disablables = (\n";
2434 foreach (@disablables) {
2435 print OUT " ", quotify("perl", $_), ",\n";
2436 }
2437 print OUT <<"EOF";
2438 );
2439
2440 EOF
2441 print OUT "our \%disabled = (\n";
2442 foreach (sort keys %disabled) {
2443 print OUT " ", quotify("perl", $_), " => ", quotify("perl", $disabled{$_}), ",\n";
2444 }
2445 print OUT <<"EOF";
2446 );
2447
2448 EOF
2449 print OUT "our %withargs = (\n";
2450 foreach (sort keys %withargs) {
2451 if (ref($withargs{$_}) eq "ARRAY") {
2452 print OUT " ", $_, " => [ ", join(", ",
2453 map { quotify("perl", $_) }
2454 @{$withargs{$_}}), " ],\n";
2455 } else {
2456 print OUT " ", $_, " => ", quotify("perl", $withargs{$_}), ",\n"
2457 }
2458 }
2459 print OUT <<"EOF";
2460 );
2461
2462 EOF
2463 if ($builder eq "unified") {
2464 my $recurse;
2465 $recurse = sub {
2466 my $indent = shift;
2467 foreach (@_) {
2468 if (ref $_ eq "ARRAY") {
2469 print OUT " "x$indent, "[\n";
2470 foreach (@$_) {
2471 $recurse->($indent + 4, $_);
2472 }
2473 print OUT " "x$indent, "],\n";
2474 } elsif (ref $_ eq "HASH") {
2475 my %h = %$_;
2476 print OUT " "x$indent, "{\n";
2477 foreach (sort keys %h) {
2478 if (ref $h{$_} eq "") {
2479 print OUT " "x($indent + 4), quotify("perl", $_), " => ", quotify("perl", $h{$_}), ",\n";
2480 } else {
2481 print OUT " "x($indent + 4), quotify("perl", $_), " =>\n";
2482 $recurse->($indent + 8, $h{$_});
2483 }
2484 }
2485 print OUT " "x$indent, "},\n";
2486 } else {
2487 print OUT " "x$indent, quotify("perl", $_), ",\n";
2488 }
2489 }
2490 };
2491 print OUT "our %unified_info = (\n";
2492 foreach (sort keys %unified_info) {
2493 if (ref $unified_info{$_} eq "") {
2494 print OUT " "x4, quotify("perl", $_), " => ", quotify("perl", $unified_info{$_}), ",\n";
2495 } else {
2496 print OUT " "x4, quotify("perl", $_), " =>\n";
2497 $recurse->(8, $unified_info{$_});
2498 }
2499 }
2500 print OUT <<"EOF";
2501 );
2502
2503 EOF
2504 }
2505 print OUT
2506 "# The following data is only used when this files is use as a script\n";
2507 print OUT "my \@makevars = (\n";
2508 foreach (sort keys %user) {
2509 print OUT " '",$_,"',\n";
2510 }
2511 print OUT ");\n";
2512 print OUT "my \%disabled_info = (\n";
2513 foreach my $what (sort keys %disabled_info) {
2514 print OUT " '$what' => {\n";
2515 foreach my $info (sort keys %{$disabled_info{$what}}) {
2516 if (ref $disabled_info{$what}->{$info} eq 'ARRAY') {
2517 print OUT " $info => [ ",
2518 join(', ', map { "'$_'" } @{$disabled_info{$what}->{$info}}),
2519 " ],\n";
2520 } else {
2521 print OUT " $info => '", $disabled_info{$what}->{$info},
2522 "',\n";
2523 }
2524 }
2525 print OUT " },\n";
2526 }
2527 print OUT ");\n";
2528 print OUT 'my @user_crossable = qw( ', join (' ', @user_crossable), " );\n";
2529 print OUT << 'EOF';
2530 # If run directly, we can give some answers, and even reconfigure
2531 unless (caller) {
2532 use Getopt::Long;
2533 use File::Spec::Functions;
2534 use File::Basename;
2535 use Pod::Usage;
2536
2537 my $here = dirname($0);
2538
2539 my $dump = undef;
2540 my $cmdline = undef;
2541 my $options = undef;
2542 my $target = undef;
2543 my $envvars = undef;
2544 my $makevars = undef;
2545 my $buildparams = undef;
2546 my $reconf = undef;
2547 my $verbose = undef;
2548 my $help = undef;
2549 my $man = undef;
2550 GetOptions('dump|d' => \$dump,
2551 'command-line|c' => \$cmdline,
2552 'options|o' => \$options,
2553 'target|t' => \$target,
2554 'environment|e' => \$envvars,
2555 'make-variables|m' => \$makevars,
2556 'build-parameters|b' => \$buildparams,
2557 'reconfigure|reconf|r' => \$reconf,
2558 'verbose|v' => \$verbose,
2559 'help' => \$help,
2560 'man' => \$man)
2561 or die "Errors in command line arguments\n";
2562
2563 unless ($dump || $cmdline || $options || $target || $envvars || $makevars
2564 || $buildparams || $reconf || $verbose || $help || $man) {
2565 print STDERR <<"_____";
2566 You must give at least one option.
2567 For more information, do '$0 --help'
2568 _____
2569 exit(2);
2570 }
2571
2572 if ($help) {
2573 pod2usage(-exitval => 0,
2574 -verbose => 1);
2575 }
2576 if ($man) {
2577 pod2usage(-exitval => 0,
2578 -verbose => 2);
2579 }
2580 if ($dump || $cmdline) {
2581 print "\nCommand line (with current working directory = $here):\n\n";
2582 print ' ',join(' ',
2583 $config{PERL},
2584 catfile($config{sourcedir}, 'Configure'),
2585 @{$config{perlargv}}), "\n";
2586 print "\nPerl information:\n\n";
2587 print ' ',$config{perl_cmd},"\n";
2588 print ' ',$config{perl_version},' for ',$config{perl_archname},"\n";
2589 }
2590 if ($dump || $options) {
2591 my $longest = 0;
2592 my $longest2 = 0;
2593 foreach my $what (@disablables) {
2594 $longest = length($what) if $longest < length($what);
2595 $longest2 = length($disabled{$what})
2596 if $disabled{$what} && $longest2 < length($disabled{$what});
2597 }
2598 print "\nEnabled features:\n\n";
2599 foreach my $what (@disablables) {
2600 print " $what\n" unless $disabled{$what};
2601 }
2602 print "\nDisabled features:\n\n";
2603 foreach my $what (@disablables) {
2604 if ($disabled{$what}) {
2605 print " $what", ' ' x ($longest - length($what) + 1),
2606 "[$disabled{$what}]", ' ' x ($longest2 - length($disabled{$what}) + 1);
2607 print $disabled_info{$what}->{macro}
2608 if $disabled_info{$what}->{macro};
2609 print ' (skip ',
2610 join(', ', @{$disabled_info{$what}->{skipped}}),
2611 ')'
2612 if $disabled_info{$what}->{skipped};
2613 print "\n";
2614 }
2615 }
2616 }
2617 if ($dump || $target) {
2618 print "\nConfig target attributes:\n\n";
2619 foreach (sort keys %target) {
2620 next if $_ =~ m|^_| || $_ eq 'template';
2621 my $quotify = sub {
2622 map { (my $x = $_) =~ s|([\\\$\@"])|\\$1|g; "\"$x\""} @_;
2623 };
2624 print ' ', $_, ' => ';
2625 if (ref($target{$_}) eq "ARRAY") {
2626 print '[ ', join(', ', $quotify->(@{$target{$_}})), " ],\n";
2627 } else {
2628 print $quotify->($target{$_}), ",\n"
2629 }
2630 }
2631 }
2632 if ($dump || $envvars) {
2633 print "\nRecorded environment:\n\n";
2634 foreach (sort keys %{$config{perlenv}}) {
2635 print ' ',$_,' = ',($config{perlenv}->{$_} || ''),"\n";
2636 }
2637 }
2638 if ($dump || $makevars) {
2639 print "\nMakevars:\n\n";
2640 foreach my $var (@makevars) {
2641 my $prefix = '';
2642 $prefix = $config{CROSS_COMPILE}
2643 if grep { $var eq $_ } @user_crossable;
2644 $prefix //= '';
2645 print ' ',$var,' ' x (16 - length $var),'= ',
2646 (ref $config{$var} eq 'ARRAY'
2647 ? join(' ', @{$config{$var}})
2648 : $prefix.$config{$var}),
2649 "\n"
2650 if defined $config{$var};
2651 }
2652
2653 my @buildfile = ($config{builddir}, $config{build_file});
2654 unshift @buildfile, $here
2655 unless file_name_is_absolute($config{builddir});
2656 my $buildfile = canonpath(catdir(@buildfile));
2657 print <<"_____";
2658
2659 NOTE: These variables only represent the configuration view. The build file
2660 template may have processed these variables further, please have a look at the
2661 build file for more exact data:
2662 $buildfile
2663 _____
2664 }
2665 if ($dump || $buildparams) {
2666 my @buildfile = ($config{builddir}, $config{build_file});
2667 unshift @buildfile, $here
2668 unless file_name_is_absolute($config{builddir});
2669 print "\nbuild file:\n\n";
2670 print " ", canonpath(catfile(@buildfile)),"\n";
2671
2672 print "\nbuild file templates:\n\n";
2673 foreach (@{$config{build_file_templates}}) {
2674 my @tmpl = ($_);
2675 unshift @tmpl, $here
2676 unless file_name_is_absolute($config{sourcedir});
2677 print ' ',canonpath(catfile(@tmpl)),"\n";
2678 }
2679 }
2680 if ($reconf) {
2681 if ($verbose) {
2682 print 'Reconfiguring with: ', join(' ',@{$config{perlargv}}), "\n";
2683 foreach (sort keys %{$config{perlenv}}) {
2684 print ' ',$_,' = ',($config{perlenv}->{$_} || ""),"\n";
2685 }
2686 }
2687
2688 chdir $here;
2689 exec $^X,catfile($config{sourcedir}, 'Configure'),'reconf';
2690 }
2691 }
2692
2693 1;
2694
2695 __END__
2696
2697 =head1 NAME
2698
2699 configdata.pm - configuration data for OpenSSL builds
2700
2701 =head1 SYNOPSIS
2702
2703 Interactive:
2704
2705 perl configdata.pm [options]
2706
2707 As data bank module:
2708
2709 use configdata;
2710
2711 =head1 DESCRIPTION
2712
2713 This module can be used in two modes, interactively and as a module containing
2714 all the data recorded by OpenSSL's Configure script.
2715
2716 When used interactively, simply run it as any perl script, with at least one
2717 option, and you will get the information you ask for. See L</OPTIONS> below.
2718
2719 When loaded as a module, you get a few databanks with useful information to
2720 perform build related tasks. The databanks are:
2721
2722 %config Configured things.
2723 %target The OpenSSL config target with all inheritances
2724 resolved.
2725 %disabled The features that are disabled.
2726 @disablables The list of features that can be disabled.
2727 %withargs All data given through --with-THING options.
2728 %unified_info All information that was computed from the build.info
2729 files.
2730
2731 =head1 OPTIONS
2732
2733 =over 4
2734
2735 =item B<--help>
2736
2737 Print a brief help message and exit.
2738
2739 =item B<--man>
2740
2741 Print the manual page and exit.
2742
2743 =item B<--dump> | B<-d>
2744
2745 Print all relevant configuration data. This is equivalent to B<--command-line>
2746 B<--options> B<--target> B<--environment> B<--make-variables>
2747 B<--build-parameters>.
2748
2749 =item B<--command-line> | B<-c>
2750
2751 Print the current configuration command line.
2752
2753 =item B<--options> | B<-o>
2754
2755 Print the features, both enabled and disabled, and display defined macro and
2756 skipped directories where applicable.
2757
2758 =item B<--target> | B<-t>
2759
2760 Print the config attributes for this config target.
2761
2762 =item B<--environment> | B<-e>
2763
2764 Print the environment variables and their values at the time of configuration.
2765
2766 =item B<--make-variables> | B<-m>
2767
2768 Print the main make variables generated in the current configuration
2769
2770 =item B<--build-parameters> | B<-b>
2771
2772 Print the build parameters, i.e. build file and build file templates.
2773
2774 =item B<--reconfigure> | B<--reconf> | B<-r>
2775
2776 Redo the configuration.
2777
2778 =item B<--verbose> | B<-v>
2779
2780 Verbose output.
2781
2782 =back
2783
2784 =cut
2785
2786 EOF
2787 close(OUT);
2788 if ($builder_platform eq 'unix') {
2789 my $mode = (0755 & ~umask);
2790 chmod $mode, 'configdata.pm'
2791 or warn sprintf("WARNING: Couldn't change mode for 'configdata.pm' to 0%03o: %s\n",$mode,$!);
2792 }
2793
2794 my %builders = (
2795 unified => sub {
2796 print 'Creating ',$target{build_file},"\n";
2797 run_dofile(catfile($blddir, $target{build_file}),
2798 @{$config{build_file_templates}});
2799 },
2800 );
2801
2802 $builders{$builder}->($builder_platform, @builder_opts);
2803
2804 $SIG{__DIE__} = $orig_death_handler;
2805
2806 print <<"EOF" if ($disabled{threads} eq "unavailable");
2807
2808 The library could not be configured for supporting multi-threaded
2809 applications as the compiler options required on this system are not known.
2810 See file INSTALL for details if you need multi-threading.
2811 EOF
2812
2813 print <<"EOF" if ($no_shared_warn);
2814
2815 The options 'shared', 'pic' and 'dynamic-engine' aren't supported on this
2816 platform, so we will pretend you gave the option 'no-pic', which also disables
2817 'shared' and 'dynamic-engine'. If you know how to implement shared libraries
2818 or position independent code, please let us know (but please first make sure
2819 you have tried with a current version of OpenSSL).
2820 EOF
2821
2822 print <<"EOF";
2823
2824 **********************************************************************
2825 *** ***
2826 *** OpenSSL has been successfully configured ***
2827 *** ***
2828 *** If you encounter a problem while building, please open an ***
2829 *** issue on GitHub <https://github.com/openssl/openssl/issues> ***
2830 *** and include the output from the following command: ***
2831 *** ***
2832 *** perl configdata.pm --dump ***
2833 *** ***
2834 *** (If you are new to OpenSSL, you might want to consult the ***
2835 *** 'Troubleshooting' section in the INSTALL file first) ***
2836 *** ***
2837 **********************************************************************
2838 EOF
2839
2840 exit(0);
2841
2842 ######################################################################
2843 #
2844 # Helpers and utility functions
2845 #
2846
2847 # Death handler, to print a helpful message in case of failure #######
2848 #
2849 sub death_handler {
2850 die @_ if $^S; # To prevent the added message in eval blocks
2851 my $build_file = $target{build_file} // "build file";
2852 my @message = ( <<"_____", @_ );
2853
2854 Failure! $build_file wasn't produced.
2855 Please read INSTALL and associated NOTES files. You may also have to look over
2856 your available compiler tool chain or change your configuration.
2857
2858 _____
2859
2860 # Dying is terminal, so it's ok to reset the signal handler here.
2861 $SIG{__DIE__} = $orig_death_handler;
2862 die @message;
2863 }
2864
2865 # Configuration file reading #########################################
2866
2867 # Note: All of the helper functions are for lazy evaluation. They all
2868 # return a CODE ref, which will return the intended value when evaluated.
2869 # Thus, whenever there's mention of a returned value, it's about that
2870 # intended value.
2871
2872 # Helper function to implement conditional inheritance depending on the
2873 # value of $disabled{asm}. Used in inherit_from values as follows:
2874 #
2875 # inherit_from => [ "template", asm("asm_tmpl") ]
2876 #
2877 sub asm {
2878 my @x = @_;
2879 sub {
2880 $disabled{asm} ? () : @x;
2881 }
2882 }
2883
2884 # Helper function to implement conditional value variants, with a default
2885 # plus additional values based on the value of $config{build_type}.
2886 # Arguments are given in hash table form:
2887 #
2888 # picker(default => "Basic string: ",
2889 # debug => "debug",
2890 # release => "release")
2891 #
2892 # When configuring with --debug, the resulting string will be
2893 # "Basic string: debug", and when not, it will be "Basic string: release"
2894 #
2895 # This can be used to create variants of sets of flags according to the
2896 # build type:
2897 #
2898 # cflags => picker(default => "-Wall",
2899 # debug => "-g -O0",
2900 # release => "-O3")
2901 #
2902 sub picker {
2903 my %opts = @_;
2904 return sub { add($opts{default} || (),
2905 $opts{$config{build_type}} || ())->(); }
2906 }
2907
2908 # Helper function to combine several values of different types into one.
2909 # This is useful if you want to combine a string with the result of a
2910 # lazy function, such as:
2911 #
2912 # cflags => combine("-Wall", sub { $disabled{zlib} ? () : "-DZLIB" })
2913 #
2914 sub combine {
2915 my @stuff = @_;
2916 return sub { add(@stuff)->(); }
2917 }
2918
2919 # Helper function to implement conditional values depending on the value
2920 # of $disabled{threads}. Can be used as follows:
2921 #
2922 # cflags => combine("-Wall", threads("-pthread"))
2923 #
2924 sub threads {
2925 my @flags = @_;
2926 return sub { add($disabled{threads} ? () : @flags)->(); }
2927 }
2928
2929 sub shared {
2930 my @flags = @_;
2931 return sub { add($disabled{shared} ? () : @flags)->(); }
2932 }
2933
2934 our $add_called = 0;
2935 # Helper function to implement adding values to already existing configuration
2936 # values. It handles elements that are ARRAYs, CODEs and scalars
2937 sub _add {
2938 my $separator = shift;
2939
2940 # If there's any ARRAY in the collection of values OR the separator
2941 # is undef, we will return an ARRAY of combined values, otherwise a
2942 # string of joined values with $separator as the separator.
2943 my $found_array = !defined($separator);
2944
2945 my @values =
2946 map {
2947 my $res = $_;
2948 while (ref($res) eq "CODE") {
2949 $res = $res->();
2950 }
2951 if (defined($res)) {
2952 if (ref($res) eq "ARRAY") {
2953 $found_array = 1;
2954 @$res;
2955 } else {
2956 $res;
2957 }
2958 } else {
2959 ();
2960 }
2961 } (@_);
2962
2963 $add_called = 1;
2964
2965 if ($found_array) {
2966 [ @values ];
2967 } else {
2968 join($separator, grep { defined($_) && $_ ne "" } @values);
2969 }
2970 }
2971 sub add_before {
2972 my $separator = " ";
2973 if (ref($_[$#_]) eq "HASH") {
2974 my $opts = pop;
2975 $separator = $opts->{separator};
2976 }
2977 my @x = @_;
2978 sub { _add($separator, @x, @_) };
2979 }
2980 sub add {
2981 my $separator = " ";
2982 if (ref($_[$#_]) eq "HASH") {
2983 my $opts = pop;
2984 $separator = $opts->{separator};
2985 }
2986 my @x = @_;
2987 sub { _add($separator, @_, @x) };
2988 }
2989
2990 sub read_eval_file {
2991 my $fname = shift;
2992 my $content;
2993 my @result;
2994
2995 open F, "< $fname" or die "Can't open '$fname': $!\n";
2996 {
2997 undef local $/;
2998 $content = <F>;
2999 }
3000 close F;
3001 {
3002 local $@;
3003
3004 @result = ( eval $content );
3005 warn $@ if $@;
3006 }
3007 return wantarray ? @result : $result[0];
3008 }
3009
3010 # configuration reader, evaluates the input file as a perl script and expects
3011 # it to fill %targets with target configurations. Those are then added to
3012 # %table.
3013 sub read_config {
3014 my $fname = shift;
3015 my %targets;
3016
3017 {
3018 # Protect certain tables from tampering
3019 local %table = ();
3020
3021 %targets = read_eval_file($fname);
3022 }
3023 my %preexisting = ();
3024 foreach (sort keys %targets) {
3025 $preexisting{$_} = 1 if $table{$_};
3026 }
3027 die <<"EOF",
3028 The following config targets from $fname
3029 shadow pre-existing config targets with the same name:
3030 EOF
3031 map { " $_\n" } sort keys %preexisting
3032 if %preexisting;
3033
3034
3035 # For each target, check that it's configured with a hash table.
3036 foreach (keys %targets) {
3037 if (ref($targets{$_}) ne "HASH") {
3038 if (ref($targets{$_}) eq "") {
3039 warn "Deprecated target configuration for $_, ignoring...\n";
3040 } else {
3041 warn "Misconfigured target configuration for $_ (should be a hash table), ignoring...\n";
3042 }
3043 delete $targets{$_};
3044 } else {
3045 $targets{$_}->{_conf_fname_int} = add([ $fname ]);
3046 }
3047 }
3048
3049 %table = (%table, %targets);
3050
3051 }
3052
3053 # configuration resolver. Will only resolve all the lazy evaluation
3054 # codeblocks for the chosen target and all those it inherits from,
3055 # recursively
3056 sub resolve_config {
3057 my $target = shift;
3058 my @breadcrumbs = @_;
3059
3060 # my $extra_checks = defined($ENV{CONFIGURE_EXTRA_CHECKS});
3061
3062 if (grep { $_ eq $target } @breadcrumbs) {
3063 die "inherit_from loop! target backtrace:\n "
3064 ,$target,"\n ",join("\n ", @breadcrumbs),"\n";
3065 }
3066
3067 if (!defined($table{$target})) {
3068 warn "Warning! target $target doesn't exist!\n";
3069 return ();
3070 }
3071 # Recurse through all inheritances. They will be resolved on the
3072 # fly, so when this operation is done, they will all just be a
3073 # bunch of attributes with string values.
3074 # What we get here, though, are keys with references to lists of
3075 # the combined values of them all. We will deal with lists after
3076 # this stage is done.
3077 my %combined_inheritance = ();
3078 if ($table{$target}->{inherit_from}) {
3079 my @inherit_from =
3080 map { ref($_) eq "CODE" ? $_->() : $_ } @{$table{$target}->{inherit_from}};
3081 foreach (@inherit_from) {
3082 my %inherited_config = resolve_config($_, $target, @breadcrumbs);
3083
3084 # 'template' is a marker that's considered private to
3085 # the config that had it.
3086 delete $inherited_config{template};
3087
3088 foreach (keys %inherited_config) {
3089 if (!$combined_inheritance{$_}) {
3090 $combined_inheritance{$_} = [];
3091 }
3092 push @{$combined_inheritance{$_}}, $inherited_config{$_};
3093 }
3094 }
3095 }
3096
3097 # We won't need inherit_from in this target any more, since we've
3098 # resolved all the inheritances that lead to this
3099 delete $table{$target}->{inherit_from};
3100
3101 # Now is the time to deal with those lists. Here's the place to
3102 # decide what shall be done with those lists, all based on the
3103 # values of the target we're currently dealing with.
3104 # - If a value is a coderef, it will be executed with the list of
3105 # inherited values as arguments.
3106 # - If the corresponding key doesn't have a value at all or is the
3107 # empty string, the inherited value list will be run through the
3108 # default combiner (below), and the result becomes this target's
3109 # value.
3110 # - Otherwise, this target's value is assumed to be a string that
3111 # will simply override the inherited list of values.
3112 my $default_combiner = add();
3113
3114 my %all_keys =
3115 map { $_ => 1 } (keys %combined_inheritance,
3116 keys %{$table{$target}});
3117
3118 sub process_values {
3119 my $object = shift;
3120 my $inherited = shift; # Always a [ list ]
3121 my $target = shift;
3122 my $entry = shift;
3123
3124 $add_called = 0;
3125
3126 while(ref($object) eq "CODE") {
3127 $object = $object->(@$inherited);
3128 }
3129 if (!defined($object)) {
3130 return ();
3131 }
3132 elsif (ref($object) eq "ARRAY") {
3133 local $add_called; # To make sure recursive calls don't affect it
3134 return [ map { process_values($_, $inherited, $target, $entry) }
3135 @$object ];
3136 } elsif (ref($object) eq "") {
3137 return $object;
3138 } else {
3139 die "cannot handle reference type ",ref($object)
3140 ," found in target ",$target," -> ",$entry,"\n";
3141 }
3142 }
3143
3144 foreach (sort keys %all_keys) {
3145 my $previous = $combined_inheritance{$_};
3146
3147 # Current target doesn't have a value for the current key?
3148 # Assign it the default combiner, the rest of this loop body
3149 # will handle it just like any other coderef.
3150 if (!exists $table{$target}->{$_}) {
3151 $table{$target}->{$_} = $default_combiner;
3152 }
3153
3154 $table{$target}->{$_} = process_values($table{$target}->{$_},
3155 $combined_inheritance{$_},
3156 $target, $_);
3157 unless(defined($table{$target}->{$_})) {
3158 delete $table{$target}->{$_};
3159 }
3160 # if ($extra_checks &&
3161 # $previous && !($add_called || $previous ~~ $table{$target}->{$_})) {
3162 # warn "$_ got replaced in $target\n";
3163 # }
3164 }
3165
3166 # Finally done, return the result.
3167 return %{$table{$target}};
3168 }
3169
3170 sub usage
3171 {
3172 print STDERR $usage;
3173 print STDERR "\npick os/compiler from:\n";
3174 my $j=0;
3175 my $i;
3176 my $k=0;
3177 foreach $i (sort keys %table)
3178 {
3179 next if $table{$i}->{template};
3180 next if $i =~ /^debug/;
3181 $k += length($i) + 1;
3182 if ($k > 78)
3183 {
3184 print STDERR "\n";
3185 $k=length($i);
3186 }
3187 print STDERR $i . " ";
3188 }
3189 foreach $i (sort keys %table)
3190 {
3191 next if $table{$i}->{template};
3192 next if $i !~ /^debug/;
3193 $k += length($i) + 1;
3194 if ($k > 78)
3195 {
3196 print STDERR "\n";
3197 $k=length($i);
3198 }
3199 print STDERR $i . " ";
3200 }
3201 print STDERR "\n\nNOTE: If in doubt, on Unix-ish systems use './config'.\n";
3202 exit(1);
3203 }
3204
3205 sub run_dofile
3206 {
3207 my $out = shift;
3208 my @templates = @_;
3209
3210 unlink $out || warn "Can't remove $out, $!"
3211 if -f $out;
3212 foreach (@templates) {
3213 die "Can't open $_, $!" unless -f $_;
3214 }
3215 my $perlcmd = (quotify("maybeshell", $config{PERL}))[0];
3216 my $cmd = "$perlcmd \"-I.\" \"-Mconfigdata\" \"$dofile\" -o\"Configure\" \"".join("\" \"",@templates)."\" > \"$out.new\"";
3217 #print STDERR "DEBUG[run_dofile]: \$cmd = $cmd\n";
3218 system($cmd);
3219 exit 1 if $? != 0;
3220 rename("$out.new", $out) || die "Can't rename $out.new, $!";
3221 }
3222
3223 sub compiler_predefined {
3224 state %predefined;
3225 my $cc = shift;
3226
3227 return () if $^O eq 'VMS';
3228
3229 die 'compiler_predefined called without a compiler command'
3230 unless $cc;
3231
3232 if (! $predefined{$cc}) {
3233
3234 $predefined{$cc} = {};
3235
3236 # collect compiler pre-defines from gcc or gcc-alike...
3237 open(PIPE, "$cc -dM -E -x c /dev/null 2>&1 |");
3238 while (my $l = <PIPE>) {
3239 $l =~ m/^#define\s+(\w+(?:\(\w+\))?)(?:\s+(.+))?/ or last;
3240 $predefined{$cc}->{$1} = $2 // '';
3241 }
3242 close(PIPE);
3243 }
3244
3245 return %{$predefined{$cc}};
3246 }
3247
3248 sub which
3249 {
3250 my ($name)=@_;
3251
3252 if (eval { require IPC::Cmd; 1; }) {
3253 IPC::Cmd->import();
3254 return scalar IPC::Cmd::can_run($name);
3255 } else {
3256 # if there is $directories component in splitpath,
3257 # then it's not something to test with $PATH...
3258 return $name if (File::Spec->splitpath($name))[1];
3259
3260 foreach (File::Spec->path()) {
3261 my $fullpath = catfile($_, "$name$target{exe_extension}");
3262 if (-f $fullpath and -x $fullpath) {
3263 return $fullpath;
3264 }
3265 }
3266 }
3267 }
3268
3269 sub env
3270 {
3271 my $name = shift;
3272 my %opts = @_;
3273
3274 unless ($opts{cacheonly}) {
3275 # Note that if $ENV{$name} doesn't exist or is undefined,
3276 # $config{perlenv}->{$name} will be created with the value
3277 # undef. This is intentional.
3278
3279 $config{perlenv}->{$name} = $ENV{$name}
3280 if ! exists $config{perlenv}->{$name};
3281 }
3282 return $config{perlenv}->{$name};
3283 }
3284
3285 # Configuration printer ##############################################
3286
3287 sub print_table_entry
3288 {
3289 local $now_printing = shift;
3290 my %target = resolve_config($now_printing);
3291 my $type = shift;
3292
3293 # Don't print the templates
3294 return if $target{template};
3295
3296 my @sequence = (
3297 "sys_id",
3298 "cpp",
3299 "cppflags",
3300 "defines",
3301 "includes",
3302 "cc",
3303 "cflags",
3304 "unistd",
3305 "ld",
3306 "lflags",
3307 "loutflag",
3308 "ex_libs",
3309 "bn_ops",
3310 "apps_aux_src",
3311 "cpuid_asm_src",
3312 "uplink_aux_src",
3313 "bn_asm_src",
3314 "ec_asm_src",
3315 "des_asm_src",
3316 "aes_asm_src",
3317 "bf_asm_src",
3318 "md5_asm_src",
3319 "cast_asm_src",
3320 "sha1_asm_src",
3321 "rc4_asm_src",
3322 "rmd160_asm_src",
3323 "rc5_asm_src",
3324 "wp_asm_src",
3325 "cmll_asm_src",
3326 "modes_asm_src",
3327 "padlock_asm_src",
3328 "chacha_asm_src",
3329 "poly1035_asm_src",
3330 "thread_scheme",
3331 "perlasm_scheme",
3332 "dso_scheme",
3333 "shared_target",
3334 "shared_cflag",
3335 "shared_defines",
3336 "shared_ldflag",
3337 "shared_rcflag",
3338 "shared_extension",
3339 "dso_extension",
3340 "obj_extension",
3341 "exe_extension",
3342 "ranlib",
3343 "ar",
3344 "arflags",
3345 "aroutflag",
3346 "rc",
3347 "rcflags",
3348 "rcoutflag",
3349 "mt",
3350 "mtflags",
3351 "mtinflag",
3352 "mtoutflag",
3353 "multilib",
3354 "build_scheme",
3355 );
3356
3357 if ($type eq "TABLE") {
3358 print "\n";
3359 print "*** $now_printing\n";
3360 foreach (@sequence) {
3361 if (ref($target{$_}) eq "ARRAY") {
3362 printf "\$%-12s = %s\n", $_, join(" ", @{$target{$_}});
3363 } else {
3364 printf "\$%-12s = %s\n", $_, $target{$_};
3365 }
3366 }
3367 } elsif ($type eq "HASH") {
3368 my $largest =
3369 length((sort { length($a) <=> length($b) } @sequence)[-1]);
3370 print " '$now_printing' => {\n";
3371 foreach (@sequence) {
3372 if ($target{$_}) {
3373 if (ref($target{$_}) eq "ARRAY") {
3374 print " '",$_,"'"," " x ($largest - length($_))," => [ ",join(", ", map { "'$_'" } @{$target{$_}})," ],\n";
3375 } else {
3376 print " '",$_,"'"," " x ($largest - length($_))," => '",$target{$_},"',\n";
3377 }
3378 }
3379 }
3380 print " },\n";
3381 }
3382 }
3383
3384 # Utility routines ###################################################
3385
3386 # On VMS, if the given file is a logical name, File::Spec::Functions
3387 # will consider it an absolute path. There are cases when we want a
3388 # purely syntactic check without checking the environment.
3389 sub isabsolute {
3390 my $file = shift;
3391
3392 # On non-platforms, we just use file_name_is_absolute().
3393 return file_name_is_absolute($file) unless $^O eq "VMS";
3394
3395 # If the file spec includes a device or a directory spec,
3396 # file_name_is_absolute() is perfectly safe.
3397 return file_name_is_absolute($file) if $file =~ m|[:\[]|;
3398
3399 # Here, we know the given file spec isn't absolute
3400 return 0;
3401 }
3402
3403 # Makes a directory absolute and cleans out /../ in paths like foo/../bar
3404 # On some platforms, this uses rel2abs(), while on others, realpath() is used.
3405 # realpath() requires that at least all path components except the last is an
3406 # existing directory. On VMS, the last component of the directory spec must
3407 # exist.
3408 sub absolutedir {
3409 my $dir = shift;
3410
3411 # realpath() is quite buggy on VMS. It uses LIB$FID_TO_NAME, which
3412 # will return the volume name for the device, no matter what. Also,
3413 # it will return an incorrect directory spec if the argument is a
3414 # directory that doesn't exist.
3415 if ($^O eq "VMS") {
3416 return rel2abs($dir);
3417 }
3418
3419 # We use realpath() on Unix, since no other will properly clean out
3420 # a directory spec.
3421 use Cwd qw/realpath/;
3422
3423 return realpath($dir);
3424 }
3425
3426 sub quotify {
3427 my %processors = (
3428 perl => sub { my $x = shift;
3429 $x =~ s/([\\\$\@"])/\\$1/g;
3430 return '"'.$x.'"'; },
3431 maybeshell => sub { my $x = shift;
3432 (my $y = $x) =~ s/([\\\"])/\\$1/g;
3433 if ($x ne $y || $x =~ m|\s|) {
3434 return '"'.$y.'"';
3435 } else {
3436 return $x;
3437 }
3438 },
3439 );
3440 my $for = shift;
3441 my $processor =
3442 defined($processors{$for}) ? $processors{$for} : sub { shift; };
3443
3444 return map { $processor->($_); } @_;
3445 }
3446
3447 # collect_from_file($filename, $line_concat_cond_re, $line_concat)
3448 # $filename is a file name to read from
3449 # $line_concat_cond_re is a regexp detecting a line continuation ending
3450 # $line_concat is a CODEref that takes care of concatenating two lines
3451 sub collect_from_file {
3452 my $filename = shift;
3453 my $line_concat_cond_re = shift;
3454 my $line_concat = shift;
3455
3456 open my $fh, $filename || die "unable to read $filename: $!\n";
3457 return sub {
3458 my $saved_line = "";
3459 $_ = "";
3460 while (<$fh>) {
3461 s|\R$||;
3462 if (defined $line_concat) {
3463 $_ = $line_concat->($saved_line, $_);
3464 $saved_line = "";
3465 }
3466 if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
3467 $saved_line = $_;
3468 next;
3469 }
3470 return $_;
3471 }
3472 die "$filename ending with continuation line\n" if $_;
3473 close $fh;
3474 return undef;
3475 }
3476 }
3477
3478 # collect_from_array($array, $line_concat_cond_re, $line_concat)
3479 # $array is an ARRAYref of lines
3480 # $line_concat_cond_re is a regexp detecting a line continuation ending
3481 # $line_concat is a CODEref that takes care of concatenating two lines
3482 sub collect_from_array {
3483 my $array = shift;
3484 my $line_concat_cond_re = shift;
3485 my $line_concat = shift;
3486 my @array = (@$array);
3487
3488 return sub {
3489 my $saved_line = "";
3490 $_ = "";
3491 while (defined($_ = shift @array)) {
3492 s|\R$||;
3493 if (defined $line_concat) {
3494 $_ = $line_concat->($saved_line, $_);
3495 $saved_line = "";
3496 }
3497 if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
3498 $saved_line = $_;
3499 next;
3500 }
3501 return $_;
3502 }
3503 die "input text ending with continuation line\n" if $_;
3504 return undef;
3505 }
3506 }
3507
3508 # collect_information($lineiterator, $line_continue, $regexp => $CODEref, ...)
3509 # $lineiterator is a CODEref that delivers one line at a time.
3510 # All following arguments are regex/CODEref pairs, where the regexp detects a
3511 # line and the CODEref does something with the result of the regexp.
3512 sub collect_information {
3513 my $lineiterator = shift;
3514 my %collectors = @_;
3515
3516 while(defined($_ = $lineiterator->())) {
3517 s|\R$||;
3518 my $found = 0;
3519 if ($collectors{"BEFORE"}) {
3520 $collectors{"BEFORE"}->($_);
3521 }
3522 foreach my $re (keys %collectors) {
3523 if ($re !~ /^OTHERWISE|BEFORE|AFTER$/ && /$re/) {
3524 $collectors{$re}->($lineiterator);
3525 $found = 1;
3526 };
3527 }
3528 if ($collectors{"OTHERWISE"}) {
3529 $collectors{"OTHERWISE"}->($lineiterator, $_)
3530 unless $found || !defined $collectors{"OTHERWISE"};
3531 }
3532 if ($collectors{"AFTER"}) {
3533 $collectors{"AFTER"}->($_);
3534 }
3535 }
3536 }
3537
3538 # tokenize($line)
3539 # tokenize($line,$separator)
3540 # $line is a line of text to split up into tokens
3541 # $separator [optional] is a regular expression that separates the tokens,
3542 # the default being spaces. Do not use quotes of any kind as separators,
3543 # that will give undefined results.
3544 # Returns a list of tokens.
3545 #
3546 # Tokens are divided by separator (spaces by default). If the tokens include
3547 # the separators, they have to be quoted with single or double quotes.
3548 # Double quotes inside a double quoted token must be escaped. Escaping is done
3549 # with backslash.
3550 # Basically, the same quoting rules apply for " and ' as in any
3551 # Unix shell.
3552 sub tokenize {
3553 my $line = my $debug_line = shift;
3554 my $separator = shift // qr|\s+|;
3555 my @result = ();
3556
3557 if ($ENV{CONFIGURE_DEBUG_TOKENIZE}) {
3558 print STDERR "DEBUG[tokenize]: \$separator = $separator\n";
3559 }
3560
3561 while ($line =~ s|^${separator}||, $line ne "") {
3562 my $token = "";
3563 again:
3564 $line =~ m/^(.*?)(${separator}|"|'|$)/;
3565 $token .= $1;
3566 $line = $2.$';
3567
3568 if ($line =~ m/^"((?:[^"\\]+|\\.)*)"/) {
3569 $token .= $1;
3570 $line = $';
3571 goto again;
3572 } elsif ($line =~ m/^'([^']*)'/) {
3573 $token .= $1;
3574 $line = $';
3575 goto again;
3576 }
3577 push @result, $token;
3578 }
3579
3580 if ($ENV{CONFIGURE_DEBUG_TOKENIZE}) {
3581 print STDERR "DEBUG[tokenize]: Parsed '$debug_line' into:\n";
3582 print STDERR "DEBUG[tokenize]: ('", join("', '", @result), "')\n";
3583 }
3584 return @result;
3585 }