blob: 1c7a75a03b5120b10833ac2174e68f1941a83560 [file] [log] [blame]
Rich Lanea06d0c32013-03-25 08:52:03 -07001:: # Copyright 2013, Big Switch Networks, Inc.
2:: #
3:: # LoxiGen is licensed under the Eclipse Public License, version 1.0 (EPL), with
4:: # the following special exception:
5:: #
6:: # LOXI Exception
7:: #
8:: # As a special exception to the terms of the EPL, you may distribute libraries
9:: # generated by LoxiGen (LoxiGen Libraries) under the terms of your choice, provided
10:: # that copyright and licensing notices generated by LoxiGen are not altered or removed
11:: # from the LoxiGen Libraries and the notice provided below is (i) included in
12:: # the LoxiGen Libraries, if distributed in source code form and (ii) included in any
13:: # documentation for the LoxiGen Libraries, if distributed in binary form.
14:: #
15:: # Notice: "Copyright 2013, Big Switch Networks, Inc. This library was generated by the LoxiGen Compiler."
16:: #
17:: # You may not use this file except in compliance with the EPL or LOXI Exception. You may obtain
18:: # a copy of the EPL at:
19:: #
20:: # http://www.eclipse.org/legal/epl-v10.html
21:: #
22:: # Unless required by applicable law or agreed to in writing, software
23:: # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
24:: # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
25:: # EPL for the specific language governing permissions and limitations
26:: # under the EPL.
27::
28:: import itertools
Rich Lane3f075972013-03-15 22:56:29 -070029:: import of_g
Rich Lanea06d0c32013-03-25 08:52:03 -070030:: include('_copyright.py')
31
32:: include('_autogen.py')
33
34import struct
35import const
36import util
Rich Lane15cbe842013-04-26 16:04:11 -070037import loxi.generic_util
Rich Lanea06d0c32013-03-25 08:52:03 -070038import loxi
39
40def unpack_list(buf):
41 if len(buf) % 8 != 0: raise loxi.ProtocolError("action list length not a multiple of 8")
Rich Laned4e08692013-02-20 17:40:33 -080042 def deserializer(buf):
43 type, length = struct.unpack_from("!HH", buf)
Rich Lanea06d0c32013-03-25 08:52:03 -070044 if length % 8 != 0: raise loxi.ProtocolError("action length not a multiple of 8")
Rich Lanea06d0c32013-03-25 08:52:03 -070045 parser = parsers.get(type)
46 if not parser: raise loxi.ProtocolError("unknown action type %d" % type)
Rich Laned4e08692013-02-20 17:40:33 -080047 return parser(buf)
Rich Lane15cbe842013-04-26 16:04:11 -070048 return loxi.generic_util.unpack_list(deserializer, "!2xH", buf)
Rich Lanea06d0c32013-03-25 08:52:03 -070049
50class Action(object):
51 type = None # override in subclass
52 pass
53
54:: for ofclass in ofclasses:
Rich Laned1dd9e62013-02-20 18:28:01 -080055:: include('_ofclass.py', ofclass=ofclass, superclass="Action")
Rich Lanea06d0c32013-03-25 08:52:03 -070056
57:: #endfor
58
Rich Lane3f075972013-03-15 22:56:29 -070059:: if version == of_g.VERSION_1_0:
Rich Lanea06d0c32013-03-25 08:52:03 -070060def parse_vendor(buf):
Rich Lane3f075972013-03-15 22:56:29 -070061:: else:
62def parse_experimenter(buf):
63:: #endif
Rich Lanea06d0c32013-03-25 08:52:03 -070064 if len(buf) < 16:
65 raise loxi.ProtocolError("experimenter action too short")
66
67 experimenter, = struct.unpack_from("!L", buf, 4)
68 if experimenter == 0x005c16c7: # Big Switch Networks
69 subtype, = struct.unpack_from("!L", buf, 8)
70 elif experimenter == 0x00002320: # Nicira
71 subtype, = struct.unpack_from("!H", buf, 8)
72 else:
73 raise loxi.ProtocolError("unexpected experimenter id %#x" % experimenter)
74
75 if subtype in experimenter_parsers[experimenter]:
76 return experimenter_parsers[experimenter][subtype](buf)
77 else:
78 raise loxi.ProtocolError("unexpected BSN experimenter subtype %#x" % subtype)
79
80parsers = {
81:: sort_key = lambda x: x.type_members[0].value
82:: msgtype_groups = itertools.groupby(sorted(ofclasses, key=sort_key), sort_key)
83:: for (k, v) in msgtype_groups:
84:: v = list(v)
85:: if len(v) == 1:
86 ${k} : ${v[0].pyname}.unpack,
87:: else:
88 ${k} : parse_${k[12:].lower()},
89:: #endif
90:: #endfor
91}
92
93:: experimenter_ofclasses = [x for x in ofclasses if x.type_members[0].value == 'const.OFPAT_VENDOR']
94:: sort_key = lambda x: x.type_members[1].value
95:: experimenter_ofclasses.sort(key=sort_key)
96:: grouped = itertools.groupby(experimenter_ofclasses, sort_key)
97experimenter_parsers = {
98:: for (experimenter, v) in grouped:
99 ${experimenter} : {
100:: for ofclass in v:
101 ${ofclass.type_members[2].value}: ${ofclass.pyname}.unpack,
102:: #endfor
103 },
104:: #endfor
105}