497 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			497 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*-
 | 
						|
 * Copyright (c) 1991, 1993
 | 
						|
 *	The Regents of the University of California.  All rights reserved.
 | 
						|
 *
 | 
						|
 * This code is derived from software contributed to Berkeley by
 | 
						|
 * Kenneth Almquist.
 | 
						|
 *
 | 
						|
 * Redistribution and use in source and binary forms, with or without
 | 
						|
 * modification, are permitted provided that the following conditions
 | 
						|
 * are met:
 | 
						|
 * 1. Redistributions of source code must retain the above copyright
 | 
						|
 *    notice, this list of conditions and the following disclaimer.
 | 
						|
 * 2. Redistributions in binary form must reproduce the above copyright
 | 
						|
 *    notice, this list of conditions and the following disclaimer in the
 | 
						|
 *    documentation and/or other materials provided with the distribution.
 | 
						|
 * 4. Neither the name of the University nor the names of its contributors
 | 
						|
 *    may be used to endorse or promote products derived from this software
 | 
						|
 *    without specific prior written permission.
 | 
						|
 *
 | 
						|
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
 | 
						|
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | 
						|
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 | 
						|
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
 | 
						|
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 | 
						|
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 | 
						|
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 | 
						|
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 | 
						|
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 | 
						|
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 | 
						|
 * SUCH DAMAGE.
 | 
						|
 */
 | 
						|
 | 
						|
#ifndef lint
 | 
						|
static char const copyright[] =
 | 
						|
"@(#) Copyright (c) 1991, 1993\n\
 | 
						|
	The Regents of the University of California.  All rights reserved.\n";
 | 
						|
#endif /* not lint */
 | 
						|
 | 
						|
#ifndef lint
 | 
						|
#if 0
 | 
						|
static char sccsid[] = "@(#)mkinit.c	8.2 (Berkeley) 5/4/95";
 | 
						|
#endif
 | 
						|
#endif /* not lint */
 | 
						|
#include <sys/cdefs.h>
 | 
						|
/*
 | 
						|
__FBSDID("$FreeBSD: src/bin/sh/mkinit.c,v 1.17 2004/04/06 20:06:51 markm Exp $");
 | 
						|
*/
 | 
						|
 | 
						|
/*
 | 
						|
 * This program scans all the source files for code to handle various
 | 
						|
 * special events and combines this code into one file.  This (allegedly)
 | 
						|
 * improves the structure of the program since there is no need for
 | 
						|
 * anyone outside of a module to know that that module performs special
 | 
						|
 * operations on particular events.
 | 
						|
 *
 | 
						|
 * Usage:  mkinit sourcefile...
 | 
						|
 */
 | 
						|
 | 
						|
 | 
						|
#include <sys/types.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
#include <fcntl.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <errno.h>
 | 
						|
 | 
						|
/*
 | 
						|
 * OUTFILE is the name of the output file.  Output is initially written
 | 
						|
 * to the file OUTTEMP, which is then moved to OUTFILE.
 | 
						|
 */
 | 
						|
 | 
						|
#define OUTFILE "init.c"
 | 
						|
#define OUTTEMP "init.c.new"
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * A text structure is basicly just a string that grows as more characters
 | 
						|
 * are added onto the end of it.  It is implemented as a linked list of
 | 
						|
 * blocks of characters.  The routines addstr and addchar append a string
 | 
						|
 * or a single character, respectively, to a text structure.  Writetext
 | 
						|
 * writes the contents of a text structure to a file.
 | 
						|
 */
 | 
						|
 | 
						|
#define BLOCKSIZE 512
 | 
						|
 | 
						|
struct text {
 | 
						|
	char *nextc;
 | 
						|
	int nleft;
 | 
						|
	struct block *start;
 | 
						|
	struct block *last;
 | 
						|
};
 | 
						|
 | 
						|
