Skip to content

Commit b6a9778

Browse files
jimmodpgeorge
authored andcommitted
py/misc: Change sizeof to offsetof for variable-length alloc.
This fixes the case where e.g. struct foo_t { mp_obj_t x; uint16_t y; char buf[]; }; will have `sizeof(struct foo_t)==8`, but `offsetof(struct foo_t, buf)==6`. When computing the size to allocate for `m_new_obj_var` we need to use offsetof to avoid over-allocating. This is important especially when it might cause it to spill over into another GC block. This work was funded through GitHub Sponsors. Signed-off-by: Jim Mussared <[email protected]>
1 parent c85db05 commit b6a9778

File tree

11 files changed

+24
-24
lines changed

11 files changed

+24
-24
lines changed

extmod/modre.c

+2-2
Original file line numberDiff line numberDiff line change
@@ -207,12 +207,12 @@ STATIC mp_obj_t re_exec(bool is_anchored, uint n_args, const mp_obj_t *args) {
207207
subj.begin_line = subj.begin = mp_obj_str_get_data(args[1], &len);
208208
subj.end = subj.begin + len;
209209
int caps_num = (self->re.sub + 1) * 2;
210-
mp_obj_match_t *match = m_new_obj_var(mp_obj_match_t, char *, caps_num);
210+
mp_obj_match_t *match = m_new_obj_var(mp_obj_match_t, caps, char *, caps_num);
211211
// cast is a workaround for a bug in msvc: it treats const char** as a const pointer instead of a pointer to pointer to const char
212212
memset((char *)match->caps, 0, caps_num * sizeof(char *));
213213
int res = re1_5_recursiveloopprog(&self->re, &subj, match->caps, caps_num, is_anchored);
214214
if (res == 0) {
215-
m_del_var(mp_obj_match_t, char *, caps_num, match);
215+
m_del_var(mp_obj_match_t, caps, char *, caps_num, match);
216216
return mp_const_none;
217217
}
218218

extmod/nimble/modbluetooth_nimble.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -1701,7 +1701,7 @@ STATIC int create_l2cap_channel(uint16_t mtu, mp_bluetooth_nimble_l2cap_channel_
17011701
// multiply that by the "MTUs per channel" (set to 3 above).
17021702
const size_t buf_blocks = MP_CEIL_DIVIDE(mtu, L2CAP_BUF_BLOCK_SIZE) * L2CAP_BUF_SIZE_MTUS_PER_CHANNEL;
17031703

1704-
mp_bluetooth_nimble_l2cap_channel_t *chan = m_new_obj_var(mp_bluetooth_nimble_l2cap_channel_t, uint8_t, OS_MEMPOOL_SIZE(buf_blocks, L2CAP_BUF_BLOCK_SIZE) * sizeof(os_membuf_t));
1704+
mp_bluetooth_nimble_l2cap_channel_t *chan = m_new_obj_var(mp_bluetooth_nimble_l2cap_channel_t, sdu_mem, uint8_t, OS_MEMPOOL_SIZE(buf_blocks, L2CAP_BUF_BLOCK_SIZE) * sizeof(os_membuf_t));
17051705
MP_STATE_PORT(bluetooth_nimble_root_pointers)->l2cap_chan = chan;
17061706

17071707
// Will be set in BLE_L2CAP_EVENT_COC_CONNECTED or BLE_L2CAP_EVENT_COC_ACCEPT.

extmod/vfs_lfsx_file.c

+2-2
Original file line numberDiff line numberDiff line change
@@ -90,9 +90,9 @@ mp_obj_t MP_VFS_LFSx(file_open)(mp_obj_t self_in, mp_obj_t path_in, mp_obj_t mod
9090
}
9191

9292
#if LFS_BUILD_VERSION == 1
93-
MP_OBJ_VFS_LFSx_FILE *o = m_new_obj_var_with_finaliser(MP_OBJ_VFS_LFSx_FILE, uint8_t, self->lfs.cfg->prog_size);
93+
MP_OBJ_VFS_LFSx_FILE *o = m_new_obj_var_with_finaliser(MP_OBJ_VFS_LFSx_FILE, file_buffer, uint8_t, self->lfs.cfg->prog_size);
9494
#else
95-
MP_OBJ_VFS_LFSx_FILE *o = m_new_obj_var_with_finaliser(MP_OBJ_VFS_LFSx_FILE, uint8_t, self->lfs.cfg->cache_size);
95+
MP_OBJ_VFS_LFSx_FILE *o = m_new_obj_var_with_finaliser(MP_OBJ_VFS_LFSx_FILE, file_buffer, uint8_t, self->lfs.cfg->cache_size);
9696
#endif
9797
o->base.type = type;
9898
o->vfs = self;

ports/unix/coverage.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -545,7 +545,7 @@ STATIC mp_obj_t extra_coverage(void) {
545545
fun_bc.context = &context;
546546
fun_bc.child_table = NULL;
547547
fun_bc.bytecode = (const byte *)"\x01"; // just needed for n_state
548-
mp_code_state_t *code_state = m_new_obj_var(mp_code_state_t, mp_obj_t, 1);
548+
mp_code_state_t *code_state = m_new_obj_var(mp_code_state_t, state, mp_obj_t, 1);
549549
code_state->fun_bc = &fun_bc;
550550
code_state->ip = (const byte *)"\x00"; // just needed for an invalid opcode
551551
code_state->sp = &code_state->state[0];

py/misc.h

+7-7
Original file line numberDiff line numberDiff line change
@@ -71,26 +71,26 @@ typedef unsigned int uint;
7171
#define m_new0(type, num) ((type *)(m_malloc0(sizeof(type) * (num))))
7272
#define m_new_obj(type) (m_new(type, 1))
7373
#define m_new_obj_maybe(type) (m_new_maybe(type, 1))
74-
#define m_new_obj_var(obj_type, var_type, var_num) ((obj_type *)m_malloc(sizeof(obj_type) + sizeof(var_type) * (var_num)))
75-
#define m_new_obj_var0(obj_type, var_type, var_num) ((obj_type *)m_malloc0(sizeof(obj_type) + sizeof(var_type) * (var_num)))
76-
#define m_new_obj_var_maybe(obj_type, var_type, var_num) ((obj_type *)m_malloc_maybe(sizeof(obj_type) + sizeof(var_type) * (var_num)))
74+
#define m_new_obj_var(obj_type, var_field, var_type, var_num) ((obj_type *)m_malloc(offsetof(obj_type, var_field) + sizeof(var_type) * (var_num)))
75+
#define m_new_obj_var0(obj_type, var_field, var_type, var_num) ((obj_type *)m_malloc0(offsetof(obj_type, var_field) + sizeof(var_type) * (var_num)))
76+
#define m_new_obj_var_maybe(obj_type, var_field, var_type, var_num) ((obj_type *)m_malloc_maybe(offsetof(obj_type, var_field) + sizeof(var_type) * (var_num)))
7777
#if MICROPY_ENABLE_FINALISER
7878
#define m_new_obj_with_finaliser(type) ((type *)(m_malloc_with_finaliser(sizeof(type))))
79-
#define m_new_obj_var_with_finaliser(type, var_type, var_num) ((type *)m_malloc_with_finaliser(sizeof(type) + sizeof(var_type) * (var_num)))
79+
#define m_new_obj_var_with_finaliser(type, var_field, var_type, var_num) ((type *)m_malloc_with_finaliser(offsetof(type, var_field) + sizeof(var_type) * (var_num)))
8080
#else
8181
#define m_new_obj_with_finaliser(type) m_new_obj(type)
82-
#define m_new_obj_var_with_finaliser(type, var_type, var_num) m_new_obj_var(type, var_type, var_num)
82+
#define m_new_obj_var_with_finaliser(type, var_field, var_type, var_num) m_new_obj_var(type, var_field, var_type, var_num)
8383
#endif
8484
#if MICROPY_MALLOC_USES_ALLOCATED_SIZE
8585
#define m_renew(type, ptr, old_num, new_num) ((type *)(m_realloc((ptr), sizeof(type) * (old_num), sizeof(type) * (new_num))))
8686
#define m_renew_maybe(type, ptr, old_num, new_num, allow_move) ((type *)(m_realloc_maybe((ptr), sizeof(type) * (old_num), sizeof(type) * (new_num), (allow_move))))
8787
#define m_del(type, ptr, num) m_free(ptr, sizeof(type) * (num))
88-
#define m_del_var(obj_type, var_type, var_num, ptr) (m_free(ptr, sizeof(obj_type) + sizeof(var_type) * (var_num)))
88+
#define m_del_var(obj_type, var_field, var_type, var_num, ptr) (m_free(ptr, offsetof(obj_type, var_field) + sizeof(var_type) * (var_num)))
8989
#else
9090
#define m_renew(type, ptr, old_num, new_num) ((type *)(m_realloc((ptr), sizeof(type) * (new_num))))
9191
#define m_renew_maybe(type, ptr, old_num, new_num, allow_move) ((type *)(m_realloc_maybe((ptr), sizeof(type) * (new_num), (allow_move))))
9292
#define m_del(type, ptr, num) ((void)(num), m_free(ptr))
93-
#define m_del_var(obj_type, var_type, var_num, ptr) ((void)(var_num), m_free(ptr))
93+
#define m_del_var(obj_type, var_field, var_type, var_num, ptr) ((void)(var_num), m_free(ptr))
9494
#endif
9595
#define m_del_obj(type, ptr) (m_del(type, ptr, 1))
9696

py/modthread.c

+2-2
Original file line numberDiff line numberDiff line change
@@ -235,15 +235,15 @@ STATIC mp_obj_t mod_thread_start_new_thread(size_t n_args, const mp_obj_t *args)
235235
// check for keyword arguments
236236
if (n_args == 2) {
237237
// just position arguments
238-
th_args = m_new_obj_var(thread_entry_args_t, mp_obj_t, pos_args_len);
238+
th_args = m_new_obj_var(thread_entry_args_t, args, mp_obj_t, pos_args_len);
239239
th_args->n_kw = 0;
240240
} else {
241241
// positional and keyword arguments
242242
if (mp_obj_get_type(args[2]) != &mp_type_dict) {
243243
mp_raise_TypeError(MP_ERROR_TEXT("expecting a dict for keyword args"));
244244
}
245245
mp_map_t *map = &((mp_obj_dict_t *)MP_OBJ_TO_PTR(args[2]))->map;
246-
th_args = m_new_obj_var(thread_entry_args_t, mp_obj_t, pos_args_len + 2 * map->used);
246+
th_args = m_new_obj_var(thread_entry_args_t, args, mp_obj_t, pos_args_len + 2 * map->used);
247247
th_args->n_kw = map->used;
248248
// copy across the keyword arguments
249249
for (size_t i = 0, n = pos_args_len; i < map->alloc; ++i) {

py/objexcept.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -221,7 +221,7 @@ mp_obj_t mp_obj_exception_make_new(const mp_obj_type_t *type, size_t n_args, siz
221221
o_tuple = (mp_obj_tuple_t *)&mp_const_empty_tuple_obj;
222222
} else {
223223
// Try to allocate memory for the tuple containing the args
224-
o_tuple = m_new_obj_var_maybe(mp_obj_tuple_t, mp_obj_t, n_args);
224+
o_tuple = m_new_obj_var_maybe(mp_obj_tuple_t, items, mp_obj_t, n_args);
225225

226226
#if MICROPY_ENABLE_EMERGENCY_EXCEPTION_BUF
227227
// If we are called by mp_obj_new_exception_msg_varg then it will have

py/objfun.c

+5-5
Original file line numberDiff line numberDiff line change
@@ -215,7 +215,7 @@ mp_code_state_t *mp_obj_fun_bc_prepare_codestate(mp_obj_t self_in, size_t n_args
215215
// RuntimeError should be raised instead. So, we use m_new_obj_var_maybe(),
216216
// return NULL, then vm.c takes the needed action (either raise
217217
// RuntimeError or fallback to stack allocation).
218-
code_state = m_new_obj_var_maybe(mp_code_state_t, byte, state_size);
218+
code_state = m_new_obj_var_maybe(mp_code_state_t, state, byte, state_size);
219219
if (!code_state) {
220220
return NULL;
221221
}
@@ -247,18 +247,18 @@ STATIC mp_obj_t fun_bc_call(mp_obj_t self_in, size_t n_args, size_t n_kw, const
247247
// allocate state for locals and stack
248248
mp_code_state_t *code_state = NULL;
249249
#if MICROPY_ENABLE_PYSTACK
250-
code_state = mp_pystack_alloc(sizeof(mp_code_state_t) + state_size);
250+
code_state = mp_pystack_alloc(offsetof(mp_code_state_t, state) + state_size);
251251
#else
252252
if (state_size > VM_MAX_STATE_ON_STACK) {
253-
code_state = m_new_obj_var_maybe(mp_code_state_t, byte, state_size);
253+
code_state = m_new_obj_var_maybe(mp_code_state_t, state, byte, state_size);
254254
#if MICROPY_DEBUG_VM_STACK_OVERFLOW
255255
if (code_state != NULL) {
256256
memset(code_state->state, 0, state_size);
257257
}
258258
#endif
259259
}
260260
if (code_state == NULL) {
261-
code_state = alloca(sizeof(mp_code_state_t) + state_size);
261+
code_state = alloca(offsetof(mp_code_state_t, state) + state_size);
262262
#if MICROPY_DEBUG_VM_STACK_OVERFLOW
263263
memset(code_state->state, 0, state_size);
264264
#endif
@@ -320,7 +320,7 @@ STATIC mp_obj_t fun_bc_call(mp_obj_t self_in, size_t n_args, size_t n_kw, const
320320
#else
321321
// free the state if it was allocated on the heap
322322
if (state_size != 0) {
323-
m_del_var(mp_code_state_t, byte, state_size, code_state);
323+
m_del_var(mp_code_state_t, state, byte, state_size, code_state);
324324
}
325325
#endif
326326

py/objnamedtuple.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -143,7 +143,7 @@ STATIC mp_obj_t namedtuple_make_new(const mp_obj_type_t *type_in, size_t n_args,
143143
}
144144

145145
mp_obj_namedtuple_type_t *mp_obj_new_namedtuple_base(size_t n_fields, mp_obj_t *fields) {
146-
mp_obj_namedtuple_type_t *o = m_new_obj_var0(mp_obj_namedtuple_type_t, qstr, n_fields);
146+
mp_obj_namedtuple_type_t *o = m_new_obj_var0(mp_obj_namedtuple_type_t, fields, qstr, n_fields);
147147
o->n_fields = n_fields;
148148
for (size_t i = 0; i < n_fields; i++) {
149149
o->fields[i] = mp_obj_str_get_qstr(fields[i]);

py/objtuple.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -264,7 +264,7 @@ void mp_obj_tuple_get(mp_obj_t self_in, size_t *len, mp_obj_t **items) {
264264
void mp_obj_tuple_del(mp_obj_t self_in) {
265265
assert(mp_obj_is_type(self_in, &mp_type_tuple));
266266
mp_obj_tuple_t *self = MP_OBJ_TO_PTR(self_in);
267-
m_del_var(mp_obj_tuple_t, mp_obj_t, self->len, self);
267+
m_del_var(mp_obj_tuple_t, items, mp_obj_t, self->len, self);
268268
}
269269

270270
/******************************************************************************/

py/objtype.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -1155,7 +1155,7 @@ mp_obj_t mp_obj_new_type(qstr name, mp_obj_t bases_tuple, mp_obj_t locals_dict)
11551155
// (currently 10, plus 1 for base, plus 1 for base-protocol).
11561156
// Note: mp_obj_type_t is (2 + 3 + #slots) words, so going from 11 to 12 slots
11571157
// moves from 4 to 5 gc blocks.
1158-
mp_obj_type_t *o = m_new_obj_var0(mp_obj_type_t, void *, 10 + (bases_len ? 1 : 0) + (base_protocol ? 1 : 0));
1158+
mp_obj_type_t *o = m_new_obj_var0(mp_obj_type_t, slots, void *, 10 + (bases_len ? 1 : 0) + (base_protocol ? 1 : 0));
11591159
o->base.type = &mp_type_type;
11601160
o->flags = base_flags;
11611161
o->name = name;

0 commit comments

Comments
 (0)