#!/bin/bash # Grab default values for $CFLAGS and such. source ./configure if [ -z ".config" ] then echo "No .config (see "make help" for configuration options)." exit 1 fi echo "Make generated/config.h from .config." # This long and roundabout sed invocation is to make old versions of sed happy. # New ones have '\n' so can replace one line with two without all the branches # and tedious mucking about with hold space. sed -n \ -e 's/^# CONFIG_\(.*\) is not set.*/\1/' \ -e 't notset' \ -e 's/^CONFIG_\(.*\)=y.*/\1/' \ -e 't isset' \ -e 's/^CONFIG_\([^=]*\)=\(.*\)/#define CFG_\1 \2/p' \ -e 'd' \ -e ':notset' \ -e 'h' \ -e 's/.*/#define CFG_& 0/p' \ -e 'g' \ -e 's/.*/#define USE_&(...)/p' \ -e 'd' \ -e ':isset' \ -e 'h' \ -e 's/.*/#define CFG_& 1/p' \ -e 'g' \ -e 's/.*/#define USE_&(...) __VA_ARGS__/p' \ .config > generated/config.h || exit 1 echo "Extract configuration information from toys/*.c files..." scripts/genconfig.sh echo "Generate headers from toys/*/*.c..." # Create a list of all the applets toybox can provide. Note that the first # entry is out of order on purpose (the toybox multiplexer applet must be the # first element of the array). The rest must be sorted in alphabetical order # for fast binary search. echo "generated/newtoys.h" echo "NEWTOY(toybox, NULL, TOYFLAG_STAYROOT)" > generated/newtoys.h sed -n -e 's/^USE_[A-Z0-9_]*(/&/p' toys/*/*.c \ | sed 's/\(.*TOY(\)\([^,]*\),\(.*\)/\2 \1\2,\3/' | sort -k 1,1 \ | sed 's/[^ ]* //' >> generated/newtoys.h # Extract list of command letters from processed header file function getflags() { sed -n -e "s/.*TOY($1"',[ \t]*"\([^"]*\)"[ \t]*,.*)/\1/' \ -e 't keep;d;:keep' -e 's/^[<>=][0-9]//' -e 's/[?&^]//' \ -e 't keep' -e 's/[><=][0-9][0-9]*//g' -e 's/+.//g' \ -e 's/([^)]*)//g' -e 's/\[[^]]*\]//g' -e 's/[-?^:&#|@*]//g' -e 'p' } # Extract global structure definitions and flag definitions from toys/*/*.c function getglobals() { # Run newtoys.h through the compiler's preprocessor to resolve USE macros # against current config. NEWTOYS="$(cat generated/config.h generated/newtoys.h | $CC -E - | sed 's/" *"//g')" # Grab allyesconfig for comparison ALLTOYS="$((sed '/USE_.*([^)]*)$/s/$/ __VA_ARGS__/' generated/config.h && cat generated/newtoys.h) | $CC -E - | sed 's/" *"//g')" for i in toys/*/*.c do NAME="$(echo $i | sed 's@.*/\(.*\)\.c@\1@')" echo -e "// $i\n" sed -n -e '/^GLOBALS(/,/^)/b got;b;:got' \ -e 's/^GLOBALS(/struct '"$NAME"'_data {/' \ -e 's/^)/};/' -e 'p' $i FLAGS="$(echo "$NEWTOYS" | getflags "$NAME")" ZFLAGS="$(echo "$ALLTOYS" | getflags "$NAME" | sed 's/[-'"$FLAGS"']//g')" echo "#ifdef FOR_${NAME}" X=0 while [ $X -lt ${#FLAGS} ] do echo -ne "#define FLAG_${FLAGS:$X:1}\t" X=$(($X+1)) echo "(1<<$((${#FLAGS}-$X)))" done X=0 while [ $X -lt ${#ZFLAGS} ] do echo "#define FLAG_${ZFLAGS:$X:1} 0" X=$(($X+1)) done echo "#define TT this.${NAME}" echo "#endif" done } echo "generated/globals.h" GLOBSTRUCT="$(getglobals)" ( echo "$GLOBSTRUCT" echo echo "extern union global_union {" echo "$GLOBSTRUCT" | sed -n 's/struct \(.*\)_data {/ struct \1_data \1;/p' echo "} this;" ) > generated/globals.h echo "generated/help.h" # Only recreate generated/help.h if python is installed if [ ! -z "$(which python)" ] && [ ! -z "$(grep 'CONFIG_HELP=y' .config)" ] then echo "Extract help text from Config.in." scripts/config2help.py Config.in > generated/help.h || exit 1 fi # Extract a list of toys/*/*.c files to compile from the data in ".config": # 1) Get a list of C files in toys/* and glue them together into a regex we can # feed to grep that will match any one of them (whole word, not substring). TOYFILES="^$(ls toys/*/*.c | sed -n 's@^.*/\(.*\)\.c$@\1@;s/-/_/g;H;${g;s/\n//;s/\n/$|^/gp}')\$" # 2) Grab the XXX part of all CONFIG_XXX entries, removing everything after the # second underline # 3) Sort the list, keeping only one of each entry. # 4) Convert to lower case. # 5) Remove any config symbol not recognized as a filename from step 1. # 6) Add "toys/*/" prefix and ".c" suffix. TOYFILES=$(sed -nre 's/^CONFIG_(.*)=y/\1/p' < .config \ | sort -u | tr A-Z a-z | grep -E "$TOYFILES" | sed 's@\(.*\)@toys/\*/\1.c@') echo "Library probe..." # We trust --as-needed to remove each library if we don't use any symbols # out of it, this loop is because the compiler has no way to ignore a library # that doesn't exist, so we have to detect and skip nonexistent libraries # for it. OPTLIBS="$(for i in util crypt m; do echo "int main(int argc, char *argv[]) {return 0;}" | ${CROSS_COMPILE}${CC} -xc - -o /dev/null -Wl,--as-needed -l$i > /dev/null 2>/dev/null && echo -l$i; done)" echo "Compile toybox..." do_loudly() { [ ! -z "$V" ] && echo "$@" "$@" } do_loudly ${CROSS_COMPILE}${CC} $CFLAGS -I . -o toybox_unstripped $OPTIMIZE \ main.c lib/*.c $TOYFILES -Wl,--as-needed $OPTLIBS || exit 1 do_loudly ${CROSS_COMPILE}${STRIP} toybox_unstripped -o toybox || exit 1 # gcc 4.4's strip command is buggy, and doesn't set the executable bit on # its output the way SUSv4 suggests it do so. do_loudly chmod +x toybox || exit 1