blob: 439e56ff183f11181c4dde8d1bfc0ca6660c9763 [file] [log] [blame]
:: # Copyright 2013, Big Switch Networks, Inc.
:: #
:: # LoxiGen is licensed under the Eclipse Public License, version 1.0 (EPL), with
:: # the following special exception:
:: #
:: # LOXI Exception
:: #
:: # As a special exception to the terms of the EPL, you may distribute libraries
:: # generated by LoxiGen (LoxiGen Libraries) under the terms of your choice, provided
:: # that copyright and licensing notices generated by LoxiGen are not altered or removed
:: # from the LoxiGen Libraries and the notice provided below is (i) included in
:: # the LoxiGen Libraries, if distributed in source code form and (ii) included in any
:: # documentation for the LoxiGen Libraries, if distributed in binary form.
:: #
:: # Notice: "Copyright 2013, Big Switch Networks, Inc. This library was generated by the LoxiGen Compiler."
:: #
:: # You may not use this file except in compliance with the EPL or LOXI Exception. You may obtain
:: # a copy of the EPL at:
:: #
:: # http://www.eclipse.org/legal/epl-v10.html
:: #
:: # 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
:: # EPL for the specific language governing permissions and limitations
:: # under the EPL.
::
:: import itertools
:: include('_copyright.py')
:: include('_autogen.py')
import struct
import const
import util
import loxi
def unpack_list(buf):
if len(buf) % 8 != 0: raise loxi.ProtocolError("action list length not a multiple of 8")
actions = []
offset = 0
while offset < len(buf):
type, length = struct.unpack_from("!HH", buf, offset)
if length == 0: raise loxi.ProtocolError("action length is 0")
if length % 8 != 0: raise loxi.ProtocolError("action length not a multiple of 8")
if offset + length > len(buf): raise loxi.ProtocolError("action length overruns list length")
parser = parsers.get(type)
if not parser: raise loxi.ProtocolError("unknown action type %d" % type)
actions.append(parser(buffer(buf, offset, length)))
offset += length
return actions
class Action(object):
type = None # override in subclass
pass
:: for ofclass in ofclasses:
:: nonskip_members = [m for m in ofclass.members if not m.skip]
class ${ofclass.pyname}(Action):
:: for m in ofclass.type_members:
${m.name} = ${m.value}
:: #endfor
def __init__(self, ${', '.join(["%s=None" % m.name for m in nonskip_members])}):
:: for m in nonskip_members:
if ${m.name} != None:
self.${m.name} = ${m.name}
else:
self.${m.name} = ${m.oftype.gen_init_expr()}
:: #endfor
def pack(self):
packed = []
:: include("_pack.py", ofclass=ofclass)
return ''.join(packed)
@staticmethod
def unpack(buf):
obj = ${ofclass.pyname}()
:: include("_unpack.py", ofclass=ofclass)
return obj
def __eq__(self, other):
if type(self) != type(other): return False
if self.type != other.type: return False
:: for m in nonskip_members:
if self.${m.name} != other.${m.name}: return False
:: #endfor
return True
def __ne__(self, other):
return not self.__eq__(other)
def show(self):
import loxi.pp
return loxi.pp.pp(self)
def pretty_print(self, q):
:: include('_pretty_print.py', ofclass=ofclass)
:: #endfor
def parse_vendor(buf):
if len(buf) < 16:
raise loxi.ProtocolError("experimenter action too short")
experimenter, = struct.unpack_from("!L", buf, 4)
if experimenter == 0x005c16c7: # Big Switch Networks
subtype, = struct.unpack_from("!L", buf, 8)
elif experimenter == 0x00002320: # Nicira
subtype, = struct.unpack_from("!H", buf, 8)
else:
raise loxi.ProtocolError("unexpected experimenter id %#x" % experimenter)
if subtype in experimenter_parsers[experimenter]:
return experimenter_parsers[experimenter][subtype](buf)
else:
raise loxi.ProtocolError("unexpected BSN experimenter subtype %#x" % subtype)
parsers = {
:: sort_key = lambda x: x.type_members[0].value
:: msgtype_groups = itertools.groupby(sorted(ofclasses, key=sort_key), sort_key)
:: for (k, v) in msgtype_groups:
:: v = list(v)
:: if len(v) == 1:
${k} : ${v[0].pyname}.unpack,
:: else:
${k} : parse_${k[12:].lower()},
:: #endif
:: #endfor
}
:: experimenter_ofclasses = [x for x in ofclasses if x.type_members[0].value == 'const.OFPAT_VENDOR']
:: sort_key = lambda x: x.type_members[1].value
:: experimenter_ofclasses.sort(key=sort_key)
:: grouped = itertools.groupby(experimenter_ofclasses, sort_key)
experimenter_parsers = {
:: for (experimenter, v) in grouped:
${experimenter} : {
:: for ofclass in v:
${ofclass.type_members[2].value}: ${ofclass.pyname}.unpack,
:: #endfor
},
:: #endfor
}