本文轉(zhuǎn)自公眾號(hào),歡迎關(guān)注
Windows下基于MSVC搭建Wintun開(kāi)發(fā)環(huán)境 (qq.com)
前言
l有這么一種場(chǎng)景,某個(gè)windows下的應(yīng)用程序底層數(shù)據(jù)接口基于以太網(wǎng),現(xiàn)在想修改為串口,那么一般來(lái)說(shuō)需要修改該應(yīng)用程序,添加對(duì)串口的支持,但是很多時(shí)候應(yīng)用程序可能是第三方開(kāi)發(fā)的并不能修改,有沒(méi)有在不修改應(yīng)用程序的情況下實(shí)現(xiàn)兼容呢,
Wintun就提供了解決方案,Wintun可以創(chuàng)建虛擬網(wǎng)卡,提供IP層的數(shù)據(jù)鏈路,那么我們只需要使用Wintun創(chuàng)建虛擬網(wǎng)卡,將串口數(shù)據(jù)組包成指定協(xié)議層的包(UDP或者TCP/IP)等進(jìn)行轉(zhuǎn)發(fā),那么該應(yīng)用軟件可以訪問(wèn)該虛擬網(wǎng)卡,無(wú)需任何修改。
由于wintun工作在IP層,所以需要根據(jù)實(shí)際情況去實(shí)現(xiàn)UDP或者TCP/IP等包的組包和解包。
l還有一種場(chǎng)景,我們希望在windows下進(jìn)行嵌入式tcp/ip協(xié)議棧的源碼級(jí)別開(kāi)發(fā)測(cè)試,那么就可以使用Wintun來(lái)模擬IP層的數(shù)據(jù)鏈路,而不直接訪問(wèn)真實(shí)的MAC和PHY,基于此進(jìn)行協(xié)議棧的開(kāi)發(fā)調(diào)試,由于Wintun是IP層的,所以無(wú)法模擬數(shù)據(jù)鏈路層的部分,真實(shí)的場(chǎng)景是以太網(wǎng)控制器實(shí)現(xiàn)幀的收發(fā),而Wintun的是更上一層基于IP包的收發(fā),也就是從IP層截?cái)嗔耍圆荒苓M(jìn)行ARP等數(shù)據(jù)鏈路層的協(xié)議的分析調(diào)試。
準(zhǔn)備
官網(wǎng)
https://www.wintun.net/
下載代碼
git clone https://git.zx2c4.com/wintun
下載編譯好的庫(kù)
https://www.wintun.net/builds/wintun-0.14.1.zip
編譯庫(kù)
參考 https: / /git.zx2c4.com/wintun /about/#building
在自己的工程中使用
解壓wintun-0.14.1.zip復(fù)制文件夾wintun到自己的工程目錄
將下載的源碼wintun\\example下的example.c復(fù)制到自己的工程中。
右鍵點(diǎn)擊項(xiàng)目名->屬性 設(shè)置相關(guān)屬性,Window2和X86都同樣設(shè)置。
設(shè)置頭文件包含路徑
$(MSBuildProjectDirectory)\\Src\\wintun\\include;
設(shè)置鏈接的庫(kù)
iphlpapi.lib;kernel32.lib;ntdll.lib;ws2_32.lib;
設(shè)置輸出路徑
$(ProjectDir)$(Platform)\\$(Configuration)\\
切換X86和X64版本,都進(jìn)行編譯
點(diǎn)擊項(xiàng)目名->重新生成構(gòu)建。
將wintun\\bin\\amd64和wintun\\bin\\x86下的wintun.dll分別放入
工程的輸出目錄\\x64\\Debug和\\Win32\\Debug下
右鍵點(diǎn)擊exe文件,以管理員身份運(yùn)行(一定要管理員權(quán)限,否則出錯(cuò))
控制面板,網(wǎng)絡(luò)和共享中心,更改適配器設(shè)置,可以看到多了網(wǎng)卡Demo
命令行ipconfig可以看到
也可以ping通
基本API的使用
上面使用example進(jìn)行了測(cè)試,為了熟悉API,可以自己寫一個(gè)測(cè)試代碼。
參考https://git.zx2c4.com/wintun/about/
加載庫(kù),解析函數(shù)地址
LoadLibraryExW
GetProcAddress
先定義14個(gè)API函數(shù)的指針,
static WINTUN_CREATE_ADAPTER_FUNC *WintunCreateAdapter;
static WINTUN_CLOSE_ADAPTER_FUNC *WintunCloseAdapter;
static WINTUN_OPEN_ADAPTER_FUNC *WintunOpenAdapter;
static WINTUN_GET_ADAPTER_LUID_FUNC *WintunGetAdapterLUID;
static WINTUN_GET_RUNNING_DRIVER_VERSION_FUNC *WintunGetRunningDriverVersion;
static WINTUN_DELETE_DRIVER_FUNC *WintunDeleteDriver;
static WINTUN_SET_LOGGER_FUNC *WintunSetLogger;
static WINTUN_START_SESSION_FUNC *WintunStartSession;
static WINTUN_END_SESSION_FUNC *WintunEndSession;
static WINTUN_GET_READ_WAIT_EVENT_FUNC *WintunGetReadWaitEvent;
static WINTUN_RECEIVE_PACKET_FUNC *WintunReceivePacket;
static WINTUN_RELEASE_RECEIVE_PACKET_FUNC *WintunReleaseReceivePacket;
static WINTUN_ALLOCATE_SEND_PACKET_FUNC *WintunAllocateSendPacket;
static WINTUN_SEND_PACKET_FUNC *WintunSendPacket;
加載庫(kù)
HMODULE Wintun = LoadLibraryExW(L"wintun.dll", NULL, LOAD_LIBRARY_SEARCH_APPLICATION_DIR | LOAD_LIBRARY_SEARCH_SYSTEM32);
if (!Wintun)
{
return -1;
}
解析函數(shù)地址,其他13個(gè)API類似
if ((*(FARPROC*)&WintunCreateAdapter = GetProcAddress(Wintun, "WintunCreateAdapter")) == NULL)
{
FreeLibrary(Wintun);
return -2;
}
以上為了方便理解分開(kāi)了寫,example.c中用宏的形式更簡(jiǎn)潔。
創(chuàng)建適配器
GUID ExampleGuid = { 0xdeadbabe, 0xcafe, 0xbeef, { 0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef } };
WINTUN_ADAPTER_HANDLE Adapter = WintunCreateAdapter(L"Demo", L"Example", &ExampleGuid);
if (!Adapter)
{
FreeLibrary(Wintun);
return -3;
}
獲取版本
DWORD Version = WintunGetRunningDriverVersion();
printf("Wintun v%u.%u loaded", (Version > > 16) & 0xff, (Version > > 0) & 0xff);
啟動(dòng)會(huì)話
MIB_UNICASTIPADDRESS_ROW AddressRow;
InitializeUnicastIpAddressEntry(&AddressRow); /* 單播IP地址 */
WintunGetAdapterLUID(Adapter, &AddressRow.InterfaceLuid); /* 獲取LUID */
AddressRow.Address.Ipv4.sin_family = AF_INET;
AddressRow.Address.Ipv4.sin_addr.S_un.S_addr = htonl(ip); /* IP地址 16進(jìn)制表示的xxx.xxx.xxx.xxx 高8位表示第一個(gè)xxx的16進(jìn)制值 */
AddressRow.OnLinkPrefixLength = 24; /* /24 網(wǎng)絡(luò),即子網(wǎng)掩碼為255.255.255.000 */
AddressRow.DadState = IpDadStatePreferred;
DWORD LastError = CreateUnicastIpAddressEntry(&AddressRow);
if (LastError != ERROR_SUCCESS && LastError != ERROR_OBJECT_ALREADY_EXISTS)
{
printf("Failed to set IP address %d", LastError);
WintunCloseAdapter(Adapter);
FreeLibrary(Wintun);
}
WINTUN_SESSION_HANDLE Session = WintunStartSession(Adapter, 0x400000);
if (!Session)
{
printf("Failed to create adapter");
WintunCloseAdapter(Adapter);
FreeLibrary(Wintun);
}
發(fā)送包
Wintun工作在第三層IP層,發(fā)送的是IP包。
BYTE* Packet = WintunAllocateSendPacket(Session, len); /* 先分配空間 */
if (Packet)
{
memcpy(Packet,buffer,len);
WintunSendPacket(Session, Packet); /* Packet中包含了長(zhǎng)度信息,所以不需要再添加長(zhǎng)度 */
}
接收包
DWORD PacketSize;
BYTE* Packet = WintunReceivePacket(Session, &PacketSize);
if (Packet)
{
/* 使用接收到的數(shù)據(jù) */
/* 釋放 */
WintunReleaseReceivePacket(Session, Packet);
}
總結(jié)
Wintun提供了簡(jiǎn)潔的接口,在用戶空間即可創(chuàng)建虛擬網(wǎng)卡,進(jìn)行IP層的數(shù)據(jù)傳輸,基于此可以應(yīng)用于很多應(yīng)用場(chǎng)景,比如tcp/ip協(xié)議棧的移植,以太網(wǎng)和其他接口的轉(zhuǎn)換等。
審核編輯:湯梓紅
-
嵌入式
+關(guān)注
關(guān)注
5087文章
19150瀏覽量
306357 -
WINDOWS
+關(guān)注
關(guān)注
4文章
3553瀏覽量
88980 -
開(kāi)發(fā)環(huán)境
+關(guān)注
關(guān)注
1文章
226瀏覽量
16655
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論