1 #! @PERL@ 2 3 # This script handles linking the tool executables on Linux, 4 # statically and at an alternative load address. 5 # 6 # Linking statically sidesteps all sorts of complications to do with 7 # having two copies of the dynamic linker (valgrind's and the 8 # client's) coexisting in the same process. The alternative load 9 # address is needed because Valgrind itself will load the client at 10 # whatever address it specifies, which is almost invariably the 11 # default load address. Hence we can't allow Valgrind itself (viz, 12 # the tool executable) to be loaded at that address. 13 # 14 # Unfortunately there's no standard way to do 'static link at 15 # alternative address', so these link_tool_exe_*.in scripts handle 16 # the per-platform hoop-jumping. 17 # 18 # What we get passed here is: 19 # first arg 20 # the alternative load address 21 # all the rest of the args 22 # the gcc invokation to do the final link, that 23 # the build system would have done, left to itself 24 # 25 # We just let the script 'die' if something is wrong, rather than do 26 # proper error reporting. We don't expect the users to run this 27 # directly. It is only run as part of the build process, with 28 # carefully constrained inputs. 29 # 30 # 31 # So: what we actually do is: 32 # 33 # Look at the specified gcc invokation. Ignore all parts of it except 34 # the *.a, *.o and -o outfile parts. Wrap them up in a new command 35 # which looks (eg) as follows: 36 # 37 # (64-bit): 38 # 39 # /usr/bin/ld -static -arch x86_64 -macosx_version_min 10.5 \ 40 # -o memcheck-amd64-darwin -u __start -e __start \ 41 # -image_base 0x138000000 -stack_addr 0x13c000000 \ 42 # -stack_size 0x800000 \ 43 # memcheck_amd*.o \ 44 # ../coregrind/libcoregrind-amd64-darwin.a \ 45 # ../VEX/libvex-amd64-darwin.a 46 # 47 # (32-bit) 48 # 49 # /usr/bin/ld -static -arch i386 -macosx_version_min 10.5 \ 50 # -o memcheck-x86-darwin -u __start -e __start \ 51 # -image_base 0x38000000 -stack_addr 0x3c000000 \ 52 # -stack_size 0x800000 \ 53 # memcheck_x86*.o \ 54 # ../coregrind/libcoregrind-x86-darwin.a \ 55 # ../VEX/libvex-x86-darwin.a 56 # 57 # The addresses shown above will actually work, although "for real" we 58 # of course need to take it from argv[1]. In these examples the stack 59 # is placed 64M after the executable start. It is probably safer to 60 # place it 64M before the executable's start point, so the executable 61 # + data + bss can grow arbitrarily in future without colliding with 62 # the stack. 63 # 64 # There's one more twist: we need to know the word size of the 65 # executable for which we are linking. We need to know this because 66 # we must tell the linker that, by handing it either "-arch x86_64" or 67 # "-arch i386". Fortunately we can figure this out by scanning the 68 # gcc invokation, which itself must contain either "-arch x86_64" or 69 # "-arch i386". 70 71 use warnings; 72 use strict; 73 # we need to be able to do 64-bit arithmetic: 74 use Math::BigInt; 75 76 77 # User configurable constants: how far before the exe should we 78 # place the stack? 79 my $TX_STACK_OFFSET_BEFORE_TEXT = 64 * 1024 * 1024; 80 81 # and how big should the stack be? 82 my $TX_STACK_SIZE = 8 * 1024 * 1024; 83 84 85 # string -> bool 86 sub is_dota_or_doto($) 87 { 88 my ($str) = @_; 89 if ($str =~ /.\.a$/ || $str =~ /.\.o$/) { 90 return 1; 91 } else { 92 return 0; 93 } 94 } 95 96 97 # expect at least: alt-load-address gcc -o foo bar.o 98 die "Not enough arguments" 99 if (($#ARGV + 1) < 5); 100 101 my $ala = $ARGV[0]; 102 103 # check for plausible-ish alt load address 104 die "Bogus alt-load address (1)" 105 if (length($ala) < 3 || index($ala, "0x") != 0); 106 107 die "Bogus alt-load address (2)" 108 if ($ala !~ /^0x[0-9a-fA-F]+$/); 109 110 111 # get hold of the outfile name (following "-o") 112 my $outname = ""; 113 114 foreach my $n (2 .. $#ARGV - 1) { 115 my $str = $ARGV[$n]; 116 if ($str eq "-o" && $outname eq "") { 117 $outname = $ARGV[$n + 1]; 118 } 119 } 120 121 die "Can't find '-o outfilename' in command line" 122 if ($outname eq ""); 123 124 125 # get hold of the string following "-arch" 126 my $archstr = ""; 127 128 foreach my $n (2 .. $#ARGV - 1) { 129 my $str = $ARGV[$n]; 130 if ($str eq "-arch" && $archstr eq "") { 131 $archstr = $ARGV[$n + 1]; 132 } 133 } 134 135 die "Can't find '-arch archstr' in command line" 136 if ($archstr eq ""); 137 138 139 # build the command line 140 my $cmd = "/usr/bin/ld"; 141 142 $cmd = "$cmd -static"; 143 $cmd = "$cmd -arch $archstr"; 144 $cmd = "$cmd -macosx_version_min 10.5"; 145 $cmd = "$cmd -o $outname"; 146 $cmd = "$cmd -u __start -e __start"; 147 148 my $stack_addr = Math::BigInt->new( $ala ) - $TX_STACK_OFFSET_BEFORE_TEXT; 149 my $stack_addr_str = $stack_addr->as_hex(); 150 my $stack_size_str = Math::BigInt::as_hex($TX_STACK_SIZE); 151 152 $cmd = "$cmd -image_base $ala"; 153 $cmd = "$cmd -stack_addr $stack_addr_str"; 154 $cmd = "$cmd -stack_size $stack_size_str"; 155 156 foreach my $n (2 .. $#ARGV) { 157 my $str = $ARGV[$n]; 158 if (is_dota_or_doto($str)) { 159 $cmd = "$cmd $str"; 160 } 161 } 162 163 #print "link_tool_exe_darwin: $cmd\n"; 164 165 166 # Execute the command: 167 my $r = system("$cmd"); 168 169 if ($r == 0) { 170 exit 0; 171 } else { 172 exit 1; 173 } 174