GetIfEntry和netsh之间的MTU不匹配

3

我正在开发一款基于UDP运行的伪传输层软件,提供可靠的面向连接传输,作为TCP的替代选择。为了最大化网络效率,在初始化时我们会查询“最佳”网络适配器的MTU。

MIB_IFROW row = {0};
row.dwIndex = dwBestIfIndex;

dwRes = GetIfEntry(&row);

在网上搜索后,我发现您可以使用以下netsh命令从命令提示符(而不是C++ API)查询相同的值:

netsh interface ipv4 show interfaces
netsh interface ipv4 show subinterfaces

令人困扰的问题在于,尽管row.dwMtu可能设置为1500,但是通过在发送笔记本电脑上窥视网络流量,我们发现我们的数据包被分成了1300字节的数据包。netsh也报告MTU为1300。
显然,netsh命令报告的值是实际使用的值。有人知道我可以调用哪个API来获得与netsh相同的值吗?
1个回答

3

有本地MTU大小和路径MTU(参见RFC 1191)。您可以使用以下命令:

ping -f -l 1464 www.stackoverflow.com
ping -f -l 1465 www.stackoverflow.com

确定路径 MTU。考虑 ICMP 包头大小和 IP 头大小。有关 API 解决方案,请参见 Path MTU Discovery

更新:请不要再提出此类问题!对我来说,找到答案非常有趣,我花了几个小时的时间。但是...我找到了!要接收正确的 MTU 大小值,您应该使用 GetIpInterfaceTable API,它返回 PMIB_IPINTERFACE_TABLE 结构,该结构具有 MIB_IPINTERFACE_ROW 结构的数组。 MIB_IPINTERFACE_ROW 具有 InterfaceIndex,可帮助您识别 IP 接口,与其他众所周知的 IP Helper 函数中的相同。您还可以使用 ConvertInterfaceLuidToNameW 函数从另一个字段 InterfaceLuid 中接收接口名称。

但最有趣的是 NlMtu 字段。

typedef struct _MIB_IPINTERFACE_ROW {
    ADDRESS_FAMILY Family;
    NET_LUID InterfaceLuid;
    NET_IFINDEX InterfaceIndex;
    // ...
    ULONG NlMtu;
    // ...
};

在文档中(请参见http://msdn.microsoft.com/en-us/library/aa814496(v=VS.85).aspx),它被描述为“网络层MTU大小,以字节为单位。”。Windows Driver Kit文档中也可以找到同样的文本,没有更多的内容(请参见http://msdn.microsoft.com/en-us/library/ff559254(VS.85).aspx)。这个字段NlMtu就是你要找的。例如,我的电脑通过DSL路由器连接到互联网,NlMtu不像没有路由器时那样是1500,而是正确的1492。在您的情况下,它必须是1300。
如果您安装了最新版本的Microsoft SDK,则会在C:\Program Files\Microsoft SDKs\Windows\v7.0\Samples\netds\iphelp\Netinfo目录中找到一个使用GetIpInterfaceTable API的示例。只需在netinfo.c文件中第270行设置断点,您将在InterfaceTable->Table[i].NlMtu中看到相应接口适配器的IP MTU的正确值。如果您在注册表中验证HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\services\Tcpip\Parameters\Interfaces{34407201-997C-41FF-9EBF-1B7D6DF92B38}下的MTU REG_DWORD值,该值必须相同作为您的接口卡(在我的情况下为{34407201-997C-41FF-9EBF-1B7D6DF92B38})。
函数GetIpInterfaceTable从Vista开始存在,但是正如您可以从名称PMIB_IPINTERFACE_TABLE中看到的那样,这些值来自TCP适配器的MIB信息,请参见(Windows DDK)。几年前根本没有IP helper DLL,并且要接收显示IpConfig.exe的信息,必须使用CreateFile和DeviceIoControl从TCP TDI驱动程序中提供此类信息(请参见tcpioctl.h中的L"\Device\Tcp"和IOCTL_TCP_QUERY_INFORMATION_EX等常量)。因此,在Windows XP上可能可以提供相同的信息,但这并不重要。

谢谢您的回复,Oleg。我不是在寻找确定MTU的方法,因为我认为已经有至少两个代理已经确定了MTU,并且每个代理都提供了一种查询该值的方法:1\ GetIfEntry() 和 2\ netsh使用的任何内容。我正在寻找的是netsh使用的API。 - ChrisJ
@ChrisJ 我在我的回答中写道,你不应该再问这样的问题。当然,我并不是认真的 :-). 这是一个非常好的问题,但不是一个简单的问题。 - Oleg
感谢您提供的解决方案。我还没有测试过,但我已经转发给了我的同事。我们会尽快更新。 - ChrisJ

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接