1 |
/* Copyright (c) 1997 Regents of the University of California */ |
2 |
|
3 |
#ifndef lint |
4 |
static char SCCSid[] = "$SunId$ LBL"; |
5 |
#endif |
6 |
|
7 |
/* |
8 |
* Handle data stream to and from GL display process. |
9 |
*/ |
10 |
|
11 |
#include "standard.h" |
12 |
#include <string.h> |
13 |
#include "gldisp.h" |
14 |
|
15 |
|
16 |
#define GD_TYPELEN 1 /* a type is 1 byte in length */ |
17 |
|
18 |
/* argument types */ |
19 |
#define GD_TY_END 0 /* argument list terminator */ |
20 |
#define GD_TY_NAM 1 /* 8-byte (max.) identifier */ |
21 |
#define GD_TY_INT 2 /* 4-byte integer */ |
22 |
#define GD_TY_FLT 3 /* 4-byte IEEE float */ |
23 |
#define GD_TY_DBL 4 /* 8-byte IEEE double */ |
24 |
#define GD_TY_CLR 5 /* 4-byte RGBE color value */ |
25 |
#define GD_TY_ARR 6 /* 5-byte array prefix */ |
26 |
#define GD_TY_STR 7 /* nul-terminated string */ |
27 |
#define GD_TY_ARG 8 /* 1-byte argument list prefix */ |
28 |
#define GD_TY_ERR 9 /* 1-byte error code */ |
29 |
|
30 |
#define GD_NTYPES 10 /* number of argument types */ |
31 |
|
32 |
/* argument lengths */ |
33 |
#define GD_ARGLEN {0,8,4,4,8,4,5,-1,1,1} |
34 |
|
35 |
#define GD_MAXID 8 /* maximum id. length (must be 8) */ |
36 |
|
37 |
/* |
38 |
* A request consists of an argument list, the first of |
39 |
* which is always the request name as an 8-char (max.) string. |
40 |
* The types of the following arguments must match the required |
41 |
* arguments of the display request, or an error will result. |
42 |
* |
43 |
* Only functions return values, and they return them as a argument |
44 |
* list on the client's receiving connection. It is up to the client |
45 |
* program to keep track of its function calls and which values correspond |
46 |
* to which request functions. |
47 |
* |
48 |
* An error is indicated with a special GD_TY_ERR code on the receiving |
49 |
* connection, and usually indicates something fatal. |
50 |
*/ |
51 |
|
52 |
/* error codes */ |
53 |
#define GD_ER_UNRECOG 0 /* unrecognized request */ |
54 |
#define GD_ER_ARGTYPE 1 /* argument type mismatch */ |
55 |
#define GD_ER_ARGMISS 2 /* argument(s) missing */ |
56 |
|
57 |
#define GD_NERRS 3 /* number of errors */ |
58 |
|
59 |
/* request argument */ |
60 |
typedef struct { |
61 |
BYTE typ; /* argument type */ |
62 |
BYTE atyp; /* array subtype if typ==GD_TY_ARR */ |
63 |
union { |
64 |
char n[GD_MAXID]; /* 8-char (max.) id. */ |
65 |
int4 n1, n2; /* used for ID comparison */ |
66 |
int4 i; /* 4-byte integer */ |
67 |
float f; /* 4-byte IEEE float */ |
68 |
double d; /* 8-byte IEEE double */ |
69 |
COLR c; /* RGBE */ |
70 |
struct array { |
71 |
int4 l; /* length */ |
72 |
MEM_PTR p; /* values */ |
73 |
} a; /* array */ |
74 |
char *s; /* nul-terminated string */ |
75 |
} v; /* argument value */ |
76 |
} GDarg; |
77 |
|
78 |
/* a request and its arguments */ |
79 |
typedef struct gdreq { |
80 |
struct gdreq *next; /* next request in list */ |
81 |
short argc; /* number of arguments */ |
82 |
GDarg argv[1]; /* argument list (expandable) */ |
83 |
} GDrequest; |
84 |
|
85 |
GDrequest *gdProTab[GD_HSIZ]; /* registered prototypes */ |
86 |
|
87 |
|
88 |
gdRegProto(r) /* register a request prototype */ |
89 |
register GDrequest *r; |
90 |
{ |
91 |
register int4 hval; |
92 |
|
93 |
hval = gdHash(r->argv); |
94 |
r->next = gdProTab[hval]; |
95 |
gdProTab[hval] = r; |
96 |
} |
97 |
|
98 |
|
99 |
GDrequest * |
100 |
gdReqAlloc(nm, ac) /* allocate a request and its arguments */ |
101 |
char *nm; |
102 |
int ac; |
103 |
{ |
104 |
register GDrequest *nr; |
105 |
|
106 |
if (ac < 1) |
107 |
return(NULL); |
108 |
nr = (GDrequest *)malloc(sizeof(GDrequest)+(ac-1)*sizeof(GDarg)); |
109 |
if (nr == NULL) |
110 |
return(NULL); |
111 |
nr->next = NULL; |
112 |
nr->argc = ac; |
113 |
if (nm != NULL) |
114 |
(void)strncmp(rn->argv[0].v.n, nm, GD_MAXID); |
115 |
return(nr); |
116 |
} |
117 |
|
118 |
|
119 |
char * |
120 |
gdStrAlloc(a, str) /* allocate and save a string value */ |
121 |
register GDarg *a; |
122 |
char *str; |
123 |
{ |
124 |
a->typ = GD_TY_STR; |
125 |
a->v.s = (char *)malloc(strlen(str)+1); |
126 |
if (a->v.s == NULL) |
127 |
return(NULL); |
128 |
return(strcpy(a->v.s, str)); |
129 |
} |
130 |
|
131 |
|
132 |
MEM_PTR |
133 |
gdArrAlloc(a, typ, len, arr) /* allocate and assign an array */ |
134 |
register GDarg *a; |
135 |
int typ, len; |
136 |
MEM_PTR arr; |
137 |
{ |
138 |
static short esiz[GD_NTYPES] = GD_ELELEN; |
139 |
|
140 |
if (esiz[typ] <= 0) |
141 |
return(NULL); |
142 |
a->v.a.p = (MEM_PTR)malloc(len*esiz[typ]); |
143 |
if (a->v.a.p == NULL) |
144 |
return(NULL); |
145 |
a->typ = GD_TY_ARR; |
146 |
a->atyp = typ; |
147 |
a->v.a.l = len; |
148 |
if (arr != NULL) |
149 |
(void)memcpy((char *)a->v.a.p, (char *)arr, len*esiz[typ]); |
150 |
return(a->v.a.p); |
151 |
} |
152 |
|
153 |
|
154 |
gdDoneArg(a) /* free any argument data */ |
155 |
register GDarg *a; |
156 |
{ |
157 |
register int j; |
158 |
/* free allocated arrays */ |
159 |
switch (a->typ) { |
160 |
case GD_TY_ARR: /* array of... */ |
161 |
if (a->atyp == GD_TY_ARR) { /* arrays */ |
162 |
for (j = a->v.a.l; j--; ) |
163 |
gdDoneArg((GDarg *)a->v.a.p + j); |
164 |
} else if (a->atyp == GD_TY_STR) { /* strings */ |
165 |
for (j = r->v.a.l; j--; ) |
166 |
gdFree((char **)r->v.a.p + j); |
167 |
} |
168 |
gdFree(r->argv[i].v.a.p); |
169 |
break; |
170 |
case GD_TY_STR: /* string value */ |
171 |
gdFree(a->v.s); |
172 |
break; |
173 |
} |
174 |
} |
175 |
|
176 |
|
177 |
gdFreeReq(r) /* free a request */ |
178 |
register GDrequest *r; |
179 |
{ |
180 |
register int i; |
181 |
/* free any argument data */ |
182 |
for (i = r->argc; i--; ) |
183 |
gdDoneArg(r->argv + i); |
184 |
gdFree(r); /* free basic structure */ |
185 |
} |