1#!/bin/bash
2
3# This has to be a separate file from scripts/make.sh so it can be called
4# before menuconfig.  (It's called again from scripts/make.sh just to be sure.)
5
6mkdir -p generated
7
8source configure
9
10probecc()
11{
12  ${CROSS_COMPILE}${CC} $CFLAGS -xc -o /dev/null $1 -
13}
14
15# Probe for a single config symbol with a "compiles or not" test.
16# Symbol name is first argument, flags second, feed C file to stdin
17probesymbol()
18{
19  probecc $2 2>/dev/null && DEFAULT=y || DEFAULT=n
20  rm a.out 2>/dev/null
21  echo -e "config $1\n\tbool" || exit 1
22  echo -e "\tdefault $DEFAULT\n" || exit 1
23}
24
25probeconfig()
26{
27  > generated/cflags
28  # llvm produces its own really stupid warnings about things that aren't wrong,
29  # and although you can turn the warning off, gcc reacts badly to command line
30  # arguments it doesn't understand. So probe.
31  [ -z "$(probecc -Wno-string-plus-int <<< \#warn warn 2>&1 | grep string-plus-int)" ] &&
32    echo -Wno-string-plus-int >> generated/cflags
33
34  # Probe for container support on target
35  probesymbol TOYBOX_CONTAINER << EOF
36    #include <linux/sched.h>
37    int x=CLONE_NEWNS|CLONE_NEWUTS|CLONE_NEWIPC|CLONE_NEWNET;
38
39    int main(int argc, char *argv[]) { setns(0,0); return unshare(x); }
40EOF
41
42  probesymbol TOYBOX_FIFREEZE -c << EOF
43    #include <linux/fs.h>
44    #ifndef FIFREEZE
45    #error nope
46    #endif
47EOF
48
49  # Work around some uClibc limitations
50  probesymbol TOYBOX_ICONV -c << EOF
51    #include "iconv.h"
52EOF
53  probesymbol TOYBOX_FALLOCATE << EOF
54    #include <fcntl.h>
55
56    int main(int argc, char *argv[]) { return posix_fallocate(0,0,0); }
57EOF
58
59  # Android and some other platforms miss utmpx
60  probesymbol TOYBOX_UTMPX -c << EOF
61    #include <utmpx.h>
62    #ifndef BOOT_TIME
63    #error nope
64    #endif
65    int main(int argc, char *argv[]) {
66      struct utmpx *a; 
67      if (0 != (a = getutxent())) return 0;
68      return 1;
69    }
70EOF
71
72  # Android is missing shadow.h
73  probesymbol TOYBOX_SHADOW -c << EOF
74    #include <shadow.h>
75    int main(int argc, char *argv[]) {
76      struct spwd *a = getspnam("root"); return 0;
77    }
78EOF
79
80  # Some commands are android-specific
81  probesymbol TOYBOX_ON_ANDROID -c << EOF
82    #ifndef __ANDROID__
83    #error nope
84    #endif
85EOF
86
87  # nommu support
88  probesymbol TOYBOX_FORK << EOF
89    #include <unistd.h>
90    int main(int argc, char *argv[]) { return fork(); }
91EOF
92  echo -e '\tdepends on !TOYBOX_MUSL_NOMMU_IS_BROKEN'
93
94  probesymbol TOYBOX_PRLIMIT << EOF
95    #include <sys/time.h>
96    #include <sys/resource.h>
97
98    int main(int argc, char *argv[]) { prlimit(0, 0, 0, 0); }
99EOF
100}
101
102genconfig()
103{
104  # Reverse sort puts posix first, examples last.
105  for j in $(ls toys/*/README | sort -s -r)
106  do
107    DIR="$(dirname "$j")"
108
109    [ $(ls "$DIR" | wc -l) -lt 2 ] && continue
110
111    echo "menu \"$(head -n 1 $j)\""
112    echo
113
114    # extract config stanzas from each source file, in alphabetical order
115    for i in $(ls -1 $DIR/*.c)
116    do
117      # Grab the config block for Config.in
118      echo "# $i"
119      sed -n '/^\*\//q;/^config [A-Z]/,$p' $i || return 1
120      echo
121    done
122
123    echo endmenu
124  done
125}
126
127probeconfig > generated/Config.probed || rm generated/Config.probed
128genconfig > generated/Config.in || rm generated/Config.in
129
130# Find names of commands that can be built standalone in these C files
131toys()
132{
133  grep 'TOY(.*)' "$@" | grep -v TOYFLAG_NOFORK | grep -v "0))" | \
134    sed -rn 's/([^:]*):.*(OLD|NEW)TOY\( *([a-zA-Z][^,]*) *,.*/\1:\3/p'
135}
136
137WORKING=
138PENDING=
139toys toys/*/*.c | (
140while IFS=":" read FILE NAME
141do
142  [ "$NAME" == help ] && continue
143  [ "$NAME" == install ] && continue
144  echo -e "$NAME: $FILE *.[ch] lib/*.[ch]\n\tscripts/single.sh $NAME\n"
145  echo -e "test_$NAME:\n\tscripts/test.sh $NAME\n"
146  [ "${FILE/pending//}" != "$FILE" ] &&
147    PENDING="$PENDING $NAME" ||
148    WORKING="$WORKING $NAME"
149done &&
150echo -e "clean::\n\trm -f $WORKING $PENDING" &&
151echo -e "list:\n\t@echo $(echo $WORKING | tr ' ' '\n' | sort | xargs)" &&
152echo -e "list_pending:\n\t@echo $(echo $PENDING | tr ' ' '\n' | sort | xargs)" &&
153echo -e ".PHONY: $WORKING $PENDING" | sed 's/ \([^ ]\)/ test_\1/g'
154) > .singlemake
155