OpenSolaris_b135/cmd/geniconvtbl/itmcomp.c

Compare this file to the similar file:
Show the results in this format:

/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License, Version 1.0 only
 * (the "License").  You may not use this file except in compliance
 * with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */
/*
 * Copyright (c) 1999 by Sun Microsystems, Inc.
 * All rights reserved.
 */

#pragma ident	"%Z%%M%	%I%	%E% SMI"

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <libgen.h>
#include <fcntl.h>
#include <locale.h>
#include <string.h>
#include <sys/wait.h>
#include <sys/param.h>
#include <stdarg.h>
#include <errno.h>

#include "itmcomp.h"
#include "maptype.h"

#if !defined(TEXT_DOMAIN)
#define	TEXT_DOMAIN	"SYS_TEST"
#endif
#define	ITMSUFFIX	".bt"
#define	ME_DEFAULT	"geniconvtbl"
#define	CPP_PATH	"/usr/lib/cpp"

itmc_ref_t	*ref_first[ITMC_OBJ_LAST + 1];
itmc_ref_t	*ref_last[ITMC_OBJ_LAST + 1];

itmc_name_t	*name_first;
itmc_name_t	*name_last;

char		*itm_input_file;		/* referred in itm_comp.l */
char		*itm_output_file;

cmd_opt_t	cmd_opt;
itm_num_t	name_id;
itm_num_t	reg_id;

itmc_name_t	name_lookup_error;
int		error_deferred;

char *itm_name_type_name[] = {
	"UNKNOWN",
	"ITM",
	"STRING",
	"DIRECTION",
	"CONDITION",
	"MAP",
	"OPERATION",
	"EXPRESSION",
	"DATA",
	"NAME",
	"RANGE",
	"REGISTER",
};


static void	usage(int status);
static int	cpp_opt_append(char	*opt, char	*arg);
static void	cpp_opt_trunc(int num);
static int	parse_opts(int argc, char	**argv);
static char	*prog_path_expand(const char	*base_name);
static void	map_name_type_append(char	*optarg);
static char	*map_type_name_str(itmc_map_type_t);
static char	*strdup_vital(const char *);

#if defined(ENABLE_TRACE)
static void	trace_option(void);
#endif /* ENABLE_TRACE */
static FILE	*cpp_open(void);
static void	cpp_close(FILE		*fp);
static int	itm_compile(char	*file);
static void	wait_child(pid_t pid);
static int	fork_error(void);




int
main(int argc, char **argv)
{
	char	**pp;
	pid_t	pid;

	(void) setlocale(LC_ALL, "");

	(void) textdomain(TEXT_DOMAIN);

	(void) parse_opts(argc, argv);

#if defined(ENABLE_TRACE)
	trace_option();
#endif /* ENABLE_TRACE */

	if (NULL != cmd_opt.disassemble) {
		disassemble(cmd_opt.disassemble);
	} else if (NULL == cmd_opt.input_file) {
		(void) itm_compile(NULL);
	} else {
		if (1 < cmd_opt.input_file_num) {
			for (pp = cmd_opt.input_file; *pp; pp++) {
				(void) printf("%s:\n", *pp);
				pid = fork();
				switch (pid) {
				case 0:
					exit(itm_compile(*pp));
					break;
				case -1:
					(void) fork_error();
					break;
				default:
					wait_child(pid);
				}
			}
		} else {
			(void) itm_compile(*(cmd_opt.input_file));
		}
	}

	return (0);
}


