blob: cc0d61ce96cb1cb530a3b43ea20a1301d281b57a [file] [log] [blame]
Damien George110ba352014-08-28 23:37:02 +01001/*
2 * This file is part of the Micro Python project, http://micropython.org/
3 *
4 * The MIT License (MIT)
5 *
6 * Copyright (c) 2014 Damien P. George
7 *
8 * Permission is hereby granted, free of charge, to any person obtaining a copy
9 * of this software and associated documentation files (the "Software"), to deal
10 * in the Software without restriction, including without limitation the rights
11 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
12 * copies of the Software, and to permit persons to whom the Software is
13 * furnished to do so, subject to the following conditions:
14 *
15 * The above copyright notice and this permission notice shall be included in
16 * all copies or substantial portions of the Software.
17 *
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
21 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
23 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
24 * THE SOFTWARE.
25 */
26
27#include <stdio.h>
28#include <string.h>
29#include <assert.h>
30
Damien George51dfcb42015-01-01 20:27:54 +000031#include "py/nlr.h"
32#include "py/runtime0.h"
33#include "py/runtime.h"
34#include "py/emitglue.h"
Damien George99886182015-04-06 22:38:53 +010035#include "py/bc.h"
Damien George110ba352014-08-28 23:37:02 +010036
Damien George110ba352014-08-28 23:37:02 +010037#if 0 // print debugging info
38#define DEBUG_printf DEBUG_printf
39#else // don't print debugging info
40#define DEBUG_printf(...) (void)0
41#endif
42
Damien George8f54c082016-01-15 15:20:43 +000043#if MICROPY_EMIT_NATIVE
44
Damien George110ba352014-08-28 23:37:02 +010045// convert a Micro Python object to a valid native value based on type
46mp_uint_t mp_convert_obj_to_native(mp_obj_t obj, mp_uint_t type) {
47 DEBUG_printf("mp_convert_obj_to_native(%p, " UINT_FMT ")\n", obj, type);
48 switch (type & 3) {
49 case MP_NATIVE_TYPE_OBJ: return (mp_uint_t)obj;
50 case MP_NATIVE_TYPE_BOOL:
Damien Georgedd0a0f72016-01-07 16:48:20 +000051 case MP_NATIVE_TYPE_INT: return mp_obj_get_int_truncated(obj);
Damien Georgee9dac3b2014-09-29 22:10:41 +010052 case MP_NATIVE_TYPE_UINT: {
53 mp_buffer_info_t bufinfo;
54 if (mp_get_buffer(obj, &bufinfo, MP_BUFFER_RW)) {
55 return (mp_uint_t)bufinfo.buf;
56 } else {
Damien Georgedd0a0f72016-01-07 16:48:20 +000057 return mp_obj_get_int_truncated(obj);
Damien Georgee9dac3b2014-09-29 22:10:41 +010058 }
59 }
Damien George110ba352014-08-28 23:37:02 +010060 default: assert(0); return 0;
61 }
62}
63
Damien George8f54c082016-01-15 15:20:43 +000064#endif
65
66#if MICROPY_EMIT_NATIVE || MICROPY_EMIT_INLINE_THUMB
67
Damien George110ba352014-08-28 23:37:02 +010068// convert a native value to a Micro Python object based on type
69mp_obj_t mp_convert_native_to_obj(mp_uint_t val, mp_uint_t type) {
70 DEBUG_printf("mp_convert_native_to_obj(" UINT_FMT ", " UINT_FMT ")\n", val, type);
71 switch (type & 3) {
72 case MP_NATIVE_TYPE_OBJ: return (mp_obj_t)val;
Paul Sokolovsky1b586f32015-10-11 12:09:43 +030073 case MP_NATIVE_TYPE_BOOL: return mp_obj_new_bool(val);
Damien George110ba352014-08-28 23:37:02 +010074 case MP_NATIVE_TYPE_INT: return mp_obj_new_int(val);
75 case MP_NATIVE_TYPE_UINT: return mp_obj_new_int_from_uint(val);
76 default: assert(0); return mp_const_none;
77 }
78}
79
Damien George8f54c082016-01-15 15:20:43 +000080#endif
81
82#if MICROPY_EMIT_NATIVE
83
Damien George110ba352014-08-28 23:37:02 +010084// wrapper that accepts n_args and n_kw in one argument
85// (native emitter can only pass at most 3 arguments to a function)
Damien George4abff752014-08-30 14:59:21 +010086mp_obj_t mp_native_call_function_n_kw(mp_obj_t fun_in, mp_uint_t n_args_kw, const mp_obj_t *args) {
Damien George110ba352014-08-28 23:37:02 +010087 return mp_call_function_n_kw(fun_in, n_args_kw & 0xff, (n_args_kw >> 8) & 0xff, args);
88}
89
90// wrapper that makes raise obj and raises it
Damien Georgeb6e6b522015-01-21 17:00:01 +000091// END_FINALLY opcode requires that we don't raise if o==None
92void mp_native_raise(mp_obj_t o) {
93 if (o != mp_const_none) {
94 nlr_raise(mp_make_raise_obj(o));
95 }
Damien George110ba352014-08-28 23:37:02 +010096}
97
98// these must correspond to the respective enum in runtime0.h
99void *const mp_fun_table[MP_F_NUMBER_OF] = {
100 mp_convert_obj_to_native,
101 mp_convert_native_to_obj,
Damien George110ba352014-08-28 23:37:02 +0100102 mp_load_name,
103 mp_load_global,
104 mp_load_build_class,
105 mp_load_attr,
106 mp_load_method,
107 mp_store_name,
108 mp_store_global,
109 mp_store_attr,
110 mp_obj_subscr,
111 mp_obj_is_true,
112 mp_unary_op,
113 mp_binary_op,
114 mp_obj_new_tuple,
115 mp_obj_new_list,
116 mp_obj_list_append,
117 mp_obj_new_dict,
118 mp_obj_dict_store,
119#if MICROPY_PY_BUILTINS_SET
120 mp_obj_new_set,
121 mp_obj_set_store,
122#endif
123 mp_make_function_from_raw_code,
124 mp_native_call_function_n_kw,
125 mp_call_method_n_kw,
Damien George78772ad2015-04-06 22:48:21 +0100126 mp_call_method_n_kw_var,
Damien George110ba352014-08-28 23:37:02 +0100127 mp_getiter,
128 mp_iternext,
129 nlr_push,
130 nlr_pop,
131 mp_native_raise,
132 mp_import_name,
133 mp_import_from,
134 mp_import_all,
135#if MICROPY_PY_BUILTINS_SLICE
136 mp_obj_new_slice,
137#endif
138 mp_unpack_sequence,
139 mp_unpack_ex,
Damien Georgee6ce10a2014-09-06 18:38:20 +0100140 mp_delete_name,
141 mp_delete_global,
Damien George4cd9ced2015-01-15 14:41:41 +0000142 mp_obj_new_cell,
143 mp_make_closure_from_raw_code,
Damien George99886182015-04-06 22:38:53 +0100144 mp_setup_code_state,
Damien George110ba352014-08-28 23:37:02 +0100145};
146
147/*
148void mp_f_vector(mp_fun_kind_t fun_kind) {
149 (mp_f_table[fun_kind])();
150}
151*/
152
153#endif // MICROPY_EMIT_NATIVE