Hello all,
I ran into an issue where I was seeing a long delay in the scripts
called in udhcp_run_script. I was using an old version of OpenWrt
(kamikaze) and a satellite modem. An NTP script was being called and the
modem would sometimes take a long time to respond to the DNS lookup when
it was offline.
This delay started affecting my lease time. The lease that I would get
from my satellite modem before it was online would be short: only 60
seconds. The delay with NTP and the modem would typically be about 18
seconds. This would cause the first DHCP renew request from dhcpc to be
a little late. Under certain circumstances, I could even see the first
DHCP renew to occur after the lease had expired!
I patched dhcpc.c in my build using the following patch below. I used
the existing variable already_waited_sec to calculate the number of
seconds it took to complete running the scripts. If the delay was not
greater than the timeout (i.e. half the lease time), then the code would
simply send the renew using the already_waited_sec (e.g. 30 sec - 18 sec
= 12 sec to send the renew). If the delay was greater than the timeout
but less than the lease time, the code subtracts the time out from the
already_waited_sec, halves the time out, and continues as before. (E.g.
already_waited_sec is 40 sec, then the timeout would be 15 sec and
already_waited_sec would be changed to 10 sec. This would cause the
renew to be sent out 5 seconds later at the 45 second mark.) This
happens in a while loop while the timeout is greater than 0. (Note that
if a renew event is missed, it is simply skipped. E.g. In a 60 sec
lease, the 30 sec mark could be skipped and the next renew sent at 45
seconds.) If the already_waited_sec ends up being greater than the
timeout, the code will not send a renew and re-enter the requesting state.
Some improvements: If a renew event is missed, the code waits until the
next timeout. An improvement could be to send a renew immediately.
Another possible improvement would be to goto the renew event using a
goto rather than continuing. Both of these issues I did not see as a big
issue since I think both of these are unlikely events and are handled
well enough. I mostly saw issues with the scripts taking a few seconds
to run and wanted to account for the delay. I also did not want to send
a renew after the lease had expired.
I also put in some comments but I am not sure if they are up to par for
busybox. I would appreciate any feedback. Thank you.
--John Schroeder
--- a/networking/udhcp/dhcpc.c
+++ b/networking/udhcp/dhcpc.c
@@ -1752,6 +1752,7 @@
#endif
/* enter bound state */
timeout = lease_seconds / 2;
+ timestamp_before_wait = (unsigned)monotonic_sec();
temp_addr.s_addr = packet.yiaddr;
bb_info_msg("Lease of %s obtained, lease time %u",
inet_ntoa(temp_addr), (unsigned)lease_seconds);
@@ -1774,7 +1775,21 @@
#endif
/* make future renew packets use different xid */
/* xid = random_xid(); ...but why bother? */
- already_waited_sec = 0;
+ already_waited_sec = (unsigned)monotonic_sec() -
timestamp_before_wait;
+ /* If already beyond timeout, skip the next renew
+ * event. Continue skipping renews and halving the
+ * timeout until the next timeout is less that the
+ * already waited time or the timeout is 0.
+ */
+ while ((already_waited_sec > timeout) && (timeout > 0)) {
+ already_waited_sec -= timeout;
+ timeout >>= 1;
+ }
+ /* If timeout is 0, then the lease has expired. Set
+ * already_waited_sec to 0.
+ */
+ if (timeout == 0)
+ already_waited_sec = 0;
continue; /* back to main loop */
}
if (*message == DHCPNAK) {
_______________________________________________
busybox mailing list
[email protected]
http://lists.busybox.net/mailman/listinfo/busybox