static int
itm_compile(char *file)
{
	char	*cmd_line;
	char	*command;
	char	*p;
	size_t	length;
	FILE	*fp;

	extern int	yyparse();
	extern FILE *yyin;

	if (NULL == file) {
		itm_input_file = gettext("*stdin*");
	} else {
		if (0 != access(file, R_OK)) {
			int	e = errno;
			itm_error(
				gettext("%1$s: can not access %2$s: "),
				cmd_opt.my_name, file);
			errno = e;
			PERROR(NULL);
			exit(ITMC_STATUS_CMD2);
		}
		itm_input_file = file;
	}

	if ((NULL == cmd_opt.output_file) &&
	    (0 == cmd_opt.no_output)) {
		p = strrchr(file, '.');
		if (NULL == p) {
			length = strlen(file);
		} else {
			length = p - file;
		}
		itm_output_file = malloc_vital(length + 5);
		(void) memcpy(itm_output_file, file, length);
		(void) memcpy(itm_output_file + length, ITMSUFFIX, 5);
	} else {
		itm_output_file = cmd_opt.output_file;
	}

	if (0 != cmd_opt.preprocess) {
		if (NULL == file) {
			fp = cpp_open();
			cmd_line = cmd_opt.preprocess;
		} else {
			(void) cpp_opt_append(file, NULL);
			fp = cpp_open();
			cpp_opt_trunc(1);
		}
		if (NULL == fp) {
			p = strchr(cmd_line, ' ');
			if (NULL == p) {
				length = strlen(cmd_line);
			} else {
				length = (p - cmd_line);
			}
			command = malloc_vital((sizeof (char)) * (length + 1));
			(void) memcpy(command, cmd_line, length);
			*(command + length) = '\0';
			PERROR(command);
			itm_error(
				gettext("%1$s: can not start %2$s on %3$s\n"),
				cmd_opt.my_name, command, itm_input_file);
			exit(ITMC_STATUS_SYS);
		} else {
			yyin = fp;
		}

		(void) yyparse();
		if (NULL == cmd_opt.preprocess_specified) {
			cpp_close(fp);
		}
	} else {
		if ((NULL == file) || (0 != strcmp("-", file))) {
			yyin = stdin;
		} else {
			yyin = fopen(file, "r");
			if (NULL == yyin) {
				itm_error(
					gettext("%1$s: can not open %2$s\n"),
					cmd_opt.my_name, itm_input_file);
				exit(ITMC_STATUS_CMD2);
			}
		}
		(void) yyparse();
		if (stdin != yyin) {
			(void) fclose(yyin);
		}
	}

	return (ITMC_STATUS_SUCCESS);
}




static void
wait_child(pid_t pid)
{
	int	stat_loc;
	char *msgstr;

	(void) waitpid(pid, &stat_loc, 0);
	if (WTERMSIG(stat_loc)) {
		if (WCOREDUMP(stat_loc)) {
			msgstr = gettext("signal received: %s, core dumped\n");
		} else {
			msgstr = gettext("signal received: %s\n");
		}
		itm_error(msgstr, strsignal(WTERMSIG(stat_loc)));
	}
}


static int
fork_error(void)
{
	PERROR(gettext("fork"));
	exit(ITMC_STATUS_SYS);
	return (0); /* never return */
}



