IP首部校驗(yàn)和的計(jì)算方法:
湘橋網(wǎng)站建設(shè)公司創(chuàng)新互聯(lián)建站,湘橋網(wǎng)站設(shè)計(jì)制作,有大型網(wǎng)站制作公司豐富經(jīng)驗(yàn)。已為湘橋近1000家提供企業(yè)網(wǎng)站建設(shè)服務(wù)。企業(yè)網(wǎng)站搭建\外貿(mào)營(yíng)銷網(wǎng)站建設(shè)要多少錢(qián),請(qǐng)找那個(gè)售后服務(wù)好的湘橋做網(wǎng)站的公司定做!
1.把校驗(yàn)和字段清零。
2.然后對(duì)每16位(2字節(jié))進(jìn)行二進(jìn)制反碼求和,反碼求和的意思是先對(duì)每16位求和,再將得到的和轉(zhuǎn)為反碼。
接下來(lái)詳細(xì)描述反碼求和的步驟:看下面的代碼
算法:
SHORT checksum(USHORT* buffer, int size)
{
unsigned long cksum = 0;
while(size>1)
{
cksum += *buffer++;
size -= sizeof(USHORT);
}
if(size)
{
cksum += *(UCHAR*)buffer;
}
cksum = (cksum>>16) + (cksum&0xffff);
cksum += (cksum>>16);
return (USHORT)(~cksum);
}
參數(shù)buffer是指向16位整數(shù)的指針,剛開(kāi)始指向的是IP首部的起始地址,參數(shù)size是IP首部的大小。while循環(huán)是將IP首部的內(nèi)容以16位為單元加在一起,如果沒(méi)有整除(即size還有余下的不足16位的部分),則加上余下的部分,此時(shí)的cksum就是相加后的結(jié)果,這個(gè)結(jié)果往往超出了16位,因?yàn)樾r?yàn)和是16位的,所以要將高16位和計(jì)算得到的cksum再加工。
再加工第一步:cksum = (cksum>>16) + (cksum&0xffff); sum>>16是將高16位移位到低16位,sum&0xffff是取出低16位,相加得到新的cksum。
再加工第二步:cksum += (cksum>>16); 第一步相加時(shí)很可能會(huì)產(chǎn)生進(jìn)位,因此要再次把進(jìn)位移到低16位進(jìn)行相加。
這樣就加工好了,接下來(lái)就是取反,并強(qiáng)制轉(zhuǎn)換為16位,這樣就得到了最終的校驗(yàn)和。
校驗(yàn)和計(jì)算出來(lái)了,接下來(lái)就是該如何校驗(yàn):
接收方進(jìn)行校驗(yàn)時(shí),也是對(duì)每16位進(jìn)行二進(jìn)制反碼求和。接收方計(jì)算校驗(yàn)和時(shí)的首部與發(fā)送方計(jì)算校驗(yàn)和時(shí)的首部相比,多了一個(gè)發(fā)送方計(jì)算出來(lái)的校驗(yàn)和。因此,如果首部在傳輸過(guò)程中沒(méi)有發(fā)生差錯(cuò),那么接收方計(jì)算的結(jié)果應(yīng)該為全一,因?yàn)榻邮辗接?jì)算除校驗(yàn)和以外的部分得到值是校驗(yàn)和的反碼,再加多出來(lái)的校驗(yàn)和當(dāng)然是全一了。
附加:
IP頭:
4500 0046
17d9 0000
4011 ec1d(校驗(yàn)字段)
ac1c 0f3b
ac1c 0f3d
計(jì)算:4500 + 0046 +17d9 + 0000 + 4011+ ec1d +ac1c + 0f3b + ac1c + 0f3d
取出的和相加再取反->即為應(yīng)填充的校驗(yàn)和
當(dāng)接受到IP數(shù)據(jù)包時(shí),要檢查IP頭是否正確,則對(duì)IP頭進(jìn)行檢驗(yàn),方法同上:
計(jì)算:
4500 + 0046 +17d9 + 0000 + 4011+ ec1d +ac1c + 0f3b + ac1c + 0f3d 它們的和相加得出的一個(gè)數(shù)再次相加為FFFF,得到的結(jié)果是全一,正確。
得到的結(jié)果是全一,正確。