freebsd revision 1.1 1 #!/bin/sh
2 #
3 # Id: freebsd,v 1.24 2011/05/18 19:55:44 sar Exp
4 #
5 # $FreeBSD$
6
7 if [ -x /usr/bin/logger ]; then
8 LOGGER="/usr/bin/logger -s -p user.notice -t dhclient"
9 else
10 LOGGER=echo
11 fi
12
13 make_resolv_conf() {
14 if [ x"$new_domain_name_servers" != x ]; then
15 ( cat /dev/null > /etc/resolv.conf.dhclient )
16 exit_status=$?
17 if [ $exit_status -ne 0 ]; then
18 $LOGGER "Unable to create /etc/resolv.conf.dhclient: Error $exit_status"
19 else
20 if [ "x$new_domain_search" != x ]; then
21 ( echo search $new_domain_search >> /etc/resolv.conf.dhclient )
22 exit_status=$?
23 elif [ "x$new_domain_name" != x ]; then
24 # Note that the DHCP 'Domain Name Option' is really just a domain
25 # name, and that this practice of using the domain name option as
26 # a search path is both nonstandard and deprecated.
27 ( echo search $new_domain_name >> /etc/resolv.conf.dhclient )
28 exit_status=$?
29 fi
30 for nameserver in $new_domain_name_servers; do
31 if [ $exit_status -ne 0 ]; then
32 break
33 fi
34 ( echo nameserver $nameserver >>/etc/resolv.conf.dhclient )
35 exit_status=$?
36 done
37
38 # If there were no errors, attempt to mv the new file into place.
39 if [ $exit_status -eq 0 ]; then
40 ( mv /etc/resolv.conf.dhclient /etc/resolv.conf )
41 exit_status=$?
42 fi
43
44 if [ $exit_status -ne 0 ]; then
45 $LOGGER "Error while writing new /etc/resolv.conf."
46 fi
47 fi
48 elif [ "x${new_dhcp6_name_servers}" != x ] ; then
49 ( cat /dev/null > /etc/resolv.conf.dhclient6 )
50 exit_status=$?
51 if [ $exit_status -ne 0 ] ; then
52 $LOGGER "Unable to create /etc/resolv.conf.dhclient6: Error $exit_status"
53 else
54 if [ "x${new_dhcp6_domain_search}" != x ] ; then
55 ( echo search ${new_dhcp6_domain_search} >> /etc/resolv.conf.dhclient6 )
56 exit_status=$?
57 fi
58 for nameserver in ${new_dhcp6_name_servers} ; do
59 if [ $exit_status -ne 0 ] ; then
60 break
61 fi
62 # If the nameserver has a link-local address
63 # add a <zone_id> (interface name) to it.
64 case $nameserver in
65 fe80:*) zone_id="%$interface";;
66 FE80:*) zone_id="%$interface";;
67 *) zone_id="";;
68 esac
69 ( echo nameserver ${nameserver}$zone_id >> /etc/resolv.conf.dhclient6 )
70 exit_status=$?
71 done
72
73 if [ $exit_status -eq 0 ] ; then
74 ( mv /etc/resolv.conf.dhclient6 /etc/resolv.conf )
75 exit_status=$?
76 fi
77
78 if [ $exit_status -ne 0 ] ; then
79 $LOGGER "Error while writing new /etc/resolv.conf."
80 fi
81 fi
82 fi
83 }
84
85 # Must be used on exit. Invokes the local dhcp client exit hooks, if any.
86 exit_with_hooks() {
87 exit_status=$1
88 if [ -f /etc/dhclient-exit-hooks ]; then
89 . /etc/dhclient-exit-hooks
90 fi
91 # probably should do something with exit status of the local script
92 exit $exit_status
93 }
94
95 # This function was largely borrowed from dhclient-script that
96 # ships with Centos, authored by Jiri Popelka and David Cantrell
97 # of Redhat. Thanks guys.
98 add_ipv6_addr_with_DAD() {
99 ifconfig ${interface} inet6 ${new_ip6_address}/${new_ip6_prefixlen} alias
100
101 if [ ${dad_wait_time} -le 0 ]
102 then
103 # if we're not waiting for DAD, assume we're good
104 return 0
105 fi
106
107 # Repeatedly test whether newly added address passed
108 # duplicate address detection (DAD)
109 for i in $(seq 1 ${dad_wait_time}); do
110 sleep 1 # give the DAD some time
111
112 addr=$(ifconfig ${interface} \
113 | grep "${new_ip6_address} prefixlen ${new_ip6_prefixlen}")
114
115 # tentative flag == DAD is still not complete
116 tentative=$(echo "${addr}" | grep tentative)
117 # dadfailed flag == address is already in use somewhere else
118 dadfailed=$(echo "${addr}" | grep duplicated)
119
120 if [ -n "${dadfailed}" ] ; then
121 # dad failed, remove the address
122 ifconfig ${interface} inet6 ${new_ip6_address}/${new_ip6_prefixlen} -alias
123 exit_with_hooks 3
124 fi
125
126 if [ -z "${tentative}" ] ; then
127 if [ -n "${addr}" ]; then
128 # DAD is over
129 return 0
130 else
131 # address was auto-removed (or not added at all)
132 exit_with_hooks 3
133 fi
134 fi
135 done
136
137 return 0
138 }
139
140 # Invoke the local dhcp client enter hooks, if they exist.
141 if [ -f /etc/dhclient-enter-hooks ]; then
142 exit_status=0
143 . /etc/dhclient-enter-hooks
144 # allow the local script to abort processing of this state
145 # local script must set exit_status variable to nonzero.
146 if [ $exit_status -ne 0 ]; then
147 exit $exit_status
148 fi
149 fi
150
151 if [ x$new_network_number != x ]; then
152 $LOGGER New Network Number: $new_network_number
153 fi
154
155 if [ x$new_broadcast_address != x ]; then
156 $LOGGER New Broadcast Address: $new_broadcast_address
157 new_broadcast_arg="broadcast $new_broadcast_address"
158 fi
159 if [ x$old_broadcast_address != x ]; then
160 old_broadcast_arg="broadcast $old_broadcast_address"
161 fi
162 if [ x$new_subnet_mask != x ]; then
163 new_netmask_arg="netmask $new_subnet_mask"
164 fi
165 if [ x$old_subnet_mask != x ]; then
166 old_netmask_arg="netmask $old_subnet_mask"
167 fi
168 if [ x$alias_subnet_mask != x ]; then
169 alias_subnet_arg="netmask $alias_subnet_mask"
170 fi
171 if [ x$new_interface_mtu != x ]; then
172 mtu_arg="mtu $new_interface_mtu"
173 fi
174 if [ x$IF_METRIC != x ]; then
175 metric_arg="metric $IF_METRIC"
176 fi
177
178 if [ x$reason = xMEDIUM ]; then
179 eval "ifconfig $interface $medium"
180 eval "ifconfig $interface inet -alias 0.0.0.0 $medium" >/dev/null 2>&1
181 sleep 1
182 exit_with_hooks 0
183 fi
184
185 ###
186 ### DHCPv4 Handlers
187 ###
188
189 if [ x$reason = xPREINIT ]; then
190 if [ x$alias_ip_address != x ]; then
191 ifconfig $interface inet -alias $alias_ip_address > /dev/null 2>&1
192 route delete $alias_ip_address 127.0.0.1 > /dev/null 2>&1
193 fi
194 ifconfig $interface inet 0.0.0.0 netmask 0.0.0.0 \
195 broadcast 255.255.255.255 up
196 exit_with_hooks 0
197 fi
198
199 if [ x$reason = xARPCHECK ] || [ x$reason = xARPSEND ]; then
200 exit_with_hooks 0;
201 fi
202
203 if [ x$reason = xBOUND ] || [ x$reason = xRENEW ] || \
204 [ x$reason = xREBIND ] || [ x$reason = xREBOOT ]; then
205 current_hostname=`/bin/hostname`
206 if [ x$current_hostname = x ] || \
207 [ x$current_hostname = x$old_host_name ]; then
208 if [ x$current_hostname = x ] || \
209 [ x$new_host_name != x$old_host_name ]; then
210 $LOGGER "New Hostname: $new_host_name"
211 hostname $new_host_name
212 fi
213 fi
214 if [ x$old_ip_address != x ] && [ x$alias_ip_address != x ] && \
215 [ x$alias_ip_address != x$old_ip_address ]; then
216 ifconfig $interface inet -alias $alias_ip_address > /dev/null 2>&1
217 route delete $alias_ip_address 127.0.0.1 > /dev/null 2>&1
218 fi
219 if [ x$old_ip_address != x ] && [ x$old_ip_address != x$new_ip_address ]
220 then
221 eval "ifconfig $interface inet -alias $old_ip_address $medium"
222 route delete $old_ip_address 127.1 >/dev/null 2>&1
223 for router in $old_routers; do
224 route delete default $router >/dev/null 2>&1
225 done
226 if [ -n "$old_static_routes" ]; then
227 set -- $old_static_routes
228 while [ $# -gt 1 ]; do
229 route delete $1 $2
230 shift; shift
231 done
232 fi
233 arp -n -a | sed -n -e 's/^.*(\(.*\)) at .*$/arp -d \1/p' |sh
234 fi
235 if [ x$old_ip_address = x ] || [ x$old_ip_address != x$new_ip_address ] || \
236 [ x$reason = xBOUND ] || [ x$reason = xREBOOT ]; then
237 eval "ifconfig $interface inet $new_ip_address $new_netmask_arg \
238 $new_broadcast_arg $mtu_arg $metric_arg $medium"
239 $LOGGER "New IP Address ($interface): $new_ip_address"
240 $LOGGER "New Subnet Mask ($interface): $new_subnet_mask"
241 $LOGGER "New Broadcast Address ($interface): $new_broadcast_address"
242 if [ -n "$new_routers" ]; then
243 $LOGGER "New Routers: $new_routers"
244 fi
245 route add $new_ip_address 127.1 >/dev/null 2>&1
246 for router in $new_routers; do
247 # If the subnet is captive, eg the netmask is /32 but the default
248 # gateway is (obviously) outside of this, then we need to produce a
249 # host route to reach the gateway.
250 if [ "x$new_subnet_mask" = "x255.255.255.255" ] ; then
251 route add -host $router -interface $interface
252 fi
253 route add default $router >/dev/null 2>&1
254 done
255 if [ -n "$new_static_routes" ]; then
256 $LOGGER "New Static Routes: $new_static_routes"
257 set -- $new_static_routes
258 while [ $# -gt 1 ]; do
259 route add $1 $2
260 shift; shift
261 done
262 fi
263 else
264 # we haven't changed the address, have we changed other options
265 # that we wish to update?
266 if [ x$new_routers != x ] && [ x$new_routers != x$old_routers ] ; then
267 # if we've changed routers delete the old and add the new.
268 $LOGGER "New Routers: $new_routers"
269 for router in $old_routers; do
270 route delete default $router >/dev/null 2>&1
271 done
272 for router in $new_routers; do
273 # If the subnet is captive, eg the netmask is /32 but the default
274 # gateway is (obviously) outside of this, then we need to produce a
275 # host route to reach the gateway.
276 if [ "x$new_subnet_mask" = "x255.255.255.255" ] ; then
277 route add -host $router -interface $interface
278 fi
279 route add default $router >/dev/null 2>&1
280 done
281 fi
282 fi
283 if [ x$new_ip_address != x$alias_ip_address ] && [ x$alias_ip_address != x ];
284 then
285 ifconfig $interface inet alias $alias_ip_address $alias_subnet_arg
286 route add $alias_ip_address 127.0.0.1
287 fi
288 make_resolv_conf
289 exit_with_hooks 0
290 fi
291
292 if [ x$reason = xEXPIRE ] || [ x$reason = xFAIL ] || [ x$reason = xRELEASE ] \
293 || [ x$reason = xSTOP ]; then
294 if [ x$alias_ip_address != x ]; then
295 ifconfig $interface inet -alias $alias_ip_address > /dev/null 2>&1
296 route delete $alias_ip_address 127.0.0.1 > /dev/null 2>&1
297 fi
298 if [ x$old_ip_address != x ]; then
299 eval "ifconfig $interface inet -alias $old_ip_address $medium"
300 route delete $old_ip_address 127.1 >/dev/null 2>&1
301 for router in $old_routers; do
302 route delete default $router >/dev/null 2>&1
303 done
304 if [ -n "$old_static_routes" ]; then
305 set -- $old_static_routes
306 while [ $# -gt 1 ]; do
307 route delete $1 $2
308 shift; shift
309 done
310 fi
311 arp -n -a | sed -n -e 's/^.*(\(.*\)) at .*$/arp -d \1/p' \
312 |sh >/dev/null 2>&1
313 fi
314 if [ x$alias_ip_address != x ]; then
315 ifconfig $interface inet alias $alias_ip_address $alias_subnet_arg
316 route add $alias_ip_address 127.0.0.1
317 fi
318 exit_with_hooks 0
319 fi
320
321 if [ x$reason = xTIMEOUT ]; then
322 if [ x$alias_ip_address != x ]; then
323 ifconfig $interface inet -alias $alias_ip_address > /dev/null 2>&1
324 route delete $alias_ip_address 127.0.0.1 > /dev/null 2>&1
325 fi
326 eval "ifconfig $interface inet $new_ip_address $new_netmask_arg \
327 $new_broadcast_arg $mtu_arg $metric_arg $medium"
328 $LOGGER "New IP Address ($interface): $new_ip_address"
329 $LOGGER "New Subnet Mask ($interface): $new_subnet_mask"
330 $LOGGER "New Broadcast Address ($interface): $new_broadcast_address"
331 sleep 1
332 if [ -n "$new_routers" ]; then
333 $LOGGER "New Routers: $new_routers"
334 set -- $new_routers
335 if ping -q -c 1 $1; then
336 if [ x$new_ip_address != x$alias_ip_address ] && \
337 [ x$alias_ip_address != x ]; then
338 ifconfig $interface inet alias $alias_ip_address $alias_subnet_arg
339 route add $alias_ip_address 127.0.0.1
340 fi
341 route add $new_ip_address 127.1 >/dev/null 2>&1
342 for router in $new_routers; do
343 if [ "x$new_subnet_mask" = "x255.255.255.255" ] ; then
344 route add -host $router -interface $interface
345 fi
346 route add default $router >/dev/null 2>&1
347 done
348 set -- $new_static_routes
349 while [ $# -gt 1 ]; do
350 route add $1 $2
351 shift; shift
352 done
353 make_resolv_conf
354 exit_with_hooks 0
355 fi
356 fi
357 eval "ifconfig $interface inet -alias $new_ip_address $medium"
358 for router in $old_routers; do
359 route delete default $router >/dev/null 2>&1
360 done
361 if [ -n "$old_static_routes" ]; then
362 set -- $old_static_routes
363 while [ $# -gt 1 ]; do
364 route delete $1 $2
365 shift; shift
366 done
367 fi
368 arp -n -a | sed -n -e 's/^.*(\(.*\)) at .*$/arp -d \1/p' \
369 |sh >/dev/null 2>&1
370 exit_with_hooks 1
371 fi
372
373 ###
374 ### DHCPv6 Handlers
375 ###
376
377 if [ ${reason} = PREINIT6 ] ; then
378 # Ensure interface is up.
379 ifconfig ${interface} up
380
381 # XXX: Remove any stale addresses from aborted clients.
382
383 # We need to give the kernel some time to active interface
384 interface_up_wait_time=5
385 for i in $(seq 0 ${interface_up_wait_time})
386 do
387 ifconfig ${interface} | grep inactive >/dev/null 2>&1
388 if [ $? -ne 0 ]; then
389 break;
390 fi
391 sleep 1
392 done
393
394 # Wait for duplicate address detection for this interface if the
395 # --dad-wait-time parameter has been specified and is greater than
396 # zero.
397 if [ ${dad_wait_time} -gt 0 ]; then
398 # Check if any IPv6 address on this interface is marked as
399 # tentative.
400 ifconfig ${interface} | grep inet6 | grep tentative \
401 >/dev/null 2>&1
402 if [ $? -eq 0 ]; then
403 # Wait for duplicate address detection to complete or for
404 # the timeout specified as --dad-wait-time.
405 for i in $(seq 0 $dad_wait_time)
406 do
407 # We're going to poll for the tentative flag every second.
408 sleep 1
409 ifconfig ${interface} | grep inet6 | grep tentative \
410 >/dev/null 2>&1
411 if [ $? -ne 0 ]; then
412 break;
413 fi
414 done
415 fi
416 fi
417
418
419 exit_with_hooks 0
420 fi
421
422 if [ x${old_ip6_prefix} != x ] || [ x${new_ip6_prefix} != x ] ; then
423 echo Prefix ${reason} old=${old_ip6_prefix} new=${new_ip6_prefix}
424
425 exit_with_hooks 0
426 fi
427
428 if [ ${reason} = BOUND6 ] ; then
429 if [ x${new_ip6_address} = x ] || [ x${new_ip6_prefixlen} = x ] ; then
430 exit_with_hooks 2;
431 fi
432
433 # Add address to interface, check for DAD if dad_wait_time > 0
434 add_ipv6_addr_with_DAD
435
436 # Check for nameserver options.
437 make_resolv_conf
438
439 exit_with_hooks 0
440 fi
441
442 if [ ${reason} = RENEW6 ] || [ ${reason} = REBIND6 ] ; then
443 # Make sure nothing has moved around on us.
444
445 # Nameservers/domains/etc.
446 if [ "x${new_dhcp6_name_servers}" != "x${old_dhcp6_name_servers}" ] ||
447 [ "x${new_dhcp6_domain_search}" != "x${old_dhcp6_domain_search}" ] ; then
448 make_resolv_conf
449 fi
450
451 exit_with_hooks 0
452 fi
453
454 if [ ${reason} = DEPREF6 ] ; then
455 if [ x${new_ip6_address} = x ] ; then
456 exit_with_hooks 2;
457 fi
458
459 ifconfig ${interface} inet6 ${new_ip6_address} deprecated
460
461 exit_with_hooks 0
462 fi
463
464 if [ ${reason} = EXPIRE6 -o ${reason} = RELEASE6 -o ${reason} = STOP6 ] ; then
465 if [ x${old_ip6_address} = x ] || [ x${old_ip6_prefixlen} = x ] ; then
466 exit_with_hooks 2;
467 fi
468
469 ifconfig ${interface} inet6 ${old_ip6_address}/${old_ip6_prefixlen} -alias
470
471 exit_with_hooks 0
472 fi
473
474 exit_with_hooks 0
475