static int
parse_opts(int argc, char **argv)
{
	int		c;
	int		i;
	char		*p;
	int		error_num = 0;

#ifdef YYDEBUG
	extern int	yydebug;
#endif /* YYDEBUG */

	extern char	*optarg;
	extern int	optind;


	cmd_opt.my_name = basename(*(argv + 0));
	if ('\0' == *(cmd_opt.my_name)) {
		cmd_opt.my_name = ME_DEFAULT;
	}

	cmd_opt.preprocess_default = CPP_PATH;
	cmd_opt.preprocess = cmd_opt.preprocess_default;
	cmd_opt.strip = 1; /* stripped by default */
	while ((c = getopt(argc, argv, "d:i:p:W:D:I:U:fnsM:lo:qX:h")) != EOF) {
		switch (c) {
		case 'd':
			cmd_opt.disassemble = optarg;
			break;
		case 'i':
			cmd_opt.interpreter = optarg;
			break;
		case 'p':
			if (NULL != cmd_opt.preprocess_specified) {
				(void) fprintf(stderr,
				gettext("multiple -p options are specified\n"));
				error_num += 1;
			}
			cmd_opt.preprocess_specified =
				prog_path_expand(optarg);
			cmd_opt.preprocess = cmd_opt.preprocess_specified;
			if (NULL == cmd_opt.preprocess) {
				(void) fprintf(stderr,
				gettext("cannot find preprocessor \"%s\"\n"),
					optarg);
				error_num += 1;
			}
			(void) cpp_opt_append(NULL, NULL);
			p = basename(optarg);
			if (NULL == p) {
				*(cmd_opt.cpp_opt + 0) = strdup_vital(optarg);
			} else {
				*(cmd_opt.cpp_opt + 0) = strdup_vital(p);
			}
			break;
		case 'W':
			if (cpp_opt_append(optarg, NULL)) {
				error_num += 1;
			}
			break;
		case 'I':
			if (cpp_opt_append("-I", optarg)) {
				error_num += 1;
			}
			break;
		case 'D':
			if (cpp_opt_append("-D", optarg)) {
				error_num += 1;
			}
			break;
		case 'U':
			if (cpp_opt_append("-U", optarg)) {
				error_num += 1;
			}
			break;
		case 'f':
			cmd_opt.force_overwrite = 1;
			break;
		case 'n':
			cmd_opt.no_output = 1;
			break;
		case 'M':
			map_name_type_append(optarg);
			break;
		case 'l':
			cmd_opt.large_table = 1;
			break;
		case 'o':
			cmd_opt.output_file = optarg;
			break;
		case 's':
			cmd_opt.strip = 0;
			break;
		case 'q':
			cmd_opt.quiet = 1;
			break;
#if defined(ENABLE_TRACE)
		case 'X':
			cmd_opt.trace = malloc_vital((sizeof (char)) * 128);
			(void) memset(cmd_opt.trace, 0, (sizeof (char)) * 128);
			for (p = optarg; *p; p++) {
				*(cmd_opt.trace + ((*p) & 0x007f)) = 1;
			}
#ifdef YYDEBUG
			if (TRACE('Y'))	yydebug = 1;
#endif /* YYDEBUG */
			break;
#endif /* ENABLE_TRACE */
		case 'h':
			usage(ITMC_STATUS_SUCCESS);
			break;
		default:
			usage(ITMC_STATUS_CMD);
		}
	}

	if (optind < argc) {
		cmd_opt.input_file_num = (argc - optind);
		cmd_opt.input_file =
			malloc_vital((sizeof (char *)) *
					(argc - optind + 1));
		*(cmd_opt.input_file + (argc - optind)) = NULL;
	}

	for (i = 0; optind < argc; optind++, i++) {
		*(cmd_opt.input_file + i) = argv[optind];
	}

	/* check conflict */

	if ((1 < cmd_opt.input_file_num) && (NULL != cmd_opt.output_file)) {
		itm_error(gettext("use -o with single input file\n"));
		error_num++;
	}

	if ((cmd_opt.input_file_num <= 0) &&
	    (NULL == cmd_opt.output_file) &&
	    (NULL == cmd_opt.disassemble) &&
	    (0 == cmd_opt.no_output)) {
		itm_error(gettext(
			"output file is unnamed. "
			"use -o to specify output file\n"));
		error_num++;
	}

	if (cmd_opt.disassemble &&
	    (cmd_opt.interpreter ||
	    cmd_opt.cpp_opt ||
	    cmd_opt.preprocess_specified ||
	    cmd_opt.input_file ||
	    cmd_opt.force_overwrite ||
	    cmd_opt.no_output ||
	    cmd_opt.map_name_type ||
	    cmd_opt.large_table ||
	    cmd_opt.output_file)) {
		itm_error(
			gettext("-d may not specified with other options\n"));
		error_num++;
	}

	if (error_num) {
		usage(ITMC_STATUS_CMD);
	}

	/*
	 * do not move upward
	 * may conflict with -d option
	 */
	if ((NULL == cmd_opt.preprocess_specified) &&
	    (NULL != cmd_opt.preprocess_default)) {
		(void) cpp_opt_append(NULL, NULL);
		p = basename(cmd_opt.preprocess_default);
		if (NULL == p) {
			*(cmd_opt.cpp_opt + 0) =
				strdup_vital(cmd_opt.preprocess_default);
		} else {
			*(cmd_opt.cpp_opt + 0) = strdup_vital(p);
		}
	}
	return (0);
}


