Use pyplnet to bring up the network.
[bootcd.git] / initscripts / pl_netinit
1 #!/bin/sh
2
3 # the name of the floppy based network configuration
4 # files (checked first). the name planet.cnf is kept
5 # for backward compatibility with old nodes, and only
6 # the floppy disk is searched for files with this name.
7 # new files are named plnode.txt and can be located on
8 # a floppy or usb device or on the cdrom
9 OLD_NODE_CONF_NAME=planet.cnf
10 NEW_NODE_CONF_NAME=plnode.txt
11
12 # one location of cd-based network configuration file
13 # (checked if floppy conf file missing and usb
14 # configuration file is missing)
15 CD_NET_CONF_BOOT=/usr/boot/$NEW_NODE_CONF_NAME
16
17 # the other location of cd-based network configuration file
18 CD_NET_CONF_ROOT=/usr/$NEW_NODE_CONF_NAME
19
20 # if all other network configuration file sources 
21 # don't exist, fall back to this one (always on the cd)
22 FALLBACK_NET_CONF=/usr/boot/default-node.txt
23
24 # a temporary place to hold the old configuration file
25 # off of the floppy disk if we find it (so we don't have
26 # to remount the floppy later)
27 TMP_OLD_FLOPPY_CONF_FILE=/tmp/oldfloppy_planet.cnf
28
29 # once a configuration file is found, save it in /tmp
30 # (may be used later by boot scripts)
31 USED_NET_CONF=/tmp/planet.cnf
32
33 # default device to use for contacting PLC if not specified
34 # in the configuration file
35 DEFAULT_NET_DEV=eth0
36
37 # where to store the temporary dhclient conf file
38 DHCLIENT_CONF_FILE=/tmp/dhclient.conf
39
40 # which fs types we support finding node configuration files on
41 # (will be based as a -t parameter to mount)
42 NODE_CONF_DEVICE_FS_TYPES="msdos,ext2"
43
44 # a temporary place to mount devices that might contain configuration
45 # files on
46 CONF_DEVICE_MOUNT_POINT=/mnt/confdevice
47 /bin/mkdir -p $CONF_DEVICE_MOUNT_POINT
48
49 # for some backwards compatibility, save the ifconfig <device>
50 # output to this file after everything is online
51 IFCONFIG_OUTPUT=/tmp/ifconfig
52
53 # set to 1 if the default network configuration was loaded off the cd
54 # (no other configuration file found)
55 DEFAULT_NET_CONF=0
56
57
58 net_init_failed()
59 {
60     echo
61     echo $(date "+%H:%M:%S") " pl_netinit: network initialization failed,"
62     echo $(date "+%H:%M:%S") " pl_netinit: shutting down machine in two hours"
63     /bin/sleep 2h
64     /sbin/shutdown -h now
65     exit 1
66 }
67
68 # Function for checking the IP address to see if its sensible.
69 check_ip()
70 {
71     case "$*" in
72         "" | *[!0-9.]* | *[!0-9]) return 1 ;;
73     esac
74     local IFS=.
75     set -- $*
76     [ $# -eq 4 ] &&
77     [ ${1:-666} -le 255 ] && [ ${2:-666} -le 255 ] &&
78     [ ${3:-666} -le 255 ] && [ ${4:-666} -le 255 ]
79 }
80
81 # find and parse a node network configuration file. return 0 if not found,
82 # return 1 if found and parsed. if this is the case, DEFAULT_NET_CONF will 
83 # be set to 1. For any found configuration file, $USED_NET_CONF will
84 # contain the validated contents
85 find_node_config()
86 {
87     /bin/rm -f $TMP_OLD_FLOPPY_CONF_FILE 2>&1 > /dev/null
88
89     echo $(date "+%H:%M:%S") " pl_netinit: looking for node configuration file on floppy"
90     
91     /bin/mount -o ro -t $NODE_CONF_DEVICE_FS_TYPES /dev/fd0 \
92         $CONF_DEVICE_MOUNT_POINT 2>&1 > /dev/null
93     if [[ $? -eq 0 ]]; then
94
95         # 1. check for new named file first on the floppy disk
96         if [ -r "$CONF_DEVICE_MOUNT_POINT/$NEW_NODE_CONF_NAME" ]; then
97             conf_file="$CONF_DEVICE_MOUNT_POINT/$NEW_NODE_CONF_NAME"
98             echo $(date "+%H:%M:%S") " pl_netinit: found node configuration file $conf_file"
99             /etc/init.d/pl_validateconf < $conf_file > $USED_NET_CONF
100             /bin/umount $CONF_DEVICE_MOUNT_POINT
101             return 1
102
103         # since we have the floppy mounted already, see if an old file
104         # exists there so we don't have to remount the floppy when we need
105         # to check for an old file on it (later in the order). if it does
106         # just copy it off to a special location
107         elif [ -r "$CONF_DEVICE_MOUNT_POINT/$OLD_NODE_CONF_NAME" ]; then
108             conf_file="$CONF_DEVICE_MOUNT_POINT/$OLD_NODE_CONF_NAME"
109             /bin/cp -f $conf_file $TMP_OLD_FLOPPY_CONF_FILE
110             echo $(date "+%H:%M:%S") " pl_netinit: found old named configuration file, checking later."
111         else
112             echo $(date "+%H:%M:%S") " pl_netinit: floppy mounted, but no configuration file."
113         fi
114
115         /bin/umount $CONF_DEVICE_MOUNT_POINT
116     else
117         echo $(date "+%H:%M:%S") " pl_netinit: no floppy could be mounted, continuing search."
118     fi
119
120     # 2. check for a new named file on removable flash devices (those 
121     # that start with sd*, because usb_storage emulates scsi devices).
122     # to prevent checking normal scsi disks, also make sure
123     # /sys/block/<dev>/removable is set to 1
124
125     echo $(date "+%H:%M:%S") " pl_netinit: looking for node configuration file on flash based devices"
126
127     # make the sd* hd* expansion fail to an empty string if there are no sd
128     # devices
129     shopt -s nullglob
130
131     for device in /sys/block/[hs]d*; do
132         removable=`cat $device/removable`
133         if [[ $removable -ne 1 ]]; then
134             continue
135         fi
136
137         partitions=$(/bin/awk "\$4 ~ /`basename $device`[0-9]*/ { print \$4 }" /proc/partitions)
138         for partition in $partitions ; do
139             check_dev=/dev/$partition
140
141             echo $(date "+%H:%M:%S") " pl_netinit: looking for node configuration file on device $check_dev"
142             /bin/mount -o ro -t $NODE_CONF_DEVICE_FS_TYPES $check_dev \
143                 $CONF_DEVICE_MOUNT_POINT 2>&1 > /dev/null
144             if [[ $? -eq 0 ]]; then
145                 if [ -r "$CONF_DEVICE_MOUNT_POINT/$NEW_NODE_CONF_NAME" ]; then
146                     conf_file="$CONF_DEVICE_MOUNT_POINT/$NEW_NODE_CONF_NAME"
147                     echo $(date "+%H:%M:%S") " pl_netinit: found node configuration file $conf_file"
148                     /etc/init.d/pl_validateconf < $conf_file > $USED_NET_CONF
149                     echo $(date "+%H:%M:%S") " pl_netinit: found configuration"
150                     /bin/umount $CONF_DEVICE_MOUNT_POINT
151                     return 1
152                 fi
153                 
154                 echo $(date "+%H:%M:%S") " pl_netinit: not found"
155
156                 /bin/umount $CONF_DEVICE_MOUNT_POINT
157             fi
158         done
159     done
160
161     # normal filename expansion setting
162     shopt -u nullglob
163
164     # 3. see if there is an old file on the floppy disk. if there was,
165     # the file $TMP_OLD_FLOPPY_CONF_FILE will be readable.
166     if [ -r "$TMP_OLD_FLOPPY_CONF_FILE" ]; then
167         conf_file=$TMP_OLD_FLOPPY_CONF_FILE
168         echo $(date "+%H:%M:%S") " pl_netinit: found node configuration file $conf_file"
169         /etc/init.d/pl_validateconf < $conf_file > $USED_NET_CONF
170         return 1
171     fi
172
173
174     # 4. check for plnode.txt on the cd at /usr/boot
175     echo $(date "+%H:%M:%S") " pl_netinit: looking for network configuration on cd in /usr/boot"
176     if [ -r "$CD_NET_CONF_BOOT" ]; then
177         
178         echo $(date "+%H:%M:%S") " pl_netinit: found cd configuration file $CD_NET_BOOT_CONF"
179         /etc/init.d/pl_validateconf < $CD_NET_CONF_BOOT > $USED_NET_CONF
180         return 1
181     fi
182     
183
184     # 5. check for plnode.txt on the cd at /usr
185     echo $(date "+%H:%M:%S") " pl_netinit: looking for network configuration on cd in /usr"
186     if [ -r "$CD_NET_CONF_ROOT" ]; then
187         
188         echo $(date "+%H:%M:%S") " pl_netinit: found cd configuration file $CD_NET_CONF_ROOT"
189         /etc/init.d/pl_validateconf < $CD_NET_CONF_ROOT > $USED_NET_CONF
190         return 1
191     fi
192
193
194     # 6. no node configuration file could be found, fall back to
195     # builtin default. this can't be used to install a machine, but
196     # will at least let it download and run the boot manager, which
197     # can inform the users appropriately.
198     echo $(date "+%H:%M:%S") " pl_netinit: using default network configuration"
199     if [ -r "$FALLBACK_NET_CONF" ]; then
200         echo $(date "+%H:%M:%S") " pl_netinit: found cd default configuration file $FALLBACK_NET_CONF"
201         /etc/init.d/pl_validateconf < $FALLBACK_NET_CONF > $USED_NET_CONF
202         DEFAULT_NET_CONF=1
203         return 1
204     fi
205
206     return 0
207 }
208
209
210 echo $(date "+%H:%M:%S") " pl_netinit: bringing loopback network device up"
211 /sbin/ifconfig lo 127.0.0.1 up
212
213 find_node_config
214 if [ $? -eq 0 ]; then
215     # no network configuration file found. this should not happen as the
216     # default cd image has a backup one. halt.
217     echo $(date "+%H:%M:%S") " pl_netinit: unable to find even a default network configuration"
218     echo $(date "+%H:%M:%S") " pl_netinit: file, this cd may be corrupt."
219     net_init_failed
220 fi
221
222 # load the configuration file. if it was a default one (not user specified),
223 # then remove the saved copy from /tmp, but continue on. since a network 
224 # configuration file is required and boot scripts only know about this location
225 # they will fail (as they should) - but the network will be up if dhcp is
226 # available
227
228 echo $(date "+%H:%M:%S") " pl_netinit: loading network configuration"
229 . $USED_NET_CONF
230
231 if [[ $DEFAULT_NET_CONF -eq 1 ]]; then
232     /bin/rm -f $USED_NET_CONF
233 fi
234
235 # initialize IPMI device
236 if [[ -n "$IPMI_ADDRESS" ]] ; then
237     echo -n "pl_netinit: initializing IPMI: "
238     cmd="ipnmac -i $IPMI_ADDRESS"
239     if [[ -n "$IPMI_MAC" ]] ; then
240         cmd="$cmd -m $IPMI_MAC"
241     fi
242     echo $cmd
243     $cmd
244 fi
245
246 # now, we need to find which device to use (ie, eth0 or eth1). start out
247 # by defaulting to eth0, then see if the network configuration file specified
248 # either a mac address (in which case we will need to find the device), or
249 # the device itself
250
251 ETH_DEVICE=
252 if [[ -n "$NET_DEVICE" ]]; then
253     # the user specified a mac address we should use. find the network
254     # device for it.
255     NET_DEVICE=$(tr A-Z a-z <<<$NET_DEVICE)
256
257     pushd /sys/class/net
258     for device in *; do
259         dev_address=`cat $device/address | tr A-Z a-z`
260         if [ "$device" == "$NET_DEVICE" -o "$dev_address" == "$NET_DEVICE" ]; then
261             ETH_DEVICE=$device
262             echo $(date "+%H:%M:%S") " pl_netinit: found device $ETH_DEVICE with mac address $dev_address"
263             break
264         fi
265     done
266     popd
267 fi
268
269 # if we didn't find a device yet, check which is the primary
270 if [[ -z "$ETH_DEVICE" ]]; then
271     pushd /etc/sysconfig/network-scripts > /dev/null
272     for conf in ifcfg-*; do
273         egrep -q '^PRIMARY=["'"'"']?[yY1t]' $conf || continue
274         ETH_DEVICE=${conf#ifcfg-}
275         break
276     done
277     popd > /dev/null
278 fi
279
280 # still nothing? fail the boot.
281 if [[ -z "$ETH_DEVICE" ]]; then
282     echo $(date "+%H:%M:%S") " pl_netinit: unable to find a usable device, check to make sure"
283     echo $(date "+%H:%M:%S") " pl_netinit: the NET_DEVICE field in the configuration file"
284     echo $(date "+%H:%M:%S") " pl_netinit: corresponds with a network adapter on this system"
285     net_init_failed
286 fi
287
288
289 # actually check to make sure ifconfig <device> succeeds
290 /sbin/ifconfig $ETH_DEVICE up 2>&1 > /dev/null
291 if [[ $? -ne 0 ]]; then
292     echo $(date "+%H:%M:%S") " pl_netinit: device $ETH_DEVICE does not exist, most likely"
293     echo $(date "+%H:%M:%S") " pl_netinit: this cd does not have hardware support for your"
294     echo $(date "+%H:%M:%S") " pl_netinit: network adapter. please send the following lines"
295     echo $(date "+%H:%M:%S") " pl_netinit: to PlanetLab Support: support@planet-lab.org"
296     echo $(date "+%H:%M:%S") " pl_netinit: for further assistance"
297     echo
298     /sbin/lspci -n | /bin/grep "Class 0200"    
299     echo
300
301     net_init_failed
302 fi
303
304 echo $(date "+%H:%M:%S") " pl_netinit: attempting to start networking"
305 /sbin/service network start
306
307 # for backwards compatibility
308 /sbin/ifconfig $ETH_DEVICE > $IFCONFIG_OUTPUT
309
310 echo $(date "+%H:%M:%S") " pl_netinit: network online"
311