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-04-01 11:58:54 +09:00
|
|
|
import array
|
2013-03-29 12:50:58 +09:00
|
|
|
import socket
|
2013-03-29 12:50:59 +09:00
|
|
|
import struct
|
2013-08-01 13:54:57 +09:00
|
|
|
from ryu.lib import addrconv
|
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'
|
|
|
|
|
2013-04-01 11:58:54 +09:00
|
|
|
data = str(data) # input can be bytearray.
|
|
|
|
s = sum(array.array('H', data))
|
|
|
|
s = (s & 0xffff) + (s >> 16)
|
|
|
|
s += (s >> 16)
|
2013-03-29 12:50:58 +09:00
|
|
|
return socket.ntohs(~s & 0xffff)
|
2013-03-29 12:50:59 +09:00
|
|
|
|
|
|
|
|
|
|
|
# avoid circular import
|
2013-07-23 12:54:10 +09:00
|
|
|
_IPV4_PSEUDO_HEADER_PACK_STR = '!4s4sxBH'
|
2013-03-29 12:50:59 +09:00
|
|
|
_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,
|
2013-08-01 13:54:57 +09:00
|
|
|
addrconv.ipv4.text_to_bin(ipvx.src),
|
|
|
|
addrconv.ipv4.text_to_bin(ipvx.dst),
|
|
|
|
ipvx.proto, length)
|
2013-03-29 12:50:59 +09:00
|
|
|
elif ipvx.version == 6:
|
|
|
|
header = struct.pack(_IPV6_PSEUDO_HEADER_PACK_STR,
|
2013-08-01 13:54:57 +09:00
|
|
|
addrconv.ipv6.text_to_bin(ipvx.src),
|
|
|
|
addrconv.ipv6.text_to_bin(ipvx.dst),
|
|
|
|
length, ipvx.nxt)
|
2013-03-29 12:50:59 +09:00
|
|
|
else:
|
|
|
|
raise ValueError('Unknown IP version %d' % ipvx.version)
|
|
|
|
|
|
|
|
buf = header + payload
|
|
|
|
return checksum(buf)
|