1
Fork 0
mirror of https://git.savannah.gnu.org/git/guile.git synced 2025-06-16 00:30:21 +02:00

* struct.c (scm_make_struct_layout, init_struct, scm_struct_ref,

scm_struct_set_x), struct.h, gc.c (scm_gc_mark): Completed Tom
Lord's implementation of structs, allowing for tail arrays as
described in the manual. Also fixed some bugs. (Both the interface
and the implementation should be improved.)
This commit is contained in:
Mikael Djurfeldt 1996-09-22 22:38:56 +00:00
parent 90b826c925
commit 2c36c351d0
2 changed files with 99 additions and 46 deletions

View file

@ -98,12 +98,22 @@ scm_make_struct_layout (fields)
switch (field_desc[x + 1])
{
case 'w':
SCM_ASSERT ((field_desc[x] != 's'), SCM_MAKICHR (field_desc[x + 1]),
SCM_ASSERT (field_desc[x] != 's', SCM_MAKICHR (field_desc[x + 1]),
"self fields not writable", s_struct_make_layout);
case 'r':
case 'o':
break;
case 'R':
case 'W':
case 'O':
SCM_ASSERT (field_desc[x] != 's', SCM_MAKICHR (field_desc[x + 1]),
"self fields not allowed in tail array",
s_struct_make_layout);
SCM_ASSERT (x == len - 2, SCM_MAKICHR (field_desc[x + 1]),
"tail array field must be last field in layout",
s_struct_make_layout);
break;
default:
SCM_ASSERT (0, SCM_MAKICHR (field_desc[x]) , "unrecognized ref specification", s_struct_make_layout);
}
@ -126,34 +136,50 @@ scm_make_struct_layout (fields)
#ifdef __STDC__
static void
init_struct (SCM handle, SCM tail_elts, SCM inits)
init_struct (SCM handle, int tail_elts, SCM inits)
#else
static void
init_struct (handle, tail_elts, inits)
SCM handle;
SCM tail_elts;
int tail_elts;
SCM inits;
#endif
{
SCM layout;
SCM * data;
unsigned char * fields_desc;
unsigned char prot;
int n_fields;
SCM * mem;
int tailp = 0;
layout = SCM_STRUCT_LAYOUT (handle);
data = SCM_STRUCT_DATA (handle);
fields_desc = (unsigned char *)SCM_CHARS (layout);
fields_desc = (unsigned char *) SCM_CHARS (layout) - 2;
n_fields = SCM_LENGTH (layout) / 2;
mem = SCM_STRUCT_DATA (handle);
while (n_fields)
{
if (!tailp)
{
fields_desc += 2;
prot = fields_desc[1];
if (SCM_LAYOUT_TAILP (prot))
{
tailp = 1;
prot = prot == 'R' ? 'r' : prot == 'W' ? 'w' : 'o';
*mem++ = tail_elts;
n_fields += tail_elts - 1;
if (n_fields == 0)
break;
}
}
switch (*fields_desc)
{
#if 0
case 'i':
if ( ((fields_desc[1] != 'r') && (fields_desc[1] != 'w'))
|| ((inits == SCM_EOL) || !SCM_NUMBERP (SCM_CAR (inits))))
if ((prot != 'r' && prot != 'w') || inits == SCM_EOL)
*mem = 0;
else
{
@ -164,8 +190,7 @@ init_struct (handle, tail_elts, inits)
#endif
case 'u':
if ( ((fields_desc[1] != 'r') && (fields_desc[1] != 'w'))
|| ((inits == SCM_EOL) || !SCM_NUMBERP (SCM_CAR (inits))))
if ((prot != 'r' && prot != 'w') || inits == SCM_EOL)
*mem = 0;
else
{
@ -175,8 +200,7 @@ init_struct (handle, tail_elts, inits)
break;
case 'p':
if ( ((fields_desc[1] != 'r') && (fields_desc[1] != 'w'))
|| (inits == SCM_EOL))
if ((prot != 'r' && prot != 'w') || inits == SCM_EOL)
*mem = SCM_EOL;
else
{
@ -188,8 +212,7 @@ init_struct (handle, tail_elts, inits)
#if 0
case 'd':
if ( ((fields_desc[1] != 'r') && (fields_desc[1] != 'w'))
|| ((inits == SCM_EOL) || !SCM_NUMBERP (SCM_CAR (inits))))
if ((prot != 'r' && prot != 'w') || inits == SCM_EOL)
*((double *)mem) = 0.0;
else
{
@ -205,7 +228,6 @@ init_struct (handle, tail_elts, inits)
break;
}
fields_desc += 2;
n_fields--;
mem++;
}
@ -295,10 +317,15 @@ scm_make_struct (vtable, tail_array_size, init)
tail_elts = SCM_INUM (tail_array_size);
SCM_NEWCELL (handle);
SCM_DEFER_INTS;
data = (SCM*)scm_must_malloc (sizeof (SCM) * (2 + basic_size + tail_elts), "structure");
*data = (SCM)(2 + basic_size + tail_elts);
data[1] = struct_num++;
data += 2;
data = (SCM*)scm_must_malloc (sizeof (SCM) * (scm_struct_n_extra_words
+ basic_size
+ tail_elts),
"structure");
data += scm_struct_n_extra_words;
data[scm_struct_i_n_words] = (SCM) (scm_struct_n_extra_words
+ basic_size
+ tail_elts);
data[scm_struct_i_tag] = struct_num++;
SCM_SETCDR (handle, data);
SCM_SETCAR (handle, ((SCM)SCM_STRUCT_DATA (vtable)) + 1);
init_struct (handle, tail_elts, init);
@ -340,10 +367,15 @@ scm_make_vtable_vtable (extra_fields, tail_array_size, init)
tail_elts = SCM_INUM (tail_array_size);
SCM_NEWCELL (handle);
SCM_DEFER_INTS;
data = (SCM*)scm_must_malloc (sizeof (SCM) * (2 + basic_size + tail_elts), "structure");
*data = (SCM)(2 + basic_size + tail_elts);
data[1] = struct_num++;
data += 2;
data = (SCM *) scm_must_malloc (sizeof (SCM) * (scm_struct_n_extra_words
+ basic_size
+ tail_elts),
"structure");
data += scm_struct_n_extra_words;
data[scm_struct_i_n_words] = (SCM) (scm_struct_n_extra_words
+ basic_size
+ tail_elts);
data[scm_struct_i_tag] = struct_num++;
SCM_SETCDR (handle, data);
SCM_SETCAR (handle, ((SCM)data) + 1);
SCM_STRUCT_LAYOUT (handle) = layout;
@ -384,22 +416,30 @@ scm_struct_ref (handle, pos)
p = SCM_INUM (pos);
fields_desc = (unsigned char *)SCM_CHARS (layout);
n_fields = SCM_LENGTH (layout) / 2;
n_fields = data[- scm_struct_n_extra_words] - scm_struct_n_extra_words;
SCM_ASSERT (p < n_fields, pos, SCM_OUTOFRANGE, s_struct_ref);
SCM_ASSERT (p <= n_fields, pos, SCM_OUTOFRANGE, s_struct_ref);
field_type = fields_desc[p * 2];
{
unsigned char ref;
ref = fields_desc [p * 2 + 1];
if ((ref != 'r') && (ref != 'w'))
{
if ((ref == 'R') || (ref == 'W'))
field_type = 'u';
else
SCM_ASSERT (0, pos, "ref denied", s_struct_ref);
}
}
if (p * 2 < SCM_LENGTH (layout))
{
unsigned char ref;
field_type = fields_desc[p * 2];
ref = fields_desc[p * 2 + 1];
if ((ref != 'r') && (ref != 'w'))
{
if ((ref == 'R') || (ref == 'W'))
field_type = 'u';
else
SCM_ASSERT (0, pos, "ref denied", s_struct_ref);
}
}
else if (fields_desc[SCM_LENGTH (layout) - 1] != 'O')
field_type = fields_desc[SCM_LENGTH (layout) - 2];
else
{
SCM_ASSERT (0, pos, "ref denied", s_struct_ref);
}
switch (field_type)
{
case 'u':
@ -461,17 +501,25 @@ scm_struct_set_x (handle, pos, val)
p = SCM_INUM (pos);
fields_desc = (unsigned char *)SCM_CHARS (layout);
n_fields = SCM_LENGTH (layout) / 2;
n_fields = data[- scm_struct_n_extra_words] - scm_struct_n_extra_words;
SCM_ASSERT (p <= n_fields, pos, SCM_OUTOFRANGE, s_struct_set_x);
SCM_ASSERT (p < n_fields, pos, SCM_OUTOFRANGE, s_struct_set_x);
field_type = fields_desc[p * 2];
{
unsigned char set_x;
set_x = fields_desc [p * 2 + 1];
if (set_x != 'w')
SCM_ASSERT (0, pos, "set_x denied", s_struct_set_x);
}
if (p * 2 < SCM_LENGTH (layout))
{
unsigned char set_x;
field_type = fields_desc[p * 2];
set_x = fields_desc [p * 2 + 1];
if (set_x != 'w')
SCM_ASSERT (0, pos, "set_x denied", s_struct_set_x);
}
else if (fields_desc[SCM_LENGTH (layout) - 1] == 'W')
field_type = fields_desc[SCM_LENGTH (layout) - 2];
else
{
SCM_ASSERT (0, pos, "set_x denied", s_struct_ref);
}
switch (field_type)
{
case 'u':