| 1 | // This defines the bytecode instructions used by the VM. It does so by invoking | 
|---|
| 2 | // an OPCODE() macro which is expected to be defined at the point that this is | 
|---|
| 3 | // included. (See: http://en.wikipedia.org/wiki/X_Macro for more.) | 
|---|
| 4 | // | 
|---|
| 5 | // The first argument is the name of the opcode. The second is its "stack | 
|---|
| 6 | // effect" -- the amount that the op code changes the size of the stack. A | 
|---|
| 7 | // stack effect of 1 means it pushes a value and the stack grows one larger. | 
|---|
| 8 | // -2 means it pops two values, etc. | 
|---|
| 9 | // | 
|---|
| 10 | // Note that the order of instructions here affects the order of the dispatch | 
|---|
| 11 | // table in the VM's interpreter loop. That in turn affects caching which | 
|---|
| 12 | // affects overall performance. Take care to run benchmarks if you change the | 
|---|
| 13 | // order here. | 
|---|
| 14 |  | 
|---|
| 15 | // Load the constant at index [arg]. | 
|---|
| 16 | OPCODE(CONSTANT, 1) | 
|---|
| 17 |  | 
|---|
| 18 | // Push null onto the stack. | 
|---|
| 19 | OPCODE(NULL, 1) | 
|---|
| 20 |  | 
|---|
| 21 | // Push false onto the stack. | 
|---|
| 22 | OPCODE(FALSE, 1) | 
|---|
| 23 |  | 
|---|
| 24 | // Push true onto the stack. | 
|---|
| 25 | OPCODE(TRUE, 1) | 
|---|
| 26 |  | 
|---|
| 27 | // Pushes the value in the given local slot. | 
|---|
| 28 | OPCODE(LOAD_LOCAL_0, 1) | 
|---|
| 29 | OPCODE(LOAD_LOCAL_1, 1) | 
|---|
| 30 | OPCODE(LOAD_LOCAL_2, 1) | 
|---|
| 31 | OPCODE(LOAD_LOCAL_3, 1) | 
|---|
| 32 | OPCODE(LOAD_LOCAL_4, 1) | 
|---|
| 33 | OPCODE(LOAD_LOCAL_5, 1) | 
|---|
| 34 | OPCODE(LOAD_LOCAL_6, 1) | 
|---|
| 35 | OPCODE(LOAD_LOCAL_7, 1) | 
|---|
| 36 | OPCODE(LOAD_LOCAL_8, 1) | 
|---|
| 37 |  | 
|---|
| 38 | // Note: The compiler assumes the following _STORE instructions always | 
|---|
| 39 | // immediately follow their corresponding _LOAD ones. | 
|---|
| 40 |  | 
|---|
| 41 | // Pushes the value in local slot [arg]. | 
|---|
| 42 | OPCODE(LOAD_LOCAL, 1) | 
|---|
| 43 |  | 
|---|
| 44 | // Stores the top of stack in local slot [arg]. Does not pop it. | 
|---|
| 45 | OPCODE(STORE_LOCAL, 0) | 
|---|
| 46 |  | 
|---|
| 47 | // Pushes the value in upvalue [arg]. | 
|---|
| 48 | OPCODE(LOAD_UPVALUE, 1) | 
|---|
| 49 |  | 
|---|
| 50 | // Stores the top of stack in upvalue [arg]. Does not pop it. | 
|---|
| 51 | OPCODE(STORE_UPVALUE, 0) | 
|---|
| 52 |  | 
|---|
| 53 | // Pushes the value of the top-level variable in slot [arg]. | 
|---|
| 54 | OPCODE(LOAD_MODULE_VAR, 1) | 
|---|
| 55 |  | 
|---|
| 56 | // Stores the top of stack in top-level variable slot [arg]. Does not pop it. | 
|---|
| 57 | OPCODE(STORE_MODULE_VAR, 0) | 
|---|
| 58 |  | 
|---|
| 59 | // Pushes the value of the field in slot [arg] of the receiver of the current | 
|---|
| 60 | // function. This is used for regular field accesses on "this" directly in | 
|---|
| 61 | // methods. This instruction is faster than the more general CODE_LOAD_FIELD | 
|---|
| 62 | // instruction. | 
|---|
| 63 | OPCODE(LOAD_FIELD_THIS, 1) | 
|---|
| 64 |  | 
|---|
| 65 | // Stores the top of the stack in field slot [arg] in the receiver of the | 
|---|
| 66 | // current value. Does not pop the value. This instruction is faster than the | 
|---|
| 67 | // more general CODE_LOAD_FIELD instruction. | 
|---|
| 68 | OPCODE(STORE_FIELD_THIS, 0) | 
|---|
| 69 |  | 
|---|
| 70 | // Pops an instance and pushes the value of the field in slot [arg] of it. | 
|---|
| 71 | OPCODE(LOAD_FIELD, 0) | 
|---|
| 72 |  | 
|---|
| 73 | // Pops an instance and stores the subsequent top of stack in field slot | 
|---|
| 74 | // [arg] in it. Does not pop the value. | 
|---|
| 75 | OPCODE(STORE_FIELD, -1) | 
|---|
| 76 |  | 
|---|
| 77 | // Pop and discard the top of stack. | 
|---|
| 78 | OPCODE(POP, -1) | 
|---|
| 79 |  | 
|---|
| 80 | // Invoke the method with symbol [arg]. The number indicates the number of | 
|---|
| 81 | // arguments (not including the receiver). | 
|---|
| 82 | OPCODE(CALL_0, 0) | 
|---|
| 83 | OPCODE(CALL_1, -1) | 
|---|
| 84 | OPCODE(CALL_2, -2) | 
|---|
| 85 | OPCODE(CALL_3, -3) | 
|---|
| 86 | OPCODE(CALL_4, -4) | 
|---|
| 87 | OPCODE(CALL_5, -5) | 
|---|
| 88 | OPCODE(CALL_6, -6) | 
|---|
| 89 | OPCODE(CALL_7, -7) | 
|---|
| 90 | OPCODE(CALL_8, -8) | 
|---|
| 91 | OPCODE(CALL_9, -9) | 
|---|
| 92 | OPCODE(CALL_10, -10) | 
|---|
| 93 | OPCODE(CALL_11, -11) | 
|---|
| 94 | OPCODE(CALL_12, -12) | 
|---|
| 95 | OPCODE(CALL_13, -13) | 
|---|
| 96 | OPCODE(CALL_14, -14) | 
|---|
| 97 | OPCODE(CALL_15, -15) | 
|---|
| 98 | OPCODE(CALL_16, -16) | 
|---|
| 99 |  | 
|---|
| 100 | // Invoke a superclass method with symbol [arg]. The number indicates the | 
|---|
| 101 | // number of arguments (not including the receiver). | 
|---|
| 102 | OPCODE(SUPER_0, 0) | 
|---|
| 103 | OPCODE(SUPER_1, -1) | 
|---|
| 104 | OPCODE(SUPER_2, -2) | 
|---|
| 105 | OPCODE(SUPER_3, -3) | 
|---|
| 106 | OPCODE(SUPER_4, -4) | 
|---|
| 107 | OPCODE(SUPER_5, -5) | 
|---|
| 108 | OPCODE(SUPER_6, -6) | 
|---|
| 109 | OPCODE(SUPER_7, -7) | 
|---|
| 110 | OPCODE(SUPER_8, -8) | 
|---|
| 111 | OPCODE(SUPER_9, -9) | 
|---|
| 112 | OPCODE(SUPER_10, -10) | 
|---|
| 113 | OPCODE(SUPER_11, -11) | 
|---|
| 114 | OPCODE(SUPER_12, -12) | 
|---|
| 115 | OPCODE(SUPER_13, -13) | 
|---|
| 116 | OPCODE(SUPER_14, -14) | 
|---|
| 117 | OPCODE(SUPER_15, -15) | 
|---|
| 118 | OPCODE(SUPER_16, -16) | 
|---|
| 119 |  | 
|---|
| 120 | // Jump the instruction pointer [arg] forward. | 
|---|
| 121 | OPCODE(JUMP, 0) | 
|---|
| 122 |  | 
|---|
| 123 | // Jump the instruction pointer [arg] backward. | 
|---|
| 124 | OPCODE(LOOP, 0) | 
|---|
| 125 |  | 
|---|
| 126 | // Pop and if not truthy then jump the instruction pointer [arg] forward. | 
|---|
| 127 | OPCODE(JUMP_IF, -1) | 
|---|
| 128 |  | 
|---|
| 129 | // If the top of the stack is false, jump [arg] forward. Otherwise, pop and | 
|---|
| 130 | // continue. | 
|---|
| 131 | OPCODE(AND, -1) | 
|---|
| 132 |  | 
|---|
| 133 | // If the top of the stack is non-false, jump [arg] forward. Otherwise, pop | 
|---|
| 134 | // and continue. | 
|---|
| 135 | OPCODE(OR, -1) | 
|---|
| 136 |  | 
|---|
| 137 | // Close the upvalue for the local on the top of the stack, then pop it. | 
|---|
| 138 | OPCODE(CLOSE_UPVALUE, -1) | 
|---|
| 139 |  | 
|---|
| 140 | // Exit from the current function and return the value on the top of the | 
|---|
| 141 | // stack. | 
|---|
| 142 | OPCODE(RETURN, 0) | 
|---|
| 143 |  | 
|---|
| 144 | // Creates a closure for the function stored at [arg] in the constant table. | 
|---|
| 145 | // | 
|---|
| 146 | // Following the function argument is a number of arguments, two for each | 
|---|
| 147 | // upvalue. The first is true if the variable being captured is a local (as | 
|---|
| 148 | // opposed to an upvalue), and the second is the index of the local or | 
|---|
| 149 | // upvalue being captured. | 
|---|
| 150 | // | 
|---|
| 151 | // Pushes the created closure. | 
|---|
| 152 | OPCODE(CLOSURE, 1) | 
|---|
| 153 |  | 
|---|
| 154 | // Creates a new instance of a class. | 
|---|
| 155 | // | 
|---|
| 156 | // Assumes the class object is in slot zero, and replaces it with the new | 
|---|
| 157 | // uninitialized instance of that class. This opcode is only emitted by the | 
|---|
| 158 | // compiler-generated constructor metaclass methods. | 
|---|
| 159 | OPCODE(CONSTRUCT, 0) | 
|---|
| 160 |  | 
|---|
| 161 | // Creates a new instance of a foreign class. | 
|---|
| 162 | // | 
|---|
| 163 | // Assumes the class object is in slot zero, and replaces it with the new | 
|---|
| 164 | // uninitialized instance of that class. This opcode is only emitted by the | 
|---|
| 165 | // compiler-generated constructor metaclass methods. | 
|---|
| 166 | OPCODE(FOREIGN_CONSTRUCT, 0) | 
|---|
| 167 |  | 
|---|
| 168 | // Creates a class. Top of stack is the superclass. Below that is a string for | 
|---|
| 169 | // the name of the class. Byte [arg] is the number of fields in the class. | 
|---|
| 170 | OPCODE(CLASS, -1) | 
|---|
| 171 |  | 
|---|
| 172 | // Ends a class. | 
|---|
| 173 | // Atm the stack contains the class and the ClassAttributes (or null). | 
|---|
| 174 | OPCODE(END_CLASS, -2) | 
|---|
| 175 |  | 
|---|
| 176 | // Creates a foreign class. Top of stack is the superclass. Below that is a | 
|---|
| 177 | // string for the name of the class. | 
|---|
| 178 | OPCODE(FOREIGN_CLASS, -1) | 
|---|
| 179 |  | 
|---|
| 180 | // Define a method for symbol [arg]. The class receiving the method is popped | 
|---|
| 181 | // off the stack, then the function defining the body is popped. | 
|---|
| 182 | // | 
|---|
| 183 | // If a foreign method is being defined, the "function" will be a string | 
|---|
| 184 | // identifying the foreign method. Otherwise, it will be a function or | 
|---|
| 185 | // closure. | 
|---|
| 186 | OPCODE(METHOD_INSTANCE, -2) | 
|---|
| 187 |  | 
|---|
| 188 | // Define a method for symbol [arg]. The class whose metaclass will receive | 
|---|
| 189 | // the method is popped off the stack, then the function defining the body is | 
|---|
| 190 | // popped. | 
|---|
| 191 | // | 
|---|
| 192 | // If a foreign method is being defined, the "function" will be a string | 
|---|
| 193 | // identifying the foreign method. Otherwise, it will be a function or | 
|---|
| 194 | // closure. | 
|---|
| 195 | OPCODE(METHOD_STATIC, -2) | 
|---|
| 196 |  | 
|---|
| 197 | // This is executed at the end of the module's body. Pushes NULL onto the stack | 
|---|
| 198 | // as the "return value" of the import statement and stores the module as the | 
|---|
| 199 | // most recently imported one. | 
|---|
| 200 | OPCODE(END_MODULE, 1) | 
|---|
| 201 |  | 
|---|
| 202 | // Import a module whose name is the string stored at [arg] in the constant | 
|---|
| 203 | // table. | 
|---|
| 204 | // | 
|---|
| 205 | // Pushes null onto the stack so that the fiber for the imported module can | 
|---|
| 206 | // replace that with a dummy value when it returns. (Fibers always return a | 
|---|
| 207 | // value when resuming a caller.) | 
|---|
| 208 | OPCODE(IMPORT_MODULE, 1) | 
|---|
| 209 |  | 
|---|
| 210 | // Import a variable from the most recently imported module. The name of the | 
|---|
| 211 | // variable to import is at [arg] in the constant table. Pushes the loaded | 
|---|
| 212 | // variable's value. | 
|---|
| 213 | OPCODE(IMPORT_VARIABLE, 1) | 
|---|
| 214 |  | 
|---|
| 215 | // This pseudo-instruction indicates the end of the bytecode. It should | 
|---|
| 216 | // always be preceded by a `CODE_RETURN`, so is never actually executed. | 
|---|
| 217 | OPCODE(END, 0) | 
|---|
| 218 |  | 
|---|