Numworks Epsilon  1.4.1
Graphing Calculator Operating System
emitglue.c
Go to the documentation of this file.
1 /*
2  * This file is part of the MicroPython project, http://micropython.org/
3  *
4  * The MIT License (MIT)
5  *
6  * Copyright (c) 2013, 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 // This code glues the code emitters to the runtime.
28 
29 #include <stdint.h>
30 #include <stdio.h>
31 #include <string.h>
32 #include <assert.h>
33 
34 #include "py/emitglue.h"
35 #include "py/runtime0.h"
36 #include "py/bc.h"
37 
38 #if MICROPY_DEBUG_VERBOSE // print debugging info
39 #define DEBUG_PRINT (1)
40 #define WRITE_CODE (1)
41 #define DEBUG_printf DEBUG_printf
42 #define DEBUG_OP_printf(...) DEBUG_printf(__VA_ARGS__)
43 #else // don't print debugging info
44 #define DEBUG_printf(...) (void)0
45 #define DEBUG_OP_printf(...) (void)0
46 #endif
47 
48 #if MICROPY_DEBUG_PRINTERS
50 #endif
51 
54  rc->kind = MP_CODE_RESERVED;
55  return rc;
56 }
57 
59  const mp_uint_t *const_table,
61  uint16_t n_obj, uint16_t n_raw_code,
62  #endif
63  mp_uint_t scope_flags) {
64 
65  rc->kind = MP_CODE_BYTECODE;
66  rc->scope_flags = scope_flags;
67  rc->data.u_byte.bytecode = code;
68  rc->data.u_byte.const_table = const_table;
69  #if MICROPY_PERSISTENT_CODE_SAVE
70  rc->data.u_byte.bc_len = len;
71  rc->data.u_byte.n_obj = n_obj;
72  rc->data.u_byte.n_raw_code = n_raw_code;
73  #endif
74 
75 #ifdef DEBUG_PRINT
76  DEBUG_printf("assign byte code: code=%p len=" UINT_FMT " flags=%x\n", code, len, (uint)scope_flags);
77 #endif
78 #if MICROPY_DEBUG_PRINTERS
79  if (mp_verbose_flag >= 2) {
80  mp_bytecode_print(rc, code, len, const_table);
81  }
82 #endif
83 }
84 
85 #if MICROPY_EMIT_NATIVE || MICROPY_EMIT_INLINE_ASM
86 void mp_emit_glue_assign_native(mp_raw_code_t *rc, mp_raw_code_kind_t kind, void *fun_data, mp_uint_t fun_len, const mp_uint_t *const_table, mp_uint_t n_pos_args, mp_uint_t scope_flags, mp_uint_t type_sig) {
87  assert(kind == MP_CODE_NATIVE_PY || kind == MP_CODE_NATIVE_VIPER || kind == MP_CODE_NATIVE_ASM);
88  rc->kind = kind;
89  rc->scope_flags = scope_flags;
90  rc->n_pos_args = n_pos_args;
91  rc->data.u_native.fun_data = fun_data;
92  rc->data.u_native.const_table = const_table;
93  rc->data.u_native.type_sig = type_sig;
94 
95 #ifdef DEBUG_PRINT
96  DEBUG_printf("assign native: kind=%d fun=%p len=" UINT_FMT " n_pos_args=" UINT_FMT " flags=%x\n", kind, fun_data, fun_len, n_pos_args, (uint)scope_flags);
97  for (mp_uint_t i = 0; i < fun_len; i++) {
98  if (i > 0 && i % 16 == 0) {
99  DEBUG_printf("\n");
100  }
101  DEBUG_printf(" %02x", ((byte*)fun_data)[i]);
102  }
103  DEBUG_printf("\n");
104 
105 #ifdef WRITE_CODE
106  FILE *fp_write_code = fopen("out-code", "wb");
107  fwrite(fun_data, fun_len, 1, fp_write_code);
108  fclose(fp_write_code);
109 #endif
110 #else
111  (void)fun_len;
112 #endif
113 }
114 #endif
115 
117  DEBUG_OP_printf("make_function_from_raw_code %p\n", rc);
118  assert(rc != NULL);
119 
120  // def_args must be MP_OBJ_NULL or a tuple
121  assert(def_args == MP_OBJ_NULL || MP_OBJ_IS_TYPE(def_args, &mp_type_tuple));
122 
123  // def_kw_args must be MP_OBJ_NULL or a dict
124  assert(def_kw_args == MP_OBJ_NULL || MP_OBJ_IS_TYPE(def_kw_args, &mp_type_dict));
125 
126  // make the function, depending on the raw code kind
127  mp_obj_t fun;
128  switch (rc->kind) {
129  #if MICROPY_EMIT_NATIVE
130  case MP_CODE_NATIVE_PY:
131  fun = mp_obj_new_fun_native(def_args, def_kw_args, rc->data.u_native.fun_data, rc->data.u_native.const_table);
132  break;
134  fun = mp_obj_new_fun_viper(rc->n_pos_args, rc->data.u_native.fun_data, rc->data.u_native.type_sig);
135  break;
136  #endif
137  #if MICROPY_EMIT_INLINE_ASM
138  case MP_CODE_NATIVE_ASM:
139  fun = mp_obj_new_fun_asm(rc->n_pos_args, rc->data.u_native.fun_data, rc->data.u_native.type_sig);
140  break;
141  #endif
142  default:
143  // rc->kind should always be set and BYTECODE is the only remaining case
144  assert(rc->kind == MP_CODE_BYTECODE);
145  fun = mp_obj_new_fun_bc(def_args, def_kw_args, rc->data.u_byte.bytecode, rc->data.u_byte.const_table);
146  break;
147  }
148 
149  // check for generator functions and if so wrap in generator object
150  if ((rc->scope_flags & MP_SCOPE_FLAG_GENERATOR) != 0) {
151  fun = mp_obj_new_gen_wrap(fun);
152  }
153 
154  return fun;
155 }
156 
158  DEBUG_OP_printf("make_closure_from_raw_code %p " UINT_FMT " %p\n", rc, n_closed_over, args);
159  // make function object
160  mp_obj_t ffun;
161  if (n_closed_over & 0x100) {
162  // default positional and keyword args given
163  ffun = mp_make_function_from_raw_code(rc, args[0], args[1]);
164  } else {
165  // default positional and keyword args not given
167  }
168  // wrap function in closure object
169  return mp_obj_new_closure(ffun, n_closed_over & 0xff, args + ((n_closed_over >> 7) & 2));
170 }
mp_raw_code_kind_t
Definition: emitglue.h:33
uintptr_t mp_uint_t
Definition: mpconfigport.h:74
mp_obj_t mp_obj_new_closure(mp_obj_t fun, size_t n_closed, const mp_obj_t *closed)
Definition: objclosure.c:90
union _mp_raw_code_t::@15 data
struct _mp_raw_code_t::@15::@16 u_byte
#define assert(e)
Definition: assert.h:9
#define MP_OBJ_IS_TYPE(o, t)
Definition: obj.h:254
mp_obj_t mp_obj_new_gen_wrap(mp_obj_t fun)
Definition: objgenerator.c:78
unsigned short uint16_t
Definition: stdint.h:5
#define MP_SCOPE_FLAG_GENERATOR
Definition: runtime0.h:32
struct _mp_raw_code_t::@15::@17 u_native
#define DEBUG_OP_printf(...)
Definition: emitglue.c:45
void mp_emit_glue_assign_native(mp_raw_code_t *rc, mp_raw_code_kind_t kind, void *fun_data, mp_uint_t fun_len, const mp_uint_t *const_table, mp_uint_t n_pos_args, mp_uint_t scope_flags, mp_uint_t type_sig)
mp_obj_t mp_obj_new_fun_asm(size_t n_args, void *fun_data, mp_uint_t type_sig)
#define NULL
Definition: stddef.h:4
#define MP_OBJ_NULL
Definition: obj.h:73
mp_obj_t mp_make_closure_from_raw_code(const mp_raw_code_t *rc, mp_uint_t n_closed_over, const mp_obj_t *args)
Definition: emitglue.c:157
#define UINT_FMT
Definition: mpconfigport.h:71
mp_obj_t mp_make_function_from_raw_code(const mp_raw_code_t *rc, mp_obj_t def_args, mp_obj_t def_kw_args)
Definition: emitglue.c:116
mp_obj_t mp_obj_new_fun_viper(size_t n_args, void *fun_data, mp_uint_t type_sig)
mp_uint_t n_pos_args
Definition: emitglue.h:45
args
Definition: i18n.py:175
mp_uint_t mp_verbose_flag
mp_uint_t scope_flags
Definition: emitglue.h:44
void mp_bytecode_print(const void *descr, const byte *code, mp_uint_t len, const mp_uint_t *const_table)
unsigned char byte
Definition: misc.h:37
#define m_new0(type, num)
Definition: misc.h:59
mp_raw_code_t * mp_emit_glue_new_raw_code(void)
Definition: emitglue.c:52
void FILE
Definition: stdio.h:8
mp_obj_t mp_obj_new_fun_native(mp_obj_t def_args_in, mp_obj_t def_kw_args, const void *fun_data, const mp_uint_t *const_table)
const mp_obj_type_t mp_type_tuple
Definition: objtuple.c:220
const mp_obj_type_t mp_type_dict
Definition: objdict.c:552
uint64_t mp_obj_t
Definition: obj.h:39
void mp_emit_glue_assign_bytecode(mp_raw_code_t *rc, const byte *code, mp_uint_t len, const mp_uint_t *const_table, mp_uint_t scope_flags)
Definition: emitglue.c:58
#define DEBUG_printf(...)
Definition: emitglue.c:44
#define MICROPY_PERSISTENT_CODE_SAVE
Definition: mpconfig.h:246
mp_obj_t mp_obj_new_fun_bc(mp_obj_t def_args, mp_obj_t def_kw_args, const byte *code, const mp_uint_t *const_table)
Definition: objfun.c:344
mp_raw_code_kind_t kind
Definition: emitglue.h:43
unsigned int uint
Definition: misc.h:38