summaryrefslogtreecommitdiffstats
path: root/src/net
diff options
context:
space:
mode:
authorMichael Brown2016-07-19 17:57:32 +0200
committerMichael Brown2016-07-19 18:07:53 +0200
commit4ad3c73b3099cfe3b7f1c79ddfe9061809e4ac6d (patch)
tree6d576931282819e6f62db87b9c917a6740dc0176 /src/net
parent[settings] Allow settings blocks to specify a sibling ordering (diff)
downloadipxe-4ad3c73b3099cfe3b7f1c79ddfe9061809e4ac6d.tar.gz
ipxe-4ad3c73b3099cfe3b7f1c79ddfe9061809e4ac6d.tar.xz
ipxe-4ad3c73b3099cfe3b7f1c79ddfe9061809e4ac6d.zip
[ipv6] Match user expectations for IPv6 settings priorities
A reasonable user expectation is that ${net0/ip6} should show the "highest-priority" of the IPv6 addresses, even when multiple IPv6 addresses are active. The expected order of priority is likely to be manually-assigned addresses first, then stateful DHCPv6 addresses, then SLAAC addresses, and lastly link-local addresses. Using ${priority} to enforce an ordering is undesirable since that would affect the priority assigned to each of the net<N> blocks as a whole, so use the sibling ordering capability instead. Signed-off-by: Michael Brown <mcb30@ipxe.org>
Diffstat (limited to 'src/net')
-rw-r--r--src/net/ipv6.c1
-rw-r--r--src/net/ndp.c19
-rw-r--r--src/net/udp/dhcpv6.c1
3 files changed, 19 insertions, 2 deletions
diff --git a/src/net/ipv6.c b/src/net/ipv6.c
index cdd6640c..04ba3d8b 100644
--- a/src/net/ipv6.c
+++ b/src/net/ipv6.c
@@ -1185,6 +1185,7 @@ static int ipv6_register_settings ( struct net_device *netdev ) {
ref_init ( &ipv6set->refcnt, NULL );
settings_init ( &ipv6set->settings, &ipv6_settings_operations,
&ipv6set->refcnt, &ipv6_scope );
+ ipv6set->settings.order = IPV6_ORDER_LINK_LOCAL;
/* Register settings */
if ( ( rc = register_settings ( &ipv6set->settings, parent,
diff --git a/src/net/ndp.c b/src/net/ndp.c
index c488acc7..a35a1219 100644
--- a/src/net/ndp.c
+++ b/src/net/ndp.c
@@ -981,17 +981,28 @@ static int ndp_register_settings ( struct net_device *netdev,
size_t option_len;
unsigned int prefixes;
unsigned int instance;
+ int order;
int rc;
/* Count number of prefix options. We can assume that the
* options are well-formed, otherwise they would have been
* rejected prior to being stored.
*/
+ order = IPV6_ORDER_PREFIX_ONLY;
for ( prefixes = 0, offset = 0 ; offset < len ; offset += option_len ) {
+
+ /* Skip non-prefix options */
option = ( ( ( void * ) options ) + offset );
option_len = ( option->header.blocks * NDP_OPTION_BLKSZ );
- if ( option->header.type == NDP_OPT_PREFIX )
- prefixes++;
+ if ( option->header.type != NDP_OPT_PREFIX )
+ continue;
+
+ /* Count number of prefixes */
+ prefixes++;
+
+ /* Increase overall order if we have SLAAC addresses */
+ if ( option->prefix.flags & NDP_PREFIX_AUTONOMOUS )
+ order = IPV6_ORDER_SLAAC;
}
/* Allocate and initialise structure */
@@ -1004,6 +1015,7 @@ static int ndp_register_settings ( struct net_device *netdev,
ref_init ( &ndpset->refcnt, NULL );
settings_init ( &ndpset->settings, &ndp_settings_operations,
&ndpset->refcnt, &ndp_settings_scope );
+ ndpset->settings.order = order;
memcpy ( &ndpset->router, router, sizeof ( ndpset->router ) );
ndpset->lifetime = lifetime;
ndpset->len = len;
@@ -1028,6 +1040,9 @@ static int ndp_register_settings ( struct net_device *netdev,
settings_init ( &prefset->settings,
&ndp_prefix_settings_operations,
&ndpset->refcnt, &ndp_settings_scope );
+ prefset->settings.order =
+ ( ( option->prefix.flags & NDP_PREFIX_AUTONOMOUS ) ?
+ IPV6_ORDER_SLAAC : IPV6_ORDER_PREFIX_ONLY );
prefset->prefix = &option->prefix;
snprintf ( prefset->name, sizeof ( prefset->name ), "%d",
instance++ );
diff --git a/src/net/udp/dhcpv6.c b/src/net/udp/dhcpv6.c
index ac701042..a2c69aaa 100644
--- a/src/net/udp/dhcpv6.c
+++ b/src/net/udp/dhcpv6.c
@@ -375,6 +375,7 @@ static int dhcpv6_register ( struct in6_addr *lease,
ref_init ( &dhcpv6set->refcnt, NULL );
settings_init ( &dhcpv6set->settings, &dhcpv6_settings_operations,
&dhcpv6set->refcnt, &dhcpv6_scope );
+ dhcpv6set->settings.order = IPV6_ORDER_DHCPV6;
data = ( ( ( void * ) dhcpv6set ) + sizeof ( *dhcpv6set ) );
len = options->len;
memcpy ( data, options->data, len );