xref: /netbsd/usr.sbin/wsconscfg/wsconscfg.c (revision 41e309b9)
1*41e309b9Ssevan /* $NetBSD: wsconscfg.c,v 1.18 2017/05/04 16:26:10 sevan Exp $ */
2b43055f3Sdrochner 
3b43055f3Sdrochner /*
4b43055f3Sdrochner  * Copyright (c) 1999
5b43055f3Sdrochner  *	Matthias Drochner.  All rights reserved.
6b43055f3Sdrochner  *
7b43055f3Sdrochner  * Redistribution and use in source and binary forms, with or without
8b43055f3Sdrochner  * modification, are permitted provided that the following conditions
9b43055f3Sdrochner  * are met:
10b43055f3Sdrochner  * 1. Redistributions of source code must retain the above copyright
11b43055f3Sdrochner  *    notice, this list of conditions and the following disclaimer.
12b43055f3Sdrochner  * 2. Redistributions in binary form must reproduce the above copyright
13b43055f3Sdrochner  *    notice, this list of conditions and the following disclaimer in the
14b43055f3Sdrochner  *    documentation and/or other materials provided with the distribution.
15b43055f3Sdrochner  *
16b43055f3Sdrochner  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17b43055f3Sdrochner  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18b43055f3Sdrochner  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19b43055f3Sdrochner  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20b43055f3Sdrochner  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21b43055f3Sdrochner  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22b43055f3Sdrochner  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23b43055f3Sdrochner  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24b43055f3Sdrochner  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25b43055f3Sdrochner  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26b43055f3Sdrochner  *
27b43055f3Sdrochner  */
28b43055f3Sdrochner 
2936c7456dSperry #include <sys/cdefs.h>
30b43055f3Sdrochner #include <stdio.h>
313ab02a87Smatt #include <stdlib.h>
32b43055f3Sdrochner #include <fcntl.h>
33b43055f3Sdrochner #include <unistd.h>
34b43055f3Sdrochner #include <sys/types.h>
35b43055f3Sdrochner #include <sys/ioctl.h>
36b43055f3Sdrochner #include <err.h>
37b12f1e56Sdrochner #include <errno.h>
38b43055f3Sdrochner 
39b43055f3Sdrochner #include <dev/wscons/wsconsio.h>
401a8fb0adSchristos #include <dev/wscons/wsdisplay_usl_io.h>
41b43055f3Sdrochner 
42b43055f3Sdrochner #define DEFDEV "/dev/ttyEcfg"
43b43055f3Sdrochner 
448b0f9554Sperry static void usage(void) __dead;
45b43055f3Sdrochner 
46b43055f3Sdrochner static void
usage(void)471a8fb0adSchristos usage(void)
48b43055f3Sdrochner {
49ab547228Schristos 	const char *p = getprogname();
50b43055f3Sdrochner 	(void)fprintf(stderr,
51eaa6bf85Swiz 	     "Usage: %s [-e emul] [-f ctldev] [-t type] index\n"
52ab547228Schristos 	     "\t%s -d [-F] [-f ctldev] index\n"
534659bd50Swiz 	     "\t%s -g [-f ctldev]\n"
54ab547228Schristos 	     "\t%s -k | -m [-d] [-f ctldev] [index]\n"
554659bd50Swiz 	     "\t%s -s [-f ctldev] index\n", p, p, p, p, p);
56b43055f3Sdrochner 	exit(1);
57b43055f3Sdrochner }
58b43055f3Sdrochner 
59b43055f3Sdrochner int
main(int argc,char ** argv)601a8fb0adSchristos main(int argc, char **argv)
61b43055f3Sdrochner {
621a8fb0adSchristos 	const char *wsdev;
63ab547228Schristos 	int c, delete, kbd, idx, wsfd, swtch, get, mux;
64b43055f3Sdrochner 	struct wsdisplay_addscreendata asd;
65b43055f3Sdrochner 	struct wsdisplay_delscreendata dsd;
66e7af5efeSaugustss 	struct wsmux_device wmd;
67b43055f3Sdrochner 
681a8fb0adSchristos 	setprogname(argv[0]);
69b43055f3Sdrochner 	wsdev = DEFDEV;
70b43055f3Sdrochner 	delete = 0;
716f1e305bSdrochner 	kbd = 0;
72e7af5efeSaugustss 	mux = 0;
731a8fb0adSchristos 	swtch = 0;
741a8fb0adSchristos 	idx = -1;
75ab547228Schristos 	get = 0;
76b43055f3Sdrochner 	asd.screentype = 0;
77b43055f3Sdrochner 	asd.emul = 0;
78b43055f3Sdrochner 	dsd.flags = 0;
79b43055f3Sdrochner 
80ab547228Schristos 	while ((c = getopt(argc, argv, "de:Ff:gkmst:")) != -1) {
81b43055f3Sdrochner 		switch (c) {
82ab547228Schristos 		case 'd':
83ab547228Schristos 			delete++;
84ab547228Schristos 			break;
85ab547228Schristos 		case 'e':
86ab547228Schristos 			asd.emul = optarg;
87ab547228Schristos 			break;
88ab547228Schristos 		case 'F':
89ab547228Schristos 			dsd.flags |= WSDISPLAY_DELSCR_FORCE;
90ab547228Schristos 			break;
91b43055f3Sdrochner 		case 'f':
92b43055f3Sdrochner 			wsdev = optarg;
93b43055f3Sdrochner 			break;
94ab547228Schristos 		case 'g':
95ab547228Schristos 			get++;
96b43055f3Sdrochner 			break;
976f1e305bSdrochner 		case 'k':
986f1e305bSdrochner 			kbd++;
996f1e305bSdrochner 			break;
100e7af5efeSaugustss 		case 'm':
101e7af5efeSaugustss 			mux++;
102e7af5efeSaugustss 			kbd++;
103e7af5efeSaugustss 			break;
1041a8fb0adSchristos 		case 's':
1051a8fb0adSchristos 			swtch++;
1061a8fb0adSchristos 			break;
107b43055f3Sdrochner 		case 't':
108b43055f3Sdrochner 			asd.screentype = optarg;
109b43055f3Sdrochner 			break;
110b43055f3Sdrochner 		case '?':
111b43055f3Sdrochner 		default:
112b43055f3Sdrochner 			usage();
113b43055f3Sdrochner 			break;
114b43055f3Sdrochner 		}
115b43055f3Sdrochner 	}
116b43055f3Sdrochner 	argc -= optind;
117b43055f3Sdrochner 	argv += optind;
118b43055f3Sdrochner 
119ab547228Schristos 	if (!get || argc != 0) {
1201a8fb0adSchristos 		if ((kbd || swtch) ? (argc > 1) : (argc != 1))
121b43055f3Sdrochner 			usage();
122b43055f3Sdrochner 
1236f1e305bSdrochner 		if (argc > 0 && sscanf(argv[0], "%d", &idx) != 1)
124b43055f3Sdrochner 			errx(1, "invalid index");
125ab547228Schristos 	}
126ab547228Schristos 	if ((wsfd = open(wsdev, get ? O_RDONLY : O_RDWR)) == -1)
1271a8fb0adSchristos 		err(EXIT_FAILURE, "Cannot open `%s'", wsdev);
128b43055f3Sdrochner 
129ab547228Schristos 
1301a8fb0adSchristos 	if (swtch) {
1311a8fb0adSchristos 		if (ioctl(wsfd, VT_ACTIVATE, idx) == -1)
1321a8fb0adSchristos 		    err(EXIT_FAILURE, "Cannot switch to %d", idx);
133ab547228Schristos 	} else if (get) {
134ab547228Schristos 		if (ioctl(wsfd, VT_GETACTIVE, &idx) == -1)
135ab547228Schristos 		    err(EXIT_FAILURE, "Cannot get current screen");
136ab547228Schristos 		(void)printf("%d\n", idx);
1371a8fb0adSchristos 	} else if (kbd) {
1381a8fb0adSchristos 		wmd.type = mux ? WSMUX_MUX : WSMUX_KBD;
139e7af5efeSaugustss 		wmd.idx = idx;
140e7af5efeSaugustss 		if (delete) {
1411a8fb0adSchristos 			if (ioctl(wsfd, WSMUX_REMOVE_DEVICE, &wmd) == -1)
1421a8fb0adSchristos 				err(EXIT_FAILURE, "WSMUX_REMOVE_DEVICE");
143e7af5efeSaugustss 		} else {
1441a8fb0adSchristos 			if (ioctl(wsfd, WSMUX_ADD_DEVICE, &wmd) == -1)
1451a8fb0adSchristos 				err(EXIT_FAILURE, "WSMUX_ADD_DEVICE");
146e7af5efeSaugustss 		}
1476f1e305bSdrochner 	} else if (delete) {
148b43055f3Sdrochner 		dsd.idx = idx;
1491a8fb0adSchristos 		if (ioctl(wsfd, WSDISPLAYIO_DELSCREEN, &dsd) == -1)
1501a8fb0adSchristos 			err(EXIT_FAILURE, "WSDISPLAYIO_DELSCREEN");
151b43055f3Sdrochner 	} else {
152b43055f3Sdrochner 		asd.idx = idx;
1531a8fb0adSchristos 		if (ioctl(wsfd, WSDISPLAYIO_ADDSCREEN, &asd) == -1) {
154b12f1e56Sdrochner 			if (errno == EBUSY)
1551a8fb0adSchristos 				errx(EXIT_FAILURE,
1561a8fb0adSchristos 				    "screen %d is already configured", idx);
157b12f1e56Sdrochner 			else
1581a8fb0adSchristos 				err(EXIT_FAILURE, "WSDISPLAYIO_ADDSCREEN");
159b43055f3Sdrochner 		}
160b12f1e56Sdrochner 	}
161b43055f3Sdrochner 
1621a8fb0adSchristos 	return 0;
163b43055f3Sdrochner }
164