2012-08-27 11:37:28 +09:00
|
|
|
# Copyright (C) 2012 Nippon Telegraph and Telephone Corporation.
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
|
|
# implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
2013-03-29 12:50:58 +09:00
|
|
|
import socket
|
2013-03-29 12:50:59 +09:00
|
|
|
import struct
|
2013-03-29 12:50:58 +09:00
|
|
|
|
2012-08-27 11:37:28 +09:00
|
|
|
|
|
|
|
def carry_around_add(a, b):
|
|
|
|
c = a + b
|
|
|
|
return (c & 0xffff) + (c >> 16)
|
|
|
|
|
|
|
|
|
|
|
|
def checksum(data):
|
2013-03-29 12:50:57 +09:00
|
|
|
if len(data) % 2:
|
|
|
|
data += '\x00'
|
|
|
|
|
2012-08-27 11:37:28 +09:00
|
|
|
s = 0
|
|
|
|
for i in range(0, len(data), 2):
|
|
|
|
w = data[i] + (data[i + 1] << 8)
|
|
|
|
s = carry_around_add(s, w)
|
2013-03-29 12:50:58 +09:00
|
|
|
return socket.ntohs(~s & 0xffff)
|
2013-03-29 12:50:59 +09:00
|
|
|
|
|
|
|
|
|
|
|
# avoid circular import
|
|
|
|
_IPV4_PSEUDO_HEADER_PACK_STR = '!IIxBH'
|
|
|
|
_IPV6_PSEUDO_HEADER_PACK_STR = '!16s16sI3xB'
|
|
|
|
|
|
|
|
|
|
|
|
def checksum_ip(ipvx, length, payload):
|
|
|
|
"""
|
|
|
|
calculate checksum of IP pseudo header
|
|
|
|
|
|
|
|
IPv4 pseudo header
|
|
|
|
UDP RFC768
|
|
|
|
TCP RFC793 3.1
|
|
|
|
|
|
|
|
0 7 8 15 16 23 24 31
|
|
|
|
+--------+--------+--------+--------+
|
|
|
|
| source address |
|
|
|
|
+--------+--------+--------+--------+
|
|
|
|
| destination address |
|
|
|
|
+--------+--------+--------+--------+
|
|
|
|
| zero |protocol| length |
|
|
|
|
+--------+--------+--------+--------+
|
|
|
|
|
|
|
|
|
|
|
|
IPv6 pseudo header
|
|
|
|
RFC2460 8.1
|
|
|
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
| |
|
|
|
|
+ +
|
|
|
|
| |
|
|
|
|
+ Source Address +
|
|
|
|
| |
|
|
|
|
+ +
|
|
|
|
| |
|
|
|
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
| |
|
|
|
|
+ +
|
|
|
|
| |
|
|
|
|
+ Destination Address +
|
|
|
|
| |
|
|
|
|
+ +
|
|
|
|
| |
|
|
|
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
| Upper-Layer Packet Length |
|
|
|
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
| zero | Next Header |
|
|
|
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
"""
|
|
|
|
if ipvx.version == 4:
|
|
|
|
header = struct.pack(_IPV4_PSEUDO_HEADER_PACK_STR,
|
|
|
|
ipvx.src, ipvx.dst, ipvx.proto, length)
|
|
|
|
elif ipvx.version == 6:
|
|
|
|
header = struct.pack(_IPV6_PSEUDO_HEADER_PACK_STR,
|
|
|
|
ipvx.src, ipvx.dst, length, ipvx.nxt)
|
|
|
|
else:
|
|
|
|
raise ValueError('Unknown IP version %d' % ipvx.version)
|
|
|
|
|
|
|
|
buf = header + payload
|
|
|
|
return checksum(buf)
|