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