static FILE *
cpp_open(void)
{
	pid_t	pid;
	int	filedes[2];
	int	i;

	for (i = 0; i < cmd_opt.cpp_opt_num; i++) {
		TRACE_MESSAGE('C', ("%s\n", *(cmd_opt.cpp_opt + i)));
	}

	if (pipe(filedes)) {
		PERROR(gettext("pipe"));
		itm_error(gettext("failed to open pipe\n"));
		exit(ITMC_STATUS_SYS);
	}
	pid = fork();
	if (pid == 0) {	/* child */
		(void) close(filedes[0]);
		(void) close(1);
		(void) dup2(filedes[1], 1);
		(void) execv(cmd_opt.preprocess, cmd_opt.cpp_opt);
		exit(0);
	} else if (pid == (pid_t)(-1)) {	/* error */
		return	(NULL);
	} else {
		(void) close(filedes[1]);
		return (fdopen(filedes[0], "r"));
	}
	return	(NULL); /* NEVER */
}


static int
cpp_opt_append(char	*opt, char	*arg)
{
	size_t	opt_len;
	size_t	arg_len;
	char	*new_opt;
	char	**new_opt_list;

	opt_len = ((NULL == opt) ? 0 : strlen(opt));
	arg_len = ((NULL == arg) ? 0 : strlen(arg));
	if (0 < (opt_len + arg_len)) {
		new_opt = malloc_vital(opt_len + arg_len + 1);
		if (NULL != opt) {
			(void) memcpy(new_opt, opt, opt_len + 1);
		}
		if (NULL != arg) {
			(void) memcpy(new_opt + opt_len, arg, arg_len + 1);
		}
	} else {
		new_opt = NULL;
	}

	if (0 == cmd_opt.cpp_opt_reserved) {
		cmd_opt.cpp_opt_reserved = 32;
		cmd_opt.cpp_opt = malloc_vital((sizeof (char *)) * 32);
		*(cmd_opt.cpp_opt + 0) = "cpp";
		cmd_opt.cpp_opt_num = 1;
	} else if ((cmd_opt.cpp_opt_reserved - 2) <= cmd_opt.cpp_opt_num) {
		cmd_opt.cpp_opt_reserved += 32;
		new_opt_list = malloc_vital((sizeof (char *)) *
					    cmd_opt.cpp_opt_reserved);
		(void) memcpy(new_opt_list, cmd_opt.cpp_opt,
			(sizeof (char *)) * cmd_opt.cpp_opt_num);
		(void) memset(new_opt_list + cmd_opt.cpp_opt_num, 0, 32);
		free(cmd_opt.cpp_opt);
		cmd_opt.cpp_opt = new_opt_list;
	}
	if (NULL != new_opt) {
		*(cmd_opt.cpp_opt + cmd_opt.cpp_opt_num) = new_opt;
		cmd_opt.cpp_opt_num += 1;
	}
	return (0);
}


static void
cpp_opt_trunc(int num)
{
	if (cmd_opt.cpp_opt_num < num) {
		num = cmd_opt.cpp_opt_num;
	}
	for (; 0 < num; --num) {
		free(cmd_opt.cpp_opt + cmd_opt.cpp_opt_num);
		--(cmd_opt.cpp_opt_num);
	}
}


static void
cpp_close(FILE *fp)
{
	(void) fclose(fp);
	(void) wait_child(0);
}




