| parrotcode: List aka array routines | |
| Contents | C |

src/list.c - List aka array routines

List is roughly based on concepts of IntList (thanks to Steve), so I don't repeat them here.
Especially the same invariants hold, except an empty list is really empty, meaning, push does first check for space.
The main differences are:
- List can hold items of different size, it's suitable for ints and PMCs ..., calculations are still done in terms of items. The item_size is specified at list creation time with the "type" argument.
If you later store different item types in the list, as stated initially, you'll get probably not what you want - so don't do this.
- List does auto grow. The caller may implement a different behaviour if she likes.
- Error checking for out of bounds access is minimal, caller knows better, what should be done.
- List structure itself is different from List_chunk, implying:
list->prev but list->end- list chunks don't have ->start and ->end fields.
Instead the list has ->start,
which is start of first chunk,
and ->cap,
the total usable capacity in the list.
- number of items in chunks are not fixed, but there is a mode using same sized chunks

enum_grow_fixedMAX_ITEMS size,
chosen,
when the first access to the array is indexed and beyond MIN_ITEMS and below 10 * MAX_ITEMSMAX_ITEMS a sparse chunk will be created. new P0, .PerlArray
set P0, 100000 # sets fixed sized, no sparse
enum_grow_growingMIN_ITEMS to MAX_ITEMS, this will be selected for pushing data on an empty array.enum_grow_mixedget_chunk below for details.
To save memory, List can handle sparse arrays. This code snippet:
new P0, .IntList set P0[1000000], 42
generates 3 List_chunks, one at the beginning of the array, a big sparse chunk and a chunk for the actual data.
Setting values inside sparse chunks changes them to real chunks. For poping/shifting inside sparse chunks, s. return value below.

fixed_itemsgrow_itemsno_power_2sparsechunk->items holds the items of this sparse hole.
A List can hold various datatypes. See src/datatypes.h for the enumeration of types.
Not all are yet implemented in list_set/list_item, see the switch().
Arbitrary length data:
Construct initializer with:
In list_assign the values are copied into the array, list_get returns a pointer as for all other data types.
See src/list_2.t and list_new_init().

List get functions return a (void*) pointer to the location of the stored data. The caller has to extract the value from this pointer.
For non existent data beyond the dimensions of the array a NULL pointer is returned.
For non existing data inside sparse holes, a pointer (void*)-1 is returned.
The caller can decide to assume these data as undef or 0 or whatever is appropriate.

See t/src/{int,}list.c and t/pmc/{int,}list.t.
Also all array usage depends on list.

static List_chunk *allocate_chunk(Interp *interpreter, List *list, UINTVAL items, UINTVAL size)static void list_dump(FILE *fp, List *list, INTVAL type)static void rebuild_chunk_ptrs(List *list, int cut)static void rebuild_sparse(List *list)static void rebuild_other(Interp *interpreter, List *list)static void rebuild_fix_ends(Interp *interpreter, List *list)rebuild_chunk_list().static UINTVAL rebuild_chunk_list(Interp *interpreter, List *list)static List_chunk *alloc_next_size(Interp *interpreter, List *list, int where, UINTVAL idx)static List_chunk *add_chunk(Interp *interpreter, List *list, int where, UINTVAL idx)UINTVAL ld(UINTVAL x)static List_chunk *get_chunk(Interp *interpreter, List *list, UINTVAL *idx)idx, also update the idx to point into the chunk.rebuild_chunk_list does provide. if all_chunks_are_MAX_ITEMS
chunk = chunk_list[ idx / MAX_ITEMS ]
idx = idx % MAX_ITEMS
done.
chunk = first
repeat
if (index < chunk->items)
done.
if (index >= items_in_chunk_block)
index -= items_in_chunk_block
chunk += chunks_in_chunk_block
continue
calc chunk and index in this block
done.
rebuild_chunk_list tries to reduce the complexity, but may fail, if you e.g. do a prime sieve by actually list_deleting the none prime numbers.chunk_blocks are there. They come from:MAX_ITEMS sized, when this would improve performance.static void split_chunk(Interp *interpreter, List *list, List_chunk *chunk, UINTVAL ix)idxMAX_ITEMS near idx and if there is still sparse space after the real chunk, this also n*MAX_ITEMS sized, so that consecutive writing would make MAX_ITEMS sized real chunks.static void list_set(Interp *interpreter, List *list, void *item, INTVAL type, INTVAL idx)item of type type in chunk at idx.static void *list_item(Interp *interpreter, List *list, int type, INTVAL idx)type in the chunk at idx.static void list_append(Interp *interpreter, List *list, void *item, int type, UINTVAL idx)
List *list_new(Interp *interpreter, INTVAL type)type.void list_pmc_new(Interp *interpreter, PMC *container)List *list_new_init(Interp *interpreter, INTVAL type, PMC *init)list_new_init() uses these initializers: 0 ... size (set initial size of list)
1 ... array dimensions (multiarray)
2 ... type (overriding type parameter)
3 ... item_size for enum_type_sized
4 ... items_per_chunk
void list_pmc_new_init(Interp *interpreter, PMC *container, PMC *init)List *list_clone(Interp *interpreter, List *other)void list_mark(Interp *interpreter, List *list)void list_visit(Interp *interpreter, List *list, void *pinfo)pinfo is the visit info, (see include/parrot/pmc_freeze.h>).INTVAL list_length(Interp *interpreter, List *list)void list_set_length(Interp *interpreter, List *list, INTVAL len)len.void list_insert(Interp *interpreter, List *list, INTVAL idx, INTVAL n_items)n_items at idx.void list_delete(Interp *interpreter, List *list, INTVAL idx, INTVAL n_items)n_items at idx.void list_push(Interp *interpreter, List *list, void *item, int type)item of type type on to the end of the list.void list_unshift(Interp *interpreter, List *list, void *item, int type)item of type type on to the start of the list.void *list_pop(Interp *interpreter, List *list, int type)type from the end of the list.void *list_shift(Interp *interpreter, List *list, int type)type from the start of the list.void list_assign(Interp *interpreter, List *list, INTVAL idx, void *item, int type)item of type type to index idx.void *list_get(Interp *interpreter, List *list, INTVAL idx, int type)type at index idx.void list_splice(Interp *interpreter, List *list, PMC *value, INTVAL offset, INTVAL count)count items starting at offset with the items in value.count is 0 then the items in value will be inserted after offset.
10.10.2002 Initial version.
11.10.2002 More documentation, optimized irregular chunk blocks, fixed indexed access WRT list->start, cosmetics.
13.10.2002 Put intlist_length into src/intlist.c.
16.10.2002 Integrated list in parrot/arrays.
17.10.2002 Clone integral data (intlist).
18.10.2002 Moved tests to t/src/list.t.
19.10.2002 Set intial length (new_init).
21.10.2002 gc_debug stuff.
21.10.2002 splice.
22.10.2002 Update comment WRT clone in splice.
26.10.2002 user_data
Fixes.
08.11.2002 arbitrary sized items (enum_type_sized).
08.01.2003 move Chunk_list flags out of buffer header.
Join chunks > MAX_ITEMS (Matt Fowles)
Greater threshold before do_sparse(). Setting initial size to avoid sparse
04.07.2003 Use a SArray for user_data
|
|
|