2014-05-03 18:27:38 -04:00
|
|
|
/*
|
2017-06-30 03:22:17 -04:00
|
|
|
* This file is part of the MicroPython project, http://micropython.org/
|
2014-05-03 18:27:38 -04:00
|
|
|
*
|
|
|
|
* The MIT License (MIT)
|
|
|
|
*
|
|
|
|
* Copyright (c) 2013, 2014 Damien P. George
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
all: Unify header guard usage.
The code conventions suggest using header guards, but do not define how
those should look like and instead point to existing files. However, not
all existing files follow the same scheme, sometimes omitting header guards
altogether, sometimes using non-standard names, making it easy to
accidentally pick a "wrong" example.
This commit ensures that all header files of the MicroPython project (that
were not simply copied from somewhere else) follow the same pattern, that
was already present in the majority of files, especially in the py folder.
The rules are as follows.
Naming convention:
* start with the words MICROPY_INCLUDED
* contain the full path to the file
* replace special characters with _
In addition, there are no empty lines before #ifndef, between #ifndef and
one empty line before #endif. #endif is followed by a comment containing
the name of the guard macro.
py/grammar.h cannot use header guards by design, since it has to be
included multiple times in a single C file. Several other files also do not
need header guards as they are only used internally and guaranteed to be
included only once:
* MICROPY_MPHALPORT_H
* mpconfigboard.h
* mpconfigport.h
* mpthreadport.h
* pin_defs_*.h
* qstrdefs*.h
2017-06-29 17:14:58 -04:00
|
|
|
#ifndef MICROPY_INCLUDED_PY_BC0_H
|
|
|
|
#define MICROPY_INCLUDED_PY_BC0_H
|
2014-05-03 18:27:38 -04:00
|
|
|
|
2019-09-02 06:24:01 -04:00
|
|
|
// MicroPython bytecode opcodes, grouped based on the format of the opcode
|
2014-01-04 13:44:46 -05:00
|
|
|
|
2022-03-15 18:37:58 -04:00
|
|
|
// All opcodes are encoded as a byte with an optional argument. Arguments are
|
|
|
|
// variable-length encoded so they can be as small as possible. The possible
|
|
|
|
// encodings for arguments are (ip[0] is the opcode):
|
|
|
|
//
|
|
|
|
// - unsigned relative bytecode offset:
|
|
|
|
// - if ip[1] high bit is clear then: arg = ip[1]
|
|
|
|
// - if ip[1] high bit is set then: arg = ip[1] & 0x7f | ip[2] << 7
|
|
|
|
//
|
|
|
|
// - signed relative bytecode offset:
|
|
|
|
// - if ip[1] high bit is clear then: arg = ip[1] - 0x40
|
|
|
|
// - if ip[1] high bit is set then: arg = (ip[1] & 0x7f | ip[2] << 7) - 0x4000
|
|
|
|
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_MASK_FORMAT (0xf0)
|
|
|
|
#define MP_BC_MASK_EXTRA_BYTE (0x9e)
|
2013-12-21 13:17:45 -05:00
|
|
|
|
2019-09-02 07:35:26 -04:00
|
|
|
#define MP_BC_FORMAT_BYTE (0)
|
|
|
|
#define MP_BC_FORMAT_QSTR (1)
|
|
|
|
#define MP_BC_FORMAT_VAR_UINT (2)
|
|
|
|
#define MP_BC_FORMAT_OFFSET (3)
|
|
|
|
|
|
|
|
// Nibbles in magic number are: BB BB BB BB BB BO VV QU
|
|
|
|
#define MP_BC_FORMAT(op) ((0x000003a4 >> (2 * ((op) >> 4))) & 3)
|
|
|
|
|
2019-09-02 06:24:01 -04:00
|
|
|
// Load, Store, Delete, Import, Make, Build, Unpack, Call, Jump, Exception, For, sTack, Return, Yield, Op
|
|
|
|
#define MP_BC_BASE_RESERVED (0x00) // ----------------
|
|
|
|
#define MP_BC_BASE_QSTR_O (0x10) // LLLLLLSSSDDII---
|
|
|
|
#define MP_BC_BASE_VINT_E (0x20) // MMLLLLSSDDBBBBBB
|
|
|
|
#define MP_BC_BASE_VINT_O (0x30) // UUMMCCCC--------
|
|
|
|
#define MP_BC_BASE_JUMP_E (0x40) // J-JJJJJEEEEF----
|
|
|
|
#define MP_BC_BASE_BYTE_O (0x50) // LLLLSSDTTTTTEEFF
|
2019-08-21 22:39:07 -04:00
|
|
|
#define MP_BC_BASE_BYTE_E (0x60) // --BREEEYYI------
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_LOAD_CONST_SMALL_INT_MULTI (0x70) // LLLLLLLLLLLLLLLL
|
2020-02-25 20:07:17 -05:00
|
|
|
// (0x80) // LLLLLLLLLLLLLLLL
|
|
|
|
// (0x90) // LLLLLLLLLLLLLLLL
|
|
|
|
// (0xa0) // LLLLLLLLLLLLLLLL
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_LOAD_FAST_MULTI (0xb0) // LLLLLLLLLLLLLLLL
|
|
|
|
#define MP_BC_STORE_FAST_MULTI (0xc0) // SSSSSSSSSSSSSSSS
|
|
|
|
#define MP_BC_UNARY_OP_MULTI (0xd0) // OOOOOOO
|
|
|
|
#define MP_BC_BINARY_OP_MULTI (0xd7) // OOOOOOOOO
|
2020-02-25 20:07:17 -05:00
|
|
|
// (0xe0) // OOOOOOOOOOOOOOOO
|
|
|
|
// (0xf0) // OOOOOOOOOO------
|
2013-12-21 13:17:45 -05:00
|
|
|
|
2019-09-11 22:19:49 -04:00
|
|
|
#define MP_BC_LOAD_CONST_SMALL_INT_MULTI_NUM (64)
|
|
|
|
#define MP_BC_LOAD_CONST_SMALL_INT_MULTI_EXCESS (16)
|
|
|
|
#define MP_BC_LOAD_FAST_MULTI_NUM (16)
|
|
|
|
#define MP_BC_STORE_FAST_MULTI_NUM (16)
|
|
|
|
#define MP_BC_UNARY_OP_MULTI_NUM (MP_UNARY_OP_NUM_BYTECODE)
|
|
|
|
#define MP_BC_BINARY_OP_MULTI_NUM (MP_BINARY_OP_NUM_BYTECODE)
|
|
|
|
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_LOAD_CONST_FALSE (MP_BC_BASE_BYTE_O + 0x00)
|
|
|
|
#define MP_BC_LOAD_CONST_NONE (MP_BC_BASE_BYTE_O + 0x01)
|
|
|
|
#define MP_BC_LOAD_CONST_TRUE (MP_BC_BASE_BYTE_O + 0x02)
|
|
|
|
#define MP_BC_LOAD_CONST_SMALL_INT (MP_BC_BASE_VINT_E + 0x02) // signed var-int
|
|
|
|
#define MP_BC_LOAD_CONST_STRING (MP_BC_BASE_QSTR_O + 0x00) // qstr
|
|
|
|
#define MP_BC_LOAD_CONST_OBJ (MP_BC_BASE_VINT_E + 0x03) // ptr
|
|
|
|
#define MP_BC_LOAD_NULL (MP_BC_BASE_BYTE_O + 0x03)
|
2013-12-21 13:17:45 -05:00
|
|
|
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_LOAD_FAST_N (MP_BC_BASE_VINT_E + 0x04) // uint
|
|
|
|
#define MP_BC_LOAD_DEREF (MP_BC_BASE_VINT_E + 0x05) // uint
|
|
|
|
#define MP_BC_LOAD_NAME (MP_BC_BASE_QSTR_O + 0x01) // qstr
|
|
|
|
#define MP_BC_LOAD_GLOBAL (MP_BC_BASE_QSTR_O + 0x02) // qstr
|
|
|
|
#define MP_BC_LOAD_ATTR (MP_BC_BASE_QSTR_O + 0x03) // qstr
|
|
|
|
#define MP_BC_LOAD_METHOD (MP_BC_BASE_QSTR_O + 0x04) // qstr
|
|
|
|
#define MP_BC_LOAD_SUPER_METHOD (MP_BC_BASE_QSTR_O + 0x05) // qstr
|
|
|
|
#define MP_BC_LOAD_BUILD_CLASS (MP_BC_BASE_BYTE_O + 0x04)
|
|
|
|
#define MP_BC_LOAD_SUBSCR (MP_BC_BASE_BYTE_O + 0x05)
|
2013-12-21 13:17:45 -05:00
|
|
|
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_STORE_FAST_N (MP_BC_BASE_VINT_E + 0x06) // uint
|
|
|
|
#define MP_BC_STORE_DEREF (MP_BC_BASE_VINT_E + 0x07) // uint
|
|
|
|
#define MP_BC_STORE_NAME (MP_BC_BASE_QSTR_O + 0x06) // qstr
|
|
|
|
#define MP_BC_STORE_GLOBAL (MP_BC_BASE_QSTR_O + 0x07) // qstr
|
|
|
|
#define MP_BC_STORE_ATTR (MP_BC_BASE_QSTR_O + 0x08) // qstr
|
|
|
|
#define MP_BC_STORE_SUBSCR (MP_BC_BASE_BYTE_O + 0x06)
|
2013-12-21 13:17:45 -05:00
|
|
|
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_DELETE_FAST (MP_BC_BASE_VINT_E + 0x08) // uint
|
|
|
|
#define MP_BC_DELETE_DEREF (MP_BC_BASE_VINT_E + 0x09) // uint
|
|
|
|
#define MP_BC_DELETE_NAME (MP_BC_BASE_QSTR_O + 0x09) // qstr
|
|
|
|
#define MP_BC_DELETE_GLOBAL (MP_BC_BASE_QSTR_O + 0x0a) // qstr
|
2013-12-21 13:17:45 -05:00
|
|
|
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_DUP_TOP (MP_BC_BASE_BYTE_O + 0x07)
|
|
|
|
#define MP_BC_DUP_TOP_TWO (MP_BC_BASE_BYTE_O + 0x08)
|
|
|
|
#define MP_BC_POP_TOP (MP_BC_BASE_BYTE_O + 0x09)
|
|
|
|
#define MP_BC_ROT_TWO (MP_BC_BASE_BYTE_O + 0x0a)
|
|
|
|
#define MP_BC_ROT_THREE (MP_BC_BASE_BYTE_O + 0x0b)
|
2013-12-21 13:17:45 -05:00
|
|
|
|
2022-03-15 18:37:58 -04:00
|
|
|
#define MP_BC_UNWIND_JUMP (MP_BC_BASE_JUMP_E + 0x00) // signed relative bytecode offset; then a byte
|
|
|
|
#define MP_BC_JUMP (MP_BC_BASE_JUMP_E + 0x02) // signed relative bytecode offset
|
|
|
|
#define MP_BC_POP_JUMP_IF_TRUE (MP_BC_BASE_JUMP_E + 0x03) // signed relative bytecode offset
|
|
|
|
#define MP_BC_POP_JUMP_IF_FALSE (MP_BC_BASE_JUMP_E + 0x04) // signed relative bytecode offset
|
2022-03-21 01:36:13 -04:00
|
|
|
#define MP_BC_JUMP_IF_TRUE_OR_POP (MP_BC_BASE_JUMP_E + 0x05) // unsigned relative bytecode offset
|
|
|
|
#define MP_BC_JUMP_IF_FALSE_OR_POP (MP_BC_BASE_JUMP_E + 0x06) // unsigned relative bytecode offset
|
2022-03-15 18:37:58 -04:00
|
|
|
#define MP_BC_SETUP_WITH (MP_BC_BASE_JUMP_E + 0x07) // unsigned relative bytecode offset
|
|
|
|
#define MP_BC_SETUP_EXCEPT (MP_BC_BASE_JUMP_E + 0x08) // unsigned relative bytecode offset
|
|
|
|
#define MP_BC_SETUP_FINALLY (MP_BC_BASE_JUMP_E + 0x09) // unsigned relative bytecode offset
|
|
|
|
#define MP_BC_POP_EXCEPT_JUMP (MP_BC_BASE_JUMP_E + 0x0a) // unsigned relative bytecode offset
|
|
|
|
#define MP_BC_FOR_ITER (MP_BC_BASE_JUMP_E + 0x0b) // unsigned relative bytecode offset
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_WITH_CLEANUP (MP_BC_BASE_BYTE_O + 0x0c)
|
|
|
|
#define MP_BC_END_FINALLY (MP_BC_BASE_BYTE_O + 0x0d)
|
|
|
|
#define MP_BC_GET_ITER (MP_BC_BASE_BYTE_O + 0x0e)
|
|
|
|
#define MP_BC_GET_ITER_STACK (MP_BC_BASE_BYTE_O + 0x0f)
|
2013-12-21 13:17:45 -05:00
|
|
|
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_BUILD_TUPLE (MP_BC_BASE_VINT_E + 0x0a) // uint
|
|
|
|
#define MP_BC_BUILD_LIST (MP_BC_BASE_VINT_E + 0x0b) // uint
|
|
|
|
#define MP_BC_BUILD_MAP (MP_BC_BASE_VINT_E + 0x0c) // uint
|
|
|
|
#define MP_BC_STORE_MAP (MP_BC_BASE_BYTE_E + 0x02)
|
|
|
|
#define MP_BC_BUILD_SET (MP_BC_BASE_VINT_E + 0x0d) // uint
|
|
|
|
#define MP_BC_BUILD_SLICE (MP_BC_BASE_VINT_E + 0x0e) // uint
|
|
|
|
#define MP_BC_STORE_COMP (MP_BC_BASE_VINT_E + 0x0f) // uint
|
|
|
|
#define MP_BC_UNPACK_SEQUENCE (MP_BC_BASE_VINT_O + 0x00) // uint
|
|
|
|
#define MP_BC_UNPACK_EX (MP_BC_BASE_VINT_O + 0x01) // uint
|
2013-12-21 13:17:45 -05:00
|
|
|
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_RETURN_VALUE (MP_BC_BASE_BYTE_E + 0x03)
|
2019-08-21 22:39:07 -04:00
|
|
|
#define MP_BC_RAISE_LAST (MP_BC_BASE_BYTE_E + 0x04)
|
|
|
|
#define MP_BC_RAISE_OBJ (MP_BC_BASE_BYTE_E + 0x05)
|
|
|
|
#define MP_BC_RAISE_FROM (MP_BC_BASE_BYTE_E + 0x06)
|
|
|
|
#define MP_BC_YIELD_VALUE (MP_BC_BASE_BYTE_E + 0x07)
|
|
|
|
#define MP_BC_YIELD_FROM (MP_BC_BASE_BYTE_E + 0x08)
|
py: Compress load-int, load-fast, store-fast, unop, binop bytecodes.
There is a lot potential in compress bytecodes and make more use of the
coding space. This patch introduces "multi" bytecodes which have their
argument included in the bytecode (by addition).
UNARY_OP and BINARY_OP now no longer take a 1 byte argument for the
opcode. Rather, the opcode is included in the first byte itself.
LOAD_FAST_[0,1,2] and STORE_FAST_[0,1,2] are removed in favour of their
multi versions, which can take an argument between 0 and 15 inclusive.
The majority of LOAD_FAST/STORE_FAST codes fit in this range and so this
saves a byte for each of these.
LOAD_CONST_SMALL_INT_MULTI is used to load small ints between -16 and 47
inclusive. Such ints are quite common and now only need 1 byte to
store, and now have much faster decoding.
In all this patch saves about 2% RAM for typically bytecode (1.8% on
64-bit test, 2.5% on pyboard test). It also reduces the binary size
(because bytecodes are simplified) and doesn't harm performance.
2014-10-25 11:43:46 -04:00
|
|
|
|
2019-09-02 06:24:01 -04:00
|
|
|
#define MP_BC_MAKE_FUNCTION (MP_BC_BASE_VINT_O + 0x02) // uint
|
|
|
|
#define MP_BC_MAKE_FUNCTION_DEFARGS (MP_BC_BASE_VINT_O + 0x03) // uint
|
|
|
|
#define MP_BC_MAKE_CLOSURE (MP_BC_BASE_VINT_E + 0x00) // uint; extra byte
|
|
|
|
#define MP_BC_MAKE_CLOSURE_DEFARGS (MP_BC_BASE_VINT_E + 0x01) // uint; extra byte
|
|
|
|
#define MP_BC_CALL_FUNCTION (MP_BC_BASE_VINT_O + 0x04) // uint
|
|
|
|
#define MP_BC_CALL_FUNCTION_VAR_KW (MP_BC_BASE_VINT_O + 0x05) // uint
|
|
|
|
#define MP_BC_CALL_METHOD (MP_BC_BASE_VINT_O + 0x06) // uint
|
|
|
|
#define MP_BC_CALL_METHOD_VAR_KW (MP_BC_BASE_VINT_O + 0x07) // uint
|
|
|
|
|
|
|
|
#define MP_BC_IMPORT_NAME (MP_BC_BASE_QSTR_O + 0x0b) // qstr
|
|
|
|
#define MP_BC_IMPORT_FROM (MP_BC_BASE_QSTR_O + 0x0c) // qstr
|
2019-08-21 22:39:07 -04:00
|
|
|
#define MP_BC_IMPORT_STAR (MP_BC_BASE_BYTE_E + 0x09)
|
2015-01-01 15:27:54 -05:00
|
|
|
|
all: Unify header guard usage.
The code conventions suggest using header guards, but do not define how
those should look like and instead point to existing files. However, not
all existing files follow the same scheme, sometimes omitting header guards
altogether, sometimes using non-standard names, making it easy to
accidentally pick a "wrong" example.
This commit ensures that all header files of the MicroPython project (that
were not simply copied from somewhere else) follow the same pattern, that
was already present in the majority of files, especially in the py folder.
The rules are as follows.
Naming convention:
* start with the words MICROPY_INCLUDED
* contain the full path to the file
* replace special characters with _
In addition, there are no empty lines before #ifndef, between #ifndef and
one empty line before #endif. #endif is followed by a comment containing
the name of the guard macro.
py/grammar.h cannot use header guards by design, since it has to be
included multiple times in a single C file. Several other files also do not
need header guards as they are only used internally and guaranteed to be
included only once:
* MICROPY_MPHALPORT_H
* mpconfigboard.h
* mpconfigport.h
* mpthreadport.h
* pin_defs_*.h
* qstrdefs*.h
2017-06-29 17:14:58 -04:00
|
|
|
#endif // MICROPY_INCLUDED_PY_BC0_H
|