211 lines
5.2 KiB
C
211 lines
5.2 KiB
C
/*
|
|
* $Id: object.c,v 1.31 2005-07-05 06:26:33 bacon Exp $
|
|
*/
|
|
|
|
#include <xp/stx/object.h>
|
|
#include <xp/stx/memory.h>
|
|
#include <xp/stx/symbol.h>
|
|
#include <xp/stx/class.h>
|
|
#include <xp/stx/hash.h>
|
|
#include <xp/stx/misc.h>
|
|
|
|
/* n: number of instance variables */
|
|
xp_word_t xp_stx_alloc_word_object (
|
|
xp_stx_t* stx, const xp_word_t* data, xp_word_t n)
|
|
{
|
|
xp_word_t idx;
|
|
xp_stx_word_object_t* obj;
|
|
|
|
/* bytes to allocidxed =
|
|
* number of instance variables * word_size
|
|
*/
|
|
idx = xp_stx_memory_alloc (&stx->memory,
|
|
n * xp_sizeof(xp_word_t) + xp_sizeof(xp_stx_object_t));
|
|
if (idx >= stx->memory.capacity) return idx; /* failed */
|
|
|
|
xp_assert (stx->nil == XP_STX_NIL);
|
|
|
|
/*
|
|
XP_STX_CLASS(stx,idx) = stx->nil;
|
|
XP_STX_ACCESS(stx,idx) = (n << 2) | XP_STX_WORD_INDEXED;
|
|
while (n-- > 0) XP_STX_AT(stx,idx,n) = stx->nil;
|
|
*/
|
|
obj = XP_STX_WORD_OBJECT(stx,idx);
|
|
obj->header.class = stx->nil;
|
|
obj->header.access = (n << 2) | XP_STX_WORD_INDEXED;
|
|
if (data == XP_NULL) {
|
|
while (n-- > 0) obj->data[n] = stx->nil;
|
|
}
|
|
else {
|
|
while (n-- > 0) obj->data[n] = data[n];
|
|
}
|
|
|
|
return idx;
|
|
}
|
|
|
|
/* n: number of bytes */
|
|
xp_word_t xp_stx_alloc_byte_object (
|
|
xp_stx_t* stx, const xp_byte_t* data, xp_word_t n)
|
|
{
|
|
xp_word_t idx;
|
|
xp_stx_byte_object_t* obj;
|
|
|
|
idx = xp_stx_memory_alloc (
|
|
&stx->memory, n + xp_sizeof(xp_stx_object_t));
|
|
if (idx >= stx->memory.capacity) return idx; /* failed */
|
|
|
|
xp_assert (stx->nil == XP_STX_NIL);
|
|
|
|
/*
|
|
XP_STX_CLASS(stx,idx) = stx->nil;
|
|
XP_STX_ACCESS(stx,idx) = (n << 2) | XP_STX_BYTE_INDEXED;
|
|
while (n-- > 0) XP_STX_BYTEAT(stx,idx,n) = 0;
|
|
*/
|
|
obj = XP_STX_BYTE_OBJECT(stx,idx);
|
|
obj->header.class = stx->nil;
|
|
obj->header.access = (n << 2) | XP_STX_BYTE_INDEXED;
|
|
|
|
if (data == XP_NULL) {
|
|
while (n-- > 0) obj->data[n] = 0;
|
|
}
|
|
else {
|
|
while (n-- > 0) obj->data[n] = data[n];
|
|
}
|
|
|
|
return idx;
|
|
}
|
|
|
|
xp_word_t xp_stx_alloc_char_object (
|
|
xp_stx_t* stx, const xp_char_t* str)
|
|
{
|
|
return (str == XP_NULL)?
|
|
xp_stx_alloc_char_objectx (stx, XP_NULL, 0):
|
|
xp_stx_alloc_char_objectx (stx, str, xp_strlen(str));
|
|
}
|
|
|
|
xp_word_t xp_stx_alloc_char_objectx (
|
|
xp_stx_t* stx, const xp_char_t* str, xp_word_t n)
|
|
{
|
|
xp_word_t idx;
|
|
xp_stx_char_object_t* obj;
|
|
|
|
idx = xp_stx_memory_alloc (&stx->memory,
|
|
(n + 1) * xp_sizeof(xp_char_t) + xp_sizeof(xp_stx_object_t));
|
|
if (idx >= stx->memory.capacity) return idx; /* failed */
|
|
|
|
xp_assert (stx->nil == XP_STX_NIL);
|
|
|
|
/*
|
|
XP_STX_CLASS(stx,idx) = stx->nil;
|
|
XP_STX_ACCESS(stx,idx) = (n << 2) | XP_STX_CHAR_INDEXED;
|
|
XP_STX_CHARAT(stx,idx,n) = XP_CHAR('\0');
|
|
while (n-- > 0) XP_STX_CHARAT(stx,idx,n) = str[n];
|
|
*/
|
|
obj = XP_STX_CHAR_OBJECT(stx,idx);
|
|
obj->header.class = stx->nil;
|
|
obj->header.access = (n << 2) | XP_STX_CHAR_INDEXED;
|
|
obj->data[n] = XP_CHAR('\0');
|
|
|
|
if (str == XP_NULL) {
|
|
while (n-- > 0) obj->data[n] = XP_CHAR('\0');
|
|
}
|
|
else {
|
|
while (n-- > 0) obj->data[n] = str[n];
|
|
}
|
|
|
|
return idx;
|
|
}
|
|
|
|
xp_word_t xp_stx_allocn_char_object (xp_stx_t* stx, ...)
|
|
{
|
|
xp_word_t idx, n = 0;
|
|
const xp_char_t* p;
|
|
xp_va_list ap;
|
|
xp_stx_char_object_t* obj;
|
|
|
|
xp_va_start (ap, stx);
|
|
while ((p = xp_va_arg(ap, const xp_char_t*)) != XP_NULL) {
|
|
n += xp_strlen(p);
|
|
}
|
|
xp_va_end (ap);
|
|
|
|
idx = xp_stx_memory_alloc (&stx->memory,
|
|
(n + 1) * xp_sizeof(xp_char_t) + xp_sizeof(xp_stx_object_t));
|
|
if (idx >= stx->memory.capacity) return idx; /* failed */
|
|
|
|
xp_assert (stx->nil == XP_STX_NIL);
|
|
|
|
/*
|
|
XP_STX_CLASS(stx,idx) = stx->nil;
|
|
XP_STX_ACCESS(stx,idx) = (n << 2) | XP_STX_CHAR_INDEXED;
|
|
XP_STX_CHARAT(stx,idx,n) = XP_CHAR('\0');
|
|
*/
|
|
obj = XP_STX_CHAR_OBJECT(stx,idx);
|
|
obj->header.class = stx->nil;
|
|
obj->header.access = (n << 2) | XP_STX_CHAR_INDEXED;
|
|
obj->data[n] = XP_CHAR('\0');
|
|
|
|
xp_va_start (ap, stx);
|
|
n = 0;
|
|
while ((p = xp_va_arg(ap, const xp_char_t*)) != XP_NULL) {
|
|
while (*p != XP_CHAR('\0')) {
|
|
/*XP_STX_CHARAT(stx,idx,n++) = *p++;*/
|
|
obj->data[n++] = *p++;
|
|
}
|
|
}
|
|
xp_va_end (ap);
|
|
|
|
return idx;
|
|
}
|
|
|
|
xp_word_t xp_stx_hash_char_object (xp_stx_t* stx, xp_word_t idx)
|
|
{
|
|
xp_assert (XP_STX_TYPE(stx,idx) == XP_STX_CHAR_INDEXED);
|
|
return xp_stx_strxhash (
|
|
XP_STX_DATA(stx,idx), XP_STX_SIZE(stx,idx));
|
|
}
|
|
|
|
xp_word_t xp_stx_instantiate (
|
|
xp_stx_t* stx, xp_word_t class_index, const void* data, xp_word_t size)
|
|
{
|
|
xp_stx_class_t* class_obj;
|
|
xp_word_t spec, nfields, new;
|
|
int indexable;
|
|
|
|
class_obj = (xp_stx_class_t*)
|
|
XP_STX_WORD_OBJECT(stx, class_index);
|
|
|
|
/* don't instantiate a metaclass whose instance must be
|
|
created in a different way */
|
|
/* TODO: maybe delete the following line */
|
|
xp_assert (class_obj->header.class != stx->class_metaclass);
|
|
|
|
spec = XP_STX_FROM_SMALLINT(class_obj->spec);
|
|
nfields = (spec >> XP_STX_SPEC_INDEXABLE_BITS);
|
|
indexable = spec & XP_STX_SPEC_INDEXABLE_MASK;
|
|
|
|
if (indexable == XP_STX_SPEC_BYTE_INDEXABLE) {
|
|
xp_assert (nfields == 0);
|
|
new = xp_stx_alloc_byte_object (
|
|
stx, data, nfields + size);
|
|
}
|
|
else if (indexable == XP_STX_SPEC_CHAR_INDEXABLE) {
|
|
xp_assert (nfields == 0);
|
|
new = xp_stx_alloc_char_objectx (
|
|
stx, data, nfields + size);
|
|
}
|
|
else if (indexable == XP_STX_SPEC_WORD_INDEXABLE) {
|
|
new = xp_stx_alloc_word_object (
|
|
stx, data, nfields + size);
|
|
}
|
|
else {
|
|
xp_assert (indexable == XP_STX_SPEC_NOT_INDEXABLE);
|
|
xp_assert (size == 0);
|
|
new = xp_stx_alloc_word_object (
|
|
stx, data, nfields + size);
|
|
}
|
|
|
|
XP_STX_CLASS(stx, new) = class_index;
|
|
return new;
|
|
}
|