blob: a8b0b2e0342e279577f50cb17171195511ccedfc [file] [log] [blame]
Denis Vlasenko7d219aa2006-10-05 10:17:08 +00001#!/bin/sh
2
3debug=false
4
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +00005# Linker flags used:
6#
7# Informational:
8# --warn-common
9# -Map $EXE.map
10# --verbose
11#
12# Optimizations:
13# --sort-common reduces padding
14# --sort-section alignment reduces padding
15# --gc-sections throws out unused sections,
16# does not work for shared libs
Denis Vlasenko42e41822007-10-09 18:01:13 +000017# -On Not used, maybe useful?
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +000018#
19# List of files to link:
20# $l_list == --start-group -llib1 -llib2 --end-group
21# --start-group $O_FILES $A_FILES --end-group
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +000022#
23# Shared library link:
24# -shared self-explanatory
25# -fPIC position-independent code
Denis Vlasenko724d1962007-10-10 14:41:07 +000026# --enable-new-dtags ?
27# -z,combreloc ?
28# -soname="libbusybox.so.$BB_VER"
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +000029# --undefined=lbb_main Seed name to start pulling from
Denis Vlasenko724d1962007-10-10 14:41:07 +000030# (otherwise we'll need --whole-archive)
31# -static Not used, but may be useful! manpage:
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +000032# "... This option can be used with -shared.
33# Doing so means that a shared library
34# is being created but that all of the library's
35# external references must be resolved by pulling
36# in entries from static libraries."
37
38
Denis Vlasenko018e0852007-02-25 00:40:37 +000039try() {
Denis Vlasenko32404742007-10-07 17:05:22 +000040 printf "%s\n" "Output of:" >$EXE.out
41 printf "%s\n" "$*" >>$EXE.out
42 printf "%s\n" "==========" >>$EXE.out
43 $debug && echo "Trying: $*"
Bernhard Reutner-Fischer2dfd2952008-10-23 13:49:21 +000044 $@ >>$EXE.out 2>&1
45 return $?
Denis Vlasenko7d219aa2006-10-05 10:17:08 +000046}
47
Denis Vlasenko130f5592007-11-13 17:36:12 +000048check_cc() {
Denis Vlasenko0a4624a2008-02-13 07:47:37 +000049 local tempname="/tmp/temp.$$.$RANDOM"
50 # Can use "-o /dev/null", but older gcc tend to *unlink it* on failure! :(
51 # "-xc": C language. "/dev/null" is an empty source file.
52 if $CC $1 -shared -xc /dev/null -o "$tempname".o >/dev/null 2>&1; then
Denis Vlasenko3f9c8482007-12-28 17:04:42 +000053 echo "$1";
Denis Vlasenko130f5592007-11-13 17:36:12 +000054 else
Denis Vlasenko3f9c8482007-12-28 17:04:42 +000055 echo "$2";
Denis Vlasenko130f5592007-11-13 17:36:12 +000056 fi
Denis Vlasenko0a4624a2008-02-13 07:47:37 +000057 rm "$tempname".o 2>/dev/null
Denis Vlasenko130f5592007-11-13 17:36:12 +000058}
59
Denis Vlasenko3f9c8482007-12-28 17:04:42 +000060check_libc_is_glibc() {
Denis Vlasenko0a4624a2008-02-13 07:47:37 +000061 local tempname="/tmp/temp.$$.$RANDOM"
Denis Vlasenko3f9c8482007-12-28 17:04:42 +000062 echo "\
63 #include <stdlib.h>
64 /* Apparently uclibc defines __GLIBC__ (compat trick?). Oh well. */
65 #if defined(__GLIBC__) && !defined(__UCLIBC__)
66 syntax error here
67 #endif
Denis Vlasenkoa2dcb502008-04-30 00:15:56 +000068 " >"$tempname".c
Denis Vlasenko0a4624a2008-02-13 07:47:37 +000069 if $CC "$tempname".c -c -o "$tempname".o >/dev/null 2>&1; then
Denis Vlasenko3f9c8482007-12-28 17:04:42 +000070 echo "$2";
71 else
72 echo "$1";
73 fi
Denis Vlasenko0a4624a2008-02-13 07:47:37 +000074 rm "$tempname".c "$tempname".o 2>/dev/null
Denis Vlasenko3f9c8482007-12-28 17:04:42 +000075}
76
Denis Vlasenko32404742007-10-07 17:05:22 +000077EXE="$1"
78CC="$2"
Denis Vlasenkof1d93ec2008-02-14 12:24:14 +000079CFLAGS="$3"
80LDFLAGS="$4"
81O_FILES="$5"
82A_FILES="$6"
83LDLIBS="$7"
Denis Vlasenko32404742007-10-07 17:05:22 +000084
Denis Vlasenkoa2dcb502008-04-30 00:15:56 +000085# The --sort-section option is not supported by older versions of ld
86SORT_SECTION=`check_cc "-Wl,--sort-section,alignment" ""`
Denis Vlasenko130f5592007-11-13 17:36:12 +000087
Dan Fandrichebeac162010-06-18 22:36:10 -070088START_GROUP="-Wl,--start-group"
89END_GROUP="-Wl,--end-group"
Dan Fandrichebeac162010-06-18 22:36:10 -070090INFO_OPTS="-Wl,--warn-common -Wl,-Map,$EXE.map -Wl,--verbose"
91
Bernhard Reutner-Fischera88585a2010-05-21 12:11:34 +020092# gold may not support --sort-common (yet)
93SORT_COMMON=`check_cc "-Wl,--sort-common" ""`
94
Denis Vlasenko3f9c8482007-12-28 17:04:42 +000095# Static linking against glibc produces buggy executables
96# (glibc does not cope well with ld --gc-sections).
97# See sources.redhat.com/bugzilla/show_bug.cgi?id=3400
98# Note that glibc is unsuitable for static linking anyway.
99# We are removing -Wl,--gc-sections from link command line.
Denis Vlasenkoa2dcb502008-04-30 00:15:56 +0000100GC_SECTIONS=`(
Denis Vlasenko3f9c8482007-12-28 17:04:42 +0000101. ./.config
102if test x"$CONFIG_STATIC" = x"y"; then
103 check_libc_is_glibc "" "-Wl,--gc-sections"
104else
105 echo "-Wl,--gc-sections"
106fi
107)`
108
Denys Vlasenkoeb1cda22009-06-27 00:24:35 +0200109# The --gc-sections option is not supported by older versions of ld
110if test -n "$GC_SECTIONS"; then
111 GC_SECTIONS=`check_cc "$GC_SECTIONS" ""`
112fi
113
Denis Vlasenko150d2fa2007-07-17 20:39:27 +0000114# Sanitize lib list (dups, extra spaces etc)
Denis Vlasenko76a6b232007-10-07 17:05:42 +0000115LDLIBS=`echo "$LDLIBS" | xargs -n1 | sort | uniq | xargs`
Denis Vlasenko150d2fa2007-07-17 20:39:27 +0000116
117# First link with all libs. If it fails, bail out
Denis Vlasenko76a6b232007-10-07 17:05:42 +0000118echo "Trying libraries: $LDLIBS"
Denis Vlasenko32404742007-10-07 17:05:22 +0000119# "lib1 lib2 lib3" -> "-llib1 -llib2 -llib3"
Denis Vlasenko76a6b232007-10-07 17:05:42 +0000120l_list=`echo "$LDLIBS" | sed -e 's/ / -l/g' -e 's/^/-l/' -e 's/^-l$//'`
Dan Fandrichebeac162010-06-18 22:36:10 -0700121test "x$l_list" != "x" && l_list="$START_GROUP $l_list $END_GROUP"
Denis Vlasenkof1d93ec2008-02-14 12:24:14 +0000122try $CC $CFLAGS $LDFLAGS \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000123 -o $EXE \
Bernhard Reutner-Fischera88585a2010-05-21 12:11:34 +0200124 $SORT_COMMON \
Denis Vlasenko130f5592007-11-13 17:36:12 +0000125 $SORT_SECTION \
Denis Vlasenkoa2dcb502008-04-30 00:15:56 +0000126 $GC_SECTIONS \
Dan Fandrichebeac162010-06-18 22:36:10 -0700127 $START_GROUP $O_FILES $A_FILES $END_GROUP \
Denis Vlasenko32404742007-10-07 17:05:22 +0000128 $l_list \
Denis Vlasenko150d2fa2007-07-17 20:39:27 +0000129|| {
Denis Vlasenko4824cca2008-03-21 18:29:01 +0000130 echo "Failed: $l_list"
Denis Vlasenko32404742007-10-07 17:05:22 +0000131 cat $EXE.out
Denis Vlasenko150d2fa2007-07-17 20:39:27 +0000132 exit 1
133}
Denis Vlasenko6ca409e2007-08-12 20:58:27 +0000134
Denis Vlasenko8274e062007-08-06 03:41:08 +0000135# Now try to remove each lib and build without it.
Denis Vlasenko150d2fa2007-07-17 20:39:27 +0000136# Stop when no lib can be removed.
Denis Vlasenko76a6b232007-10-07 17:05:42 +0000137while test "$LDLIBS"; do
138 $debug && echo "Trying libraries: $LDLIBS"
Denis Vlasenko150d2fa2007-07-17 20:39:27 +0000139 all_needed=true
Denis Vlasenkodd316dd2008-06-14 15:50:55 +0000140 last_needed=false
Denis Vlasenko76a6b232007-10-07 17:05:42 +0000141 for one in $LDLIBS; do
142 without_one=`echo " $LDLIBS " | sed "s/ $one / /g" | xargs`
Denis Vlasenko32404742007-10-07 17:05:22 +0000143 # "lib1 lib2 lib3" -> "-llib1 -llib2 -llib3"
Bernhard Reutner-Fischerd2bb2d32007-09-02 15:28:30 +0000144 l_list=`echo "$without_one" | sed -e 's/ / -l/g' -e 's/^/-l/' -e 's/^-l$//'`
Dan Fandrichebeac162010-06-18 22:36:10 -0700145 test x"$l_list" != x"" && l_list="$START_GROUP $l_list $END_GROUP"
Bernhard Reutner-Fischer8d91c132007-09-02 14:51:54 +0000146 $debug && echo "Trying -l options: '$l_list'"
Denis Vlasenkof1d93ec2008-02-14 12:24:14 +0000147 try $CC $CFLAGS $LDFLAGS \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000148 -o $EXE \
Bernhard Reutner-Fischera88585a2010-05-21 12:11:34 +0200149 $SORT_COMMON \
Denis Vlasenko130f5592007-11-13 17:36:12 +0000150 $SORT_SECTION \
Denis Vlasenkoa2dcb502008-04-30 00:15:56 +0000151 $GC_SECTIONS \
Dan Fandrichebeac162010-06-18 22:36:10 -0700152 $START_GROUP $O_FILES $A_FILES $END_GROUP \
Denis Vlasenkod19b87e2007-10-09 13:08:02 +0000153 $l_list
Denis Vlasenko32404742007-10-07 17:05:22 +0000154 if test $? = 0; then
Denis Vlasenkob522d692008-08-26 20:09:08 +0000155 echo " Library $one is not needed, excluding it"
Denis Vlasenko76a6b232007-10-07 17:05:42 +0000156 LDLIBS="$without_one"
Denis Vlasenko32404742007-10-07 17:05:22 +0000157 all_needed=false
Denis Vlasenkodd316dd2008-06-14 15:50:55 +0000158 last_needed=false
Denis Vlasenko150d2fa2007-07-17 20:39:27 +0000159 else
Denis Vlasenkob522d692008-08-26 20:09:08 +0000160 echo " Library $one is needed, can't exclude it (yet)"
Denis Vlasenkodd316dd2008-06-14 15:50:55 +0000161 last_needed=true
Denis Vlasenko150d2fa2007-07-17 20:39:27 +0000162 fi
163 done
164 # All libs were needed, can't remove any
165 $all_needed && break
Denis Vlasenkodd316dd2008-06-14 15:50:55 +0000166 # Optimization: was the last tried lib needed?
167 if $last_needed; then
168 # Was it the only one lib left? Don't test again then.
169 { echo "$LDLIBS" | grep -q ' '; } || break
170 fi
Denis Vlasenko150d2fa2007-07-17 20:39:27 +0000171done
172
Denis Vlasenko9862e6b2007-09-03 11:28:14 +0000173# Make the binary with final, minimal list of libs
Mike Frysinger49d15892007-11-18 06:42:56 +0000174echo "Final link with: ${LDLIBS:-<none>}"
Denis Vlasenko76a6b232007-10-07 17:05:42 +0000175l_list=`echo "$LDLIBS" | sed -e 's/ / -l/g' -e 's/^/-l/' -e 's/^-l$//'`
Dan Fandrichebeac162010-06-18 22:36:10 -0700176test "x$l_list" != "x" && l_list="$START_GROUP $l_list $END_GROUP"
Denis Vlasenko9862e6b2007-09-03 11:28:14 +0000177# --verbose gives us gobs of info to stdout (e.g. linker script used)
178if ! test -f busybox_ldscript; then
Denis Vlasenkof1d93ec2008-02-14 12:24:14 +0000179 try $CC $CFLAGS $LDFLAGS \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000180 -o $EXE \
Bernhard Reutner-Fischera88585a2010-05-21 12:11:34 +0200181 $SORT_COMMON \
Denis Vlasenko130f5592007-11-13 17:36:12 +0000182 $SORT_SECTION \
Denis Vlasenkoa2dcb502008-04-30 00:15:56 +0000183 $GC_SECTIONS \
Dan Fandrichebeac162010-06-18 22:36:10 -0700184 $START_GROUP $O_FILES $A_FILES $END_GROUP \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000185 $l_list \
Dan Fandrichebeac162010-06-18 22:36:10 -0700186 $INFO_OPTS \
Denis Vlasenkod19b87e2007-10-09 13:08:02 +0000187 || {
188 cat $EXE.out
189 exit 1
190 }
Denis Vlasenko9862e6b2007-09-03 11:28:14 +0000191else
192 echo "Custom linker script 'busybox_ldscript' found, using it"
Denis Vlasenko76a6b232007-10-07 17:05:42 +0000193 # Add SORT_BY_ALIGNMENT to linker script (found in $EXE.out):
Denis Vlasenko9862e6b2007-09-03 11:28:14 +0000194 # .rodata : { *(.rodata SORT_BY_ALIGNMENT(.rodata.*) .gnu.linkonce.r.*) }
195 # *(.data SORT_BY_ALIGNMENT(.data.*) .gnu.linkonce.d.*)
196 # *(.bss SORT_BY_ALIGNMENT(.bss.*) .gnu.linkonce.b.*)
Denis Vlasenko76a6b232007-10-07 17:05:42 +0000197 # This will eliminate most of the padding (~3kb).
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000198 # Hmm, "ld --sort-section alignment" should do it too.
Denis Vlasenkof1d93ec2008-02-14 12:24:14 +0000199 try $CC $CFLAGS $LDFLAGS \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000200 -o $EXE \
Bernhard Reutner-Fischera88585a2010-05-21 12:11:34 +0200201 $SORT_COMMON \
Denis Vlasenko130f5592007-11-13 17:36:12 +0000202 $SORT_SECTION \
Denis Vlasenkoa2dcb502008-04-30 00:15:56 +0000203 $GC_SECTIONS \
Bernhard Reutner-Fischer50dbed92008-05-09 12:43:04 +0000204 -Wl,-T,busybox_ldscript \
Dan Fandrichebeac162010-06-18 22:36:10 -0700205 $START_GROUP $O_FILES $A_FILES $END_GROUP \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000206 $l_list \
Dan Fandrichebeac162010-06-18 22:36:10 -0700207 $INFO_OPTS \
Denis Vlasenkod19b87e2007-10-09 13:08:02 +0000208 || {
209 cat $EXE.out
210 exit 1
211 }
Denis Vlasenko9862e6b2007-09-03 11:28:14 +0000212fi
Denis Vlasenko32404742007-10-07 17:05:22 +0000213
Denis Vlasenko42e41822007-10-09 18:01:13 +0000214. ./.config
Denis Vlasenko32404742007-10-07 17:05:22 +0000215
Denis Vlasenkodef88982007-10-07 17:06:01 +0000216sharedlib_dir="0_lib"
217
218if test "$CONFIG_BUILD_LIBBUSYBOX" = y; then
219 mkdir "$sharedlib_dir" 2>/dev/null
220 test -d "$sharedlib_dir" || {
221 echo "Cannot make directory $sharedlib_dir"
222 exit 1
223 }
224 ln -s "libbusybox.so.$BB_VER" "$sharedlib_dir"/libbusybox.so 2>/dev/null
225
226 EXE="$sharedlib_dir/libbusybox.so.${BB_VER}_unstripped"
Denis Vlasenkof1d93ec2008-02-14 12:24:14 +0000227 try $CC $CFLAGS $LDFLAGS \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000228 -o $EXE \
229 -shared -fPIC \
230 -Wl,--enable-new-dtags \
231 -Wl,-z,combreloc \
232 -Wl,-soname="libbusybox.so.$BB_VER" \
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +0000233 -Wl,--undefined=lbb_main \
Bernhard Reutner-Fischera88585a2010-05-21 12:11:34 +0200234 $SORT_COMMON \
Denis Vlasenko130f5592007-11-13 17:36:12 +0000235 $SORT_SECTION \
Dan Fandrichebeac162010-06-18 22:36:10 -0700236 $START_GROUP $A_FILES $END_GROUP \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000237 $l_list \
Dan Fandrichebeac162010-06-18 22:36:10 -0700238 $INFO_OPTS \
Denis Vlasenkodef88982007-10-07 17:06:01 +0000239 || {
240 echo "Linking $EXE failed"
241 cat $EXE.out
242 exit 1
243 }
Denis Vlasenko141750e2007-10-10 10:05:35 +0000244 $STRIP -s --remove-section=.note --remove-section=.comment $EXE -o "$sharedlib_dir/libbusybox.so.$BB_VER"
Denis Vlasenkod62fd842007-10-07 20:46:34 +0000245 chmod a+x "$sharedlib_dir/libbusybox.so.$BB_VER"
Denis Vlasenkodef88982007-10-07 17:06:01 +0000246 echo "libbusybox: $sharedlib_dir/libbusybox.so.$BB_VER"
247fi
Denis Vlasenko32404742007-10-07 17:05:22 +0000248
Denis Vlasenkodef88982007-10-07 17:06:01 +0000249if test "$CONFIG_FEATURE_SHARED_BUSYBOX" = y; then
250 EXE="$sharedlib_dir/busybox_unstripped"
Denis Vlasenkof1d93ec2008-02-14 12:24:14 +0000251 try $CC $CFLAGS $LDFLAGS \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000252 -o $EXE \
Bernhard Reutner-Fischera88585a2010-05-21 12:11:34 +0200253 $SORT_COMMON \
Denis Vlasenko130f5592007-11-13 17:36:12 +0000254 $SORT_SECTION \
Denis Vlasenkoa2dcb502008-04-30 00:15:56 +0000255 $GC_SECTIONS \
Dan Fandrichebeac162010-06-18 22:36:10 -0700256 $START_GROUP $O_FILES $END_GROUP \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000257 -L"$sharedlib_dir" -lbusybox \
Steve Iribarneed607a82011-05-09 01:42:12 +0200258 $l_list \
Dan Fandrichebeac162010-06-18 22:36:10 -0700259 $INFO_OPTS \
Denis Vlasenkodef88982007-10-07 17:06:01 +0000260 || {
261 echo "Linking $EXE failed"
262 cat $EXE.out
263 exit 1
264 }
Denis Vlasenko141750e2007-10-10 10:05:35 +0000265 $STRIP -s --remove-section=.note --remove-section=.comment $EXE -o "$sharedlib_dir/busybox"
Denis Vlasenkodef88982007-10-07 17:06:01 +0000266 echo "busybox linked against libbusybox: $sharedlib_dir/busybox"
267fi
Denis Vlasenkof545be02007-10-07 17:06:26 +0000268
269if test "$CONFIG_FEATURE_INDIVIDUAL" = y; then
Denis Vlasenkoe9fd69c2007-10-08 22:16:14 +0000270 echo "Linking individual applets against libbusybox (see $sharedlib_dir/*)"
Denis Vlasenkof545be02007-10-07 17:06:26 +0000271 gcc -DNAME_MAIN_CNAME -E -include include/autoconf.h include/applets.h \
272 | grep -v "^#" \
273 | grep -v "^$" \
Denis Vlasenkoe9fd69c2007-10-08 22:16:14 +0000274 > applet_lst.tmp
Denis Vlasenko642a52d2007-10-07 21:00:41 +0000275 while read name main junk; do
Denis Vlasenkof545be02007-10-07 17:06:26 +0000276
277 echo "\
Denis Vlasenko15cb4a42007-10-11 10:06:26 +0000278void lbb_prepare(const char *applet, char **argv);
Denis Vlasenkod62fd842007-10-07 20:46:34 +0000279int $main(int argc, char **argv);
Denis Vlasenkof545be02007-10-07 17:06:26 +0000280
Denis Vlasenkof545be02007-10-07 17:06:26 +0000281int main(int argc, char **argv)
282{
Denis Vlasenko15cb4a42007-10-11 10:06:26 +0000283 lbb_prepare(\"$name\", argv);
Denis Vlasenkod62fd842007-10-07 20:46:34 +0000284 return $main(argc, argv);
Denis Vlasenkof545be02007-10-07 17:06:26 +0000285}
286" >"$sharedlib_dir/applet.c"
287
288 EXE="$sharedlib_dir/$name"
Denis Vlasenkof1d93ec2008-02-14 12:24:14 +0000289 try $CC $CFLAGS $LDFLAGS "$sharedlib_dir/applet.c" \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000290 -o $EXE \
Bernhard Reutner-Fischera88585a2010-05-21 12:11:34 +0200291 $SORT_COMMON \
Denis Vlasenko130f5592007-11-13 17:36:12 +0000292 $SORT_SECTION \
Denis Vlasenkoa2dcb502008-04-30 00:15:56 +0000293 $GC_SECTIONS \
Denis Vlasenko01f3b2c2007-10-09 13:49:26 +0000294 -L"$sharedlib_dir" -lbusybox \
295 -Wl,--warn-common \
Denis Vlasenkof545be02007-10-07 17:06:26 +0000296 || {
297 echo "Linking $EXE failed"
Denis Vlasenko8d755ad2007-10-09 10:15:41 +0000298 cat $EXE.out
Denis Vlasenkof545be02007-10-07 17:06:26 +0000299 exit 1
300 }
301 rm -- "$sharedlib_dir/applet.c" $EXE.out
Denis Vlasenko141750e2007-10-10 10:05:35 +0000302 $STRIP -s --remove-section=.note --remove-section=.comment $EXE
Denis Vlasenkof545be02007-10-07 17:06:26 +0000303
Denis Vlasenkoe9fd69c2007-10-08 22:16:14 +0000304 done <applet_lst.tmp
Denis Vlasenkof545be02007-10-07 17:06:26 +0000305fi
Denis Vlasenkod62fd842007-10-07 20:46:34 +0000306
307# libbusybox.so is needed only for -lbusybox at link time,
308# it is not needed at runtime. Deleting to reduce confusion.
Denis Vlasenko8d755ad2007-10-09 10:15:41 +0000309rm "$sharedlib_dir"/libbusybox.so 2>/dev/null
310exit 0 # or else we may confuse make