blob: 01d59f508f2464d891dd5e8682bde8e9c81f4b55 [file] [log] [blame]
#!/usr/bin/env python
# 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 sys
import os
import unittest
root_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), '..')
sys.path.insert(0, root_dir)
import loxi_front_end.parser as parser
import loxi_front_end.frontend as frontend
from loxi_front_end.frontend_ir import *
class FrontendTests(unittest.TestCase):
maxDiff = None
def test_simple(self):
ast = parser.parse("""
#version 1
enum ofp_port_config {
OFPPC_PORT_DOWN = 0x1,
OFPPC_NO_STP = 0x2,
OFPPC_NO_RECV = 0x4,
OFPPC_NO_RECV_STP = 0x8,
OFPPC_NO_FLOOD = 0x10,
OFPPC_NO_FWD = 0x20,
OFPPC_NO_PACKET_IN = 0x40,
};
#version 2
struct of_echo_reply(align=8) {
uint8_t version;
uint8_t type == 3;
uint16_t length;
uint32_t xid;
of_octets_t data;
};
enum ofp_queue_op_failed_code(wire_type=uint32, bitmask=False, complete=True) {
OFPQOFC_BAD_PORT = 0,
OFPQOFC_BAD_QUEUE = 1,
OFPQOFC_EPERM = 2,
};
struct of_packet_queue {
uint32_t queue_id;
uint16_t len;
pad(2);
list(of_queue_prop_t) properties;
};
""")
# Not testing the parser, just making sure the AST is what we expect
expected_ast = [
['metadata', 'version', '1'],
['enum', 'ofp_port_config', [], [
['OFPPC_PORT_DOWN', [], 1],
['OFPPC_NO_STP', [], 2],
['OFPPC_NO_RECV', [], 4],
['OFPPC_NO_RECV_STP', [], 8],
['OFPPC_NO_FLOOD', [], 16],
['OFPPC_NO_FWD', [], 32],
['OFPPC_NO_PACKET_IN', [], 64]]],
['metadata', 'version', '2'],
['struct', 'of_echo_reply', [['align', '8']], None, [
['data', ['scalar', 'uint8_t'], 'version'],
['type', ['scalar', 'uint8_t'], 'type', 3],
['data', ['scalar', 'uint16_t'], 'length'],
['data', ['scalar', 'uint32_t'], 'xid'],
['data', ['scalar', 'of_octets_t'], 'data']]],
['enum', 'ofp_queue_op_failed_code',
[['wire_type', 'uint32'], ['bitmask','False'], ['complete', 'True']], [
['OFPQOFC_BAD_PORT', [], 0],
['OFPQOFC_BAD_QUEUE', [], 1],
['OFPQOFC_EPERM', [], 2]]],
['struct', 'of_packet_queue', [], None, [
['data', ['scalar', 'uint32_t'], 'queue_id'],
['data', ['scalar', 'uint16_t'], 'len'],
['pad', 2],
['data', ['list', 'list(of_queue_prop_t)'], 'properties']]],
]
self.assertEquals(expected_ast, ast)
ofinput = frontend.create_ofinput("standard-1.0", ast)
self.assertEquals(set([1, 2]), ofinput.wire_versions)
expected_classes = [
OFClass(name='of_echo_reply', superclass=None, members=[
OFDataMember('version', 'uint8_t'), # XXX
OFTypeMember('type', 'uint8_t', 3),
OFLengthMember('length', 'uint16_t'),
OFDataMember('xid', 'uint32_t'),
OFDataMember('data', 'of_octets_t')], virtual=False,
params={'align': '8'}),
OFClass(name='of_packet_queue', superclass=None, members=[
OFDataMember('queue_id', 'uint32_t'),
OFLengthMember('len', 'uint16_t'),
OFPadMember(2),
OFDataMember('properties', 'list(of_queue_prop_t)')], virtual=False, params={}),
]
self.assertEquals(expected_classes, ofinput.classes)
expected_enums = [
OFEnum(name='ofp_port_config', entries=[
OFEnumEntry('OFPPC_PORT_DOWN', 1, {}),
OFEnumEntry('OFPPC_NO_STP', 2, {}),
OFEnumEntry('OFPPC_NO_RECV', 4, {}),
OFEnumEntry('OFPPC_NO_RECV_STP', 8, {}),
OFEnumEntry('OFPPC_NO_FLOOD', 16, {}),
OFEnumEntry('OFPPC_NO_FWD', 32, {}),
OFEnumEntry('OFPPC_NO_PACKET_IN', 64, {})], params={}),
OFEnum(name='ofp_queue_op_failed_code', entries=[
OFEnumEntry('OFPQOFC_BAD_PORT', 0, {}),
OFEnumEntry('OFPQOFC_BAD_QUEUE', 1, {}),
OFEnumEntry('OFPQOFC_EPERM', 2, {})],
params={'wire_type': 'uint32', 'bitmask': 'False', 'complete': 'True'}),
]
self.assertEquals(expected_enums, ofinput.enums)
def test_inheritance(self):
ast = parser.parse("""
#version 1
struct of_queue_prop {
uint16_t type == ?;
uint16_t len;
pad(4);
};
struct of_queue_prop_min_rate : of_queue_prop {
uint16_t type == 1;
uint16_t len;
pad(4);
uint16_t rate;
pad(6);
};
""")
# Not testing the parser, just making sure the AST is what we expect
expected_ast = [
['metadata', 'version', '1'],
['struct', 'of_queue_prop', [], None, [
['discriminator', ['scalar', 'uint16_t'], 'type'],
['data', ['scalar', 'uint16_t'], 'len'],
['pad', 4]]],
['struct', 'of_queue_prop_min_rate', [], 'of_queue_prop', [
['type', ['scalar', 'uint16_t'], 'type', 1],
['data', ['scalar', 'uint16_t'], 'len'],
['pad', 4],
['data', ['scalar', 'uint16_t'], 'rate'],
['pad', 6]]],
]
self.assertEquals(expected_ast, ast)
ofinput = frontend.create_ofinput("standard-1.0", ast)
expected_classes = [
OFClass(name='of_queue_prop', superclass=None, members=[
OFDiscriminatorMember('type', 'uint16_t'),
OFLengthMember('len', 'uint16_t'),
OFPadMember(4)], virtual=True, params={}),
OFClass(name='of_queue_prop_min_rate', superclass='of_queue_prop', members= [
OFTypeMember('type', 'uint16_t', 1),
OFLengthMember('len', 'uint16_t'),
OFPadMember(4),
OFDataMember('rate', 'uint16_t'),
OFPadMember(6)], virtual=False, params= {}),
]
self.assertEquals(expected_classes, ofinput.classes)
def test_field_length(self):
ast = parser.parse("""
#version 1
struct of_test_entry {
uint32_t x;
};
struct of_test {
uint16_t list_len == length(list);
list(of_test_entry_t) list;
};
""")
# Not testing the parser, just making sure the AST is what we expect
expected_ast = [
['metadata', 'version', '1'],
['struct', 'of_test_entry', [], None, [
['data', ['scalar', 'uint32_t'], 'x']]],
['struct', 'of_test', [], None, [
['field_length', ['scalar', 'uint16_t'], 'list_len', 'list'],
['data', ['list', 'list(of_test_entry_t)'], 'list']]]
]
self.assertEquals(expected_ast, ast)
ofinput = frontend.create_ofinput("standard-1.0", ast)
expected_classes = [
OFClass(name='of_test_entry', superclass=None, virtual=False, params={},
members=[
OFDataMember('x', 'uint32_t')]),
OFClass(name='of_test', superclass=None, virtual=False, params={},
members=[
OFFieldLengthMember('list_len', 'uint16_t', 'list'),
OFDataMember('list', 'list(of_test_entry_t)')])
]
self.assertEquals(expected_classes, ofinput.classes)
if __name__ == '__main__':
unittest.main()