Rich Lane | a06d0c3 | 2013-03-25 08:52:03 -0700 | [diff] [blame] | 1 | # 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 pyparsing as P |
| 29 | |
| 30 | kw = P.Keyword |
| 31 | s = P.Suppress |
| 32 | lit = P.Literal |
| 33 | |
| 34 | # Useful for marking the type of a parse result (matches the empty string, but |
| 35 | # shows up in the result) |
| 36 | tag = lambda name: P.Empty().setParseAction(P.replaceWith(name)) |
| 37 | |
| 38 | word = P.Word(P.alphanums + '_') |
Rich Lane | fe735bc | 2013-04-08 16:58:49 -0700 | [diff] [blame] | 39 | integer = ( |
| 40 | P.Combine('0x' - P.Word('0123456789abcdefABCDEF') | |
| 41 | P.Word('0123456789')) |
| 42 | ).setParseAction(lambda x: int(x[0], 0)) |
Rich Lane | a06d0c3 | 2013-03-25 08:52:03 -0700 | [diff] [blame] | 43 | |
| 44 | identifier = word.copy().setName("identifier") |
| 45 | |
| 46 | # Type names |
| 47 | scalar_type = word |
| 48 | array_type = P.Combine(word + lit('[') - P.Word(P.alphanums + '_') - lit(']')) |
| 49 | list_type = P.Combine(kw('list') - lit('(') - identifier - lit(')')) |
| 50 | any_type = (array_type | list_type | scalar_type).setName("type name") |
| 51 | |
| 52 | # Structs |
Rich Lane | e4d04ea | 2013-05-09 11:27:06 -0700 | [diff] [blame] | 53 | pad_member = P.Group(kw('pad') - s('(') - integer - s(')')) |
Rich Lane | 3214287 | 2013-05-09 21:16:47 -0700 | [diff] [blame] | 54 | type_member = P.Group(tag('type') + any_type + identifier + s('==') + integer) |
Rich Lane | f424e97 | 2013-05-09 21:00:13 -0700 | [diff] [blame] | 55 | data_member = P.Group(tag('data') + any_type - identifier) |
Rich Lane | 3214287 | 2013-05-09 21:16:47 -0700 | [diff] [blame] | 56 | struct_member = pad_member | type_member | data_member; |
Rich Lane | a06d0c3 | 2013-03-25 08:52:03 -0700 | [diff] [blame] | 57 | struct = kw('struct') - identifier - s('{') + \ |
Rich Lane | e4d04ea | 2013-05-09 11:27:06 -0700 | [diff] [blame] | 58 | P.Group(P.ZeroOrMore(struct_member - s(';'))) + \ |
Rich Lane | a06d0c3 | 2013-03-25 08:52:03 -0700 | [diff] [blame] | 59 | s('}') - s(';') |
| 60 | |
Rich Lane | 517506c | 2013-04-08 14:08:31 -0700 | [diff] [blame] | 61 | # Enums |
Rich Lane | fe735bc | 2013-04-08 16:58:49 -0700 | [diff] [blame] | 62 | enum_member = P.Group(identifier + s('=') + integer) |
Rich Lane | 517506c | 2013-04-08 14:08:31 -0700 | [diff] [blame] | 63 | enum_list = P.Forward() |
| 64 | enum_list << enum_member + P.Optional(s(',') + P.Optional(enum_list)) |
| 65 | enum = kw('enum') - identifier - s('{') + \ |
| 66 | P.Group(P.Optional(enum_list)) + \ |
| 67 | s('}') - s(';') |
| 68 | |
Rich Lane | a06d0c3 | 2013-03-25 08:52:03 -0700 | [diff] [blame] | 69 | # Metadata |
| 70 | metadata_key = P.Or(kw("version")).setName("metadata key") |
| 71 | metadata = tag('metadata') + s('#') - metadata_key - word |
| 72 | |
Rich Lane | 517506c | 2013-04-08 14:08:31 -0700 | [diff] [blame] | 73 | grammar = P.ZeroOrMore(P.Group(struct) | P.Group(enum) | P.Group(metadata)) |
Rich Lane | a06d0c3 | 2013-03-25 08:52:03 -0700 | [diff] [blame] | 74 | grammar.ignore(P.cppStyleComment) |
| 75 | |
| 76 | def parse(src): |
Rich Lane | 43b2a90 | 2013-05-09 13:47:07 -0700 | [diff] [blame] | 77 | """ |
| 78 | Given an input string, return the AST. |
| 79 | |
| 80 | The AST is a low-level representation of the input. It changes frequently |
| 81 | with the input file syntax. The frontend.py module transforms the AST |
| 82 | into the OFInput represntation. |
| 83 | """ |
| 84 | return grammar.parseString(src, parseAll=True).asList() |