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