Subject: Re: crunchide failure on MIPS with NEW_TOOLCHAIN (Re: build failure
To: None <port-mips@netbsd.org, tech-toolchain@netbsd.org>
From: Izumi Tsutsui <tsutsui@ceres.dti.ne.jp>
List: port-mips
Date: 02/02/2002 16:12:57
In article <200202020240.g122eIr21290@mirage.ceres.dti.ne.jp>
I wrote:

> Maybe we have to have BFD'fied crunchide(1), which already exists
> in pkgsrc/cross/bfd-crunchide, instead of simple objcopy(1).

Here is a quick (and not well-tested) diff:

--- /dev/null	Sat Feb  2 15:38:07 2002
+++ tools/crunchide/Makefile	Sat Feb  2 15:41:56 2002
@@ -0,0 +1,14 @@
+#	$NetBSD$
+
+NOMAN=	# defined
+
+.include <bsd.own.mk>
+
+HOSTPROGNAME=	${MACHINE_GNU_PLATFORM}-crunchide
+HOST_SRCDIR=	gnu/usr.bin/crunchide
+
+TOOLCHAINOBJ!=	cd ${.CURDIR}/../toolchain && ${PRINTOBJDIR}
+BFDDIR=		${TOOLCHAINOBJ}/build/bfd
+IBERTYDIR=	${TOOLCHAINOBJ}/build/libiberty
+
+.include "${.CURDIR}/../Makefile.host"
--- /dev/null	Sat Feb  2 15:38:07 2002
+++ gnu/usr.bin/crunchide/crunchide.c	Sat Feb  2 15:38:05 2002
@@ -0,0 +1,446 @@
+/* $NetBSD$ */
+/* NetBSD: crunchide.c,v 1.9 1999/01/11 22:40:00 kleink Exp 	 */
+
+/*
+ * Copyright (c) 1997 Christopher G. Demetriou.  All rights reserved.
+ * Copyright (c) 1994 University of Maryland
+ * All Rights Reserved.
+ *
+ * Permission to use, copy, modify, distribute, and sell this software and its
+ * documentation for any purpose is hereby granted without fee, provided that
+ * the above copyright notice appear in all copies and that both that
+ * copyright notice and this permission notice appear in supporting
+ * documentation, and that the name of U.M. not be used in advertising or
+ * publicity pertaining to distribution of the software without specific,
+ * written prior permission.  U.M. makes no representations about the
+ * suitability of this software for any purpose.  It is provided "as is"
+ * without express or implied warranty.
+ *
+ * U.M. DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL U.M.
+ * BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
+ * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
+ * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ *
+ * Author: James da Silva, Systems Design and Analysis Group
+ *			   Computer Science Department
+ *			   University of Maryland at College Park
+ */
+/*
+ * crunchide.c - tiptoes through an a.out symbol table, hiding all defined
+ *	global symbols.  Allows the user to supply a "keep list" of symbols
+ *	that are not to be hidden.  This program relies on the use of the
+ * 	linker's -dc flag to actually put global bss data into the file's
+ * 	bss segment (rather than leaving it as undefined "common" data).
+ *
+ * 	The point of all this is to allow multiple programs to be linked
+ *	together without getting multiple-defined errors.
+ *
+ *	For example, consider a program "foo.c".  It can be linked with a
+ *	small stub routine, called "foostub.c", eg:
+ *	    int foo_main(int argc, char **argv){ return main(argc, argv); }
+ *      like so:
+ *	    cc -c foo.c foostub.c
+ *	    ld -dc -r foo.o foostub.o -o foo.combined.o
+ *	    crunchide -k _foo_main foo.combined.o
+ *	at this point, foo.combined.o can be linked with another program
+ * 	and invoked with "foo_main(argc, argv)".  foo's main() and any
+ * 	other globals are hidden and will not conflict with other symbols.
+ *
+ * TODO:
+ *	- resolve the theoretical hanging reloc problem (see check_reloc()
+ *	  below). I have yet to see this problem actually occur in any real
+ *	  program. In what cases will gcc/gas generate code that needs a
+ *	  relative reloc from a global symbol, other than PIC?  The
+ *	  solution is to not hide the symbol from the linker in this case,
+ *	  but to generate some random name for it so that it doesn't link
+ *	  with anything but holds the place for the reloc.
+ *      - arrange that all the BSS segments start at the same address, so
+ *	  that the final crunched binary BSS size is the max of all the
+ *	  component programs' BSS sizes, rather than their sum.
+ */
+
+#include <sys/cdefs.h>
+#ifndef lint
+__RCSID("$NetBSD: crunchide.c,v 1.2 2001/03/23 13:36:18 wiz Exp $");
+#endif
+
+#include <unistd.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <fcntl.h>
+#include <limits.h>
+#include <errno.h>
+#include <a.out.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <bfd.h>
+
+extern const char *__progname;
+
+void usage      __P((void));
+void add_to_keep_list __P((char *));
+void add_file_to_keep_list __P((char *));
+int hide_syms   __P((const char *, char *));
+static void setup_section __P((bfd *, sec_ptr, PTR));
+static void copy_section __P((bfd *, sec_ptr, PTR));
+
+int verbose;
+
+struct keep {
+	struct keep *next;
+	char *sym;
+} *keep_list;
+
+struct bfd_cookie {
+	bfd *bfd;
+	asymbol **symtable;
+};
+
+int
+main(argc, argv)
+	int argc;
+	char **argv;
+{
+	int ch, errors;
+	char *bfdname = NULL;
+
+	while ((ch = getopt(argc, argv, "b:k:f:v")) != -1)
+		switch (ch) {
+		case 'b':
+			bfdname = optarg;
+			break;
+		case 'k':
+			add_to_keep_list(optarg);
+			break;
+		case 'f':
+			add_file_to_keep_list(optarg);
+			break;
+		case 'v':
+			verbose = 1;
+			break;
+		default:
+			usage();
+		}
+
+	argc -= optind;
+	argv += optind;
+
+	if (argc == 0)
+		usage();
+
+	errors = 0;
+	while (argc) {
+		if (hide_syms(*argv, bfdname))
+			errors = 1;
+		argc--, argv++;
+	}
+
+	return errors;
+}
+
+void 
+usage()
+{
+	const char **list;
+
+	fprintf(stderr,
+"Usage: %s [-b bfdname] [-k <symbol-name>] [-f <keep-list-file>] <files> ...\n",
+		__progname);
+	fprintf(stderr, "supported targets:");
+	for (list = bfd_target_list(); *list != NULL; list++)
+		fprintf(stderr, " %s", *list);
+	fprintf(stderr, "\n");
+	exit(1);
+}
+
+void 
+add_to_keep_list(symbol)
+	char *symbol;
+{
+	struct keep *newp, *prevp, *curp;
+	int cmp;
+
+	cmp = 0;
+
+	for (curp = keep_list, prevp = NULL; curp;
+	    prevp = curp, curp = curp->next)
+		if ((cmp = strcmp(symbol, curp->sym)) <= 0)
+			break;
+
+	if (curp && cmp == 0)
+		return;		/* already in table */
+
+	newp = (struct keep *)malloc(sizeof (struct keep));
+	if (newp)
+		newp->sym = strdup(symbol);
+	if (newp == NULL || newp->sym == NULL) {
+		fprintf(stderr,
+			"%s: out of memory for keep list\n", __progname);
+		exit(1);
+	}
+	newp->next = curp;
+	if (prevp)
+		prevp->next = newp;
+	else
+		keep_list = newp;
+}
+
+int 
+in_keep_list(symbol)
+	const char *symbol;
+{
+	struct keep *curp;
+	int cmp;
+
+	cmp = 0;
+
+	for (curp = keep_list; curp; curp = curp->next)
+		if ((cmp = strcmp(symbol, curp->sym)) <= 0)
+			break;
+
+	return curp && cmp == 0;
+}
+
+void 
+add_file_to_keep_list(filename)
+	char *filename;
+{
+	FILE *keepf;
+	char symbol[1024];
+	int len;
+
+	if ((keepf = fopen(filename, "r")) == NULL) {
+		perror(filename);
+		usage();
+	}
+	while (fgets(symbol, 1024, keepf)) {
+		len = strlen(symbol);
+		if (len && symbol[len - 1] == '\n')
+			symbol[len - 1] = '\0';
+
+		add_to_keep_list(symbol);
+	}
+	fclose(keepf);
+}
+
+int 
+hide_syms(filename, bfdname)
+	const char *filename;
+	char *bfdname;
+{
+	int i, n, rv = 0;
+	bfd *org_bfd = NULL, *new_bfd = NULL;
+	char tempname[16];
+	char **name;
+	long storage_needed, number_of_symbols;
+	size_t fn_size;
+	asymbol **org_symtable, **new_symtable;
+
+	fn_size = strlen(filename);
+
+	bfd_init();
+	if ((org_bfd = bfd_openr(filename, bfdname)) == NULL) {
+		bfd_perror(filename);
+		return 1;
+	}
+	if (!bfd_check_format(org_bfd, bfd_object)) {
+		bfd_perror(filename);
+		goto err;
+	}
+
+	bfdname = bfd_get_target(org_bfd);
+	strcpy(tempname, "tmp.XXXXXXX");
+	mktemp(tempname);
+	if ((new_bfd = bfd_openw(tempname, bfdname)) == NULL) {
+		bfd_perror(tempname);
+		goto err;
+	}
+
+	if (!bfd_set_format(new_bfd, bfd_get_format(org_bfd)) ||
+	    !bfd_set_start_address(new_bfd, bfd_get_start_address(org_bfd)) ||
+	    !bfd_set_file_flags(new_bfd, (bfd_get_file_flags(org_bfd) &
+	      bfd_applicable_file_flags(new_bfd))) ||
+	    !bfd_set_arch_mach (new_bfd, bfd_get_arch (org_bfd),
+	      bfd_get_mach (org_bfd))) {
+		bfd_perror(tempname);
+		goto err;
+	}
+	bfd_map_over_sections(org_bfd, setup_section, (void *)new_bfd);
+
+	storage_needed = bfd_get_symtab_upper_bound(org_bfd);
+	if (storage_needed < 0)
+		goto err;
+	if (storage_needed == 0)
+		goto out;
+
+	org_symtable = (asymbol **)malloc(storage_needed);
+	number_of_symbols = bfd_canonicalize_symtab(org_bfd, org_symtable);
+	if (number_of_symbols < 0)
+		goto err;
+
+	new_symtable = (asymbol **)malloc(storage_needed);
+	if (new_symtable == NULL)
+		goto err;
+	name = (char **)calloc(sizeof (char *), number_of_symbols);
+	if (name == NULL)
+		goto err;
+
+	n = 0;
+	for (i = 0; i < number_of_symbols; i++) {
+		const char *symname;
+
+		new_symtable[i] = org_symtable[i];
+		if (!(new_symtable[i]->flags & BSF_GLOBAL))
+			continue;
+
+		symname = bfd_asymbol_name(new_symtable[i]);
+#if 0
+		if (symname[0] == '_')
+			symname++;
+#endif
+		if (in_keep_list(symname))
+			continue;
+
+		n++;
+
+		/*
+		 * make sure there's size for the next entry, even if it's
+		 * as large as it can be.
+		 *
+		 * "_$$hide$$ <filename> <symname><NUL>" ->
+		 *    9 + 3 + sizes of fn and sym name 
+		 */
+		name[i] = (char *)malloc(12 + fn_size + strlen(symname));
+		sprintf(name[i], "_$$hide$$ %s %s", filename, symname);
+		new_symtable[i]->name = name[i];
+	}
+
+	if (n > 0) {
+		struct bfd_cookie cookie;
+
+		new_symtable[number_of_symbols] = NULL;
+		bfd_set_symtab(new_bfd, new_symtable, number_of_symbols);
+
+		cookie.bfd = new_bfd;
+		cookie.symtable = org_symtable;
+		bfd_map_over_sections(org_bfd, copy_section, (void *)&cookie);
+
+		if (!bfd_copy_private_bfd_data(org_bfd, new_bfd)) {
+			bfd_perror("bfd_copy_private_bfd_data");
+			goto err;
+		}
+	}
+
+	bfd_close(new_bfd);
+	bfd_close(org_bfd);
+
+	if (rename(tempname, filename) < 0)
+		perror("rename");
+	unlink(tempname);
+
+	for (i = 0; i < number_of_symbols; i++)
+		if (name[i])
+			free(name[i]);
+	free(new_symtable);
+	free(name);
+
+out:
+	return (rv);
+
+err:
+	rv = 1;
+	if (org_bfd)
+		bfd_close(org_bfd);
+	if (new_bfd) {
+		unlink(tempname);
+		bfd_close(new_bfd);
+	}
+	goto out;
+}
+
+static void
+setup_section(ibfd, isection, arg)
+	bfd *ibfd;
+	sec_ptr isection;
+	PTR arg;
+{
+	sec_ptr osection;
+	bfd *obfd = (bfd *)arg;
+
+	if ((osection = bfd_make_section_anyway(obfd,
+	      bfd_section_name(ibfd, isection))) == NULL ||
+	    !bfd_set_section_size(obfd, osection,
+	      bfd_section_size(ibfd, isection)) ||
+	    !bfd_set_section_vma(obfd, osection,
+	      bfd_section_vma(ibfd, isection)) ||
+	    !bfd_set_section_alignment(obfd, osection,
+	      bfd_section_alignment(ibfd, isection)) ||
+	    !bfd_set_section_flags(obfd, osection,
+	      bfd_get_section_flags(ibfd, isection))) {
+		bfd_perror("setup_section");
+		return;
+	}
+
+	isection->output_section = osection;
+	isection->output_offset = 0;
+	if (!bfd_copy_private_section_data(ibfd, isection, obfd, osection)) {
+		bfd_perror("setup_section");
+		return;
+	}
+}
+
+static void
+copy_section(ibfd, isection, arg)
+	bfd *ibfd;
+	sec_ptr isection;
+	PTR arg;
+{
+	struct bfd_cookie *bc = (struct bfd_cookie *)arg;
+	bfd *obfd = bc->bfd;
+	asymbol **isym = bc->symtable;
+	arelent **relpp;
+	sec_ptr osection;
+	PTR memhunk;
+	bfd_size_type size;
+	long relcount, relsize;
+
+	osection = isection->output_section;
+	size = bfd_get_section_size_before_reloc(isection);
+	if (size == 0 || osection == 0)
+		return;
+
+	relsize = bfd_get_reloc_upper_bound(ibfd, isection);
+	if (relsize < 0) {
+		bfd_perror(bfd_get_filename(ibfd));
+	}
+	if (relsize == 0) {
+		bfd_set_reloc(obfd, osection, NULL, 0);
+	} else {
+		relpp = (arelent **)malloc(relsize);
+		relcount = bfd_canonicalize_reloc(ibfd, isection, relpp, isym);
+		if (relcount < 0)
+			bfd_perror(bfd_get_filename(ibfd));
+		bfd_set_reloc(obfd, osection, relpp, relcount);
+	}
+
+	isection->_cooked_size = isection->_raw_size;
+	isection->reloc_done = true;
+
+	if (bfd_get_section_flags(ibfd, isection) & SEC_HAS_CONTENTS) {
+		memhunk = (PTR)malloc((unsigned int)size);
+		if (!bfd_get_section_contents(ibfd, isection, memhunk, 0, size))
+			bfd_perror(bfd_get_filename(ibfd));
+		if (!bfd_set_section_contents(obfd, osection, memhunk, 0, size))
+			bfd_perror(bfd_get_filename(obfd));
+		free(memhunk);
+	} else if (bfd_get_section_flags(ibfd, isection) & SEC_LOAD) {
+		memhunk = (PTR)malloc((unsigned int)size);
+		memset(memhunk, 0, size);
+		if (!bfd_set_section_contents(obfd, osection, memhunk, 0, size))
+			bfd_perror(bfd_get_filename(obfd));
+		free(memhunk);
+	}
+}
--- /dev/null	Sat Feb  2 15:38:07 2002
+++ gnu/usr.bin/crunchide/crunchide.1	Sat Feb  2 11:08:54 2002
@@ -0,0 +1,94 @@
+.\"	$NetBSD$
+.\"
+.\" Copyright (c) 1994 University of Maryland
+.\" All Rights Reserved.
+.\"
+.\" Permission to use, copy, modify, distribute, and sell this software and its
+.\" documentation for any purpose is hereby granted without fee, provided that
+.\" the above copyright notice appear in all copies and that both that
+.\" copyright notice and this permission notice appear in supporting
+.\" documentation, and that the name of U.M. not be used in advertising or
+.\" publicity pertaining to distribution of the software without specific,
+.\" written prior permission.  U.M. makes no representations about the
+.\" suitability of this software for any purpose.  It is provided "as is"
+.\" without express or implied warranty.
+.\"
+.\" U.M. DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL
+.\" IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL U.M.
+.\" BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+.\" WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+.\" ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR
+.\" IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+.\"
+.\" Author: James da Silva, Systems Design and Analysis Group
+.\"			   Computer Science Department
+.\"			   University of Maryland at College Park
+.\"
+.Dd June 14, 1994
+.Dt CRUNCHIDE 1
+.Os
+.Sh NAME
+.Nm crunchide
+.Nd hides symbol names from ld, for crunching programs together
+.Sh SYNOPSIS
+.Nm ""
+.Bk -words
+.Op Fl b Ar bfdname
+.Ek
+.Bk -words
+.Op Fl f Ar keep-list-file
+.Ek
+.Bk -words
+.Op Fl k Ar keep-symbol
+.Ek
+.Bk -words
+.Op Ar object-file ...
+.Ek
+.Sh DESCRIPTION
+.Nm Crunchide
+hides the global symbols of
+.Ar object-file
+such that they are ignored by subsequent runs of the linker,
+.Xr ld 1 .
+Some symbols may be left visible via the
+.Fl k Ar keep-symbol
+and
+.Fl f Ar keep-list-file
+options.  The
+.Ar keep-list-file
+must contain a list of symbols to keep visible, one symbol per line.
+The names given by
+.Ar keep-symbol
+or in
+.Ar keep-list-file
+should be C names.  For example,
+to keep the C function
+.Dq foo
+visible, the option
+.Dq -k foo
+should be used.
+.Pp
+To recognize kernel executable format, the
+.Fl b
+flag specifies BFD name of kernel.
+.Pp
+.Nm Crunchide
+is designed as a companion program for
+.Xr crunchgen 1 ,
+which automates the process of creating crunched binaries from
+multiple component programs.
+.Sh SEE ALSO
+.Xr crunchgen 1 ,
+.Xr ld 1
+.Sh AUTHORS
+.Nm Crunch
+was written by James da Silva <jds@cs.umd.edu>.
+.sp 0
+Copyright (c) 1994 University of Maryland.  All Rights Reserved.
+.Pp
+Chris Demetriou <cgd@netbsd.org> reorganized
+.Nm
+so that it supported multiple object formats, and added
+ELF object support and ECOFF object recognition.
+.sp 0
+Copyright (c) 1997 Christopher G. Demetriou.  All Rights Reserved.
--- /dev/null	Sat Feb  2 15:38:07 2002
+++ gnu/usr.bin/crunchide/Makefile	Sat Feb  2 11:10:05 2002
@@ -0,0 +1,34 @@
+#	$NetBSD$
+
+.include <bsd.own.mk>
+
+.if defined(USE_NEW_TOOLCHAIN) && ${MKBFD} != "no"
+TOP=		${.CURDIR}/../..
+DIST=		${TOP}/dist/toolchain
+
+PROG=		crunchide
+MAN=		crunchide.1
+
+CPPFLAGS+=	-I${BFDDIR} -I${DIST}/bfd -I${DIST}/binutils -I${DIST}/include
+
+DPADD+=		${BFDDIR}/libbfd.a
+LDADD+=		-L${BFDDIR} -lbfd
+
+DPADD+=		${IBERTYDIR}/libiberty.a
+LDADD+=		-L${IBERTYDIR} -liberty
+
+.ifndef HOSTPROG
+.include	"${TOP}/lib/libbfd/arch/${MACHINE_ARCH}/defs.mk"
+
+BFDDIR!=	cd ${.CURDIR}/../../lib/libbfd && ${PRINTOBJDIR}
+IBERTYDIR!=	cd ${TOP}/lib/libiberty && ${PRINTOBJDIR}
+
+CPPFLAGS+=	-I${TOP}/lib/libbfd/arch/${MACHINE_ARCH} ${G_INCLUDES}
+LDADD+=		-lintl
+DPADD+=		${LIBINTL}
+.endif # HOSTPROG
+.endif # USE_NEW_TOOLCHAIN && MKBFD != no
+
+.ifndef HOSTPROG
+.include <bsd.prog.mk>
+.endif # HOSTPROG

Index: tools/Makefile
===================================================================
RCS file: /cvsroot/basesrc/tools/Makefile,v
retrieving revision 1.32
diff -u -r1.32 Makefile
--- tools/Makefile	2002/01/31 22:43:43	1.32
+++ tools/Makefile	2002/02/02 06:45:35
@@ -7,7 +7,7 @@
 
 # XXX We can't currently build the toolchain for x86-64.
 .if ${MACHINE_ARCH} != "x86_64"
-TOOLCHAIN_BITS= toolchain .WAIT ${NO_DBSYM:D:Udbsym} mdsetimage
+TOOLCHAIN_BITS= toolchain .WAIT ${NO_DBSYM:D:Udbsym} crunchide mdsetimage
 .endif
 
 # Dependencies in SUBDIR below ordered to maximize parallel ability.
Index: usr.bin/crunch/crunchgen/crunchgen.c
===================================================================
RCS file: /cvsroot/basesrc/usr.bin/crunch/crunchgen/crunchgen.c,v
retrieving revision 1.31
diff -u -r1.31 crunchgen.c
--- usr.bin/crunch/crunchgen/crunchgen.c	2002/01/31 19:24:47	1.31
+++ usr.bin/crunch/crunchgen/crunchgen.c	2002/02/02 06:45:37
@@ -819,7 +819,7 @@
     fprintf(outmk, "DBG=%s\n", dbg);
     fprintf(outmk, "STRIP?=strip\n");
     fprintf(outmk, "MAKE?=make\n");
-#ifdef NEW_TOOLCHAIN
+#if 0 /* ifdef NEW_TOOLCHAIN */
     fprintf(outmk, "OBJCOPY?=objcopy\n");
 #else
     fprintf(outmk, "CRUNCHIDE?=crunchide\n");
@@ -891,7 +891,7 @@
 		p->name, p->name, p->ident);
     fprintf(outmk, "\t${LD} -dc -r -o %s.cro %s_stub.o $(%s_OBJPATHS)\n", 
 	    p->name, p->name, p->ident);
