iPXE uses DHCP timeouts loosely based on values recommended by the specification, but often abbreviated to reduce timeouts for reliable and/or simple network topologies. Previous attempts to change the defaults to more spec-compliant values have met resistance and apathy, therefore this patch simply tries to extract the timing parameters to a config file and document them. The resulting default iPXE behavior is exactly the same, but downstreams are now afforded the opportunity to implement spec compliant behavior via config file overrides.
I believe the following overrides defined in config/local/general.h provide sufficiently spec compliant DHCP timeouts: /* * PXE spec defines timeouts of 4, 8, 16, 32 seconds */ #undef DHCP_DISC_START_TIMEOUT_SEC #define DHCP_DISC_START_TIMEOUT_SEC 4 #undef DHCP_DISC_END_TIMEOUT_SEC #define DHCP_DISC_END_TIMEOUT_SEC 32 /* * Elapsed time used for early break waiting for ProxyDHCP, this therefore * needs to be less than the cumulative time for the first 2 timeouts. */ #undef DHCP_DISC_PROXY_TIMEOUT_SEC #define DHCP_DISC_PROXY_TIMEOUT_SEC 11 /* * Approximate PXE spec requirement using minimum timeout (0.25s) for * timeouts of 0.25, 0.5, 1, 2, 4 */ #undef DHCP_REQ_START_TIMEOUT_SEC #define DHCP_REQ_START_TIMEOUT_SEC 0 #undef DHCP_REQ_END_TIMEOUT_SEC #define DHCP_REQ_END_TIMEOUT_SEC 4 /* * Same as normal request phase, except non-fatal, so we extend the timer * to 8 and set the early timeout to an elapsed time value that causes a * break after the 4 second timeout. */ #undef DHCP_PROXY_START_TIMEOUT_SEC #define DHCP_PROXY_START_TIMEOUT_SEC 0 #undef DHCP_PROXY_END_TIMEOUT_SEC #define DHCP_PROXY_END_TIMEOUT_SEC 8 #undef DHCP_REQ_PROXY_TIMEOUT_SEC #define DHCP_REQ_PROXY_TIMEOUT_SEC 7 /* * Same as above, retry each server using standard timeouts, extended by * one so that we can increment to the next before a timer induced failure. */ #undef PXEBS_START_TIMEOUT_SEC #define PXEBS_START_TIMEOUT_SEC 0 #undef PXEBS_END_TIMEOUT_SEC #define PXEBS_END_TIMEOUT_SEC 8 #undef PXEBS_MAX_TIMEOUT_SEC #define PXEBS_MAX_TIMEOUT_SEC 7 Signed-off-by: Alex Williamson <[email protected]> --- src/config/general.h | 61 +++++++++++++++++++++++++++++++++++++++++++++++ src/include/ipxe/dhcp.h | 11 +------- src/net/udp/dhcp.c | 31 ++++++++++++++---------- 3 files changed, 80 insertions(+), 23 deletions(-) diff --git a/src/config/general.h b/src/config/general.h index 5392034..bead2ac 100644 --- a/src/config/general.h +++ b/src/config/general.h @@ -182,6 +182,67 @@ FILE_LICENCE ( GPL2_OR_LATER ); #undef GDBUDP /* Remote GDB debugging over UDP * (both may be set) */ +/* + * DHCP and PXE Boot Server timeout parameters + * + * Initial and final timeout for DHCP discovery + * + * The PXE spec indicates discover request are sent 4 times, with timeouts + * of 4, 8, 16, 32 seconds. iPXE by default uses 1, 2, 4, 8. + */ +#define DHCP_DISC_START_TIMEOUT_SEC 1 +#define DHCP_DISC_END_TIMEOUT_SEC 10 + +/* + * ProxyDHCP offers are given precedence by continue to wait for them after + * a valid DHCPOFFER is received. We'll wait through this timeout for it. + * The PXE spec indicates waiting through the 4 & 8 second timeouts, iPXE + * by default stops after 2. + */ +#define DHCP_DISC_PROXY_TIMEOUT_SEC 2 + +/* + * Per the PXE spec, requests are also tried 4 times, but at timeout intervals + * of 1, 2, 3, 4 seconds. To adapt this to an exponential backoff timer, we + * can either do 1, 2, 4, 8, ie. 4 retires with a longer interval or start at + * 0 (0.25s) for 0.25, 0.5, 1, 2, 4, ie. one extra try and shorter initial + * timeouts. iPXE by default does a combination of both, starting at 0 and + * going through the 8 second timeout. + */ +#define DHCP_REQ_START_TIMEOUT_SEC 0 +#define DHCP_REQ_END_TIMEOUT_SEC 10 + +/* + * A ProxyDHCP offer without PXE options also goes through a request phase + * using these same parameters, but note the early break below. + */ +#define DHCP_PROXY_START_TIMEOUT_SEC 0 +#define DHCP_PROXY_END_TIMEOUT_SEC 10 + +/* + * A ProxyDHCP request timeout should not induce a failure condition, so we + * always want to break before the above set of timers expire. The iPXE + * default value of 2 breaks at the first timeout after 2 seconds, which will + * be after the 2 second timeout. + */ +#define DHCP_REQ_PROXY_TIMEOUT_SEC 2 + +/* + * Per the PXE spec, a PXE boot server request is also be retried 4 times + * at timeouts of 1, 2, 3, 4. iPXE uses the same timeouts as discovery, + * 1, 2, 4, 8, but will move on to the next server if available after an + * elapsed time greater than 3 seconds, therefore effectively only sending + * 3 tries at timeouts of 1, 2, 4. + */ +#define PXEBS_START_TIMEOUT_SEC 1 +#define PXEBS_END_TIMEOUT_SEC 10 + +/* + * Increment to the next PXE Boot server, if available, after this this much + * time has elapsed. + */ +#define PXEBS_MAX_TIMEOUT_SEC 3 + #include <config/named.h> #include NAMED_CONFIG(general.h) #include <config/local/general.h> diff --git a/src/include/ipxe/dhcp.h b/src/include/ipxe/dhcp.h index bcfb85c..d1dc5bc 100644 --- a/src/include/ipxe/dhcp.h +++ b/src/include/ipxe/dhcp.h @@ -18,6 +18,7 @@ FILE_LICENCE ( GPL2_OR_LATER ); #include <ipxe/uuid.h> #include <ipxe/netdevice.h> #include <ipxe/uaccess.h> +#include <config/general.h> struct interface; struct dhcp_options; @@ -639,16 +640,6 @@ struct dhcphdr { */ #define DHCP_MIN_LEN 552 -/** Timeouts for sending DHCP packets */ -#define DHCP_MIN_TIMEOUT ( 1 * TICKS_PER_SEC ) -#define DHCP_MAX_TIMEOUT ( 10 * TICKS_PER_SEC ) - -/** Maximum time that we will wait for ProxyDHCP responses */ -#define PROXYDHCP_MAX_TIMEOUT ( 2 * TICKS_PER_SEC ) - -/** Maximum time that we will wait for Boot Server responses */ -#define PXEBS_MAX_TIMEOUT ( 3 * TICKS_PER_SEC ) - /** Settings block name used for DHCP responses */ #define DHCP_SETTINGS_NAME "dhcp" diff --git a/src/net/udp/dhcp.c b/src/net/udp/dhcp.c index 04fad04..3527c44 100644 --- a/src/net/udp/dhcp.c +++ b/src/net/udp/dhcp.c @@ -171,8 +171,9 @@ struct dhcp_session_state { void ( * expired ) ( struct dhcp_session *dhcp ); /** Transmitted message type */ uint8_t tx_msgtype; - /** Apply minimum timeout */ - uint8_t apply_min_timeout; + /** Timeout parameters */ + uint8_t min_timeout_sec; + uint8_t max_timeout_sec; }; static struct dhcp_session_state dhcp_state_discover; @@ -272,9 +273,8 @@ static void dhcp_set_state ( struct dhcp_session *dhcp, dhcp->state = state; dhcp->start = currticks(); stop_timer ( &dhcp->timer ); - dhcp->timer.min_timeout = - ( state->apply_min_timeout ? DHCP_MIN_TIMEOUT : 0 ); - dhcp->timer.max_timeout = DHCP_MAX_TIMEOUT; + dhcp->timer.min_timeout = state->min_timeout_sec * TICKS_PER_SEC; + dhcp->timer.max_timeout = state->max_timeout_sec * TICKS_PER_SEC; start_timer_nodelay ( &dhcp->timer ); } @@ -415,7 +415,7 @@ static void dhcp_discovery_rx ( struct dhcp_session *dhcp, /* If we can't yet transition to DHCPREQUEST, do nothing */ elapsed = ( currticks() - dhcp->start ); if ( ! ( dhcp->no_pxedhcp || dhcp->proxy_offer || - ( elapsed > PROXYDHCP_MAX_TIMEOUT ) ) ) + ( elapsed > DHCP_DISC_PROXY_TIMEOUT_SEC * TICKS_PER_SEC ) ) ) return; /* Transition to DHCPREQUEST */ @@ -431,7 +431,8 @@ static void dhcp_discovery_expired ( struct dhcp_session *dhcp ) { unsigned long elapsed = ( currticks() - dhcp->start ); /* Give up waiting for ProxyDHCP before we reach the failure point */ - if ( dhcp->offer.s_addr && ( elapsed > PROXYDHCP_MAX_TIMEOUT ) ) { + if ( dhcp->offer.s_addr && + ( elapsed > DHCP_DISC_PROXY_TIMEOUT_SEC * TICKS_PER_SEC ) ) { dhcp_set_state ( dhcp, &dhcp_state_request ); return; } @@ -447,7 +448,8 @@ static struct dhcp_session_state dhcp_state_discover = { .rx = dhcp_discovery_rx, .expired = dhcp_discovery_expired, .tx_msgtype = DHCPDISCOVER, - .apply_min_timeout = 1, + .min_timeout_sec = DHCP_DISC_START_TIMEOUT_SEC, + .max_timeout_sec = DHCP_DISC_END_TIMEOUT_SEC, }; /** @@ -584,7 +586,8 @@ static struct dhcp_session_state dhcp_state_request = { .rx = dhcp_request_rx, .expired = dhcp_request_expired, .tx_msgtype = DHCPREQUEST, - .apply_min_timeout = 0, + .min_timeout_sec = DHCP_REQ_START_TIMEOUT_SEC, + .max_timeout_sec = DHCP_REQ_END_TIMEOUT_SEC, }; /** @@ -669,7 +672,7 @@ static void dhcp_proxy_expired ( struct dhcp_session *dhcp ) { unsigned long elapsed = ( currticks() - dhcp->start ); /* Give up waiting for ProxyDHCP before we reach the failure point */ - if ( elapsed > PROXYDHCP_MAX_TIMEOUT ) { + if ( elapsed > DHCP_REQ_PROXY_TIMEOUT_SEC * TICKS_PER_SEC ) { dhcp_finished ( dhcp, 0 ); return; } @@ -685,7 +688,8 @@ static struct dhcp_session_state dhcp_state_proxy = { .rx = dhcp_proxy_rx, .expired = dhcp_proxy_expired, .tx_msgtype = DHCPREQUEST, - .apply_min_timeout = 0, + .min_timeout_sec = DHCP_PROXY_START_TIMEOUT_SEC, + .max_timeout_sec = DHCP_PROXY_END_TIMEOUT_SEC, }; /** @@ -810,7 +814,7 @@ static void dhcp_pxebs_expired ( struct dhcp_session *dhcp ) { /* Give up waiting before we reach the failure point, and fail * over to the next server in the attempt list */ - if ( elapsed > PXEBS_MAX_TIMEOUT ) { + if ( elapsed > PXEBS_MAX_TIMEOUT_SEC * TICKS_PER_SEC ) { dhcp->pxe_attempt++; if ( dhcp->pxe_attempt->s_addr ) { dhcp_set_state ( dhcp, &dhcp_state_pxebs ); @@ -832,7 +836,8 @@ static struct dhcp_session_state dhcp_state_pxebs = { .rx = dhcp_pxebs_rx, .expired = dhcp_pxebs_expired, .tx_msgtype = DHCPREQUEST, - .apply_min_timeout = 1, + .min_timeout_sec = PXEBS_START_TIMEOUT_SEC, + .max_timeout_sec = PXEBS_END_TIMEOUT_SEC, }; /**************************************************************************** _______________________________________________ ipxe-devel mailing list [email protected] https://lists.ipxe.org/mailman/listinfo.cgi/ipxe-devel

