Stella Stamenova | a27199d | 2018-08-06 22:37:44 +0000 | [diff] [blame] | 1 | // RUN: llvm-tblgen -dump-json %s | %python %S/JSON-check.py %s |
Simon Tatham | 9cfd4e5 | 2018-07-11 08:40:19 +0000 | [diff] [blame] | 2 | |
| 3 | // CHECK: data['!tablegen_json_version'] == 1 |
| 4 | |
| 5 | // CHECK: all(data[s]['!name'] == s for s in data if not s.startswith("!")) |
| 6 | |
| 7 | class Base {} |
| 8 | class Intermediate : Base {} |
| 9 | class Derived : Intermediate {} |
| 10 | |
| 11 | def D : Intermediate {} |
| 12 | // CHECK: 'D' in data['!instanceof']['Base'] |
| 13 | // CHECK: 'D' in data['!instanceof']['Intermediate'] |
| 14 | // CHECK: 'D' not in data['!instanceof']['Derived'] |
| 15 | // CHECK: 'Base' in data['D']['!superclasses'] |
| 16 | // CHECK: 'Intermediate' in data['D']['!superclasses'] |
| 17 | // CHECK: 'Derived' not in data['D']['!superclasses'] |
| 18 | |
| 19 | def ExampleDagOp; |
| 20 | |
| 21 | def FieldKeywordTest { |
| 22 | int a; |
| 23 | field int b; |
| 24 | // CHECK: 'a' not in data['FieldKeywordTest']['!fields'] |
| 25 | // CHECK: 'b' in data['FieldKeywordTest']['!fields'] |
| 26 | } |
| 27 | |
| 28 | class Variables { |
| 29 | int i; |
| 30 | string s; |
| 31 | bit b; |
| 32 | bits<8> bs; |
| 33 | code c; |
| 34 | list<int> li; |
| 35 | Base base; |
| 36 | dag d; |
| 37 | } |
| 38 | def VarNull : Variables { |
| 39 | // A variable not filled in at all has its value set to JSON |
| 40 | // 'null', which translates to Python None |
| 41 | // CHECK: data['VarNull']['i'] is None |
| 42 | } |
| 43 | def VarPrim : Variables { |
| 44 | // Test initializers that map to primitive JSON types |
| 45 | |
| 46 | int i = 3; |
| 47 | // CHECK: data['VarPrim']['i'] == 3 |
| 48 | |
| 49 | // Integer literals should be emitted in the JSON at full 64-bit |
| 50 | // precision, for the benefit of JSON readers that preserve that |
| 51 | // much information. Python's is one such. |
| 52 | int enormous_pos = 9123456789123456789; |
| 53 | int enormous_neg = -9123456789123456789; |
| 54 | // CHECK: data['VarPrim']['enormous_pos'] == 9123456789123456789 |
| 55 | // CHECK: data['VarPrim']['enormous_neg'] == -9123456789123456789 |
| 56 | |
| 57 | string s = "hello, world"; |
| 58 | // CHECK: data['VarPrim']['s'] == 'hello, world' |
| 59 | |
| 60 | bit b = 0; |
| 61 | // CHECK: data['VarPrim']['b'] == 0 |
| 62 | |
| 63 | // bits<> arrays are stored in logical order (array[i] is the same |
| 64 | // bit identified in .td files as bs{i}), which means the _visual_ |
| 65 | // order of the list (in default rendering) is reversed. |
| 66 | bits<8> bs = { 0,0,0,1,0,1,1,1 }; |
| 67 | // CHECK: data['VarPrim']['bs'] == [ 1,1,1,0,1,0,0,0 ] |
| 68 | |
| 69 | code c = [{ \" }]; |
| 70 | // CHECK: data['VarPrim']['c'] == r' \" ' |
| 71 | |
| 72 | list<int> li = [ 1, 2, 3, 4 ]; |
| 73 | // CHECK: data['VarPrim']['li'] == [ 1, 2, 3, 4 ] |
| 74 | } |
| 75 | def VarObj : Variables { |
| 76 | // Test initializers that map to JSON objects containing a 'kind' |
| 77 | // discriminator |
| 78 | |
| 79 | Base base = D; |
| 80 | // CHECK: data['VarObj']['base']['kind'] == 'def' |
| 81 | // CHECK: data['VarObj']['base']['def'] == 'D' |
| 82 | // CHECK: data['VarObj']['base']['printable'] == 'D' |
| 83 | |
| 84 | dag d = (ExampleDagOp 22, "hello":$foo); |
| 85 | // CHECK: data['VarObj']['d']['kind'] == 'dag' |
| 86 | // CHECK: data['VarObj']['d']['operator']['kind'] == 'def' |
| 87 | // CHECK: data['VarObj']['d']['operator']['def'] == 'ExampleDagOp' |
| 88 | // CHECK: data['VarObj']['d']['operator']['printable'] == 'ExampleDagOp' |
| 89 | // CHECK: data['VarObj']['d']['args'] == [[22, None], ["hello", "foo"]] |
| 90 | // CHECK: data['VarObj']['d']['printable'] == '(ExampleDagOp 22, "hello":$foo)' |
| 91 | |
| 92 | int undef_int; |
| 93 | field int ref_int = undef_int; |
| 94 | // CHECK: data['VarObj']['ref_int']['kind'] == 'var' |
| 95 | // CHECK: data['VarObj']['ref_int']['var'] == 'undef_int' |
| 96 | // CHECK: data['VarObj']['ref_int']['printable'] == 'undef_int' |
| 97 | |
| 98 | bits<2> undef_bits; |
| 99 | bits<4> ref_bits; |
| 100 | let ref_bits{3-2} = 0b10; |
| 101 | let ref_bits{1-0} = undef_bits{1-0}; |
| 102 | // CHECK: data['VarObj']['ref_bits'][3] == 1 |
| 103 | // CHECK: data['VarObj']['ref_bits'][2] == 0 |
| 104 | // CHECK: data['VarObj']['ref_bits'][1]['kind'] == 'varbit' |
| 105 | // CHECK: data['VarObj']['ref_bits'][1]['var'] == 'undef_bits' |
| 106 | // CHECK: data['VarObj']['ref_bits'][1]['index'] == 1 |
| 107 | // CHECK: data['VarObj']['ref_bits'][1]['printable'] == 'undef_bits{1}' |
| 108 | // CHECK: data['VarObj']['ref_bits'][0]['kind'] == 'varbit' |
| 109 | // CHECK: data['VarObj']['ref_bits'][0]['var'] == 'undef_bits' |
| 110 | // CHECK: data['VarObj']['ref_bits'][0]['index'] == 0 |
| 111 | // CHECK: data['VarObj']['ref_bits'][0]['printable'] == 'undef_bits{0}' |
| 112 | |
| 113 | field int complex_ref_int = !add(undef_int, 2); |
| 114 | // CHECK: data['VarObj']['complex_ref_int']['kind'] == 'complex' |
| 115 | // CHECK: data['VarObj']['complex_ref_int']['printable'] == '!add(undef_int, 2)' |
| 116 | } |
| 117 | |
| 118 | // Test the !anonymous member. This is tricky because when a def is |
| 119 | // anonymous, almost by definition, the test can't reliably predict |
| 120 | // the name it will be stored under! So we have to search all the defs |
| 121 | // in the JSON output looking for the one that has the test integer |
| 122 | // field set to the right value. |
| 123 | |
| 124 | def Named { int AnonTestField = 1; } |
| 125 | // CHECK: data['Named']['AnonTestField'] == 1 |
| 126 | // CHECK: data['Named']['!anonymous'] is False |
| 127 | |
| 128 | def { int AnonTestField = 2; } |
| 129 | // CHECK: next(rec for rec in data.values() if isinstance(rec, dict) and rec.get('AnonTestField') == 2)['!anonymous'] is True |
| 130 | |
| 131 | multiclass AnonTestMulticlass<int base> { |
| 132 | def _plus_one { int AnonTestField = !add(base,1); } |
| 133 | def { int AnonTestField = !add(base,2); } |
| 134 | } |
| 135 | |
| 136 | defm NamedDefm : AnonTestMulticlass<10>; |
| 137 | // CHECK: data['NamedDefm_plus_one']['!anonymous'] is False |
| 138 | // CHECK: data['NamedDefm_plus_one']['AnonTestField'] == 11 |
| 139 | // CHECK: next(rec for rec in data.values() if isinstance(rec, dict) and rec.get('AnonTestField') == 12)['!anonymous'] is True |
| 140 | |
| 141 | // D47431 clarifies that a named def inside a multiclass gives a |
| 142 | // *non*-anonymous output record, even if the defm that instantiates |
| 143 | // that multiclass is anonymous. |
| 144 | defm : AnonTestMulticlass<20>; |
| 145 | // CHECK: next(rec for rec in data.values() if isinstance(rec, dict) and rec.get('AnonTestField') == 21)['!anonymous'] is False |
| 146 | // CHECK: next(rec for rec in data.values() if isinstance(rec, dict) and rec.get('AnonTestField') == 22)['!anonymous'] is True |