elk/lib/xt/context.c

284 lines
8.3 KiB
C

#include "xt.h"
static SYMDESCR XtIM_Syms[] = {
{ "x-event", XtIMXEvent },
{ "timer", XtIMTimer },
{ "alternate-input", XtIMAlternateInput },
{ 0, 0 }
};
static SYMDESCR XtInputMask_Syms[] = {
{ "read", XtInputReadMask },
{ "write", XtInputWriteMask },
{ "exception", XtInputExceptMask },
{ 0, 0 }
};
static Object P_Destroy_Context();
Generic_Predicate (Context)
Generic_Equal (Context, CONTEXT, context)
Generic_Print (Context, "#[context %lu]", POINTER(x))
static Object Internal_Make_Context (finalize, context) XtAppContext context; {
Object c;
c = Find_Object (T_Context, (GENERIC)0, Match_Xt_Obj, context);
if (Nullp (c)) {
c = Alloc_Object (sizeof (struct S_Context), T_Context, 0);
CONTEXT(c)->tag = Null;
CONTEXT(c)->context = context;
CONTEXT(c)->free = 0;
Register_Object (c, (GENERIC)0,
finalize ? P_Destroy_Context : (PFO)0, 1);
XtAppSetWarningHandler (context, Xt_Warning);
XtAppAddActionHook (context, (XtActionHookProc)Action_Hook,
(XtPointer)0);
}
return c;
}
/* Backwards compatibility: */
Object Make_Context (context) XtAppContext context; {
return Internal_Make_Context (1, context);
}
Object Make_Context_Foreign (context) XtAppContext context; {
return Internal_Make_Context (0, context);
}
void Check_Context (c) Object c; {
Check_Type (c, T_Context);
if (CONTEXT(c)->free)
Primitive_Error ("invalid context: ~s", c);
}
static Object P_Create_Context () {
return Make_Context (XtCreateApplicationContext ());
}
static Object P_Destroy_Context (c) Object c; {
Check_Context (c);
Free_Actions (CONTEXT(c)->context);
XtDestroyApplicationContext (CONTEXT(c)->context);
CONTEXT(c)->free = 1;
Deregister_Object (c);
return Void;
}
static Object P_Initialize_Display (c, d, name, class)
Object c, d, name, class; {
register char *sn = 0, *sc = "", *sd = 0;
Display *dpy;
extern char **Argv;
extern First_Arg, Argc;
int argc = Argc - First_Arg + 1;
Argv[First_Arg-1] = "elk";
Check_Context (c);
if (!EQ(name, False))
sn = Get_Strsym (name);
if (!EQ(class, False))
sc = Get_Strsym (class);
if (TYPE(d) == T_Display) {
XtDisplayInitialize (CONTEXT(c)->context, DISPLAY(d)->dpy,
sn, sc, (XrmOptionDescRec *)0, 0, &argc, &Argv[First_Arg-1]);
Argc = First_Arg + argc;
return Void;
}
if (!EQ(d, False))
sd = Get_Strsym (d);
dpy = XtOpenDisplay (CONTEXT(c)->context, sd, sn, sc,
(XrmOptionDescRec *)0, 0, &argc, &Argv[First_Arg-1]);
Argc = First_Arg + argc - 1;
if (dpy == 0)
if (sd)
Primitive_Error ("cannot open display ~s", d);
else
Primitive_Error ("cannot open display");
return Make_Display (0, dpy);
}
/* Due to a bug in Xt this function drops core when invoked with a
* display not owned by Xt.
*/
static Object P_Display_To_Context (d) Object d; {
Check_Type (d, T_Display);
return
Make_Context_Foreign (XtDisplayToApplicationContext (DISPLAY(d)->dpy));
}
static Object P_Set_Context_Fallback_Resources (argc, argv) Object *argv; {
register char **p = 0;
register i;
struct S_String *sp;
Object con;
con = argv[0];
Check_Context (con);
if (argc > 1) {
argv++; argc--;
p = (char **)XtMalloc ((argc+1) * sizeof (char *));
for (i = 0; i < argc; i++) {
Check_Type (argv[i], T_String);
sp = STRING(argv[i]);
p[i] = XtMalloc (sp->size + 1);
bcopy (sp->data, p[i], sp->size);
p[i][sp->size] = 0;
}
p[i] = 0;
}
XtAppSetFallbackResources (CONTEXT(con)->context, p);
return Void;
}
static Object P_Context_Main_Loop (c) Object c; {
Check_Context (c);
XtAppMainLoop (CONTEXT(c)->context);
/*NOTREACHED*/
}
static Object P_Context_Pending (c) Object c; {
Check_Context (c);
return Bits_To_Symbols ((unsigned long)XtAppPending (CONTEXT(c)->context),
1, XtIM_Syms);
}
static Object P_Context_Process_Event (argc, argv) Object *argv; {
XtInputMask mask = XtIMAll;
Check_Context (argv[0]);
if (argc == 2)
mask = (XtInputMask)Symbols_To_Bits (argv[1], 1, XtIM_Syms);
XtAppProcessEvent (CONTEXT(argv[0])->context, mask);
return Void;
}
static Boolean Work_Proc (client_data) XtPointer client_data; {
Object ret;
ret = Funcall (Get_Function ((int)client_data), Null, 0);
if (Truep (ret))
Deregister_Function ((int)client_data);
return Truep (ret);
}
static Object P_Context_Add_Work_Proc (c, p) Object c, p; {
XtWorkProcId id;
register i;
Check_Context (c);
Check_Procedure (p);
i = Register_Function (p);
id = XtAppAddWorkProc (CONTEXT(c)->context, Work_Proc, (XtPointer)i);
return Make_Id ('w', (XtPointer)id, i);
}
static Object P_Remove_Work_Proc (id) Object id; {
XtRemoveWorkProc ((XtWorkProcId)Use_Id (id, 'w'));
Deregister_Function (IDENTIFIER(id)->num);
return Void;
}
static void Timeout_Proc (client_data, id)
XtPointer client_data; XtIntervalId *id; {
Object proc, args;
register i = (int)client_data;
args = Cons (Make_Id ('t', (XtPointer)*id, i), Null);
proc = Get_Function (i);
Deregister_Function (i);
(void)Funcall (proc, args, 0);
}
static Object P_Context_Add_Timeout (c, n, p) Object c, n, p; {
XtIntervalId id;
register i;
Check_Context (c);
Check_Procedure (p);
i = Register_Function (p);
id = XtAppAddTimeOut (CONTEXT(c)->context, (unsigned long)Get_Long (n),
Timeout_Proc, (XtPointer)i);
return Make_Id ('t', (XtPointer)id, i);
}
static Object P_Remove_Timeout (id) Object id; {
XtRemoveTimeOut ((XtIntervalId)Use_Id (id, 't'));
Deregister_Function (IDENTIFIER(id)->num);
return Void;
}
/*ARGSUSED*/
static void Input_Proc (client_data, src, id) XtPointer client_data; int *src;
XtInputId *id; {
Object p, args;
GC_Node2;
p = Get_Function ((int)client_data);
args = Null;
GC_Link2 (p, args);
args = Cons (Make_Id ('i', (XtPointer)*id, (int)client_data), Null);
args = Cons (Car (p), args);
GC_Unlink;
(void)Funcall (Cdr (p), args, 0);
}
static Object P_Context_Add_Input (argc, argv) Object *argv; {
Object c, src, p;
XtInputId id;
XtInputMask m;
register i;
c = argv[0], src = argv[1], p = argv[2];
Check_Context (c);
Check_Procedure (p);
Check_Type (src, T_Port);
if (!(PORT(src)->flags & P_OPEN))
Primitive_Error ("port has been closed: ~s", src);
if (PORT(src)->flags & P_STRING)
Primitive_Error ("invalid port: ~s", src);
if (argc == 4) {
m = Symbols_To_Bits (argv[3], 1, XtInputMask_Syms);
} else {
switch (PORT(src)->flags & (P_INPUT|P_BIDIR)) {
case 0: m = XtInputWriteMask; break;
case P_INPUT: m = XtInputReadMask; break;
default: m = XtInputReadMask|XtInputWriteMask; break;
}
}
i = Register_Function (Cons (src, p));
id = XtAppAddInput (CONTEXT(c)->context, fileno (PORT(src)->file),
(XtPointer)m, Input_Proc, (XtPointer)i);
return Make_Id ('i', (XtPointer)id, i);
}
static Object P_Remove_Input (id) Object id; {
XtRemoveInput ((XtInputId)Use_Id (id, 'i'));
Deregister_Function (IDENTIFIER(id)->num);
return Void;
}
elk_init_xt_context () {
Generic_Define (Context, "context", "context?");
Define_Primitive (P_Create_Context, "create-context", 0, 0, EVAL);
Define_Primitive (P_Destroy_Context, "destroy-context", 1, 1, EVAL);
Define_Primitive (P_Initialize_Display, "initialize-display", 4, 4, EVAL);
Define_Primitive (P_Display_To_Context, "display->context", 1, 1, EVAL);
Define_Primitive (P_Set_Context_Fallback_Resources,
"set-context-fallback-resources!", 1, MANY, VARARGS);
Define_Primitive (P_Context_Main_Loop, "context-main-loop", 1, 1, EVAL);
Define_Primitive (P_Context_Pending, "context-pending", 1, 1, EVAL);
Define_Primitive (P_Context_Process_Event,
"context-process-event", 1, 2, VARARGS);
Define_Primitive (P_Context_Add_Work_Proc,
"context-add-work-proc", 2, 2, EVAL);
Define_Primitive (P_Remove_Work_Proc, "remove-work-proc", 1, 1, EVAL);
Define_Primitive (P_Context_Add_Timeout,"context-add-timeout",3, 3, EVAL);
Define_Primitive (P_Remove_Timeout, "remove-timeout", 1, 1, EVAL);
Define_Primitive (P_Context_Add_Input, "context-add-input",3, 4, VARARGS);
Define_Primitive (P_Remove_Input, "remove-input", 1, 1, EVAL);
}