rpc_util.c revision 1.9 1 1.9 tv /* $NetBSD: rpc_util.c,v 1.9 2002/01/31 19:36:50 tv Exp $ */
2 1.1 glass /*
3 1.1 glass * Sun RPC is a product of Sun Microsystems, Inc. and is provided for
4 1.1 glass * unrestricted use provided that this legend is included on all tape
5 1.1 glass * media and as a part of the software program in whole or part. Users
6 1.1 glass * may copy or modify Sun RPC without charge, but are not authorized
7 1.1 glass * to license or distribute it to anyone else except as part of a product or
8 1.5 pk * program developed by the user or with the express written consent of
9 1.5 pk * Sun Microsystems, Inc.
10 1.5 pk *
11 1.1 glass * SUN RPC IS PROVIDED AS IS WITH NO WARRANTIES OF ANY KIND INCLUDING THE
12 1.1 glass * WARRANTIES OF DESIGN, MERCHANTIBILITY AND FITNESS FOR A PARTICULAR
13 1.1 glass * PURPOSE, OR ARISING FROM A COURSE OF DEALING, USAGE OR TRADE PRACTICE.
14 1.5 pk *
15 1.1 glass * Sun RPC is provided with no support and without any obligation on the
16 1.1 glass * part of Sun Microsystems, Inc. to assist in its use, correction,
17 1.1 glass * modification or enhancement.
18 1.5 pk *
19 1.1 glass * SUN MICROSYSTEMS, INC. SHALL HAVE NO LIABILITY WITH RESPECT TO THE
20 1.1 glass * INFRINGEMENT OF COPYRIGHTS, TRADE SECRETS OR ANY PATENTS BY SUN RPC
21 1.1 glass * OR ANY PART THEREOF.
22 1.5 pk *
23 1.1 glass * In no event will Sun Microsystems, Inc. be liable for any lost revenue
24 1.1 glass * or profits or other special, indirect and consequential damages, even if
25 1.1 glass * Sun has been advised of the possibility of such damages.
26 1.5 pk *
27 1.1 glass * Sun Microsystems, Inc.
28 1.1 glass * 2550 Garcia Avenue
29 1.1 glass * Mountain View, California 94043
30 1.1 glass */
31 1.5 pk
32 1.7 christos #include <sys/cdefs.h>
33 1.9 tv #if defined(__RCSID) && !defined(lint)
34 1.7 christos #if 0
35 1.5 pk static char sccsid[] = "@(#)rpc_util.c 1.11 89/02/22 (C) 1987 SMI";
36 1.7 christos #else
37 1.9 tv __RCSID("$NetBSD: rpc_util.c,v 1.9 2002/01/31 19:36:50 tv Exp $");
38 1.7 christos #endif
39 1.1 glass #endif
40 1.1 glass
41 1.1 glass /*
42 1.8 lukem * rpc_util.c, Utility routines for the RPC protocol compiler
43 1.1 glass */
44 1.1 glass #include <stdio.h>
45 1.6 cgd #include <stdlib.h>
46 1.7 christos #include <unistd.h>
47 1.6 cgd #include <string.h>
48 1.5 pk #include <ctype.h>
49 1.1 glass #include "rpc_scan.h"
50 1.1 glass #include "rpc_parse.h"
51 1.1 glass #include "rpc_util.h"
52 1.1 glass
53 1.5 pk #define ARGEXT "argument"
54 1.5 pk
55 1.5 pk static void printwhere __P((void));
56 1.5 pk
57 1.8 lukem char curline[MAXLINESIZE]; /* current read line */
58 1.8 lukem char *where = curline; /* current point in line */
59 1.8 lukem int linenum = 0; /* current line number */
60 1.1 glass
61 1.8 lukem char *infilename; /* input filename */
62 1.1 glass
63 1.5 pk #define NFILES 7
64 1.8 lukem char *outfiles[NFILES]; /* output file names */
65 1.8 lukem int nfiles;
66 1.1 glass
67 1.8 lukem FILE *fout; /* file pointer of current output */
68 1.8 lukem FILE *fin; /* file pointer of current input */
69 1.1 glass
70 1.8 lukem list *defined; /* list of defined things */
71 1.1 glass
72 1.7 christos static char *toktostr __P((tok_kind));
73 1.7 christos static void printbuf __P((void));
74 1.7 christos static void printwhere __P((void));
75 1.7 christos static int findit __P((definition *, char *));
76 1.7 christos static char *fixit __P((char *, char *));
77 1.7 christos static int typedefed __P((definition *, char *));
78 1.7 christos
79 1.1 glass /*
80 1.8 lukem * Reinitialize the world
81 1.1 glass */
82 1.7 christos void
83 1.1 glass reinitialize()
84 1.1 glass {
85 1.4 cgd memset(curline, 0, MAXLINESIZE);
86 1.1 glass where = curline;
87 1.1 glass linenum = 0;
88 1.1 glass defined = NULL;
89 1.1 glass }
90 1.1 glass /*
91 1.8 lukem * string equality
92 1.1 glass */
93 1.7 christos int
94 1.1 glass streq(a, b)
95 1.8 lukem char *a;
96 1.8 lukem char *b;
97 1.1 glass {
98 1.1 glass return (strcmp(a, b) == 0);
99 1.1 glass }
100 1.1 glass /*
101 1.8 lukem * find a value in a list
102 1.1 glass */
103 1.5 pk definition *
104 1.1 glass findval(lst, val, cmp)
105 1.8 lukem list *lst;
106 1.8 lukem char *val;
107 1.8 lukem int (*cmp) __P((definition *, char *));
108 1.1 glass
109 1.1 glass {
110 1.8 lukem
111 1.1 glass for (; lst != NULL; lst = lst->next) {
112 1.1 glass if ((*cmp) (lst->val, val)) {
113 1.1 glass return (lst->val);
114 1.1 glass }
115 1.1 glass }
116 1.1 glass return (NULL);
117 1.1 glass }
118 1.1 glass /*
119 1.8 lukem * store a value in a list
120 1.1 glass */
121 1.1 glass void
122 1.1 glass storeval(lstp, val)
123 1.8 lukem list **lstp;
124 1.5 pk definition *val;
125 1.1 glass {
126 1.8 lukem list **l;
127 1.8 lukem list *lst;
128 1.8 lukem
129 1.1 glass
130 1.1 glass for (l = lstp; *l != NULL; l = (list **) & (*l)->next);
131 1.1 glass lst = ALLOC(list);
132 1.1 glass lst->val = val;
133 1.1 glass lst->next = NULL;
134 1.1 glass *l = lst;
135 1.1 glass }
136 1.1 glass
137 1.7 christos static int
138 1.1 glass findit(def, type)
139 1.1 glass definition *def;
140 1.8 lukem char *type;
141 1.1 glass {
142 1.1 glass return (streq(def->def_name, type));
143 1.1 glass }
144 1.1 glass
145 1.1 glass static char *
146 1.1 glass fixit(type, orig)
147 1.8 lukem char *type;
148 1.8 lukem char *orig;
149 1.1 glass {
150 1.1 glass definition *def;
151 1.1 glass
152 1.1 glass def = (definition *) FINDVAL(defined, type, findit);
153 1.1 glass if (def == NULL || def->def_kind != DEF_TYPEDEF) {
154 1.1 glass return (orig);
155 1.1 glass }
156 1.1 glass switch (def->def.ty.rel) {
157 1.1 glass case REL_VECTOR:
158 1.1 glass return (def->def.ty.old_type);
159 1.1 glass case REL_ALIAS:
160 1.1 glass return (fixit(def->def.ty.old_type, orig));
161 1.1 glass default:
162 1.1 glass return (orig);
163 1.1 glass }
164 1.1 glass }
165 1.1 glass
166 1.8 lukem char *
167 1.1 glass fixtype(type)
168 1.8 lukem char *type;
169 1.1 glass {
170 1.1 glass return (fixit(type, type));
171 1.1 glass }
172 1.1 glass
173 1.8 lukem char *
174 1.1 glass stringfix(type)
175 1.8 lukem char *type;
176 1.1 glass {
177 1.1 glass if (streq(type, "string")) {
178 1.1 glass return ("wrapstring");
179 1.1 glass } else {
180 1.1 glass return (type);
181 1.1 glass }
182 1.1 glass }
183 1.1 glass
184 1.1 glass void
185 1.1 glass ptype(prefix, type, follow)
186 1.8 lukem char *prefix;
187 1.8 lukem char *type;
188 1.8 lukem int follow;
189 1.1 glass {
190 1.1 glass if (prefix != NULL) {
191 1.1 glass if (streq(prefix, "enum")) {
192 1.1 glass f_print(fout, "enum ");
193 1.1 glass } else {
194 1.1 glass f_print(fout, "struct ");
195 1.1 glass }
196 1.1 glass }
197 1.1 glass if (streq(type, "bool")) {
198 1.1 glass f_print(fout, "bool_t ");
199 1.8 lukem } else
200 1.8 lukem if (streq(type, "string")) {
201 1.8 lukem f_print(fout, "char *");
202 1.8 lukem } else {
203 1.8 lukem f_print(fout, "%s ", follow ? fixtype(type) : type);
204 1.8 lukem }
205 1.1 glass }
206 1.1 glass
207 1.7 christos static int
208 1.1 glass typedefed(def, type)
209 1.1 glass definition *def;
210 1.8 lukem char *type;
211 1.1 glass {
212 1.1 glass if (def->def_kind != DEF_TYPEDEF || def->def.ty.old_prefix != NULL) {
213 1.1 glass return (0);
214 1.1 glass } else {
215 1.1 glass return (streq(def->def_name, type));
216 1.1 glass }
217 1.1 glass }
218 1.1 glass
219 1.7 christos int
220 1.1 glass isvectordef(type, rel)
221 1.8 lukem char *type;
222 1.1 glass relation rel;
223 1.1 glass {
224 1.1 glass definition *def;
225 1.1 glass
226 1.1 glass for (;;) {
227 1.1 glass switch (rel) {
228 1.1 glass case REL_VECTOR:
229 1.1 glass return (!streq(type, "string"));
230 1.1 glass case REL_ARRAY:
231 1.1 glass return (0);
232 1.1 glass case REL_POINTER:
233 1.1 glass return (0);
234 1.1 glass case REL_ALIAS:
235 1.1 glass def = (definition *) FINDVAL(defined, type, typedefed);
236 1.1 glass if (def == NULL) {
237 1.1 glass return (0);
238 1.1 glass }
239 1.1 glass type = def->def.ty.old_type;
240 1.1 glass rel = def->def.ty.rel;
241 1.1 glass }
242 1.1 glass }
243 1.1 glass }
244 1.1 glass
245 1.8 lukem char *
246 1.1 glass locase(str)
247 1.8 lukem char *str;
248 1.1 glass {
249 1.8 lukem char c;
250 1.1 glass static char buf[100];
251 1.8 lukem char *p = buf;
252 1.1 glass
253 1.7 christos while ((c = *str++) != '\0') {
254 1.1 glass *p++ = (c >= 'A' && c <= 'Z') ? (c - 'A' + 'a') : c;
255 1.1 glass }
256 1.1 glass *p = 0;
257 1.1 glass return (buf);
258 1.1 glass }
259 1.1 glass
260 1.5 pk void
261 1.5 pk pvname_svc(pname, vnum)
262 1.8 lukem char *pname;
263 1.8 lukem char *vnum;
264 1.5 pk {
265 1.5 pk f_print(fout, "%s_%s_svc", locase(pname), vnum);
266 1.5 pk }
267 1.1 glass
268 1.1 glass void
269 1.1 glass pvname(pname, vnum)
270 1.8 lukem char *pname;
271 1.8 lukem char *vnum;
272 1.1 glass {
273 1.1 glass f_print(fout, "%s_%s", locase(pname), vnum);
274 1.1 glass }
275 1.1 glass /*
276 1.8 lukem * print a useful (?) error message, and then die
277 1.1 glass */
278 1.1 glass void
279 1.1 glass error(msg)
280 1.8 lukem char *msg;
281 1.1 glass {
282 1.1 glass printwhere();
283 1.1 glass f_print(stderr, "%s, line %d: ", infilename, linenum);
284 1.1 glass f_print(stderr, "%s\n", msg);
285 1.1 glass crash();
286 1.1 glass }
287 1.1 glass /*
288 1.1 glass * Something went wrong, unlink any files that we may have created and then
289 1.8 lukem * die.
290 1.1 glass */
291 1.7 christos void
292 1.1 glass crash()
293 1.1 glass {
294 1.8 lukem int i;
295 1.1 glass
296 1.1 glass for (i = 0; i < nfiles; i++) {
297 1.1 glass (void) unlink(outfiles[i]);
298 1.1 glass }
299 1.1 glass exit(1);
300 1.1 glass }
301 1.1 glass
302 1.1 glass void
303 1.1 glass record_open(file)
304 1.8 lukem char *file;
305 1.1 glass {
306 1.1 glass if (nfiles < NFILES) {
307 1.1 glass outfiles[nfiles++] = file;
308 1.1 glass } else {
309 1.1 glass f_print(stderr, "too many files!\n");
310 1.1 glass crash();
311 1.1 glass }
312 1.1 glass }
313 1.1 glass
314 1.1 glass static char expectbuf[100];
315 1.1 glass
316 1.1 glass /*
317 1.8 lukem * error, token encountered was not the expected one
318 1.1 glass */
319 1.1 glass void
320 1.1 glass expected1(exp1)
321 1.1 glass tok_kind exp1;
322 1.1 glass {
323 1.1 glass s_print(expectbuf, "expected '%s'",
324 1.8 lukem toktostr(exp1));
325 1.1 glass error(expectbuf);
326 1.1 glass }
327 1.1 glass /*
328 1.8 lukem * error, token encountered was not one of two expected ones
329 1.1 glass */
330 1.1 glass void
331 1.1 glass expected2(exp1, exp2)
332 1.1 glass tok_kind exp1, exp2;
333 1.1 glass {
334 1.1 glass s_print(expectbuf, "expected '%s' or '%s'",
335 1.8 lukem toktostr(exp1),
336 1.8 lukem toktostr(exp2));
337 1.1 glass error(expectbuf);
338 1.1 glass }
339 1.1 glass /*
340 1.8 lukem * error, token encountered was not one of 3 expected ones
341 1.1 glass */
342 1.1 glass void
343 1.1 glass expected3(exp1, exp2, exp3)
344 1.1 glass tok_kind exp1, exp2, exp3;
345 1.1 glass {
346 1.1 glass s_print(expectbuf, "expected '%s', '%s' or '%s'",
347 1.8 lukem toktostr(exp1),
348 1.8 lukem toktostr(exp2),
349 1.8 lukem toktostr(exp3));
350 1.1 glass error(expectbuf);
351 1.1 glass }
352 1.1 glass
353 1.1 glass void
354 1.1 glass tabify(f, tab)
355 1.8 lukem FILE *f;
356 1.8 lukem int tab;
357 1.1 glass {
358 1.1 glass while (tab--) {
359 1.1 glass (void) fputc('\t', f);
360 1.1 glass }
361 1.1 glass }
362 1.1 glass
363 1.1 glass
364 1.1 glass static token tokstrings[] = {
365 1.5 pk {TOK_IDENT, "identifier"},
366 1.5 pk {TOK_CONST, "const"},
367 1.5 pk {TOK_RPAREN, ")"},
368 1.5 pk {TOK_LPAREN, "("},
369 1.5 pk {TOK_RBRACE, "}"},
370 1.5 pk {TOK_LBRACE, "{"},
371 1.5 pk {TOK_LBRACKET, "["},
372 1.5 pk {TOK_RBRACKET, "]"},
373 1.5 pk {TOK_STAR, "*"},
374 1.5 pk {TOK_COMMA, ","},
375 1.5 pk {TOK_EQUAL, "="},
376 1.5 pk {TOK_COLON, ":"},
377 1.5 pk {TOK_SEMICOLON, ";"},
378 1.5 pk {TOK_UNION, "union"},
379 1.5 pk {TOK_STRUCT, "struct"},
380 1.5 pk {TOK_SWITCH, "switch"},
381 1.5 pk {TOK_CASE, "case"},
382 1.5 pk {TOK_DEFAULT, "default"},
383 1.5 pk {TOK_ENUM, "enum"},
384 1.5 pk {TOK_TYPEDEF, "typedef"},
385 1.5 pk {TOK_INT, "int"},
386 1.5 pk {TOK_SHORT, "short"},
387 1.5 pk {TOK_LONG, "long"},
388 1.5 pk {TOK_UNSIGNED, "unsigned"},
389 1.5 pk {TOK_DOUBLE, "double"},
390 1.5 pk {TOK_FLOAT, "float"},
391 1.5 pk {TOK_CHAR, "char"},
392 1.5 pk {TOK_STRING, "string"},
393 1.5 pk {TOK_OPAQUE, "opaque"},
394 1.5 pk {TOK_BOOL, "bool"},
395 1.5 pk {TOK_VOID, "void"},
396 1.5 pk {TOK_PROGRAM, "program"},
397 1.5 pk {TOK_VERSION, "version"},
398 1.5 pk {TOK_EOF, "??????"}
399 1.1 glass };
400 1.1 glass
401 1.1 glass static char *
402 1.1 glass toktostr(kind)
403 1.1 glass tok_kind kind;
404 1.1 glass {
405 1.8 lukem token *sp;
406 1.1 glass
407 1.1 glass for (sp = tokstrings; sp->kind != TOK_EOF && sp->kind != kind; sp++);
408 1.1 glass return (sp->str);
409 1.1 glass }
410 1.1 glass
411 1.7 christos static void
412 1.1 glass printbuf()
413 1.1 glass {
414 1.8 lukem char c;
415 1.8 lukem int i;
416 1.8 lukem int cnt;
417 1.1 glass
418 1.8 lukem #define TABSIZE 4
419 1.1 glass
420 1.7 christos for (i = 0; (c = curline[i]) != '\0'; i++) {
421 1.1 glass if (c == '\t') {
422 1.1 glass cnt = 8 - (i % TABSIZE);
423 1.1 glass c = ' ';
424 1.1 glass } else {
425 1.1 glass cnt = 1;
426 1.1 glass }
427 1.1 glass while (cnt--) {
428 1.1 glass (void) fputc(c, stderr);
429 1.1 glass }
430 1.1 glass }
431 1.1 glass }
432 1.1 glass
433 1.5 pk static void
434 1.1 glass printwhere()
435 1.1 glass {
436 1.8 lukem int i;
437 1.8 lukem char c;
438 1.8 lukem int cnt;
439 1.1 glass
440 1.1 glass printbuf();
441 1.1 glass for (i = 0; i < where - curline; i++) {
442 1.1 glass c = curline[i];
443 1.1 glass if (c == '\t') {
444 1.1 glass cnt = 8 - (i % TABSIZE);
445 1.1 glass } else {
446 1.1 glass cnt = 1;
447 1.1 glass }
448 1.1 glass while (cnt--) {
449 1.1 glass (void) fputc('^', stderr);
450 1.1 glass }
451 1.1 glass }
452 1.1 glass (void) fputc('\n', stderr);
453 1.1 glass }
454 1.5 pk
455 1.8 lukem char *
456 1.8 lukem make_argname(pname, vname)
457 1.8 lukem char *pname;
458 1.8 lukem char *vname;
459 1.8 lukem {
460 1.8 lukem char *name;
461 1.8 lukem
462 1.8 lukem name = (char *) malloc(strlen(pname) + strlen(vname) + strlen(ARGEXT) + 3);
463 1.5 pk if (!name) {
464 1.5 pk fprintf(stderr, "failed in malloc");
465 1.5 pk exit(1);
466 1.5 pk }
467 1.5 pk sprintf(name, "%s_%s_%s", locase(pname), vname, ARGEXT);
468 1.8 lukem return (name);
469 1.5 pk }
470 1.5 pk
471 1.5 pk bas_type *typ_list_h;
472 1.5 pk bas_type *typ_list_t;
473 1.5 pk
474 1.5 pk void
475 1.8 lukem add_type(len, type)
476 1.8 lukem int len;
477 1.8 lukem char *type;
478 1.5 pk {
479 1.5 pk bas_type *ptr;
480 1.5 pk
481 1.8 lukem if ((ptr = (bas_type *) malloc(sizeof(bas_type))) == (bas_type *) NULL) {
482 1.5 pk fprintf(stderr, "failed in malloc");
483 1.5 pk exit(1);
484 1.5 pk }
485 1.8 lukem ptr->name = type;
486 1.8 lukem ptr->length = len;
487 1.8 lukem ptr->next = NULL;
488 1.5 pk if (typ_list_t == NULL) {
489 1.8 lukem typ_list_t = ptr;
490 1.8 lukem typ_list_h = ptr;
491 1.5 pk } else {
492 1.8 lukem typ_list_t->next = ptr;
493 1.8 lukem typ_list_t = ptr;
494 1.5 pk }
495 1.5 pk }
496 1.5 pk
497 1.5 pk bas_type *
498 1.5 pk find_type(type)
499 1.8 lukem char *type;
500 1.5 pk {
501 1.8 lukem bas_type *ptr;
502 1.5 pk
503 1.8 lukem ptr = typ_list_h;
504 1.5 pk
505 1.5 pk
506 1.5 pk while (ptr != NULL) {
507 1.8 lukem if (strcmp(ptr->name, type) == 0)
508 1.8 lukem return (ptr);
509 1.5 pk else
510 1.8 lukem ptr = ptr->next;
511 1.5 pk }
512 1.8 lukem return (NULL);
513 1.5 pk }
514