OSDN Git Service

Initial revision
[pf3gnuchains/gcc-fork.git] / gcc / genmultilib
1 #!/bin/sh 
2 # Generates multilib.h.
3 #   Copyright (C) 1994 Free Software Foundation, Inc.
4
5 #This file is part of GNU CC.
6
7 #GNU CC is free software; you can redistribute it and/or modify
8 #it under the terms of the GNU General Public License as published by
9 #the Free Software Foundation; either version 2, or (at your option)
10 #any later version.
11
12 #GNU CC is distributed in the hope that it will be useful,
13 #but WITHOUT ANY WARRANTY; without even the implied warranty of
14 #MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 #GNU General Public License for more details.
16
17 #You should have received a copy of the GNU General Public License
18 #along with GNU CC; see the file COPYING.  If not, write to
19 #the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
20
21 # This shell script produces a header file which the gcc driver
22 # program uses to pick which library to use based on the machine
23 # specific options that it is given.
24
25 # The first argument is a list of sets of options.  The elements in
26 # the list are separated by spaces.  Within an element, the options
27 # are separated by slashes.  No leading dash is used on the options.
28 # Each option in a set is mutually incompatible with all other options
29 # in the set.
30
31 # The optional second argument is a list of subdirectory names.  If
32 # the second argument is non-empty, there must be as many elements in
33 # the second argument as there are options in the first argument.  The
34 # elements in the second list are separated by spaces.  If the second
35 # argument is empty, the option names will be used as the directory
36 # names.
37
38 # The optional third argument is a list of options which are
39 # identical.  The elements in the list are separated by spaces.  Each
40 # element must be of the form OPTION=OPTION.  The first OPTION should
41 # appear in the first argument, and the second should be a synonym for
42 # it.
43
44 # The output looks like
45 #   #define MULTILIB_MATCHES "\
46 #   SUBDIRECTORY OPTIONS;\
47 #   ...
48 #   "
49 # The SUBDIRECTORY is the subdirectory to use.  The OPTIONS are
50 # multiple options separated by spaces.  Each option may start with an
51 # exclamation point.  gcc will consider each line in turn.  If none of
52 # the options beginning with an exclamation point are present, and all
53 # of the other options are present, that subdirectory will be used.
54 # The order of the subdirectories is such that they can be created in
55 # order; that is, a subdirectory is preceded by all its parents.
56
57 # Here is a example (this is simplified from the actual 680x0 case):
58 #   genmultilib "m68000/m68020 msoft-float" "m68000 m68020 msoft-float"
59 #               "m68000=mc68000"
60 # This produces:
61 #   #define MULTILIB_MATCHES "\
62 #   m68000/msoft-float m68000 msoft-float;\
63 #   m68000/msoft-float mc68000 msoft-float;\
64 #   m68020/msoft-float m68020 msoft-float;\
65 #   msoft-float !m68000 !mc68000 !m68020 msoft-float;\
66 #   m68000 m68000 !msoft-float;\
67 #   m68000 mc60000 !msoft-float;\
68 #   m68020 m68020 !msoft-float;\
69 #   . !m68000 !mc68000 !m68020 !msoft-float;\
70 #   "
71 # The effect is that `gcc -msoft-float' (for example) will append
72 # msoft-float to the directory name when searching for libraries or
73 # startup files, and `gcc -m68000 -msoft-float' (for example) will
74 # append m68000/msoft-float.
75
76 # Copy the positional parameters into variables.
77 options=$1
78 dirnames=$2
79 matches=$3
80
81 # What we want to do is select all combinations of the sets in
82 # options.  Each combination which includes a set of mutually
83 # exclusive options must then be output multiple times, once for each
84 # item in the set.  Selecting combinations is a recursive process.
85 # Since not all versions of sh support functions, we achieve recursion
86 # by creating a temporary shell script which invokes itself.
87 rm -f tmpmultilib
88 cat >tmpmultilib <<\EOF
89 #!/bin/sh
90 # This recursive script basically outputs all combinations of its
91 # input arguments, handling mutually exclusive sets of options by
92 # repetition.  When the script is called, ${initial} is the list of
93 # options which should appear before all combinations this will
94 # output.  The output looks like a list of subdirectory names with
95 # leading and trailing slashes.
96 if [ "$#" != "0" ]; then
97   first=$1
98   shift
99   for opt in `echo $first | sed -e 's|/| |'g`; do
100     echo ${initial}${opt}/
101   done
102   ./tmpmultilib $@
103   for opt in `echo $first | sed -e 's|/| |'g`; do
104     initial="${initial}${opt}/" ./tmpmultilib $@
105   done
106 fi
107 EOF
108 chmod +x tmpmultilib
109
110 combinations=`initial=/ ./tmpmultilib ${options}`
111
112 rm -f tmpmultilib
113
114 # Construct a sed pattern which will convert option names to directory
115 # names.
116 todirnames=
117 if [ -n "${dirnames}" ]; then
118   set x ${dirnames}
119   shift
120   for set in ${options}; do
121     for opt in `echo ${set} | sed -e 's|/| |'g`; do
122       if [ "$1" != "${opt}" ]; then
123         todirnames="${todirnames} -e s|/${opt}/|/${1}/|g"
124       fi
125       shift
126     done
127   done
128 fi
129
130 # Construct a sed pattern which will add negations based on the
131 # matches.  The semicolons are easier than getting the shell to accept
132 # quoted spaces when expanding a variable.
133 matchnegations=
134 for i in ${matches}; do
135   l=`echo $i | sed -e 's/=.*$//'`
136   r=`echo $i | sed -e 's/^.*=//'`
137   matchnegations="${matchnegations} -e s/;!${l};/;!${l};!${r};/"
138 done
139
140 # We need another recursive shell script to correctly handle positive
141 # matches.  If we are invoked as
142 #   genmultilib "opt1 opt2" "" "opt1=nopt1 opt2=nopt2"
143 # we must output
144 #   opt1/opt2 opt1 opt2
145 #   opt1/opt2 nopt1 opt2
146 #   opt1/opt2 opt1 nopt2
147 #   opt1/opt2 nopt1 nopt2
148 # In other words, we must output all combinations of matches.
149 rm -f tmpmultilib2
150 cat >tmpmultilib2 <<\EOF
151 #!/bin/sh
152 # The positional parameters are a list of matches to consider.
153 # ${dirout} is the directory name and ${optout} is the current list of
154 # options.
155 if [ "$#" = "0" ]; then
156   echo "${dirout} ${optout};\\"
157 else
158   first=$1
159   shift
160   dirout="${dirout}" optout="${optout}" ./tmpmultilib2 $@
161   l=`echo ${first} | sed -e 's/=.*$//'`
162   r=`echo ${first} | sed -e 's/^.*=//'`
163   case " ${optout} " in
164   *" ${l} "*)
165     newopt=`echo " ${optout} " | sed -e "s/ ${l} / ${r} /" -e 's/^ //' -e 's/ $//'`
166     dirout="${dirout}" optout="${newopt}" ./tmpmultilib2 $@
167     ;;
168   esac
169 fi
170 EOF
171 chmod +x tmpmultilib2
172
173 # We are ready to start output.
174 echo '#define MULTILIB_SELECT "\'
175
176 # Start with the current directory, which includes only negations.
177 optout=
178 for set in ${options}; do
179   for opt in `echo ${set} | sed -e 's|/| |'g`; do
180     optout="${optout} !${opt}"
181   done
182 done
183 optout=`echo ${optout} | sed -e 's/^ //'`
184 if [ -n "${matchnegations}" ]; then
185   optout=`echo ";${optout};" | sed -e 's/ /;/g' ${matchnegations} -e 's/^;//' -e 's/;$//' -e 's/;/ /g'`
186 fi
187 echo ". ${optout};\\"
188
189 # Work over the list of combinations.  We have to translate each one
190 # to use the directory names rather than the option names, we have to
191 # include the information in matches, and we have to generate the
192 # correct list of options and negations.
193 for combo in ${combinations}; do
194   # Use the directory names rather than the option names.
195   if [ -n "${todirnames}" ]; then
196     dirout=`echo ${combo} | sed ${todirnames}`
197   else
198     dirout=${combo}
199   fi
200   # Remove the leading and trailing slashes.
201   dirout=`echo ${dirout} | sed -e 's|^/||' -e 's|/$||g'`
202
203   # Look through the options.  We must output each option that is
204   # present, and negate each option that is not present.  If an
205   # element of a set is present, we need not negate the other elements
206   # of the set.
207   optout=
208   for set in ${options}; do
209     setopts=`echo ${set} | sed -e 's|/| |g'`
210     found=
211     for opt in ${setopts}; do
212       case "${combo}" in
213       *"/${opt}/"*)
214         optout="${optout} ${opt}"
215         found=yes
216         ;;
217       esac
218     done
219     if [ "${found}" = "" ]; then
220       for opt in ${setopts}; do
221         optout="${optout} !${opt}"
222       done
223     fi
224   done
225   optout=`echo ${optout} | sed -e 's/^ //'`
226
227   # Add any negations of matches.
228   if [ -n "${matchnegations}" ]; then
229     optout=`echo ";${optout};" | sed -e 's/ /;/g' ${matchnegations} -e 's/^;//' -e 's/;$//' -e 's/;/ /g'`
230   fi
231
232   # Output the line with all appropriate matches.
233   dirout="${dirout}" optout="${optout}" ./tmpmultilib2 ${matches}
234 done
235
236 rm -f tmpmultilib2
237
238 # That's it.
239 echo '"'
240
241 exit 0