static char *
prog_path_expand(const char *base_name)
{
	size_t	base_len;
	size_t	dir_len;
	char	path[MAXPATHLEN];
	char	*p;
	char	*pe;

	base_len = strlen(base_name);
	path[0] = '\0';

	if (NULL != strchr(base_name, '/')) {
		if (0 == access(base_name, X_OK)) {
			return (strdup_vital(base_name));
		} else {
			return (NULL);
		}
	}

	for (p = getenv("PATH"); p; ) {
		pe = strchr(p, ':');
		dir_len = ((NULL == pe) ? strlen(p) : (pe - p));
		(void) memcpy(path, p, dir_len);
		if ((0 != dir_len) &&
		    ('/' != path[dir_len - 1])) {
			path[dir_len] = '/';
			dir_len += 1;
		}
		if ((dir_len + base_len) < MAXPATHLEN) {
			(void) memcpy(path + dir_len, base_name, base_len + 1);
			if (0 == access(path, X_OK)) {
				return (strdup_vital(path));
			}
		}
		p = ((NULL == pe) ? NULL : (pe + 1));
	}
	return	(NULL);
}


static void
usage(int status)
{

	if (ITMC_STATUS_SUCCESS == status) {
		(void) fprintf(stdout,
		gettext("Usage: %1$s [-n] [-f] [-q]\n"
		"	     [-p preprocessor] [-W argument]\n"
		"	     [-Dname] [-Dname=def] [-Idirectory] [-Uname]\n"
		"	     [file ...]\n	%2$s -h\n"),
		cmd_opt.my_name, cmd_opt.my_name);
	} else {
		(void) itm_error(
		gettext("Usage: %1$s [-n] [-f] [-q]\n"
		"	     [-p preprocessor] [-W argument]\n"
		"	     [-Dname] [-Dname=def] [-Idirectory] [-Uname]\n"
		"	     [file ...]\n	%2$s -h\n"),
		cmd_opt.my_name, cmd_opt.my_name);
	}
	exit(status);
}


static char *
map_type_name_str(itmc_map_type_t type)
{
	int	i;
	for (i = 0; NULL != map_type_name[i].name; i++) {
		if (type == map_type_name[i].type) {
			return (map_type_name[i].name);
		}
	}
	return ("");
}

static void
map_name_type_append(char *optarg)
{
	char			*oa;
	char			*oa_save;
	char			*name;
	char			*p;
	char			*phf;
	int			hash_factor = 0;
	itmc_map_type_t		type;
	itmc_map_name_type_t	*m;
	int			i;

	oa = oa_save = strdup_vital(optarg);

	while ((NULL != oa) && ('\0' != *oa)) {
		name = oa;
		oa = strchr(oa, ',');
		if (NULL != oa) {
			*(oa++) = '\0';
		}
		p = strchr(name, '=');
		if (NULL == p) {
			type = ITMC_MAP_AUTOMATIC;
		} else {
			*(p++) = '\0';
			if ('\0' == *p) {
				type = ITMC_MAP_AUTOMATIC;
			} else {
				phf = strchr(p, ':');
				if (NULL != phf) {
					*(phf++) = '\0';
					hash_factor = atoi(phf);
					if (hash_factor < 0) {
						itm_error(
						gettext(
						"invalid hash factor is "
						"specified: %s\n"),
							phf);
						hash_factor = 0;
						error_deferred += 1;
					}
				}
				for (i = 0;
				    NULL != map_type_name[i].name; i++) {
					if (0 ==
					    strcmp(p, map_type_name[i].name)) {
						type = map_type_name[i].type;
						break;
					}
				}
				if (NULL == map_type_name[i].name) {
					itm_error(
					gettext(
					"unknown map type is specified: %s\n"),
					p);
					error_deferred += 1;
					continue;
				}
			}
		}
		if (0 == strcmp(name, "default")) {
			*name = '\0';
		}
		m = cmd_opt.map_name_type;
		if (NULL == m) {
			m = malloc_vital(sizeof (itmc_map_name_type_t));
			m->name = strdup_vital(name);
			m->type = type;
			m->hash_factor = hash_factor;
			m->next = NULL;
			cmd_opt.map_name_type = m;
			continue;
		}
		for (; ; m = m->next) {
			if (0 == strcmp(name, m->name)) {
				if (type == m->type) {
					m = NULL;
					break;
				}
				if ('\0' == *name) {
					itm_error(
					gettext(
					"multiple default types are specified:"
					" \"%1$s\" and \"%2$s\"\n"),
						map_type_name_str(type),
						map_type_name_str(m->type));
				} else {
					itm_error(
					gettext("map \"%1$s\" is specified as "
					"two types \"%2$s\" and \"%3$s\"\n"),
					name,
					map_type_name_str(type),
					map_type_name_str(m->type));
				}
				error_deferred += 1;
				m = NULL;
				break;
			}
			if (NULL == m->next) {
				break;
			}
		}
		if (NULL != m) {
			m->next = malloc_vital(sizeof (itmc_map_name_type_t));
			m = m->next;
			m->name = strdup_vital(name);
			m->type = type;
			m->hash_factor = hash_factor;
			m->next = NULL;

		}
	}
	free(oa_save);
}



