请问下级路由获取到的前缀为什么不是 2001:aaaa:bbbb:789c::/62 ?
1
pcslide 2022-07-03 13:25:06 +08:00
2001:aaaa:bbbb:7890::/60 网下分配可以到 2001:aaaa:bbbb:780c::/62
???????????? |
2
87728854 OP 是的,这就是奇怪的地方,理论上 789 是不应该变动的,由于指定了委派范围 ::8 至 ::C ,应该是 7898::/62 或 789C::/62 任意一个。
|
3
87728854 OP @pcslide 是的,这就是奇怪的地方,理论上 789 是不应该变动的,由于指定了委派范围 ::8 至 ::C ,应该是 7898::/62 或 789C::/62 任意一个。
|
4
smallthing 2022-07-03 13:59:17 +08:00
1.你的一级路由有问题 分配的范围错了
2.你运营商的 dhcpv6 有问题 然后你一级路由也有问题 穿透到二级 |
5
raysonx 2022-07-03 14:06:52 +08:00 4
巧了,我以前折腾过 OPNSense 的 DHCPv6-PD 功能,也遇到过类似的问题,读过那部分代码,结论是代码里有 bug ,它直接硬编码了几个 prefix 的长度作的掩码运算,没有 honor 你的 /62.
我当时手动 hack 了那部分代码解决的问题。 时间太久了,我无法直接找到当时的修改了。修那个 bug 时让我觉得代码质量太差,后来直接换 VyOS 了。 其实最新的 OpenWRT 也不错,特别是最新的 22.03 RC ,对 IPv6 的支持我觉得是普通用户能接触到的固件里面最好的了。 |
6
raysonx 2022-07-03 14:13:23 +08:00
我印象中它的 IPv6 前缀运算那部分用的是字符串截取和拼接,而不是转为 long int 形式做数学运算。在从 WAN 口拿到 PD 前缀后截取一部分字符串,再拼上后缀。然后我记得它只处理了 /48 ,/56 等几种情况,对 /60 或者其他长度的前缀处理会出错。
|
7
87728854 OP @raysonx 我还以为我对 IPV6 前缀委派理解有问题,把 《 IPV6 技术精要》涉及的章节翻来覆去看了几遍。。。这个问题已经困扰我两周了。话说你是什么时候遇到的,这都 2022 年了还没修复,真是坑啊。。。
|
8
raysonx 2022-07-03 15:43:53 +08:00 via iPad
@87728854 有两三年了吧,不知道你用的什么版本,如果不是最新版可以尝试一下最新版本看看修了没。我不确定能否找到当时的 patch 。
|
9
87728854 OP @raysonx 因为怀疑有 BUG ,我更新的最新版,我准备测试下 pfsense ,主要是比较喜欢 freebsd 。。。
|
10
raysonx 2022-07-03 16:19:12 +08:00
pfSense 好像好多年不怎么维护了,netgate 把重点转向 TNSR 了,别抱太大指望。
我找到当时 patch 的位置了,下面两行: https://github.com/opnsense/core/blob/98fe7a38813a78ab818d215bde530711d5676818/src/etc/inc/plugins.inc.d/dhcpd.inc#L1371 https://github.com/opnsense/core/blob/98fe7a38813a78ab818d215bde530711d5676818/src/etc/inc/plugins.inc.d/dhcpd.inc#L1373 可以看到对于前缀长度 60-64 的情况下,比如你举例的 2001:aaaa:bbbb:7890::/60 ,它直接截取了 7890 中的 78 ,然后拼接你指定的 08 (一直到 0C),得到 7808 至 780C 。 一个简单的 hack 是把 substr($pd_prefix_from_array_out[3], 0, 2)改为 substr($pd_prefix_from_array_out[3], 0, 3),把 substr($pd_prefix_to_array_out[3], 0, 2)改为 substr($pd_prefix_to_array_out[3], 0, 3)。当然这只针对你的使用情况,换成其他长度的前缀或者两位数的范围会出错。 |
11
raysonx 2022-07-03 16:20:43 +08:00
哦,还有去掉这两行,防止把“范围”扩展为两位数:
$pd_prefix_from_array[2] = sprintf("%02s", $pd_prefix_from_array[2]); $pd_prefix_to_array[2] = sprintf("%02s", $pd_prefix_to_array[2]); |
12
87728854 OP @raysonx 找到了,非常感谢!
1412 switch ($pdval) { 1413 // For PD sizes of /60 through /64, the user must do the math! 1414 case 60: 1415 case 62: 1416 case 63: 1417 case 64: // 3&4th bytes on 4th array 1418 $pd_prefix_from_array_out[3] = sprintf("%04s", $ifcfgipv6arr[3]); // make it 4 bytes 1419 $pd_prefix_from_array_out[3] = substr($pd_prefix_from_array_out[3], 0, 2) . $pd_prefix_from_array[2]; 1420 $pd_prefix_to_array_out[3] = sprintf("%04s", $ifcfgipv6arr[3]); // make it 4 bytes 1421 $pd_prefix_to_array_out[3] = substr($pd_prefix_to_array_out[3], 0, 2) . $pd_prefix_to_array[2]; 1422 break; 1423 case 56: // 1st&2nd bytes on 4th array 1424 $pd_prefix_from_array[2] = str_pad($pd_prefix_from_array[2], 4, "0"); 1425 $pd_prefix_from_array_out[3] = sprintf("%s", $pd_prefix_from_array[2]); // make it 4 bytes 1426 $pd_prefix_to_array[2] = str_pad($pd_prefix_to_array[2], 4, "0"); 1427 $pd_prefix_to_array_out[3] = sprintf("%s", $pd_prefix_to_array[2]); // make it 4 bytes 1428 break; 1429 case 52: // 1st byte on 4th array only, 0 to f, we only want one byte, but lookout for the user entering more 1430 $len = strlen($pd_prefix_from_array[2]); 1431 $pd_prefix_from_array[2] = substr($pd_prefix_from_array[2], $len - 1, 1); 1432 $pd_prefix_from_array_out[3] = sprintf("%s000", substr($pd_prefix_from_array[2], 0, 1)); // first byte from entered value 1433 $len = strlen($pd_prefix_to_array[2]); 1434 $pd_prefix_to_array[2] = substr($pd_prefix_to_array[2], $len - 1, 1); 1435 $pd_prefix_to_array_out[3] = sprintf("%s000", substr($pd_prefix_to_array[2], 0, 1)); 1436 break; 1437 case 48: // 4th byte on 2nd array 1438 $pd_prefix_from_array[2] = substr($pd_prefix_from_array[2], 0, 1); 1439 $pd_prefix_from_array_out[1] = substr(sprintf("%03s", $ifcfgipv6arr[1]), 0, 3) . $pd_prefix_from_array[2]; // get 1st 3 byte + nibble 1440 $pd_prefix_to_array[2] = substr($pd_prefix_to_array[2], 0, 1); 1441 $pd_prefix_to_array_out[1] = substr(sprintf("%03s", $ifcfgipv6arr[1]), 0, 3) . $pd_prefix_to_array[2]; // get 1st 3 byte + nibble 1442 break; 1443 } |
13
87728854 OP @raysonx
我当前用的版本 22.1.9_1 ,修改 /usr/local/etc/inc/plugins.inc.d/dhcpd.inc 文件: 注释掉 1399 和 1400 行。 修改 1419 和 1421 行,经测试向下级路由委派到正确的前缀,再次感谢! |
14
cnbatch 2022-07-03 18:04:27 +08:00
我给 opnsense 仓库提交了 issue ,接下来就看他们会不会处理
https://github.com/opnsense/core/issues/5847 |
17
raysonx 2022-07-03 20:16:17 +08:00
感谢,我加了一条回复,看有没有人修吧。
我已不用 OPNsense 多年,自己是懒得修了(实在不想去写和调试 PHP 代码)。 |
19
402159806 2022-07-04 08:51:45 +08:00
我用光猫拨号 openwrt ipv6 中继 用起来很稳定
|
22
acbot 2022-07-05 11:26:36 +08:00
@raysonx “...最新的 OpenWRT 也不错...” 是的,OP 18.06 官方版本使用几年了 IPv6 都很正常, 不是需要新的软件版本完全都可以不升级
|
24
cnbatch 2022-07-17 16:57:31 +08:00
|
25
87728854 OP 最近升级新版本,发现已经彻底解决了这个问题,在不手动修改代码的情况下,前缀成功下发到下一级路由,下级路由也成功给每台电脑分配了期望的 IPV6 地址,可以成功访问互联网。不过有个新问题:下级电脑无法使用 IPv6 访问上级路由 LAN 口连接的电脑,防火墙全部放开也不行。
比如: 一级路由 WAN 口获取 IPV6 地址:240e:55e:304:f97b:62be:b4ff:fe03:a49a/64 一级路由 WN 口同时从运营商获取到前缀:240e:55f:34f:f970::/60 一级路由 LAN 口分配到 IPV6 地址:240e:55f:34f:f970:62be:b4ff:fe03:a49c/64 一级路由 LAN 口使用 DHCPv6 继续向二级路由下发前缀,长度设定为 62 ,范围为:::8 至 ::c 二级路由 WAN 口成功从一级路由获取到前缀 240e:55f:34f:f97c::/62 二级路由 LAN 接口及连接的电脑分配到前缀为 240e:55f:34f:f97c::/64 的 IPV6 地址,可以正常访问互联网。 问题: 二级路由 LAN 口连接的所有电脑均无法访问一级路由 LAN 口下接入的 240e:55f:34f:f970::/64 这个段的电脑。 |