mirror of
https://https.git.savannah.gnu.org/git/make.git
synced 2026-01-30 03:14:33 +00:00
When using Perl's backticks make sure that the path to the command invoked is quoted. On Windows, in particular, paths to commands such as diff, etc. may contain whitespace.
284 lines
8.1 KiB
Perl
284 lines
8.1 KiB
Perl
# -*-mode: perl-*-
|
|
|
|
$description = "Test GNU Make's archive management features.";
|
|
|
|
$details = "\
|
|
This only works on systems that support it.";
|
|
|
|
# If this instance of make doesn't support archives, skip it
|
|
exists $FEATURES{archives} or return -1;
|
|
|
|
# In theory archive support exists on Windows but it doesn't use ar;
|
|
# someone will need to port this test.
|
|
$port_type eq 'W32' and return -1;
|
|
|
|
# Create some .o files to work with
|
|
if ($osname eq 'VMS') {
|
|
# VMS AR needs real object files at this time.
|
|
foreach $afile ('a1', 'a2', 'a3') {
|
|
# Use non-standard extension to prevent implicit rules from recreating
|
|
# objects when the test tampers with the timestamp.
|
|
1 while unlink "$afile.c1";
|
|
1 while unlink "$afile.o";
|
|
create_file("$afile.c1", "int $afile(void) {return 1;}\n");
|
|
system("cc $afile.c1 /object=$afile.o");
|
|
}
|
|
} else {
|
|
utouch(-60, qw(a1.o a2.o a3.o));
|
|
}
|
|
|
|
# Fallback if configure did not find AR
|
|
my $ar = get_config('AR') || 'ar';
|
|
|
|
my $redir = '2>&1';
|
|
$redir = '' if $osname eq 'VMS';
|
|
|
|
# This is the value from src/default.c
|
|
my $arflags = $osname eq 'aix' ? '-Xany -rv' : '-rv';
|
|
my $arvar = "AR=\"$ar\"";
|
|
|
|
# Newer versions of binutils can be built with --enable-deterministic-archives
|
|
# which forces all timestamps (among other things) to always be 0, defeating
|
|
# GNU Make's archive support. See if ar supports the U option to disable it.
|
|
unlink('libxx.a');
|
|
$_ = `"$ar" ${arflags}U libxx.a a1.o $redir`;
|
|
if ($? == 0) {
|
|
$arflags = "${arflags}U";
|
|
$arvar = "$arvar ARFLAGS=\"$arflags\"";
|
|
}
|
|
|
|
# Some versions of ar print different things on creation. Find out.
|
|
unlink('libxx.a');
|
|
my $created = `"$ar" $arflags libxx.a a1.o $redir`;
|
|
$created =~ s/a1\.o/#OBJECT#/g;
|
|
|
|
# Some versions of ar print different things on add. Find out.
|
|
my $add = `"$ar" $arflags libxx.a a2.o $redir`;
|
|
$add =~ s/a2\.o/#OBJECT#/g;
|
|
|
|
# Some versions of ar print different things on replacement. Find out.
|
|
my $repl = `"$ar" $arflags libxx.a a2.o $redir`;
|
|
$repl =~ s/a2\.o/#OBJECT#/g;
|
|
|
|
unlink('libxx.a');
|
|
|
|
# Very simple
|
|
($_ = $created) =~ s/#OBJECT#/a1.o/g;
|
|
my $answer = "$ar $arflags libxx.a a1.o\n$_";
|
|
if ($port_type eq 'VMS-DCL') {
|
|
$answer = 'library /replace libxx.a a1.o';
|
|
}
|
|
run_make_test('all: libxx.a(a1.o)', $arvar, $answer);
|
|
|
|
# Multiple .o's. Add a new one to the existing library
|
|
($_ = $add) =~ s/#OBJECT#/a2.o/g;
|
|
|
|
$answer = "$ar $arflags libxx.a a2.o\n$_";
|
|
if ($port_type eq 'VMS-DCL') {
|
|
$answer = 'library /replace libxx.a a2.o';
|
|
}
|
|
run_make_test('all: libxx.a(a1.o a2.o)', $arvar, $answer);
|
|
|
|
# Touch one of the .o's so it's rebuilt
|
|
if ($port_type eq 'VMS-DCL') {
|
|
# utouch is not changing what VMS library compare is testing for.
|
|
# So do a real change by regenerating the file.
|
|
1 while unlink('a1.o');
|
|
# Later time stamp than last insertion.
|
|
sleep(2);
|
|
system('cc a1.c1 /object=a1.o');
|
|
# Next insertion will have a later timestamp.
|
|
sleep(2);
|
|
} else {
|
|
utouch(-40, 'a1.o');
|
|
}
|
|
|
|
($_ = $repl) =~ s/#OBJECT#/a1.o/g;
|
|
$answer = "$ar $arflags libxx.a a1.o\n$_";
|
|
if ($port_type eq 'VMS-DCL') {
|
|
$answer = 'library /replace libxx.a a1.o';
|
|
}
|
|
run_make_test(undef, $arvar, $answer);
|
|
|
|
# Use wildcards
|
|
$answer = "#MAKE#: Nothing to be done for 'all'.\n";
|
|
run_make_test('all: libxx.a(*.o)', $arvar, $answer);
|
|
|
|
# Touch one of the .o's so it's rebuilt
|
|
if ($port_type eq 'VMS-DCL') {
|
|
# utouch is not changing what VMS library compare is testing for.
|
|
# So do a real change by regenerating the file.
|
|
1 while unlink('a1.o');
|
|
# Make timestamp later than last insertion.
|
|
sleep(2);
|
|
system('cc a1.c1 /object=a1.o');
|
|
} else {
|
|
utouch(-30, 'a1.o');
|
|
}
|
|
($_ = $repl) =~ s/#OBJECT#/a1.o/g;
|
|
$answer = "$ar $arflags libxx.a a1.o\n$_";
|
|
if ($port_type eq 'VMS-DCL') {
|
|
$answer = 'library /replace libxx.a a1.o';
|
|
}
|
|
run_make_test(undef, $arvar, $answer);
|
|
|
|
# Use both wildcards and simple names
|
|
if ($port_type eq 'VMS-DCL') {
|
|
# utouch is not changing what VMS library compare is testing for.
|
|
# So do a real change by regenerating the file.
|
|
1 while unlink('a2.o');
|
|
sleep(2);
|
|
system('cc a2.c1 /object=a2.o');
|
|
} else {
|
|
utouch(-50, 'a2.o');
|
|
}
|
|
($_ = $add) =~ s/#OBJECT#/a3.o/g;
|
|
$_ .= "$ar $arflags libxx.a a2.o\n";
|
|
($_ .= $repl) =~ s/#OBJECT#/a2.o/g;
|
|
$answer = "$ar $arflags libxx.a a3.o\n$_";
|
|
if ($port_type eq 'VMS-DCL') {
|
|
$answer = 'library /replace libxx.a a3.o';
|
|
}
|
|
|
|
run_make_test('all: libxx.a(a3.o *.o)', $arvar, $answer);
|
|
|
|
# Check whitespace handling
|
|
if ($port_type eq 'VMS-DCL') {
|
|
# utouch is not changing what VMS library compare is testing for.
|
|
# So do a real change by regenerating the file.
|
|
1 while unlink('a2.o');
|
|
sleep(2);
|
|
system('cc a2.c1 /object=a2.o');
|
|
} else {
|
|
utouch(-40, 'a2.o');
|
|
}
|
|
($_ = $repl) =~ s/#OBJECT#/a2.o/g;
|
|
$answer = "$ar $arflags libxx.a a2.o\n$_";
|
|
if ($port_type eq 'VMS-DCL') {
|
|
$answer = 'library /replace libxx.a a2.o';
|
|
}
|
|
run_make_test('all: libxx.a( a3.o *.o )', $arvar, $answer);
|
|
|
|
rmfiles(qw(a1.c1 a2.c1 a3.c1 a1.o a2.o a3.o libxx.a));
|
|
|
|
# Check non-archive targets
|
|
# See Savannah bug #37878
|
|
$mk_string = q!
|
|
all: foo(bar).baz
|
|
foo(bar).baz: ; @echo '$@'
|
|
!;
|
|
|
|
if ($port_type eq 'VMS-DCL') {
|
|
$mk_string =~ s/echo/write sys\$\$output/;
|
|
$mk_string =~ s/\'/\"/g;
|
|
}
|
|
run_make_test($mk_string, $arvar, "foo(bar).baz\n");
|
|
|
|
# Check renaming of archive targets.
|
|
# See Savannah bug #38442
|
|
|
|
mkdir('artest', 0777);
|
|
touch('foo.vhd');
|
|
$mk_string = q!
|
|
DIR = artest
|
|
vpath % $(DIR)
|
|
default: lib(foo)
|
|
(%): %.vhd ; @cd $(DIR) && touch $(*F) && $(AR) $(ARFLAGS) $@ $(*F) >/dev/null 2>&1 && rm $(*F)
|
|
.PHONY: default
|
|
!;
|
|
if ($port_type eq 'VMS-DCL') {
|
|
$mk_string =~ s#= artest#= sys\$\$disk:\[.artest\]#;
|
|
$mk_string =~ s#lib\(foo\)#lib.tlb\(foo\)#;
|
|
$mk_string =~ s#; \@cd#; pipe SET DEFAULT#;
|
|
$mk_string =~
|
|
s#touch \$\(\*F\)#touch \$\(\*F\) && library/create/text sys\$\$disk:\$\@#;
|
|
$mk_string =~
|
|
s#library#if f\$\$search(\"\$\@\") \.eqs\. \"\" then library#;
|
|
# VMS needs special handling for null extension
|
|
$mk_string =~ s#\@ \$\(\*F\)#\@ \$\(\*F\)\.#;
|
|
$mk_string =~ s#>/dev/null 2>&1 ##;
|
|
}
|
|
run_make_test($mk_string, $arvar, "");
|
|
|
|
run_make_test(undef, $arvar, "#MAKE#: Nothing to be done for 'default'.\n");
|
|
|
|
unlink('foo.vhd');
|
|
if ($osname eq 'VMS') {
|
|
remove_directory_tree("$cwdpath/artest");
|
|
} else {
|
|
remove_directory_tree('artest');
|
|
}
|
|
|
|
# Check long names for archive members.
|
|
# See Savannah bug #54395
|
|
|
|
if ($osname ne 'VMS' && $osname ne 'os390') {
|
|
my $pre = '1234567890123456';
|
|
my $lib = 'libxx.a';
|
|
my $cr = $created;
|
|
$cr =~ s/#OBJECT#/${pre}a/g;
|
|
my $ad = $add;
|
|
$ad =~ s/#OBJECT#/${pre}b/g;
|
|
|
|
run_make_test(qq!
|
|
# Both member names > 16 characters long
|
|
default: $lib(${pre}a) $lib(${pre}b)
|
|
|
|
(%): % ; \$(AR) \$(ARFLAGS) \$@ \$%
|
|
|
|
$pre%: ; touch \$\@
|
|
!,
|
|
$arvar, "touch ${pre}a\n$ar $arflags $lib ${pre}a\n${cr}touch ${pre}b\n$ar $arflags $lib ${pre}b\n${ad}rm ${pre}a ${pre}b\n");
|
|
|
|
# Run it again; nothing should happen
|
|
run_make_test(undef, $arvar, "#MAKE#: Nothing to be done for 'default'.\n");
|
|
|
|
unlink($lib);
|
|
}
|
|
|
|
# SV 61436 : Allow redefining archive rules to propagate timestamps
|
|
|
|
# These don't work right on z/OS for some reason: archives not fully supported?
|
|
|
|
if ($osname ne 'os390') {
|
|
# Find the output when creating an archive from multiple files
|
|
|
|
utouch(-10, 'a.o', 'b.o');
|
|
my $create2 = `"$ar" $arflags mylib.a a.o b.o $redir`;
|
|
touch('b.o');
|
|
my $add2 = `"$ar" $arflags mylib.a b.o $redir`;
|
|
unlink('a.o', 'b.o', 'mylib.a');
|
|
|
|
# Some systems complain when compiling empty files
|
|
create_file('a.c', 'int i;');
|
|
create_file('b.c', 'int j;');
|
|
utouch(-20, 'a.c', 'b.c');
|
|
|
|
my $cc = get_config('CC') || 'cc';
|
|
my $vars = "CC=\"$cc\" $arvar";
|
|
|
|
run_make_test(q!
|
|
mylib.a: mylib.a(a.o b.o)
|
|
(%): % ;
|
|
%.a: ; $(AR) $(ARFLAGS) $@ $?
|
|
%.o : %.c ; @echo Compile $<; $(COMPILE.c) -o $@ $<
|
|
!,
|
|
$vars, "Compile a.c\nCompile b.c\n$ar $arflags mylib.a a.o b.o\n${create2}rm b.o a.o");
|
|
|
|
run_make_test(undef, $vars, "#MAKE#: 'mylib.a' is up to date.");
|
|
|
|
# Now update one of the source files and it should be compiled and archived
|
|
|
|
sleep(2);
|
|
touch('b.c');
|
|
|
|
run_make_test(undef, $vars, "Compile b.c\n$ar $arflags mylib.a b.o\n${add2}rm b.o");
|
|
|
|
run_make_test(undef, $vars, "#MAKE#: 'mylib.a' is up to date.");
|
|
|
|
unlink('a.c', 'b.c', 'a.o', 'b.o', 'mylib.a');
|
|
}
|
|
|
|
# This tells the test driver that the perl test script executed properly.
|
|
1;
|