bcd.c revision 1.13
11.13Sjsm/*	$NetBSD: bcd.c,v 1.13 2004/01/27 20:30:29 jsm Exp $	*/
21.5Scgd
31.1Scgd/*
41.5Scgd * Copyright (c) 1989, 1993
51.5Scgd *	The Regents of the University of California.  All rights reserved.
61.1Scgd *
71.1Scgd * This code is derived from software contributed to Berkeley by
81.5Scgd * Steve Hayman of the Indiana University Computer Science Dept.
91.1Scgd *
101.1Scgd * Redistribution and use in source and binary forms, with or without
111.1Scgd * modification, are permitted provided that the following conditions
121.1Scgd * are met:
131.1Scgd * 1. Redistributions of source code must retain the above copyright
141.1Scgd *    notice, this list of conditions and the following disclaimer.
151.1Scgd * 2. Redistributions in binary form must reproduce the above copyright
161.1Scgd *    notice, this list of conditions and the following disclaimer in the
171.1Scgd *    documentation and/or other materials provided with the distribution.
181.12Sagc * 3. Neither the name of the University nor the names of its contributors
191.1Scgd *    may be used to endorse or promote products derived from this software
201.1Scgd *    without specific prior written permission.
211.1Scgd *
221.1Scgd * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
231.1Scgd * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
241.1Scgd * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
251.1Scgd * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
261.1Scgd * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
271.1Scgd * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
281.1Scgd * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
291.1Scgd * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
301.1Scgd * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
311.1Scgd * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
321.1Scgd * SUCH DAMAGE.
331.1Scgd */
341.1Scgd
351.7Slukem#include <sys/cdefs.h>
361.1Scgd#ifndef lint
371.7Slukem__COPYRIGHT("@(#) Copyright (c) 1989, 1993\n\
381.7Slukem	The Regents of the University of California.  All rights reserved.\n");
391.1Scgd#endif /* not lint */
401.1Scgd
411.1Scgd#ifndef lint
421.7Slukem#if 0
431.5Scgdstatic char sccsid[] = "@(#)bcd.c	8.2 (Berkeley) 3/20/94";
441.7Slukem#else
451.13Sjsm__RCSID("$NetBSD: bcd.c,v 1.13 2004/01/27 20:30:29 jsm Exp $");
461.7Slukem#endif
471.1Scgd#endif /* not lint */
481.1Scgd
491.1Scgd/*
501.1Scgd * bcd --
511.1Scgd *
521.1Scgd * Read one line of standard input and produce something that looks like a
531.1Scgd * punch card.  An attempt to reimplement /usr/games/bcd.  All I looked at
541.1Scgd * was the man page.
551.1Scgd *
561.1Scgd * I couldn't find a BCD table handy so I wrote a shell script to deduce what
571.1Scgd * the patterns were that the old bcd was using for each possible 8-bit
581.1Scgd * character.  These are the results -- the low order 12 bits represent the
591.1Scgd * holes.  (A 1 bit is a hole.)  These may be wrong, but they match the old
601.1Scgd * program!
611.1Scgd *
621.1Scgd * Steve Hayman
631.1Scgd * sahayman@iuvax.cs.indiana.edu
641.1Scgd * 1989 11 30
651.5Scgd *
661.5Scgd *
671.5Scgd * I found an error in the table. The same error is found in the SunOS 4.1.1
681.5Scgd * version of bcd. It has apparently been around a long time. The error caused
691.5Scgd * 'Q' and 'R' to have the same punch code. I only noticed the error due to
701.5Scgd * someone pointing it out to me when the program was used to print a cover
711.5Scgd * for an APA!  The table was wrong in 4 places. The other error was masked
721.5Scgd * by the fact that the input is converted to upper case before lookup.
731.5Scgd *
741.5Scgd * Dyane Bruce
751.5Scgd * db@diana.ocunix.on.ca
761.5Scgd * Nov 5, 1993
771.1Scgd */
781.1Scgd
791.1Scgd#include <sys/types.h>
801.5Scgd
811.1Scgd#include <stdio.h>
821.11Smatt#include <stdlib.h>
831.6Scgd#include <string.h>
841.1Scgd#include <ctype.h>
851.8Shubertf#include <unistd.h>
861.1Scgd
871.9Shubertfconst u_short holes[256] = {
881.1Scgd    0x0,	 0x0,	  0x0,	   0x0,	    0x0,     0x0,     0x0,     0x0,
891.1Scgd    0x0,	 0x0,	  0x0,	   0x0,	    0x0,     0x0,     0x0,     0x0,
901.1Scgd    0x0,	 0x0,	  0x0,	   0x0,	    0x0,     0x0,     0x0,     0x0,
911.1Scgd    0x0,	 0x0,	  0x0,	   0x0,	    0x0,     0x0,     0x0,     0x0,
921.1Scgd    0x0,	 0x206,	  0x20a,   0x042,   0x442,   0x222,   0x800,   0x406,
931.1Scgd    0x812,	 0x412,	  0x422,   0xa00,   0x242,   0x400,   0x842,   0x300,
941.1Scgd    0x200,	 0x100,	  0x080,   0x040,   0x020,   0x010,   0x008,   0x004,
951.1Scgd    0x002,	 0x001,	  0x012,   0x40a,   0x80a,   0x212,   0x00a,   0x006,
961.1Scgd    0x022,	 0x900,	  0x880,   0x840,   0x820,   0x810,   0x808,   0x804,
971.1Scgd    0x802,	 0x801,	  0x500,   0x480,   0x440,   0x420,   0x410,   0x408,
981.3Sjtc    0x404,	 0x402,	  0x401,   0x280,   0x240,   0x220,   0x210,   0x208,
991.1Scgd    0x204,	 0x202,	  0x201,   0x082,   0x822,   0x600,   0x282,   0x30f,
1001.1Scgd    0x900,	 0x880,	  0x840,   0x820,   0x810,   0x808,   0x804,   0x802,
1011.1Scgd    0x801,	 0x500,	  0x480,   0x440,   0x420,   0x410,   0x408,   0x404,
1021.4Sjtc    0x402,	 0x401,	  0x280,   0x240,   0x220,   0x210,   0x208,   0x204,
1031.1Scgd    0x202,	 0x201,	  0x082,   0x806,   0x822,   0x600,   0x282,   0x0,
1041.1Scgd    0x0,	 0x0,	  0x0,	   0x0,	    0x0,     0x0,     0x0,     0x0,
1051.1Scgd    0x0,	 0x0,	  0x0,	   0x0,	    0x0,     0x0,     0x0,     0x0,
1061.1Scgd    0x0,	 0x0,	  0x0,	   0x0,	    0x0,     0x0,     0x0,     0x0,
1071.1Scgd    0x0,	 0x0,	  0x0,	   0x0,	    0x0,     0x0,     0x0,     0x0,
1081.1Scgd    0x206,	 0x20a,	  0x042,   0x442,   0x222,   0x800,   0x406,   0x812,
1091.1Scgd    0x412,	 0x422,	  0xa00,   0x242,   0x400,   0x842,   0x300,   0x200,
1101.1Scgd    0x100,	 0x080,	  0x040,   0x020,   0x010,   0x008,   0x004,   0x002,
1111.1Scgd    0x001,	 0x012,	  0x40a,   0x80a,   0x212,   0x00a,   0x006,   0x022,
1121.1Scgd    0x900,	 0x880,	  0x840,   0x820,   0x810,   0x808,   0x804,   0x802,
1131.1Scgd    0x801,	 0x500,	  0x480,   0x440,   0x420,   0x410,   0x408,   0x404,
1141.3Sjtc    0x402,	 0x401,	  0x280,   0x240,   0x220,   0x210,   0x208,   0x204,
1151.1Scgd    0x202,	 0x201,	  0x082,   0x806,   0x822,   0x600,   0x282,   0x30f,
1161.1Scgd    0x900,	 0x880,	  0x840,   0x820,   0x810,   0x808,   0x804,   0x802,
1171.1Scgd    0x801,	 0x500,	  0x480,   0x440,   0x420,   0x410,   0x408,   0x404,
1181.4Sjtc    0x402,	 0x401,	  0x280,   0x240,   0x220,   0x210,   0x208,   0x204,
1191.1Scgd    0x202,	 0x201,	  0x082,   0x806,   0x822,   0x600,   0x282,   0x0
1201.1Scgd};
1211.1Scgd
1221.1Scgd/*
1231.1Scgd * i'th bit of w.
1241.1Scgd */
1251.1Scgd#define	bit(w,i)	((w)&(1<<(i)))
1261.1Scgd
1271.13Sjsmint	main(int, char *[]);
1281.13Sjsmvoid	printcard(char *);
1291.7Slukem
1301.5Scgdint
1311.1Scgdmain(argc, argv)
1321.1Scgd	int argc;
1331.1Scgd	char **argv;
1341.1Scgd{
1351.1Scgd	char cardline[80];
1361.8Shubertf
1371.8Shubertf	/* revoke setgid privileges */
1381.10Smycroft	setgid(getgid());
1391.1Scgd
1401.1Scgd	/*
1411.1Scgd	 * The original bcd prompts with a "%" when reading from stdin,
1421.1Scgd	 * but this seems kind of silly.  So this one doesn't.
1431.1Scgd	 */
1441.1Scgd
1451.1Scgd	if (argc > 1) {
1461.1Scgd		while (--argc)
1471.1Scgd			printcard(*++argv);
1481.1Scgd	} else
1491.1Scgd		while (fgets(cardline, sizeof(cardline), stdin))
1501.1Scgd			printcard(cardline);
1511.1Scgd	exit(0);
1521.1Scgd}
1531.1Scgd
1541.1Scgd#define	COLUMNS	48
1551.1Scgd
1561.7Slukemvoid
1571.1Scgdprintcard(str)
1581.7Slukem	char *str;
1591.1Scgd{
1601.9Shubertf	static const char rowchars[] = "   123456789";
1611.7Slukem	int i, row;
1621.9Shubertf	unsigned char *p;
1631.1Scgd
1641.1Scgd	/* ruthlessly remove newlines and truncate at 48 characters. */
1651.7Slukem	if ((p = strchr(str, '\n')))
1661.1Scgd		*p = '\0';
1671.1Scgd
1681.1Scgd	if (strlen(str) > COLUMNS)
1691.1Scgd		str[COLUMNS] = '\0';
1701.1Scgd
1711.1Scgd	/* make string upper case. */
1721.1Scgd	for (p = str; *p; ++p)
1731.1Scgd		if (isascii(*p) && islower(*p))
1741.1Scgd			*p = toupper(*p);
1751.1Scgd
1761.1Scgd	 /* top of card */
1771.1Scgd	putchar(' ');
1781.1Scgd	for (i = 1; i <= COLUMNS; ++i)
1791.1Scgd		putchar('_');
1801.1Scgd	putchar('\n');
1811.1Scgd
1821.1Scgd	/*
1831.1Scgd	 * line of text.  Leave a blank if the character doesn't have
1841.1Scgd	 * a hole pattern.
1851.1Scgd	 */
1861.1Scgd	p = str;
1871.1Scgd	putchar('/');
1881.1Scgd	for (i = 1; *p; i++, p++)
1891.7Slukem		if (holes[(int)*p])
1901.1Scgd			putchar(*p);
1911.1Scgd		else
1921.1Scgd			putchar(' ');
1931.1Scgd	while (i++ <= COLUMNS)
1941.1Scgd		putchar(' ');
1951.1Scgd	putchar('|');
1961.1Scgd	putchar('\n');
1971.1Scgd
1981.1Scgd	/*
1991.1Scgd	 * 12 rows of potential holes; output a ']', which looks kind of
2001.1Scgd	 * like a hole, if the appropriate bit is set in the holes[] table.
2011.1Scgd	 * The original bcd output a '[', a backspace, five control A's,
2021.1Scgd	 * and then a ']'.  This seems a little excessive.
2031.1Scgd	 */
2041.1Scgd	for (row = 0; row <= 11; ++row) {
2051.1Scgd		putchar('|');
2061.1Scgd		for (i = 0, p = str; *p; i++, p++) {
2071.7Slukem			if (bit(holes[(int)*p], 11 - row))
2081.1Scgd				putchar(']');
2091.1Scgd			else
2101.1Scgd				putchar(rowchars[row]);
2111.1Scgd		}
2121.1Scgd		while (i++ < COLUMNS)
2131.1Scgd			putchar(rowchars[row]);
2141.1Scgd		putchar('|');
2151.1Scgd		putchar('\n');
2161.1Scgd	}
2171.1Scgd
2181.1Scgd	/* bottom of card */
2191.1Scgd	putchar('|');
2201.1Scgd	for (i = 1; i <= COLUMNS; i++)
2211.1Scgd		putchar('_');
2221.1Scgd	putchar('|');
2231.1Scgd	putchar('\n');
2241.1Scgd}
225