2012/10/3 Kei Ohmura <ohmura....@gmail.com>:
> 2012/10/3 FUJITA Tomonori <fujita.tomon...@lab.ntt.co.jp>:
>
>>
>> Do we need to duplicate the exact same code here?
>>
>> How about using echo class for both types?
>
>
> How about the following code?

I modified the previous patch.


>From 9bfc35532c74aa1d75530bb0b3856962f3dd0ba6 Mon Sep 17 00:00:00 2001
From: OHMURA Kei <ohmura....@lab.ntt.co.jp>
Date: Wed, 3 Oct 2012 16:32:40 +0900
Subject: [PATCH] packet lib: add icmp.py

Signed-off-by: OHMURA Kei <ohmura....@lab.ntt.co.jp>
---
 ryu/lib/packet/icmp.py |  112 ++++++++++++++++++++++++++++++++++++++++++++++++
 ryu/lib/packet/ipv4.py |    2 +
 2 files changed, 114 insertions(+)
 create mode 100644 ryu/lib/packet/icmp.py

diff --git a/ryu/lib/packet/icmp.py b/ryu/lib/packet/icmp.py
new file mode 100644
index 0000000..7cde057
--- /dev/null
+++ b/ryu/lib/packet/icmp.py
@@ -0,0 +1,112 @@
+# 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.
+
+import struct
+import socket
+from . import packet_base
+from . import packet_utils
+
+
+ICMP_ECHO_REPLY = 0
+ICMP_DEST_UNREACH = 3
+ICMP_SRC_QUENCH = 4
+ICMP_REDIRECT = 5
+ICMP_ECHO_REQUEST = 8
+
+
+class icmp(packet_base.PacketBase):
+    _PACK_STR = '!BBH'
+    _MIN_LEN = struct.calcsize(_PACK_STR)
+    _ICMP_TYPES = {}
+
+    @staticmethod
+    def register_icmp_type(*args):
+        def _register_icmp_type(cls):
+            for type_ in args:
+                icmp._ICMP_TYPES[type_] = cls
+            return cls
+        return _register_icmp_type
+
+    def __init__(self, type_, code, csum, data=None):
+        super(icmp, self).__init__()
+        print 'icmp._ICMP_TYPES:', icmp._ICMP_TYPES
+        self.type = type_
+        self.code = code
+        self.csum = csum
+        self.data = data
+
+    @classmethod
+    def parser(cls, buf):
+        (type_, code, csum) = struct.unpack_from(cls._PACK_STR, buf)
+        msg = cls(type_, code, csum)
+        offset = cls._MIN_LEN
+
+        if len(buf) > offset:
+            cls_ = cls._ICMP_TYPES.get(type_, None)
+            if cls_:
+                msg.data = cls_.parser(buf, offset)
+            else:
+                msg.data = buf[offset:]
+
+        return msg, None
+
+    def serialize(self, payload, prev):
+        hdr = bytearray(struct.pack(icmp._PACK_STR, self.type,
+                                    self.code, self.csum))
+
+        if self.data is not None:
+            if self.type in icmp._ICMP_TYPES:
+                hdr += self.data.serialize()
+            else:
+                hdr += self.data
+
+        if self.csum == 0:
+            if len(hdr) % 2:
+                hdr += '\0'
+            self.csum = socket.htons(packet_utils.checksum(hdr))
+            struct.pack_into('!H', hdr, 2, self.csum)
+
+        return hdr
+
+
+@icmp.register_icmp_type(ICMP_ECHO_REPLY, ICMP_ECHO_REQUEST)
+class Echo(object):
+    _PACK_STR = '!HH'
+    _MIN_LEN = struct.calcsize(_PACK_STR)
+
+    def __init__(self, id_, seq, data=None):
+        self.id = id_
+        self.seq = seq
+        self.data = data
+
+    @classmethod
+    def parser(cls, buf, offset):
+        (id_, seq) = struct.unpack_from(cls._PACK_STR, buf, offset)
+        msg = cls(id_, seq)
+        offset += cls._MIN_LEN
+
+        if len(buf) > offset:
+            msg.data = buf[offset:]
+
+        return  msg
+
+    def serialize(self):
+        hdr = bytearray(struct.pack(Echo._PACK_STR, self.id,
+                                    self.seq))
+
+        if self.data is not None:
+            hdr += self.data
+
+        return hdr
diff --git a/ryu/lib/packet/ipv4.py b/ryu/lib/packet/ipv4.py
index 21d227a..d3cc4b3 100644
--- a/ryu/lib/packet/ipv4.py
+++ b/ryu/lib/packet/ipv4.py
@@ -17,6 +17,7 @@ import struct
 import socket
 from . import packet_base
 from . import packet_utils
+from . import icmp
 from . import udp
 from . import tcp
 from ryu.ofproto import inet
@@ -79,5 +80,6 @@ class ipv4(packet_base.PacketBase):
         struct.pack_into('!H', hdr, 10, self.csum)
         return hdr

+ipv4.register_packet_type(icmp.icmp, inet.IPPROTO_ICMP)
 ipv4.register_packet_type(tcp.tcp, inet.IPPROTO_TCP)
 ipv4.register_packet_type(udp.udp, inet.IPPROTO_UDP)
-- 
1.7.9.5

------------------------------------------------------------------------------
Don't let slow site performance ruin your business. Deploy New Relic APM
Deploy New Relic app performance management and know exactly
what is happening inside your Ruby, Python, PHP, Java, and .NET app
Try New Relic at no cost today and get our sweet Data Nerd shirt too!
http://p.sf.net/sfu/newrelic-dev2dev
_______________________________________________
Ryu-devel mailing list
Ryu-devel@lists.sourceforge.net
https://lists.sourceforge.net/lists/listinfo/ryu-devel

Reply via email to