void *
malloc_vital(size_t size)
{
	void	*p;

	TRACE_MESSAGE('M', ("malloc_vital: %d\n", size));

	size = ITMROUNDUP(size);

	p = (void*) malloc(size);
	if (NULL == p) {
		PERROR(gettext("malloc"));
		exit(ITMC_STATUS_SYS);
	}

	(void) memset(p, 0, size);

	return	(p);
}


static char *
strdup_vital(const char		*str)
{
	char	*p;
	size_t	len;

	if (NULL == str) {
		return	(NULL);
	}

	len = strlen(str) + 1;
	p = malloc_vital(len);
	(void) memcpy(p, str, len);
	return	(p);
}





itm_data_t *
str_to_data(int size, char *seq)
{
	itm_data_t *data;

	data = malloc_vital(sizeof (itm_data_t));

	data->size = size;
	if (size <= sizeof (data->place)) {
		(void) memmove(&(data->place), seq, size);
	} else {
		data->place.itm_ptr = (itm_place2_t)malloc_vital(size);
		(void) memmove((char *)(data->place.itm_ptr), seq, size);
	}

	return	(data);
}


char *
name_to_str(itm_data_t *name)
{
	static char	*ptr = NULL;
	static size_t	len = 0;
	size_t		req_len;
	char		*p;

	if (NULL == name) {
		p = gettext("(no name)");
		req_len = strlen(p) + 1;
	} else {
		req_len = name->size + 1;
	}

	if (len <= req_len) {
		len += 512;
		free(ptr);
		ptr = malloc_vital(len);
	}

	if (NULL == name) {
		(void) memcpy(ptr, p, req_len);
		*(ptr + req_len) = '\0';
	} else if (name->size <= (sizeof (name->place))) {
		(void) memcpy(ptr, (char *)(&(name->place)), name->size);
		*(ptr + name->size) = '\0';
	} else {
		(void) memcpy(ptr, (char *)(name->place.itm_ptr), name->size);
		*(ptr + name->size) = '\0';
	}

	return	(ptr);
}

#define	ARGUMENTSMAX (8)
char *
data_to_hexadecimal(itm_data_t		*data)
{
	static int index = 0;
	static char	*ptr[ARGUMENTSMAX] = { NULL, NULL, NULL, NULL,
						NULL, NULL, NULL, NULL};
	static long	len[ARGUMENTSMAX] = { 0, 0, 0, 0, 0, 0, 0, 0};
	char		*hdp;
	char		*p;
	long		i;
	int		val;
	size_t		req_len;

	if (ARGUMENTSMAX <= index) index = 0;
	req_len = (2 * data->size) + 1;
	if (len[index] <= req_len) {
		len[index] += 512;
		free(ptr[index]);
		ptr[index] = malloc_vital(len[index]);
	}
	hdp = ptr[index];

	if (data->size <= (sizeof (itm_place_t))) {
		p = (char *)&(data->place);
	} else {
		p = (char *)(data->place.itm_ptr);
	}

	for (i = 0; i < data->size; i++, p++) {
		val = ((*p & 0x00f0) >> 4);
		if ((0 <= val) && (val <= 9)) {
			*hdp = '0' + val;
		} else {
			*hdp = 'a' + val - 10;
		}
		hdp++;

		val = (*p & 0x000f);
		if ((0 <= val) && (val <= 9)) {
			*hdp = '0' + val;
		} else {
			*hdp = 'a' + val - 10;
		}
		hdp++;
	}
	*hdp = '\0';
	return (ptr[index++]);
}





