diff options
author | Dmitry Kozlov <xeb@mail.ru> | 2015-01-29 16:36:27 +0300 |
---|---|---|
committer | Dmitry Kozlov <xeb@mail.ru> | 2015-01-29 16:36:27 +0300 |
commit | bb92e3829330860b75ce6a5882bf10930da9ba52 (patch) | |
tree | 9db97ea42f40670ff24910c4453d129fd2c17935 /accel-pppd/ipv6 | |
parent | a5473ad84dc884f64bab3208da7f1b30c0a5015b (diff) | |
download | accel-ppp-bb92e3829330860b75ce6a5882bf10930da9ba52.tar.gz accel-ppp-bb92e3829330860b75ce6a5882bf10930da9ba52.zip |
ipv6: don't send NotOnLink for hint address (f.e. for address in Solicit message)
Diffstat (limited to 'accel-pppd/ipv6')
-rw-r--r-- | accel-pppd/ipv6/dhcpv6.c | 92 |
1 files changed, 48 insertions, 44 deletions
diff --git a/accel-pppd/ipv6/dhcpv6.c b/accel-pppd/ipv6/dhcpv6.c index 3f9df71d..2047506e 100644 --- a/accel-pppd/ipv6/dhcpv6.c +++ b/accel-pppd/ipv6/dhcpv6.c @@ -310,31 +310,33 @@ static void dhcpv6_send_reply(struct dhcpv6_packet *req, struct dhcpv6_pd *pd, i } } - list_for_each_entry(opt2, &opt->opt_list, entry) { - if (ntohs(opt2->hdr->code) == D6_OPTION_IAADDR) { - ia_addr = (struct dhcpv6_opt_ia_addr *)opt2->hdr; + if (code == D6_REPLY) { + list_for_each_entry(opt2, &opt->opt_list, entry) { + if (ntohs(opt2->hdr->code) == D6_OPTION_IAADDR) { + ia_addr = (struct dhcpv6_opt_ia_addr *)opt2->hdr; - if (IN6_IS_ADDR_UNSPECIFIED(&ia_addr->addr)) - continue; - - f1 = 0; - list_for_each_entry(a, &ses->ipv6->addr_list, entry) { - build_addr(a, ses->ipv6->peer_intf_id, &addr); - if (memcmp(&addr, &ia_addr->addr, sizeof(addr))) + if (IN6_IS_ADDR_UNSPECIFIED(&ia_addr->addr)) continue; - f1 = 1; - break; - } - - if (!f1) { - opt3 = dhcpv6_nested_option_alloc(reply, opt1, D6_OPTION_IAADDR, sizeof(*ia_addr) - sizeof(struct dhcpv6_opt_hdr)); - memcpy(opt3->hdr->data, opt2->hdr->data, sizeof(*ia_addr) - sizeof(struct dhcpv6_opt_hdr)); - - ia_addr = (struct dhcpv6_opt_ia_addr *)opt3->hdr; - ia_addr->pref_lifetime = 0; - ia_addr->valid_lifetime = 0; - insert_status(reply, opt3, D6_STATUS_NotOnLink); + f1 = 0; + list_for_each_entry(a, &ses->ipv6->addr_list, entry) { + build_addr(a, ses->ipv6->peer_intf_id, &addr); + if (memcmp(&addr, &ia_addr->addr, sizeof(addr))) + continue; + f1 = 1; + break; + } + + if (!f1) { + opt3 = dhcpv6_nested_option_alloc(reply, opt1, D6_OPTION_IAADDR, sizeof(*ia_addr) - sizeof(struct dhcpv6_opt_hdr)); + memcpy(opt3->hdr->data, opt2->hdr->data, sizeof(*ia_addr) - sizeof(struct dhcpv6_opt_hdr)); + + ia_addr = (struct dhcpv6_opt_ia_addr *)opt3->hdr; + ia_addr->pref_lifetime = 0; + ia_addr->valid_lifetime = 0; + + insert_status(reply, opt3, D6_STATUS_NotOnLink); + } } } } @@ -381,31 +383,33 @@ static void dhcpv6_send_reply(struct dhcpv6_packet *req, struct dhcpv6_pd *pd, i ia_prefix->valid_lifetime = htonl(conf_valid_lifetime); } - list_for_each_entry(opt2, &opt->opt_list, entry) { - if (ntohs(opt2->hdr->code) == D6_OPTION_IAPREFIX) { - ia_prefix = (struct dhcpv6_opt_ia_prefix *)opt2->hdr; + if (code == D6_REPLY) { + list_for_each_entry(opt2, &opt->opt_list, entry) { + if (ntohs(opt2->hdr->code) == D6_OPTION_IAPREFIX) { + ia_prefix = (struct dhcpv6_opt_ia_prefix *)opt2->hdr; - if (ia_prefix->prefix_len == 0 || IN6_IS_ADDR_UNSPECIFIED(&ia_prefix->prefix)) - continue; - - f1 = 0; - list_for_each_entry(a, &ses->ipv6_dp->prefix_list, entry) { - if (a->prefix_len != ia_prefix->prefix_len) - continue; - if (memcmp(&a->addr, &ia_prefix->prefix, sizeof(a->addr))) + if (ia_prefix->prefix_len == 0 || IN6_IS_ADDR_UNSPECIFIED(&ia_prefix->prefix)) continue; - f1 = 1; - break; - } - - if (!f1) { - opt3 = dhcpv6_nested_option_alloc(reply, opt1, D6_OPTION_IAPREFIX, sizeof(*ia_prefix) - sizeof(struct dhcpv6_opt_hdr)); - memcpy(opt3->hdr->data, opt2->hdr->data, sizeof(*ia_prefix) - sizeof(struct dhcpv6_opt_hdr)); - ia_prefix = (struct dhcpv6_opt_ia_prefix *)opt3->hdr; - ia_prefix->pref_lifetime = 0; - ia_prefix->valid_lifetime = 0; - insert_status(reply, opt3, D6_STATUS_NotOnLink); + f1 = 0; + list_for_each_entry(a, &ses->ipv6_dp->prefix_list, entry) { + if (a->prefix_len != ia_prefix->prefix_len) + continue; + if (memcmp(&a->addr, &ia_prefix->prefix, sizeof(a->addr))) + continue; + f1 = 1; + break; + } + + if (!f1) { + opt3 = dhcpv6_nested_option_alloc(reply, opt1, D6_OPTION_IAPREFIX, sizeof(*ia_prefix) - sizeof(struct dhcpv6_opt_hdr)); + memcpy(opt3->hdr->data, opt2->hdr->data, sizeof(*ia_prefix) - sizeof(struct dhcpv6_opt_hdr)); + ia_prefix = (struct dhcpv6_opt_ia_prefix *)opt3->hdr; + ia_prefix->pref_lifetime = 0; + ia_prefix->valid_lifetime = 0; + + insert_status(reply, opt3, D6_STATUS_NotOnLink); + } } } } |