Home | History | Annotate | Line # | Download | only in wsfontload
      1  1.24       uwe /* $NetBSD: wsfontload.c,v 1.24 2022/05/12 22:08:55 uwe Exp $ */
      2   1.1  drochner 
      3   1.1  drochner /*
      4   1.1  drochner  * Copyright (c) 1999
      5   1.1  drochner  *	Matthias Drochner.  All rights reserved.
      6   1.1  drochner  *
      7   1.1  drochner  * Redistribution and use in source and binary forms, with or without
      8   1.1  drochner  * modification, are permitted provided that the following conditions
      9   1.1  drochner  * are met:
     10   1.1  drochner  * 1. Redistributions of source code must retain the above copyright
     11   1.1  drochner  *    notice, this list of conditions and the following disclaimer.
     12   1.1  drochner  * 2. Redistributions in binary form must reproduce the above copyright
     13   1.1  drochner  *    notice, this list of conditions and the following disclaimer in the
     14   1.1  drochner  *    documentation and/or other materials provided with the distribution.
     15   1.1  drochner  *
     16   1.1  drochner  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
     17   1.1  drochner  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
     18   1.1  drochner  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
     19   1.1  drochner  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
     20   1.1  drochner  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
     21   1.1  drochner  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
     22   1.1  drochner  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
     23   1.1  drochner  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     24   1.1  drochner  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
     25   1.1  drochner  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     26   1.1  drochner  *
     27   1.1  drochner  */
     28   1.1  drochner 
     29   1.1  drochner #include <stdio.h>
     30   1.1  drochner #include <fcntl.h>
     31   1.6       cgd #include <stdlib.h>
     32   1.4      matt #include <string.h>
     33   1.1  drochner #include <unistd.h>
     34   1.1  drochner #include <sys/types.h>
     35   1.1  drochner #include <sys/ioctl.h>
     36  1.19  macallan #include <sys/stat.h>
     37   1.1  drochner #include <err.h>
     38   1.1  drochner 
     39   1.1  drochner #include <dev/wscons/wsconsio.h>
     40   1.1  drochner 
     41   1.8  drochner #define DEFDEV		"/dev/wsfont"
     42   1.3        ad #define DEFWIDTH	8
     43   1.3        ad #define DEFHEIGHT	16
     44   1.3        ad #define DEFENC		WSDISPLAY_FONTENC_ISO
     45   1.3        ad #define DEFBITORDER	WSDISPLAY_FONTORDER_L2R
     46   1.3        ad #define DEFBYTEORDER	WSDISPLAY_FONTORDER_L2R
     47   1.1  drochner 
     48  1.17     joerg __dead static void usage(void);
     49  1.10   xtraeme static int getencoding(char *);
     50  1.10   xtraeme static const char *rgetencoding(int);
     51  1.10   xtraeme static const char *rgetfontorder(int);
     52   1.5   hubertf 
     53   1.5   hubertf static struct {
     54  1.10   xtraeme 	const char *name;
     55   1.5   hubertf 	int val;
     56   1.5   hubertf } fontorders[] = {
     57   1.5   hubertf 	{ "known", WSDISPLAY_FONTORDER_KNOWN},
     58   1.5   hubertf 	{ "l2r", WSDISPLAY_FONTORDER_L2R},
     59   1.5   hubertf 	{ "r2l", WSDISPLAY_FONTORDER_R2L},
     60   1.5   hubertf };
     61   1.5   hubertf 
     62   1.5   hubertf static struct {
     63  1.10   xtraeme 	const char *name;
     64   1.5   hubertf 	int val;
     65   1.5   hubertf } encodings[] = {
     66   1.5   hubertf 	{"iso", WSDISPLAY_FONTENC_ISO},
     67   1.5   hubertf 	{"ibm", WSDISPLAY_FONTENC_IBM},
     68   1.5   hubertf 	{"pcvt", WSDISPLAY_FONTENC_PCVT},
     69   1.7  drochner 	{"iso7", WSDISPLAY_FONTENC_ISO7},
     70   1.7  drochner 	{"iso2", WSDISPLAY_FONTENC_ISO2},
     71  1.15  drochner 	{"koi8r", WSDISPLAY_FONTENC_KOI8_R},
     72   1.5   hubertf };
     73   1.1  drochner 
     74   1.1  drochner static void
     75  1.10   xtraeme usage(void)
     76   1.1  drochner {
     77   1.1  drochner 
     78   1.1  drochner 	(void)fprintf(stderr,
     79  1.23       wiz 		"usage: %s [-Bbv] [-e encoding] [-f wsdev] [-h height]"
     80  1.23       wiz 		" [-N name] [-w width] [fontfile]\n"
     81  1.23       wiz 		"       %s -l\n",
     82  1.23       wiz 		      getprogname(),
     83   1.6       cgd 		      getprogname());
     84   1.1  drochner 	exit(1);
     85   1.1  drochner }
     86   1.1  drochner 
     87   1.5   hubertf /*
     88  1.11       snj  * map given fontorder to its string representation
     89   1.5   hubertf  */
     90  1.10   xtraeme static const char *
     91  1.10   xtraeme rgetfontorder(int fontorder)
     92   1.5   hubertf {
     93  1.14     lukem 	size_t i;
     94   1.5   hubertf 
     95   1.5   hubertf 	for (i = 0; i < sizeof(fontorders) / sizeof(fontorders[0]); i++)
     96   1.5   hubertf 		if (fontorders[i].val == fontorder)
     97   1.5   hubertf 			return (fontorders[i].name);
     98   1.5   hubertf 
     99   1.5   hubertf 	return "unknown";
    100   1.5   hubertf }
    101   1.5   hubertf 
    102   1.5   hubertf /*
    103  1.11       snj  * map given encoding to its string representation
    104   1.5   hubertf  */
    105  1.10   xtraeme static const char *
    106  1.10   xtraeme rgetencoding(int enc)
    107   1.5   hubertf {
    108  1.14     lukem 	size_t i;
    109   1.5   hubertf 
    110   1.5   hubertf 	for (i = 0; i < sizeof(encodings) / sizeof(encodings[0]); i++)
    111   1.5   hubertf 		if (encodings[i].val == enc)
    112   1.5   hubertf 			return (encodings[i].name);
    113   1.5   hubertf 
    114   1.5   hubertf 	return "unknown";
    115   1.5   hubertf }
    116   1.5   hubertf 
    117   1.5   hubertf /*
    118   1.5   hubertf  * map given encoding string to integer value
    119   1.5   hubertf  */
    120   1.5   hubertf static int
    121  1.10   xtraeme getencoding(char *name)
    122   1.5   hubertf {
    123  1.14     lukem 	size_t i;
    124  1.14     lukem 	int j;
    125   1.5   hubertf 
    126   1.5   hubertf 	for (i = 0; i < sizeof(encodings) / sizeof(encodings[0]); i++)
    127   1.5   hubertf 		if (!strcmp(name, encodings[i].name))
    128   1.5   hubertf 			return (encodings[i].val);
    129   1.5   hubertf 
    130  1.14     lukem 	if (sscanf(name, "%d", &j) != 1)
    131   1.5   hubertf 		errx(1, "invalid encoding");
    132  1.14     lukem 	return (j);
    133   1.5   hubertf }
    134   1.5   hubertf 
    135   1.1  drochner int
    136  1.10   xtraeme main(int argc, char **argv)
    137   1.1  drochner {
    138  1.10   xtraeme 	const char *wsdev;
    139   1.1  drochner 	struct wsdisplay_font f;
    140  1.19  macallan 	struct stat st;
    141   1.1  drochner 	size_t len;
    142  1.22  macallan 	int c, res, wsfd, ffd, verbose = 0, listfonts = 0;
    143  1.21  macallan 	int use_embedded_name = 1;
    144   1.1  drochner 	void *buf;
    145  1.19  macallan 	char nbuf[65];
    146   1.1  drochner 
    147   1.1  drochner 	wsdev = DEFDEV;
    148   1.1  drochner 	f.fontwidth = DEFWIDTH;
    149   1.1  drochner 	f.fontheight = DEFHEIGHT;
    150   1.1  drochner 	f.firstchar = 0;
    151   1.1  drochner 	f.numchars = 256;
    152   1.1  drochner 	f.stride = 0;
    153   1.1  drochner 	f.encoding = DEFENC;
    154   1.1  drochner 	f.name = 0;
    155   1.3        ad 	f.bitorder = DEFBITORDER;
    156   1.3        ad 	f.byteorder = DEFBYTEORDER;
    157   1.1  drochner 
    158  1.22  macallan 	while ((c = getopt(argc, argv, "f:w:h:e:N:bBvl")) != -1) {
    159   1.1  drochner 		switch (c) {
    160   1.1  drochner 		case 'f':
    161   1.1  drochner 			wsdev = optarg;
    162   1.1  drochner 			break;
    163  1.22  macallan 		case 'l':
    164  1.22  macallan 			listfonts = 1;
    165  1.22  macallan 			break;
    166   1.1  drochner 		case 'w':
    167   1.1  drochner 			if (sscanf(optarg, "%d", &f.fontwidth) != 1)
    168   1.1  drochner 				errx(1, "invalid font width");
    169   1.1  drochner 			break;
    170   1.1  drochner 		case 'h':
    171   1.1  drochner 			if (sscanf(optarg, "%d", &f.fontheight) != 1)
    172   1.1  drochner 				errx(1, "invalid font height");
    173   1.1  drochner 			break;
    174   1.1  drochner 		case 'e':
    175   1.1  drochner 			f.encoding = getencoding(optarg);
    176   1.1  drochner 			break;
    177   1.1  drochner 		case 'N':
    178   1.1  drochner 			f.name = optarg;
    179  1.21  macallan 			use_embedded_name = 0;
    180   1.2        ad 			break;
    181   1.2        ad 		case 'b':
    182   1.2        ad 			f.bitorder = WSDISPLAY_FONTORDER_R2L;
    183   1.2        ad 			break;
    184   1.2        ad 		case 'B':
    185   1.2        ad 			f.byteorder = WSDISPLAY_FONTORDER_R2L;
    186   1.1  drochner 			break;
    187   1.5   hubertf 		case 'v':
    188   1.5   hubertf 			verbose = 1;
    189   1.5   hubertf 			break;
    190   1.1  drochner 		case '?':
    191   1.1  drochner 		default:
    192   1.1  drochner 			usage();
    193   1.1  drochner 			break;
    194   1.1  drochner 		}
    195   1.1  drochner 	}
    196   1.1  drochner 	argc -= optind;
    197   1.1  drochner 	argv += optind;
    198   1.1  drochner 
    199   1.1  drochner 	if (argc > 1)
    200   1.1  drochner 		usage();
    201   1.1  drochner 
    202  1.24       uwe 	wsfd = open(wsdev, listfonts ? O_RDONLY : O_RDWR, 0);
    203   1.1  drochner 	if (wsfd < 0)
    204   1.5   hubertf 		err(2, "open ws-device %s", wsdev);
    205   1.1  drochner 
    206  1.22  macallan 	if (listfonts == 1) {
    207  1.22  macallan 		struct wsdisplayio_fontinfo fi;
    208  1.22  macallan 		int ret;
    209  1.22  macallan 		unsigned int i;
    210  1.22  macallan 
    211  1.22  macallan 		fi.fi_buffersize = 4096;
    212  1.22  macallan 		fi.fi_numentries = 0;
    213  1.22  macallan 		fi.fi_fonts = malloc(4096);
    214  1.22  macallan 		ret = ioctl(wsfd, WSDISPLAYIO_LISTFONTS, &fi);
    215  1.22  macallan 		if (fi.fi_fonts == NULL || ret != 0) {
    216  1.22  macallan 			err(1, "error fetching font list\n");
    217  1.22  macallan 		}
    218  1.22  macallan 		for (i = 0; i < fi.fi_numentries; i++) {
    219  1.22  macallan 			printf("%s %dx%d\n", fi.fi_fonts[i].fd_name,
    220  1.22  macallan 			    fi.fi_fonts[i].fd_width, fi.fi_fonts[i].fd_height);
    221  1.22  macallan 		}
    222  1.22  macallan 		return 0;
    223  1.22  macallan 	}
    224  1.22  macallan 
    225   1.1  drochner 	if (argc > 0) {
    226   1.1  drochner 		ffd = open(argv[0], O_RDONLY, 0);
    227   1.1  drochner 		if (ffd < 0)
    228   1.5   hubertf 			err(4, "open font %s", argv[0]);
    229   1.1  drochner 		if (!f.name)
    230   1.1  drochner 			f.name = argv[0];
    231   1.1  drochner 	} else
    232   1.1  drochner 		ffd = 0;
    233   1.1  drochner 
    234   1.1  drochner 	if (!f.stride)
    235   1.1  drochner 		f.stride = (f.fontwidth + 7) / 8;
    236   1.1  drochner 	len = f.fontheight * f.numchars * f.stride;
    237  1.21  macallan 	if ((ffd != 0) && (fstat(ffd, &st) == 0)) {
    238  1.20  macallan 		if ((off_t)len != st.st_size) {
    239  1.19  macallan 			uint32_t foo = 0;
    240  1.19  macallan 			char b[65];
    241  1.19  macallan 			len = st.st_size;
    242  1.19  macallan 			/* read header */
    243  1.19  macallan 			read(ffd, b, 4);
    244  1.19  macallan 			if (strncmp(b, "WSFT", 4) != 0)
    245  1.19  macallan 				errx(1, "invalid wsf file ");
    246  1.19  macallan 			read(ffd, b, 64);
    247  1.21  macallan 			if (use_embedded_name) {
    248  1.21  macallan 				b[64] = 0;
    249  1.21  macallan 				strcpy(nbuf, b);
    250  1.21  macallan 				f.name = nbuf;
    251  1.21  macallan 			}
    252  1.19  macallan 			read(ffd, &foo, 4);
    253  1.19  macallan 			f.firstchar = le32toh(foo);
    254  1.19  macallan 			read(ffd, &foo, 4);
    255  1.19  macallan 			f.numchars = le32toh(foo);
    256  1.19  macallan 			read(ffd, &foo, 4);
    257  1.19  macallan 			f.encoding = le32toh(foo);
    258  1.19  macallan 			read(ffd, &foo, 4);
    259  1.19  macallan 			f.fontwidth = le32toh(foo);
    260  1.19  macallan 			read(ffd, &foo, 4);
    261  1.19  macallan 			f.fontheight = le32toh(foo);
    262  1.19  macallan 			read(ffd, &foo, 4);
    263  1.19  macallan 			f.stride = le32toh(foo);
    264  1.19  macallan 			read(ffd, &foo, 4);
    265  1.19  macallan 			f.bitorder = le32toh(foo);
    266  1.19  macallan 			read(ffd, &foo, 4);
    267  1.19  macallan 			f.byteorder = le32toh(foo);
    268  1.19  macallan 			len = f.numchars * f.fontheight * f.stride;
    269  1.19  macallan 		}
    270  1.19  macallan 	}
    271  1.19  macallan 
    272   1.1  drochner 	if (!len)
    273   1.1  drochner 		errx(1, "invalid font size");
    274   1.1  drochner 
    275   1.1  drochner 	buf = malloc(len);
    276   1.1  drochner 	if (!buf)
    277   1.1  drochner 		errx(1, "malloc");
    278   1.1  drochner 	res = read(ffd, buf, len);
    279   1.1  drochner 	if (res < 0)
    280   1.1  drochner 		err(4, "read font");
    281  1.14     lukem 	if ((size_t)res != len)
    282   1.1  drochner 		errx(4, "short read");
    283   1.1  drochner 
    284   1.1  drochner 	f.data = buf;
    285   1.1  drochner 
    286   1.5   hubertf 	if (verbose) {
    287   1.5   hubertf 		printf("name:       %s\n", f.name);
    288   1.5   hubertf 		printf("firstchar:  %d\n", f.firstchar);
    289   1.5   hubertf 		printf("numchars:   %d\n", f.numchars);
    290   1.5   hubertf 		printf("encoding:   %s (%d)\n",
    291   1.5   hubertf 			rgetencoding(f.encoding), f.encoding);
    292   1.5   hubertf 		printf("fontwidth:  %d\n", f.fontwidth);
    293   1.5   hubertf 		printf("fontheight: %d\n", f.fontheight);
    294   1.5   hubertf 		printf("stride:     %d\n", f.stride);
    295   1.5   hubertf 		printf("bitorder:   %s (%d)\n",
    296   1.5   hubertf 			rgetfontorder(f.bitorder), f.bitorder);
    297   1.5   hubertf 		printf("byteorder:  %s (%d)\n",
    298   1.5   hubertf 			rgetfontorder(f.byteorder), f.byteorder);
    299   1.5   hubertf 	}
    300   1.5   hubertf 
    301   1.1  drochner 	res = ioctl(wsfd, WSDISPLAYIO_LDFONT, &f);
    302   1.1  drochner 	if (res < 0)
    303   1.1  drochner 		err(3, "WSDISPLAYIO_LDFONT");
    304   1.1  drochner 
    305   1.1  drochner 	return (0);
    306   1.1  drochner }
    307