struct block {
 | 
						|
	struct block *next;
 | 
						|
	char text[BLOCKSIZE];
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * There is one event structure for each event that mkinit handles.
 | 
						|
 */
 | 
						|
 | 
						|
struct event {
 | 
						|
	char *name;		/* name of event (e.g. INIT) */
 | 
						|
	char *routine;		/* name of routine called on event */
 | 
						|
	char *comment;		/* comment describing routine */
 | 
						|
	struct text code;	/* code for handling event */
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
char writer[] = "\
 | 
						|
/*\n\
 | 
						|
 * This file was generated by the mkinit program.\n\
 | 
						|
 */\n\
 | 
						|
\n";
 | 
						|
 | 
						|
char init[] = "\
 | 
						|
/*\n\
 | 
						|
 * Initialization code.\n\
 | 
						|
 */\n";
 | 
						|
 | 
						|
char reset[] = "\
 | 
						|
/*\n\
 | 
						|
 * This routine is called when an error or an interrupt occurs in an\n\
 | 
						|
 * interactive shell and control is returned to the main command loop.\n\
 | 
						|
 */\n";
 | 
						|
 | 
						|
char shellproc[] = "\
 | 
						|
/*\n\
 | 
						|
 * This routine is called to initialize the shell to run a shell procedure.\n\
 | 
						|
 */\n";
 | 
						|
 | 
						|
 | 
						|
struct event event[] = {
 | 
						|
	{"INIT", "init", init},
 | 
						|
	{"RESET", "reset", reset},
 | 
						|
	{"SHELLPROC", "initshellproc", shellproc},
 | 
						|
	{NULL, NULL}
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
char *curfile;				/* current file */
 | 
						|
int linno;				/* current line */
 | 
						|
char *header_files[200];		/* list of header files */
 | 
						|
struct text defines;			/* #define statements */
 | 
						|
struct text decls;			/* declarations */
 | 
						|
int amiddecls;				/* for formatting */
 | 
						|
 | 
						|
 | 
						|
static void readfile(char *);
 | 
						|
static int match(char *, char *);
 | 
						|
static int gooddefine(char *);
 | 
						|
static void doevent(struct event *, FILE *, char *);
 | 
						|
static void doinclude(char *);
 | 
						|
static void dodecl(char *, FILE *);
 | 
						|
static void output(void);
 | 
						|
static void addstr(char *, struct text *);
 | 
						|
static void addchar(int, struct text *);
 | 
						|
static void writetext(struct text *, FILE *);
 | 
						|
static FILE *ckfopen(char *, char *);
 | 
						|
static void *ckmalloc(int);
 | 
						|
static char *savestr(char *);
 | 
						|
static void error(char *);
 | 
						|
 | 
						|
#define equal(s1, s2)	(strcmp(s1, s2) == 0)
 | 
						|
 | 
						|
int
 | 
						|
main(int argc __unused, char *argv[])
 | 
						|
{
 | 
						|
	char **ap;
 | 
						|
 | 
						|
	header_files[0] = "\"shell.h\"";
 | 
						|
	header_files[1] = "\"mystring.h\"";
 | 
						|
	for (ap = argv + 1 ; *ap ; ap++)
 | 
						|
		readfile(*ap);
 | 
						|
	output();
 | 
						|
	rename(OUTTEMP, OUTFILE);
 | 
						|
	exit(0);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Parse an input file.
 | 
						|
 */
 | 
						|
 | 
						|
static void
 | 
						|
readfile(char *fname)
 | 
						|
{
 | 
						|
	FILE *fp;
 | 
						|
	char line[1024];
 | 
						|
	struct event *ep;
 | 
						|
 | 
						|
	fp = ckfopen(fname, "r");
 | 
						|
	curfile = fname;
 | 
						|
	linno = 0;
 | 
						|
	amiddecls = 0;
 | 
						|
	while (fgets(line, sizeof line, fp) != NULL) {
 | 
						|
		linno++;
 | 
						|
		for (ep = event ; ep->name ; ep++) {
 | 
						|
			if (line[0] == ep->name[0] && match(ep->name, line)) {
 | 
						|
				doevent(ep, fp, fname);
 | 
						|
				break;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if (line[0] == 'I' && match("INCLUDE", line))
 | 
						|
			doinclude(line);
 | 
						|
		if (line[0] == 'M' && match("MKINIT", line))
 | 
						|
			dodecl(line, fp);
 | 
						|
		if (line[0] == '#' && gooddefine(line)) {
 | 
						|
			char *cp;
 | 
						|
			char line2[1024];
 | 
						|
			static const char undef[] = "#undef ";
 | 
						|
 | 
						|
			strcpy(line2, line);
 | 
						|
			memcpy(line2, undef, sizeof(undef) - 1);
 | 
						|
			cp = line2 + sizeof(undef) - 1;
 | 
						|
			while(*cp && (*cp == ' ' || *cp == '\t'))
 | 
						|
			        cp++;
 | 
						|
			while(*cp && *cp != ' ' && *cp != '\t' && *cp != '\n')
 | 
						|
			        cp++;
 | 
						|
			*cp++ = '\n'; *cp = '\0';
 | 
						|
			addstr(line2, &defines);
 | 
						|
			addstr(line, &defines);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	fclose(fp);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static int
 | 
						|
match(char *name, char *line)
 | 
						|
{
 | 
						|
	char *p, *q;
 | 
						|
 | 
						|
	p = name, q = line;
 | 
						|
	while (*p) {
 | 
						|
		if (*p++ != *q++)
 | 
						|
			return 0;
 | 
						|
	}
 | 
						|
	if (*q != '{' && *q != ' ' && *q != '\t' && *q != '\n')
 | 
						|
		return 0;
 | 
						|
	return 1;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static int
 | 
						|
gooddefine(char *line)
 | 
						|
{
 | 
						|
	char *p;
 | 
						|
 | 
						|
	if (! match("#define", line))
 | 
						|
		return 0;			/* not a define */
 | 
						|
	p = line + 7;
 | 
						|
	while (*p == ' ' || *p == '\t')
 | 
						|
		p++;
 | 
						|
	while (*p != ' ' && *p != '\t') {
 | 
						|
		if (*p == '(')
 | 
						|
			return 0;		/* macro definition */
 | 
						|
		p++;
 | 
						|
	}
 | 
						|
	while (*p != '\n' && *p != '\0')
 | 
						|
		p++;
 | 
						|
	if (p[-1] == '\\')
 | 
						|
		return 0;			/* multi-line definition */
 | 
						|
	return 1;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static void
 | 
						|
doevent(struct event *ep, FILE *fp, char *fname)
 | 
						|
{
 | 
						|
	char line[1024];
 | 
						|
	int indent;
 | 
						|
	char *p;
 | 
						|
 | 
						|
	sprintf(line, "\n      /* from %s: */\n", fname);
 | 
						|
	addstr(line, &ep->code);
 | 
						|
	addstr("      {\n", &ep->code);
 | 
						|
	for (;;) {
 | 
						|
		linno++;
 | 
						|
		if (fgets(line, sizeof line, fp) == NULL)
 | 
						|
			error("Unexpected EOF");
 | 
						|
		if (equal(line, "}\n"))
 | 
						|
			break;
 | 
						|
		indent = 6;
 | 
						|
		for (p = line ; *p == '\t' ; p++)
 | 
						|
			indent += 8;
 | 
						|
		for ( ; *p == ' ' ; p++)
 | 
						|
			indent++;
 | 
						|
		if (*p == '\n' || *p == '#')
 | 
						|
			indent = 0;
 | 
						|
		while (indent >= 8) {
 | 
						|
			addchar('\t', &ep->code);
 | 
						|
			indent -= 8;
 | 
						|
		}
 | 
						|
		while (indent > 0) {
 | 
						|
			addchar(' ', &ep->code);
 | 
						|
			indent--;
 | 
						|
		}
 | 
						|
		addstr(p, &ep->code);
 | 
						|
	}
 | 
						|
	addstr("      }\n", &ep->code);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static void
 | 
						|
doinclude(char *line)
 | 
						|
{
 | 
						|
	char *p;
 | 
						|
	char *name;
 | 
						|
	char **pp;
 | 
						|
 | 
						|
	for (p = line ; *p != '"' && *p != '<' && *p != '\0' ; p++);
 | 
						|
	if (*p == '\0')
 | 
						|
		error("Expecting '\"' or '<'");
 | 
						|
	name = p;
 | 
						|
	while (*p != ' ' && *p != '\t' && *p != '\n')
 | 
						|
		p++;
 | 
						|
	if (p[-1] != '"' && p[-1] != '>')
 | 
						|
		error("Missing terminator");
 | 
						|
	*p = '\0';
 | 
						|
 | 
						|
	/* name now contains the name of the include file */
 | 
						|
	for (pp = header_files ; *pp && ! equal(*pp, name) ; pp++);
 | 
						|
	if (*pp == NULL)
 | 
						|
		*pp = savestr(name);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static void
 | 
						|
dodecl(char *line1, FILE *fp)
 | 
						|
{
 | 
						|
	char line[1024];
 | 
						|
	char *p, *q;
 | 
						|
 | 
						|
	if (strcmp(line1, "MKINIT\n") == 0) { /* start of struct/union decl */
 | 
						|
		addchar('\n', &decls);
 | 
						|
		do {
 | 
						|
			linno++;
 | 
						|
			if (fgets(line, sizeof line, fp) == NULL)
 | 
						|
				error("Unterminated structure declaration");
 | 
						|
			addstr(line, &decls);
 | 
						|
		} while (line[0] != '}');
 | 
						|
		amiddecls = 0;
 | 
						|
	} else {
 | 
						|
		if (! amiddecls)
 | 
						|
			addchar('\n', &decls);
 | 
						|
		q = NULL;
 | 
						|
		for (p = line1 + 6 ; *p && strchr("=/\n", *p) == NULL; p++)
 | 
						|
			continue;
 | 
						|
		if (*p == '=') {		/* eliminate initialization */
 | 
						|
			for (q = p ; *q && *q != ';' ; q++);
 | 
						|
			if (*q == '\0')
 | 
						|
				q = NULL;
 | 
						|
			else {
 | 
						|
				while (p[-1] == ' ')
 | 
						|
					p--;
 | 
						|
				*p = '\0';
 | 
						|
			}
 | 
						|
		}
 | 
						|
		addstr("extern", &decls);
 | 
						|
		addstr(line1 + 6, &decls);
 | 
						|
		if (q != NULL)
 | 
						|
			addstr(q, &decls);
 | 
						|
		amiddecls = 1;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Write the output to the file OUTTEMP.
 | 
						|
 */
 | 
						|
 | 
						|
static void
 | 
						|
output(void)
 | 
						|
{
 | 
						|
	FILE *fp;
 | 
						|
	char **pp;
 | 
						|
	struct event *ep;
 | 
						|
 | 
						|
	fp = ckfopen(OUTTEMP, "w");
 | 
						|
	fputs(writer, fp);
 | 
						|
	for (pp = header_files ; *pp ; pp++)
 | 
						|
		fprintf(fp, "#include %s\n", *pp);
 | 
						|
	fputs("\n\n\n", fp);
 | 
						|
	writetext(&defines, fp);
 | 
						|
	fputs("\n\n", fp);
 | 
						|
	writetext(&decls, fp);
 | 
						|
	for (ep = event ; ep->name ; ep++) {
 | 
						|
		fputs("\n\n\n", fp);
 | 
						|
		fputs(ep->comment, fp);
 | 
						|
		fprintf(fp, "\nvoid\n%s(void) {\n", ep->routine);
 | 
						|
		writetext(&ep->code, fp);
 | 
						|
		fprintf(fp, "}\n");
 | 
						|
	}
 | 
						|
	fclose(fp);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * A text structure is simply a block of text that is kept in memory.
 | 
						|
 * Addstr appends a string to the text struct, and addchar appends a single
 | 
						|
 * character.
 | 
						|
 */
 | 
						|
 | 
						|
static void
 | 
						|
addstr(char *s, struct text *text)
 | 
						|
{
 | 
						|
	while (*s) {
 | 
						|
		if (--text->nleft < 0)
 | 
						|
			addchar(*s++, text);
 | 
						|
		else
 | 
						|
			*text->nextc++ = *s++;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static void
 | 
						|
addchar(int c, struct text *text)
 | 
						|
{
 | 
						|
	struct block *bp;
 | 
						|
 | 
						|
	if (--text->nleft < 0) {
 | 
						|
		bp = ckmalloc(sizeof *bp);
 | 
						|
		if (text->start == NULL)
 | 
						|
			text->start = bp;
 | 
						|
		else
 | 
						|
			text->last->next = bp;
 | 
						|
		text->last = bp;
 | 
						|
		text->nextc = bp->text;
 | 
						|
		text->nleft = BLOCKSIZE - 1;
 | 
						|
	}
 | 
						|
	*text->nextc++ = c;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Write the contents of a text structure to a file.
 | 
						|
 */
 | 
						|
static void
 | 
						|
writetext(struct text *text, FILE *fp)
 | 
						|
{
 | 
						|
	struct block *bp;
 | 
						|
 | 
						|
	if (text->start != NULL) {
 | 
						|
		for (bp = text->start ; bp != text->last ; bp = bp->next)
 | 
						|
			fwrite(bp->text, sizeof (char), BLOCKSIZE, fp);
 | 
						|
		fwrite(bp->text, sizeof (char), BLOCKSIZE - text->nleft, fp);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static FILE *
 | 
						|
ckfopen(char *file, char *mode)
 | 
						|
{
 | 
						|
	FILE *fp;
 | 
						|
 | 
						|
	if ((fp = fopen(file, mode)) == NULL) {
 | 
						|
		fprintf(stderr, "Can't open %s: %s\n", file, strerror(errno));
 | 
						|
		exit(2);
 | 
						|
	}
 | 
						|
	return fp;
 | 
						|
}
 | 
						|
 | 
						|
static void *
 | 
						|
ckmalloc(int nbytes)
 | 
						|
{
 | 
						|
	char *p;
 | 
						|
 | 
						|
	if ((p = malloc(nbytes)) == NULL)
 | 
						|
		error("Out of space");
 | 
						|
	return p;
 | 
						|
}
 | 
						|
 | 
						|
static char *
 | 
						|
savestr(char *s)
 | 
						|
{
 | 
						|
	char *p;
 | 
						|
 | 
						|
	p = ckmalloc(strlen(s) + 1);
 | 
						|
	strcpy(p, s);
 | 
						|
	return p;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
error(char *msg)
 | 
						|
{
 | 
						|
	if (curfile != NULL)
 | 
						|
		fprintf(stderr, "%s:%d: ", curfile, linno);
 | 
						|
	fprintf(stderr, "%s\n", msg);
 | 
						|
	exit(2);
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * $PchId: mkinit.c,v 1.6 2006/05/22 12:16:50 philip Exp $
 | 
						|
 */
 |