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(')')) |
Andreas Wundsam | 780e0c9 | 2013-08-02 17:48:27 -0700 | [diff] [blame] | 54 | discriminator_member = P.Group(tag('discriminator') + any_type + identifier + s('==') + s('?')) |
Rich Lane | 3214287 | 2013-05-09 21:16:47 -0700 | [diff] [blame] | 55 | type_member = P.Group(tag('type') + any_type + identifier + s('==') + integer) |
Rich Lane | f424e97 | 2013-05-09 21:00:13 -0700 | [diff] [blame] | 56 | data_member = P.Group(tag('data') + any_type - identifier) |
Andreas Wundsam | fef7d5f | 2013-08-01 22:15:44 -0700 | [diff] [blame] | 57 | |
| 58 | struct_param_name = kw("align") |
| 59 | struct_param = P.Group(struct_param_name - s('=') - any_type) |
| 60 | struct_param_list = P.Forward() |
| 61 | struct_param_list << struct_param + P.Optional(s(',') - P.Optional(struct_param_list)) |
| 62 | |
Andreas Wundsam | 780e0c9 | 2013-08-02 17:48:27 -0700 | [diff] [blame] | 63 | struct_member = pad_member | type_member | discriminator_member | data_member; |
Rich Lane | bd43150 | 2013-06-21 16:30:20 -0700 | [diff] [blame] | 64 | parent = (s(':') - identifier) | tag(None) |
Andreas Wundsam | cd2d525 | 2013-08-02 13:35:57 -0700 | [diff] [blame] | 65 | struct = kw('struct') - identifier - P.Group(P.Optional(s('(') - struct_param_list - s(')'))) - parent - s('{') + \ |
Rich Lane | e4d04ea | 2013-05-09 11:27:06 -0700 | [diff] [blame] | 66 | P.Group(P.ZeroOrMore(struct_member - s(';'))) + \ |
Rich Lane | a06d0c3 | 2013-03-25 08:52:03 -0700 | [diff] [blame] | 67 | s('}') - s(';') |
| 68 | |
Rich Lane | 517506c | 2013-04-08 14:08:31 -0700 | [diff] [blame] | 69 | # Enums |
Andreas Wundsam | 4ee5146 | 2013-07-30 11:00:37 -0700 | [diff] [blame] | 70 | enum_param_name = kw("wire_type") | kw("bitmask") | kw("complete") |
| 71 | enum_param = P.Group(enum_param_name - s('=') - any_type) |
| 72 | enum_param_list = P.Forward() |
| 73 | enum_param_list << enum_param + P.Optional(s(',') + P.Optional(enum_param_list)) |
| 74 | |
| 75 | enum_member_param_name = kw("virtual") |
| 76 | enum_member_param = P.Group(enum_member_param_name - s('=') - any_type) |
| 77 | enum_member_param_list = P.Forward() |
| 78 | enum_member_param_list << enum_member_param + P.Optional(s(',') + P.Optional(enum_member_param_list)) |
| 79 | |
| 80 | enum_member = P.Group(identifier - P.Group(P.Optional(s('(') - enum_member_param_list - s(')'))) - s('=') + integer) |
Rich Lane | 517506c | 2013-04-08 14:08:31 -0700 | [diff] [blame] | 81 | enum_list = P.Forward() |
| 82 | enum_list << enum_member + P.Optional(s(',') + P.Optional(enum_list)) |
Andreas Wundsam | 4ee5146 | 2013-07-30 11:00:37 -0700 | [diff] [blame] | 83 | enum = kw('enum') - identifier - P.Group(P.Optional(s('(') - enum_param_list - s(')'))) - s('{') + \ |
Rich Lane | 517506c | 2013-04-08 14:08:31 -0700 | [diff] [blame] | 84 | P.Group(P.Optional(enum_list)) + \ |
| 85 | s('}') - s(';') |
| 86 | |
Rich Lane | a06d0c3 | 2013-03-25 08:52:03 -0700 | [diff] [blame] | 87 | # Metadata |
| 88 | metadata_key = P.Or(kw("version")).setName("metadata key") |
| 89 | metadata = tag('metadata') + s('#') - metadata_key - word |
| 90 | |
Rich Lane | 517506c | 2013-04-08 14:08:31 -0700 | [diff] [blame] | 91 | grammar = P.ZeroOrMore(P.Group(struct) | P.Group(enum) | P.Group(metadata)) |
Rich Lane | a06d0c3 | 2013-03-25 08:52:03 -0700 | [diff] [blame] | 92 | grammar.ignore(P.cppStyleComment) |
| 93 | |
| 94 | def parse(src): |
Rich Lane | 43b2a90 | 2013-05-09 13:47:07 -0700 | [diff] [blame] | 95 | """ |
| 96 | Given an input string, return the AST. |
| 97 | |
| 98 | The AST is a low-level representation of the input. It changes frequently |
| 99 | with the input file syntax. The frontend.py module transforms the AST |
| 100 | into the OFInput represntation. |
| 101 | """ |
| 102 | return grammar.parseString(src, parseAll=True).asList() |