為做偽IP,特地做了一個(gè)IP包,用C#改寫(xiě)IP頭的校驗(yàn)和算法。
資料一:
IP頭格式:
版本號(hào) (4位)
IP頭長(zhǎng)度 (4位)
服務(wù)類(lèi)型 (8位)
數(shù)據(jù)包長(zhǎng)度 (16位)
標(biāo)識(shí)段 (16位)
標(biāo)志段 (16位)
生存時(shí)間 (8位)
傳輸協(xié)議 (8位)
頭校驗(yàn)和 (16位)
發(fā)送地址 (16位)
目標(biāo)地址 (16位)
選項(xiàng)
填充
資料二:
IP 協(xié)議采用統(tǒng)一的校驗(yàn)算法,其計(jì)算比較簡(jiǎn)單:設(shè)校驗(yàn)和初值為0,然后對(duì)數(shù)據(jù)每16位求異或,結(jié)果取反,便得校驗(yàn)和。校驗(yàn)時(shí)將數(shù)據(jù)(含校驗(yàn)和)按同樣的算法求和,結(jié)果為0則數(shù)據(jù)正確,不為0表示通訊出錯(cuò),需要丟棄該數(shù)據(jù)包。
算法源代碼:
public static UInt16 checksum(UInt16[] buffer,int size)
{
Int32 cksum=0;
int counter;
counter=0;
while(size>0)
{
UInt16 val=buffer[counter];
cksum+=Convert.ToInt32(buffer[counter]);
counter+=1;
size=-1;
}
cksum=(cksum>>16)+(cksum&0xffff);
cksum+=(cksum>>16);
return (UInt16)(~cksum);
}
注意:buffer數(shù)組為整個(gè)ip包數(shù)組,需要轉(zhuǎn)換成UInt16[];size為buffer數(shù)組的長(zhǎng)度。
關(guān)于byte[]轉(zhuǎn)換成UInt16[]的方法比較簡(jiǎn)單,在此不介紹了。
資料一:
IP頭格式:
版本號(hào) (4位)
IP頭長(zhǎng)度 (4位)
服務(wù)類(lèi)型 (8位)
數(shù)據(jù)包長(zhǎng)度 (16位)
標(biāo)識(shí)段 (16位)
標(biāo)志段 (16位)
生存時(shí)間 (8位)
傳輸協(xié)議 (8位)
頭校驗(yàn)和 (16位)
發(fā)送地址 (16位)
目標(biāo)地址 (16位)
選項(xiàng)
填充
資料二:
IP 協(xié)議采用統(tǒng)一的校驗(yàn)算法,其計(jì)算比較簡(jiǎn)單:設(shè)校驗(yàn)和初值為0,然后對(duì)數(shù)據(jù)每16位求異或,結(jié)果取反,便得校驗(yàn)和。校驗(yàn)時(shí)將數(shù)據(jù)(含校驗(yàn)和)按同樣的算法求和,結(jié)果為0則數(shù)據(jù)正確,不為0表示通訊出錯(cuò),需要丟棄該數(shù)據(jù)包。
算法源代碼:
public static UInt16 checksum(UInt16[] buffer,int size)
{
Int32 cksum=0;
int counter;
counter=0;
while(size>0)
{
UInt16 val=buffer[counter];
cksum+=Convert.ToInt32(buffer[counter]);
counter+=1;
size=-1;
}
cksum=(cksum>>16)+(cksum&0xffff);
cksum+=(cksum>>16);
return (UInt16)(~cksum);
}
注意:buffer數(shù)組為整個(gè)ip包數(shù)組,需要轉(zhuǎn)換成UInt16[];size為buffer數(shù)組的長(zhǎng)度。
關(guān)于byte[]轉(zhuǎn)換成UInt16[]的方法比較簡(jiǎn)單,在此不介紹了。