-#ifdef NEW_TOOLCHAIN
+#if 0 /* ifdef NEW_TOOLCHAIN */
     fprintf(outmk, "\t${OBJCOPY} --keep-global-symbol _crunched_%s_stub ",
   	    p->ident);
     for (lst = p->keepsymbols; lst != NULL; lst = lst->next)
Index: share/mk/bsd.own.mk
===================================================================
RCS file: /cvsroot/sharesrc/share/mk/bsd.own.mk,v
retrieving revision 1.265
diff -u -r1.265 bsd.own.mk
--- share/mk/bsd.own.mk	2002/01/31 22:43:42	1.265
+++ share/mk/bsd.own.mk	2002/02/02 06:49:34
@@ -114,6 +114,7 @@
 COMPILE_ET=	${TOOLDIR}/bin/nbcompile_et
 CONFIG=		${TOOLDIR}/bin/nbconfig
 CRUNCHGEN=	MAKE=${.MAKE:Q} ${TOOLDIR}/bin/nbcrunchgen
+CRUNCHIDE=	${TOOLDIR}/bin/${MACHINE_GNU_PLATFORM}-crunchide
 CTAGS=		${TOOLDIR}/bin/nbctags
 DBSYM=		${TOOLDIR}/bin/${MACHINE_GNU_PLATFORM}-dbsym
 EQN=		${TOOLDIR}/bin/nbeqn

---
Izumi Tsutsui
tsutsui@ceres.dti.ne.jp