void
itm_error(char *format, ...)
{
	va_list		ap;
	va_start(ap, format);

	if (0 == cmd_opt.quiet) {
		(void) vfprintf(stderr, format, ap);
	}
	va_end(ap);
}

#if defined(ENABLE_TRACE)
static void
trace_option(void)
{
	char **pp;
	int	i;

	if (!(TRACE('o')))
		return;

	itm_error("my_name	   = %s\n", cmd_opt.my_name);
	if (NULL == cmd_opt.input_file) {
		(void) fprintf(stdout, "input_file   = (stdin)\n");
	} else {
		for (pp = cmd_opt.input_file; *pp; pp++) {
			(void) fprintf(stdout, "input_file   = %s\n", *pp);
		}
	}
	itm_error("output_file  = %s\n",
		cmd_opt.output_file ? cmd_opt.output_file : "(stdout)");
	itm_error("interpreter  = %s\n",
		cmd_opt.interpreter ? cmd_opt.interpreter : "(default)");
	if (cmd_opt.cpp_opt) {
		itm_error("cpp_opt	   = %s\n", *(cmd_opt.cpp_opt));
		for (i = 1; i < cmd_opt.cpp_opt_num; i++) {
			itm_error("\t%s\n", *(cmd_opt.cpp_opt + i));
		}
	} else {
		itm_error("cpp_opt	   = %s\n", "(none)");
	}
	itm_error("preprocess_default = %s\n",
		cmd_opt.preprocess_default ? cmd_opt.preprocess_default :
		"(no)");
	itm_error("preprocess_specified = %s\n",
		cmd_opt.preprocess_specified ? cmd_opt.preprocess_specified :
		"(no)");
	itm_error("preprocess   = %s\n",
		cmd_opt.preprocess ? cmd_opt.preprocess : "(no)");
	itm_error("disassemble  = %s\n",
		cmd_opt.disassemble ? "yes" : "no");
	itm_error("map type	   =");
	if (NULL == cmd_opt.map_name_type) {
		itm_error("\n");
	} else {
		itmc_map_name_type_t *m;
		itm_error(" ");
		m = cmd_opt.map_name_type;
		itm_error("%s=%s",
			(((NULL == m->name) || ('\0' == *(m->name))) ?
				"default" : m->name),
			map_type_name_str(m->type));
		if (0 != m->hash_factor) {
			itm_error(":%ld\n", m->hash_factor);
		} else {
			(void) fputc('\n', stderr);
		}
		for (m = m->next; NULL != m; m = m->next) {
			itm_error("		%s=%s",
				(((NULL == m->name) || ('\0' == *(m->name))) ?
					"default" : m->name),
				map_type_name_str(m->type));
			if (0 != m->hash_factor) {
				itm_error(":%ld\n", m->hash_factor);
			} else {
				(void) fputc('\n', stderr);
			}
		}
	}
	itm_error("large table  = %s\n",
		cmd_opt.large_table ? "true" : "false");
	itm_error("overwrite	   = %s\n",
		cmd_opt.force_overwrite ? "true" : "false");
	itm_error("strip	      = %s\n",
		cmd_opt.strip ? "true" : "false");
	itm_error("no_output	   = %s\n",
		cmd_opt.no_output ? "true" : "false");
	itm_error("trace	      = ");
	if (NULL == cmd_opt.trace) {
		itm_error("(no)\n");
	} else {
		for (i = 0x21; i < 0x7f; i++) {
			if (TRACE(i)) {
				(void) fputc(i, stderr);
			}
		}
		(void) fputc('\n', stderr);
	}
}
#endif /* ENABLE_TRACE */

#if defined(ENABLE_TRACE)
extern void
trace_message(char *format, ...)
{
	va_list	ap;
	va_start(ap, format);

	(void) vfprintf(stderr, format, ap);

	va_end(ap);
}
#endif /* ENABLE_TRACE */