stk/Src/extend.c

313 lines
7.7 KiB
C

/*
*
* e x t e n d . c -- All the stuff dealing with
* extended types
*
* Copyright © 1993-1999 Erick Gallesio - I3S-CNRS/ESSI <eg@unice.fr>
*
*
* Permission to use, copy, modify, distribute,and license this
* software and its documentation for any purpose is hereby granted,
* provided that existing copyright notices are retained in all
* copies and that this notice is included verbatim in any
* distributions. No written agreement, license, or royalty fee is
* required for any of the authorized uses.
* This software is provided ``AS IS'' without express or implied
* warranty.
*
* Author: Erick Gallesio [eg@kaolin.unice.fr]
* Creation date: 15-Mar-1995 11:31
* Last file update: 3-Sep-1999 20:19 (eg)
*/
#include "stk.h"
#include "stklos.h"
#include "extend.h"
#define EXT_TYPE_DESCR(x) (xtypes[TYPE(x)- tc_start_extd])
static int extended_type_stamp = tc_start_extd;
static STk_extended_scheme_type *xtypes[tc_stop_extd-tc_start_extd+1];
#ifdef USE_STKLOS
static SCM xclasses[tc_stop_extd-tc_start_extd+1];
#endif
/******************************************************************************
*
* Extended Types
*
******************************************************************************/
/***
***
*** Default functions
***
***/
static void internal_display(SCM obj, SCM port, int mode)
{
sprintf(STk_tkbuffer, "#<%s %lx>", (EXT_TYPE_DESCR(obj))->type_name,
(unsigned long) obj);
Puts(STk_tkbuffer, port);
}
static SCM internal_apply(SCM obj, SCM args, SCM env)
{
Err("apply: bad procedure", obj);
return UNDEFINED; /* to make the compiler happy */
}
static SCM internal_compare(SCM x, SCM y, int equalp)
{
return Ntruth;
}
/***
***
*** Utilities
***
***/
void STk_extended_mark(SCM x)
{
STk_extended_scheme_type *p= EXT_TYPE_DESCR(x);
if (p->gc_mark_fct) (*(p->gc_mark_fct))(x);
}
void STk_extended_sweep(SCM x)
{
STk_extended_scheme_type *p = EXT_TYPE_DESCR(x);
if (p->gc_sweep_fct) (*(p->gc_sweep_fct))(x);
}
SCM STk_extended_apply(SCM x, SCM args, SCM env)
{
return (*(EXT_TYPE_DESCR(x)->apply_fct))(x, args, env);
}
void STk_extended_display(SCM x, SCM port, int mode)
{
(*(EXT_TYPE_DESCR(x)->display_fct))(x, port, mode);
}
int STk_extended_procedurep(SCM x)
{
return (EXT_TYPE_DESCR(x)->flags && EXT_ISPROC);
}
int STk_extended_eval_parameters(SCM x)
{
return (EXT_TYPE_DESCR(x)->flags && EXT_EVALPARAM);
}
SCM STk_extended_compare(SCM x, SCM y, int equalp)
{
/* One of x or y (at least) is extended. */
return EXTENDEDP(x) ? (*(EXT_TYPE_DESCR(x)->compare_fct))(x, y, equalp)
: (*(EXT_TYPE_DESCR(y)->compare_fct))(x, y, equalp);
}
char *STk_get_extended_name(int i)
{
return ((i>=tc_start_extd) && (i<extended_type_stamp)) ?
(xtypes[i - tc_start_extd])->type_name :
NULL;
}
#ifdef USE_STKLOS
void STk_register_extended_class(SCM c, int type_index)
{
if (NNULLP(c)) {
int i = type_index - tc_start_extd;
xclasses[i] = c;
STk_gc_protect(xclasses+i);
}
}
SCM STk_extended_class_of(SCM obj) /* Warning: obj MUST be an extd type */
{
return xclasses[TYPE(obj)-tc_start_extd];
}
#endif
/******************************************************************************
*
* C-pointer
*
******************************************************************************/
typedef void (*STk_disp_function)(SCM x, SCM port, int mode);
static int Cpointer_id = ANONYMOUS_DYN_PTR_ID + 1;
static int size = 0;
static STk_disp_function *display_array = NULL;
static void Cpointer_default_display(SCM obj, SCM port, int mode)
{
sprintf(STk_tkbuffer, "#<C-pointer %d %lx>", EXTID(obj),
(unsigned long) EXTDATA(obj));
Puts(STk_tkbuffer, port);
}
void STk_Cpointer_display(SCM obj, SCM port, int mode)
{
int id = EXTID(obj);
if (id <= ANONYMOUS_DYN_PTR_ID)
Cpointer_default_display(obj, port, mode);
else
(*(display_array[id]))(obj, port, mode);
}
/******************************************************************************
*
* C variable
*
******************************************************************************/
static Tcl_HashTable Cvars;
static C_hash_table_initialized = 0;
struct get_n_set_box {
SCM (*getter)();
void (*setter)();
};
SCM STk_apply_getter_C_variable(char *var)
{
Tcl_HashEntry *entry;
if ((entry = Tcl_FindHashEntry(&Cvars, var))) {
struct get_n_set_box *p = (struct get_n_set_box *) Tcl_GetHashValue(entry);
return (*(p->getter))(var);
}
else {
Fprintf(STk_curr_eport, "internal error: %s variable has no getter!!\n", var);
return UNDEFINED;
}
}
void STk_apply_setter_C_variable(char *var, SCM value)
{
Tcl_HashEntry *entry;
if ((entry = Tcl_FindHashEntry(&Cvars, var))) {
struct get_n_set_box *p = (struct get_n_set_box *) Tcl_GetHashValue(entry);
(*(p->setter))(var, value);
}
else
Fprintf(STk_curr_eport, "internal error: %s variable has no setter!!\n", var);
}
/******************************************************************************
*
* Extended types and C-pointer User interface
*
******************************************************************************/
int STk_add_new_type(STk_extended_scheme_type *p)
{
if (!p) Err("bad new type description", NIL);
/* Set the apply procedure if not defined */
if (!p->apply_fct) p->apply_fct = internal_apply;
/* Replace NULL display function by a default function */
if (!p->display_fct) p->display_fct = internal_display;
/* Replace NULL compare function by a default function */
if (!p->compare_fct) p->compare_fct = internal_compare;
/* Store the new type descriptor in the xtypes array */
xtypes[extended_type_stamp - tc_start_extd] = p;
#ifdef USE_STKLOS
/* Create a new class for this type and register it */
STk_register_extended_class(STk_make_extended_class(p->type_name),
extended_type_stamp);
#endif
return extended_type_stamp++;
}
void STk_add_new_primitive(char *fct_name, int fct_type, struct obj * (*fct_ptr)())
{
SCM z;
NEWCELL(z, fct_type);
z->storage_as.subr0.name = fct_name;
z->storage_as.subr0.f = fct_ptr;
STk_define_public_var(STk_selected_module, Intern(fct_name), z);
}
SCM STk_eval_C_string(char *s, SCM env)
{
SCM tmp = STk_internal_eval_string(s, ERR_OK, env);
return tmp == EVAL_ERROR ? NULL: tmp;
}
/*************************/
int STk_new_Cpointer_id(void (*display_func)(SCM x, SCM port, int mode))
{
if (++Cpointer_id >= size) {
if (display_array == NULL) {
display_array = must_malloc(10*sizeof (STk_disp_function));
size = 10;
}
else {
size += size / 2;
display_array = must_realloc(display_array,
size * sizeof (STk_disp_function));
}
}
/* store function in array */
display_array[Cpointer_id]= display_func? display_func : Cpointer_default_display;
return Cpointer_id;
}
SCM STk_make_Cpointer(int Cpointer_id, void *ptr, int staticp)
{
register SCM z;
NEWCELL(z, tc_Cpointer);
EXTDATA(z) = ptr;
EXTID(z) = Cpointer_id;
EXTSTATICP(z) = staticp;
return z;
}
/*************************/
void STk_define_C_variable(char *var, SCM (*getter)(), void (*setter)())
{
Tcl_HashEntry *entry;
int new;
struct get_n_set_box *p;
if (!C_hash_table_initialized) {
/* First C-var. Create Hash table */
Tcl_InitHashTable(&Cvars, TCL_STRING_KEYS);
C_hash_table_initialized = 1;
}
p = must_malloc(sizeof(struct get_n_set_box));
p->getter = getter;
p->setter = setter;
entry = Tcl_CreateHashEntry(&Cvars, var, &new);
if (!new) {
Fprintf(STk_curr_eport, "Attempt to multi-define C variable `%s' !!\n", var);
return;
}
Tcl_SetHashValue(entry, p);
/* Now enter variable in obarray and set its info field to C variable */
Intern(var)->cell_info = CELL_INFO_C_VAR;
}