xref: /netbsd-src/usr.sbin/wsfontload/wsfontload.c (revision d5bbf827ba7a858b60d96a30d4336e0e369ea365)
1*d5bbf827Suwe /* $NetBSD: wsfontload.c,v 1.24 2022/05/12 22:08:55 uwe Exp $ */
2536eedccSdrochner 
3536eedccSdrochner /*
4536eedccSdrochner  * Copyright (c) 1999
5536eedccSdrochner  *	Matthias Drochner.  All rights reserved.
6536eedccSdrochner  *
7536eedccSdrochner  * Redistribution and use in source and binary forms, with or without
8536eedccSdrochner  * modification, are permitted provided that the following conditions
9536eedccSdrochner  * are met:
10536eedccSdrochner  * 1. Redistributions of source code must retain the above copyright
11536eedccSdrochner  *    notice, this list of conditions and the following disclaimer.
12536eedccSdrochner  * 2. Redistributions in binary form must reproduce the above copyright
13536eedccSdrochner  *    notice, this list of conditions and the following disclaimer in the
14536eedccSdrochner  *    documentation and/or other materials provided with the distribution.
15536eedccSdrochner  *
16536eedccSdrochner  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17536eedccSdrochner  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18536eedccSdrochner  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19536eedccSdrochner  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20536eedccSdrochner  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21536eedccSdrochner  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22536eedccSdrochner  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23536eedccSdrochner  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24536eedccSdrochner  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25536eedccSdrochner  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26536eedccSdrochner  *
27536eedccSdrochner  */
28536eedccSdrochner 
29536eedccSdrochner #include <stdio.h>
30536eedccSdrochner #include <fcntl.h>
3125bdbb66Scgd #include <stdlib.h>
323ab02a87Smatt #include <string.h>
33536eedccSdrochner #include <unistd.h>
34536eedccSdrochner #include <sys/types.h>
35536eedccSdrochner #include <sys/ioctl.h>
36dd033bddSmacallan #include <sys/stat.h>
37536eedccSdrochner #include <err.h>
38536eedccSdrochner 
39536eedccSdrochner #include <dev/wscons/wsconsio.h>
40536eedccSdrochner 
41b03606edSdrochner #define DEFDEV		"/dev/wsfont"
42536eedccSdrochner #define DEFWIDTH	8
43536eedccSdrochner #define DEFHEIGHT	16
44536eedccSdrochner #define DEFENC		WSDISPLAY_FONTENC_ISO
4504ddf7aeSad #define DEFBITORDER	WSDISPLAY_FONTORDER_L2R
4604ddf7aeSad #define DEFBYTEORDER	WSDISPLAY_FONTORDER_L2R
47536eedccSdrochner 
48bec77c5fSjoerg __dead static void usage(void);
49f7c9c96aSxtraeme static int getencoding(char *);
50f7c9c96aSxtraeme static const char *rgetencoding(int);
51f7c9c96aSxtraeme static const char *rgetfontorder(int);
526c1d54c4Shubertf 
536c1d54c4Shubertf static struct {
54f7c9c96aSxtraeme 	const char *name;
556c1d54c4Shubertf 	int val;
566c1d54c4Shubertf } fontorders[] = {
576c1d54c4Shubertf 	{ "known", WSDISPLAY_FONTORDER_KNOWN},
586c1d54c4Shubertf 	{ "l2r", WSDISPLAY_FONTORDER_L2R},
596c1d54c4Shubertf 	{ "r2l", WSDISPLAY_FONTORDER_R2L},
606c1d54c4Shubertf };
616c1d54c4Shubertf 
626c1d54c4Shubertf static struct {
63f7c9c96aSxtraeme 	const char *name;
646c1d54c4Shubertf 	int val;
656c1d54c4Shubertf } encodings[] = {
666c1d54c4Shubertf 	{"iso", WSDISPLAY_FONTENC_ISO},
676c1d54c4Shubertf 	{"ibm", WSDISPLAY_FONTENC_IBM},
686c1d54c4Shubertf 	{"pcvt", WSDISPLAY_FONTENC_PCVT},
69e569277fSdrochner 	{"iso7", WSDISPLAY_FONTENC_ISO7},
70e569277fSdrochner 	{"iso2", WSDISPLAY_FONTENC_ISO2},
7189c8725cSdrochner 	{"koi8r", WSDISPLAY_FONTENC_KOI8_R},
726c1d54c4Shubertf };
73536eedccSdrochner 
74536eedccSdrochner static void
usage(void)75f7c9c96aSxtraeme usage(void)
76536eedccSdrochner {
77536eedccSdrochner 
78536eedccSdrochner 	(void)fprintf(stderr,
79dfc86043Swiz 		"usage: %s [-Bbv] [-e encoding] [-f wsdev] [-h height]"
80dfc86043Swiz 		" [-N name] [-w width] [fontfile]\n"
81dfc86043Swiz 		"       %s -l\n",
82dfc86043Swiz 		      getprogname(),
8325bdbb66Scgd 		      getprogname());
84536eedccSdrochner 	exit(1);
85536eedccSdrochner }
86536eedccSdrochner 
876c1d54c4Shubertf /*
88a640fe8cSsnj  * map given fontorder to its string representation
896c1d54c4Shubertf  */
90f7c9c96aSxtraeme static const char *
rgetfontorder(int fontorder)91f7c9c96aSxtraeme rgetfontorder(int fontorder)
926c1d54c4Shubertf {
93c7dd3804Slukem 	size_t i;
946c1d54c4Shubertf 
956c1d54c4Shubertf 	for (i = 0; i < sizeof(fontorders) / sizeof(fontorders[0]); i++)
966c1d54c4Shubertf 		if (fontorders[i].val == fontorder)
976c1d54c4Shubertf 			return (fontorders[i].name);
986c1d54c4Shubertf 
996c1d54c4Shubertf 	return "unknown";
1006c1d54c4Shubertf }
1016c1d54c4Shubertf 
1026c1d54c4Shubertf /*
103a640fe8cSsnj  * map given encoding to its string representation
1046c1d54c4Shubertf  */
105f7c9c96aSxtraeme static const char *
rgetencoding(int enc)106f7c9c96aSxtraeme rgetencoding(int enc)
1076c1d54c4Shubertf {
108c7dd3804Slukem 	size_t i;
1096c1d54c4Shubertf 
1106c1d54c4Shubertf 	for (i = 0; i < sizeof(encodings) / sizeof(encodings[0]); i++)
1116c1d54c4Shubertf 		if (encodings[i].val == enc)
1126c1d54c4Shubertf 			return (encodings[i].name);
1136c1d54c4Shubertf 
1146c1d54c4Shubertf 	return "unknown";
1156c1d54c4Shubertf }
1166c1d54c4Shubertf 
1176c1d54c4Shubertf /*
1186c1d54c4Shubertf  * map given encoding string to integer value
1196c1d54c4Shubertf  */
1206c1d54c4Shubertf static int
getencoding(char * name)121f7c9c96aSxtraeme getencoding(char *name)
1226c1d54c4Shubertf {
123c7dd3804Slukem 	size_t i;
124c7dd3804Slukem 	int j;
1256c1d54c4Shubertf 
1266c1d54c4Shubertf 	for (i = 0; i < sizeof(encodings) / sizeof(encodings[0]); i++)
1276c1d54c4Shubertf 		if (!strcmp(name, encodings[i].name))
1286c1d54c4Shubertf 			return (encodings[i].val);
1296c1d54c4Shubertf 
130c7dd3804Slukem 	if (sscanf(name, "%d", &j) != 1)
1316c1d54c4Shubertf 		errx(1, "invalid encoding");
132c7dd3804Slukem 	return (j);
1336c1d54c4Shubertf }
1346c1d54c4Shubertf 
135536eedccSdrochner int
main(int argc,char ** argv)136f7c9c96aSxtraeme main(int argc, char **argv)
137536eedccSdrochner {
138f7c9c96aSxtraeme 	const char *wsdev;
139536eedccSdrochner 	struct wsdisplay_font f;
140dd033bddSmacallan 	struct stat st;
141536eedccSdrochner 	size_t len;
1421322e992Smacallan 	int c, res, wsfd, ffd, verbose = 0, listfonts = 0;
143023c9317Smacallan 	int use_embedded_name = 1;
144536eedccSdrochner 	void *buf;
145dd033bddSmacallan 	char nbuf[65];
146536eedccSdrochner 
147536eedccSdrochner 	wsdev = DEFDEV;
148536eedccSdrochner 	f.fontwidth = DEFWIDTH;
149536eedccSdrochner 	f.fontheight = DEFHEIGHT;
150536eedccSdrochner 	f.firstchar = 0;
151536eedccSdrochner 	f.numchars = 256;
152536eedccSdrochner 	f.stride = 0;
153536eedccSdrochner 	f.encoding = DEFENC;
154536eedccSdrochner 	f.name = 0;
15504ddf7aeSad 	f.bitorder = DEFBITORDER;
15604ddf7aeSad 	f.byteorder = DEFBYTEORDER;
157536eedccSdrochner 
1581322e992Smacallan 	while ((c = getopt(argc, argv, "f:w:h:e:N:bBvl")) != -1) {
159536eedccSdrochner 		switch (c) {
160536eedccSdrochner 		case 'f':
161536eedccSdrochner 			wsdev = optarg;
162536eedccSdrochner 			break;
1631322e992Smacallan 		case 'l':
1641322e992Smacallan 			listfonts = 1;
1651322e992Smacallan 			break;
166536eedccSdrochner 		case 'w':
167536eedccSdrochner 			if (sscanf(optarg, "%d", &f.fontwidth) != 1)
168536eedccSdrochner 				errx(1, "invalid font width");
169536eedccSdrochner 			break;
170536eedccSdrochner 		case 'h':
171536eedccSdrochner 			if (sscanf(optarg, "%d", &f.fontheight) != 1)
172536eedccSdrochner 				errx(1, "invalid font height");
173536eedccSdrochner 			break;
174536eedccSdrochner 		case 'e':
175536eedccSdrochner 			f.encoding = getencoding(optarg);
176536eedccSdrochner 			break;
177536eedccSdrochner 		case 'N':
178536eedccSdrochner 			f.name = optarg;
179023c9317Smacallan 			use_embedded_name = 0;
180536eedccSdrochner 			break;
181a4eb8384Sad 		case 'b':
182a4eb8384Sad 			f.bitorder = WSDISPLAY_FONTORDER_R2L;
183a4eb8384Sad 			break;
184a4eb8384Sad 		case 'B':
185a4eb8384Sad 			f.byteorder = WSDISPLAY_FONTORDER_R2L;
186a4eb8384Sad 			break;
1876c1d54c4Shubertf 		case 'v':
1886c1d54c4Shubertf 			verbose = 1;
1896c1d54c4Shubertf 			break;
190536eedccSdrochner 		case '?':
191536eedccSdrochner 		default:
192536eedccSdrochner 			usage();
193536eedccSdrochner 			break;
194536eedccSdrochner 		}
195536eedccSdrochner 	}
196536eedccSdrochner 	argc -= optind;
197536eedccSdrochner 	argv += optind;
198536eedccSdrochner 
199536eedccSdrochner 	if (argc > 1)
200536eedccSdrochner 		usage();
201536eedccSdrochner 
202*d5bbf827Suwe 	wsfd = open(wsdev, listfonts ? O_RDONLY : O_RDWR, 0);
203536eedccSdrochner 	if (wsfd < 0)
2046c1d54c4Shubertf 		err(2, "open ws-device %s", wsdev);
205536eedccSdrochner 
2061322e992Smacallan 	if (listfonts == 1) {
2071322e992Smacallan 		struct wsdisplayio_fontinfo fi;
2081322e992Smacallan 		int ret;
2091322e992Smacallan 		unsigned int i;
2101322e992Smacallan 
2111322e992Smacallan 		fi.fi_buffersize = 4096;
2121322e992Smacallan 		fi.fi_numentries = 0;
2131322e992Smacallan 		fi.fi_fonts = malloc(4096);
2141322e992Smacallan 		ret = ioctl(wsfd, WSDISPLAYIO_LISTFONTS, &fi);
2151322e992Smacallan 		if (fi.fi_fonts == NULL || ret != 0) {
2161322e992Smacallan 			err(1, "error fetching font list\n");
2171322e992Smacallan 		}
2181322e992Smacallan 		for (i = 0; i < fi.fi_numentries; i++) {
2191322e992Smacallan 			printf("%s %dx%d\n", fi.fi_fonts[i].fd_name,
2201322e992Smacallan 			    fi.fi_fonts[i].fd_width, fi.fi_fonts[i].fd_height);
2211322e992Smacallan 		}
2221322e992Smacallan 		return 0;
2231322e992Smacallan 	}
2241322e992Smacallan 
225536eedccSdrochner 	if (argc > 0) {
226536eedccSdrochner 		ffd = open(argv[0], O_RDONLY, 0);
227536eedccSdrochner 		if (ffd < 0)
2286c1d54c4Shubertf 			err(4, "open font %s", argv[0]);
229536eedccSdrochner 		if (!f.name)
230536eedccSdrochner 			f.name = argv[0];
231536eedccSdrochner 	} else
232536eedccSdrochner 		ffd = 0;
233536eedccSdrochner 
234536eedccSdrochner 	if (!f.stride)
235536eedccSdrochner 		f.stride = (f.fontwidth + 7) / 8;
236536eedccSdrochner 	len = f.fontheight * f.numchars * f.stride;
237023c9317Smacallan 	if ((ffd != 0) && (fstat(ffd, &st) == 0)) {
2386722e01bSmacallan 		if ((off_t)len != st.st_size) {
239dd033bddSmacallan 			uint32_t foo = 0;
240dd033bddSmacallan 			char b[65];
241dd033bddSmacallan 			len = st.st_size;
242dd033bddSmacallan 			/* read header */
243dd033bddSmacallan 			read(ffd, b, 4);
244dd033bddSmacallan 			if (strncmp(b, "WSFT", 4) != 0)
245dd033bddSmacallan 				errx(1, "invalid wsf file ");
246dd033bddSmacallan 			read(ffd, b, 64);
247023c9317Smacallan 			if (use_embedded_name) {
248dd033bddSmacallan 				b[64] = 0;
249dd033bddSmacallan 				strcpy(nbuf, b);
250dd033bddSmacallan 				f.name = nbuf;
251023c9317Smacallan 			}
252dd033bddSmacallan 			read(ffd, &foo, 4);
253dd033bddSmacallan 			f.firstchar = le32toh(foo);
254dd033bddSmacallan 			read(ffd, &foo, 4);
255dd033bddSmacallan 			f.numchars = le32toh(foo);
256dd033bddSmacallan 			read(ffd, &foo, 4);
257dd033bddSmacallan 			f.encoding = le32toh(foo);
258dd033bddSmacallan 			read(ffd, &foo, 4);
259dd033bddSmacallan 			f.fontwidth = le32toh(foo);
260dd033bddSmacallan 			read(ffd, &foo, 4);
261dd033bddSmacallan 			f.fontheight = le32toh(foo);
262dd033bddSmacallan 			read(ffd, &foo, 4);
263dd033bddSmacallan 			f.stride = le32toh(foo);
264dd033bddSmacallan 			read(ffd, &foo, 4);
265dd033bddSmacallan 			f.bitorder = le32toh(foo);
266dd033bddSmacallan 			read(ffd, &foo, 4);
267dd033bddSmacallan 			f.byteorder = le32toh(foo);
268dd033bddSmacallan 			len = f.numchars * f.fontheight * f.stride;
269dd033bddSmacallan 		}
270dd033bddSmacallan 	}
271dd033bddSmacallan 
272536eedccSdrochner 	if (!len)
273536eedccSdrochner 		errx(1, "invalid font size");
274536eedccSdrochner 
275536eedccSdrochner 	buf = malloc(len);
276536eedccSdrochner 	if (!buf)
277536eedccSdrochner 		errx(1, "malloc");
278536eedccSdrochner 	res = read(ffd, buf, len);
279536eedccSdrochner 	if (res < 0)
280536eedccSdrochner 		err(4, "read font");
281c7dd3804Slukem 	if ((size_t)res != len)
282536eedccSdrochner 		errx(4, "short read");
283536eedccSdrochner 
284536eedccSdrochner 	f.data = buf;
285536eedccSdrochner 
2866c1d54c4Shubertf 	if (verbose) {
2876c1d54c4Shubertf 		printf("name:       %s\n", f.name);
2886c1d54c4Shubertf 		printf("firstchar:  %d\n", f.firstchar);
2896c1d54c4Shubertf 		printf("numchars:   %d\n", f.numchars);
2906c1d54c4Shubertf 		printf("encoding:   %s (%d)\n",
2916c1d54c4Shubertf 			rgetencoding(f.encoding), f.encoding);
2926c1d54c4Shubertf 		printf("fontwidth:  %d\n", f.fontwidth);
2936c1d54c4Shubertf 		printf("fontheight: %d\n", f.fontheight);
2946c1d54c4Shubertf 		printf("stride:     %d\n", f.stride);
2956c1d54c4Shubertf 		printf("bitorder:   %s (%d)\n",
2966c1d54c4Shubertf 			rgetfontorder(f.bitorder), f.bitorder);
2976c1d54c4Shubertf 		printf("byteorder:  %s (%d)\n",
2986c1d54c4Shubertf 			rgetfontorder(f.byteorder), f.byteorder);
2996c1d54c4Shubertf 	}
3006c1d54c4Shubertf 
301536eedccSdrochner 	res = ioctl(wsfd, WSDISPLAYIO_LDFONT, &f);
302536eedccSdrochner 	if (res < 0)
303536eedccSdrochner 		err(3, "WSDISPLAYIO_LDFONT");
304536eedccSdrochner 
305536eedccSdrochner 	return (0);
306536eedccSdrochner }
307