diff options
author | Damien George <damien.p.george@gmail.com> | 2014-04-05 18:49:39 +0100 |
---|---|---|
committer | Damien George <damien.p.george@gmail.com> | 2014-04-05 18:49:39 +0100 |
commit | 8123a3339dc7381a079afa90649ce140075eb2e2 (patch) | |
tree | b41d4337c7a9e234827201be8e717b32a2ac26fe /py | |
parent | ea13f407a392593e7746131952a57bad222ee882 (diff) | |
parent | cc849f70f478f7442d19ed2f4c0b4297035b2393 (diff) | |
download | micropython-8123a3339dc7381a079afa90649ce140075eb2e2.tar.gz micropython-8123a3339dc7381a079afa90649ce140075eb2e2.zip |
Merge pull request #425 from iabdalkader/del
Implement del
Diffstat (limited to 'py')
-rw-r--r-- | py/gc.c | 46 | ||||
-rw-r--r-- | py/gc.h | 1 | ||||
-rw-r--r-- | py/malloc.c | 19 | ||||
-rw-r--r-- | py/misc.h | 2 | ||||
-rw-r--r-- | py/qstrdefs.h | 1 |
5 files changed, 67 insertions, 2 deletions
@@ -1,10 +1,16 @@ #include <stdio.h> #include <string.h> +#include <stdbool.h> #include "mpconfig.h" #include "misc.h" #include "gc.h" +#include "misc.h" +#include "qstr.h" +#include "obj.h" +#include "runtime.h" + #if MICROPY_ENABLE_GC #if 0 // print debugging info @@ -21,7 +27,9 @@ typedef unsigned char byte; #define STACK_SIZE (64) // tunable; minimum is 1 STATIC byte *gc_alloc_table_start; +STATIC byte *gc_mpobj_table_start; STATIC machine_uint_t gc_alloc_table_byte_len; +STATIC machine_uint_t gc_mpobj_table_byte_len; STATIC machine_uint_t *gc_pool_start; STATIC machine_uint_t *gc_pool_end; @@ -59,6 +67,10 @@ STATIC machine_uint_t *gc_sp; #define ATB_HEAD_TO_MARK(block) do { gc_alloc_table_start[(block) / BLOCKS_PER_ATB] |= (AT_MARK << BLOCK_SHIFT(block)); } while (0) #define ATB_MARK_TO_HEAD(block) do { gc_alloc_table_start[(block) / BLOCKS_PER_ATB] &= (~(AT_TAIL << BLOCK_SHIFT(block))); } while (0) +#define ATB_SET_MPOBJ(block) do { gc_mpobj_table_start[(block) / 8] |= (1<<(block%8)); } while (0) +#define ATB_CLR_MPOBJ(block) do { gc_mpobj_table_start[(block) / 8] &= (~(1<<(block%8))); } while (0) +#define ATB_IS_MPOBJ(block) ((gc_mpobj_table_start[(block) / 8]>>(block%8))&0x01) + #define BLOCK_FROM_PTR(ptr) (((ptr) - (machine_uint_t)gc_pool_start) / BYTES_PER_BLOCK) #define PTR_FROM_BLOCK(block) (((block) * BYTES_PER_BLOCK + (machine_uint_t)gc_pool_start)) #define ATB_FROM_BLOCK(bl) ((bl) / BLOCKS_PER_ATB) @@ -73,7 +85,11 @@ void gc_init(void *start, void *end) { machine_uint_t total_word_len = (machine_uint_t*)end - (machine_uint_t*)start; gc_alloc_table_byte_len = total_word_len * BYTES_PER_WORD / (1 + BITS_PER_BYTE / 2 * BYTES_PER_BLOCK); gc_alloc_table_start = (byte*)start; - machine_uint_t gc_pool_block_len = gc_alloc_table_byte_len * BITS_PER_BYTE / 2; + + gc_mpobj_table_byte_len = (gc_alloc_table_byte_len * BITS_PER_BYTE / 2)/8; + gc_mpobj_table_start = gc_alloc_table_start+gc_alloc_table_byte_len; + + machine_uint_t gc_pool_block_len = (gc_alloc_table_byte_len * BITS_PER_BYTE / 2) -(gc_mpobj_table_byte_len / BYTES_PER_BLOCK); machine_uint_t gc_pool_word_len = gc_pool_block_len * WORDS_PER_BLOCK; gc_pool_start = (machine_uint_t*)end - gc_pool_word_len; gc_pool_end = end; @@ -81,6 +97,9 @@ void gc_init(void *start, void *end) { // clear ATBs memset(gc_alloc_table_start, 0, gc_alloc_table_byte_len); + // clear MPOBJ flags + memset(gc_mpobj_table_start, 0, gc_mpobj_table_byte_len); + // allocate first block because gc_pool_start points there and it will never // be freed, so allocating 1 block with null pointers will minimise memory loss ATB_FREE_TO_HEAD(0); @@ -157,6 +176,16 @@ STATIC void gc_sweep(void) { for (machine_uint_t block = 0; block < gc_alloc_table_byte_len * BLOCKS_PER_ATB; block++) { switch (ATB_GET_KIND(block)) { case AT_HEAD: + if (ATB_IS_MPOBJ(block)) { + mp_obj_t dest[2]; + mp_load_method((mp_obj_t*)PTR_FROM_BLOCK(block), MP_QSTR___del__, dest); + // load_method returned a method + if (dest[1] != MP_OBJ_NULL) { + mp_call_method_n_kw(0, 0, dest); + } + // clear mpobj flag + ATB_CLR_MPOBJ(block); + } free_tail = 1; // fall through to free the head @@ -237,7 +266,7 @@ void gc_info(gc_info_t *info) { info->free *= BYTES_PER_BLOCK; } -void *gc_alloc(machine_uint_t n_bytes) { +void *_gc_alloc(machine_uint_t n_bytes, bool is_mpobj) { machine_uint_t n_blocks = ((n_bytes + BYTES_PER_BLOCK - 1) & (~(BYTES_PER_BLOCK - 1))) / BYTES_PER_BLOCK; DEBUG_printf("gc_alloc(" UINT_FMT " bytes -> " UINT_FMT " blocks)\n", n_bytes, n_blocks); @@ -286,10 +315,23 @@ found: ATB_FREE_TO_TAIL(bl); } + if (is_mpobj) { + // set mp_obj flag only if it has del + ATB_SET_MPOBJ(start_block); + } + // return pointer to first block return (void*)(gc_pool_start + start_block * WORDS_PER_BLOCK); } +void *gc_alloc(machine_uint_t n_bytes) { + return _gc_alloc(n_bytes, false); +} + +void *gc_alloc_mp_obj(machine_uint_t n_bytes) { + return _gc_alloc(n_bytes, true); +} + // force the freeing of a piece of memory void gc_free(void *ptr_in) { machine_uint_t ptr = (machine_uint_t)ptr_in; @@ -4,6 +4,7 @@ void gc_collect_root(void **ptrs, machine_uint_t len); void gc_collect_end(void); void gc_collect(void); void *gc_alloc(machine_uint_t n_bytes); +void *gc_alloc_mp_obj(machine_uint_t n_bytes); void gc_free(void *ptr); machine_uint_t gc_nbytes(void *ptr); void *gc_realloc(void *ptr, machine_uint_t n_bytes); diff --git a/py/malloc.c b/py/malloc.c index 4cf1b71db0..a07d360020 100644 --- a/py/malloc.c +++ b/py/malloc.c @@ -31,6 +31,7 @@ STATIC int peak_bytes_allocated = 0; #undef free #undef realloc #define malloc gc_alloc +#define malloc_mp_obj gc_alloc_mp_obj #define free gc_free #define realloc gc_realloc #endif // MICROPY_ENABLE_GC @@ -52,6 +53,24 @@ void *m_malloc(int num_bytes) { return ptr; } +void *m_malloc_mp_obj(int num_bytes) { + if (num_bytes == 0) { + return NULL; + } + void *ptr = malloc_mp_obj(num_bytes); + if (ptr == NULL) { + printf("could not allocate memory, allocating %d bytes\n", num_bytes); + return NULL; + } +#if MICROPY_MEM_STATS + total_bytes_allocated += num_bytes; + current_bytes_allocated += num_bytes; + UPDATE_PEAK(); +#endif + DEBUG_printf("malloc %d : %p\n", num_bytes, ptr); + return ptr; +} + void *m_malloc0(int num_bytes) { void *ptr = m_malloc(num_bytes); if (ptr != NULL) { @@ -27,12 +27,14 @@ typedef unsigned int uint; #define m_new0(type, num) ((type*)(m_malloc0(sizeof(type) * (num)))) #define m_new_obj(type) (m_new(type, 1)) #define m_new_obj_var(obj_type, var_type, var_num) ((obj_type*)m_malloc(sizeof(obj_type) + sizeof(var_type) * (var_num))) +#define m_new_mp_obj(type)((type*)(m_malloc_mp_obj(sizeof(type)))) #define m_renew(type, ptr, old_num, new_num) ((type*)(m_realloc((ptr), sizeof(type) * (old_num), sizeof(type) * (new_num)))) #define m_del(type, ptr, num) m_free(ptr, sizeof(type) * (num)) #define m_del_obj(type, ptr) (m_del(type, ptr, 1)) #define m_del_var(obj_type, var_type, var_num, ptr) (m_free(ptr, sizeof(obj_type) + sizeof(var_type) * (var_num))) void *m_malloc(int num_bytes); +void *m_malloc_mp_obj(int num_bytes); void *m_malloc0(int num_bytes); void *m_realloc(void *ptr, int old_num_bytes, int new_num_bytes); void m_free(void *ptr, int num_bytes); diff --git a/py/qstrdefs.h b/py/qstrdefs.h index 06548d0fa5..2271941a05 100644 --- a/py/qstrdefs.h +++ b/py/qstrdefs.h @@ -27,6 +27,7 @@ Q(__sub__) Q(__repr__) Q(__str__) Q(__getattr__) +Q(__del__) Q(micropython) Q(byte_code) |