display.c revision 1.19 1 /* $NetBSD: display.c,v 1.19 2024/10/20 13:49:41 mlelstv Exp $ */
2
3 /*-
4 * Copyright (c) 1998, 2004 The NetBSD Foundation, Inc.
5 * All rights reserved.
6 *
7 * This code is derived from software contributed to The NetBSD Foundation
8 * by Juergen Hannken-Illjes.
9 *
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
12 * are met:
13 * 1. Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
18 *
19 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 * POSSIBILITY OF SUCH DAMAGE.
30 */
31
32 #include <sys/ioctl.h>
33 #include <sys/time.h>
34
35 #include <dev/wscons/wsconsio.h>
36
37 #include <err.h>
38 #include <errno.h>
39 #include <stdio.h>
40 #include <stdlib.h>
41 #include <string.h>
42
43 #include "wsconsctl.h"
44
45 static int border;
46 static int dpytype;
47 static struct wsdisplay_usefontdata font;
48 static struct wsdisplay_getfont gfont;
49 static char fontname_buf[256];
50 static struct wsdisplay_param backlight;
51 static struct wsdisplay_param brightness;
52 static struct wsdisplay_param contrast;
53 static struct wsdisplay_scroll_data scroll_l;
54 static struct wsdisplayio_edid_info edid_info;
55 static int msg_default_attrs, msg_default_bg, msg_default_fg;
56 static int msg_kernel_attrs, msg_kernel_bg, msg_kernel_fg;
57 static int splash_enable, splash_progress;
58
59 struct field display_field_tab[] = {
60 { "border", &border, FMT_COLOR, 0 },
61 { "type", &dpytype, FMT_DPYTYPE, FLG_RDONLY },
62 { "font", &font.name, FMT_STRING, 0 },
63 { "backlight", &backlight.curval, FMT_UINT, 0 },
64 { "brightness", &brightness.curval, FMT_UINT, FLG_MODIFY },
65 { "contrast", &contrast.curval, FMT_UINT, FLG_MODIFY },
66 { "scroll.fastlines", &scroll_l.fastlines, FMT_UINT, FLG_MODIFY },
67 { "scroll.slowlines", &scroll_l.slowlines, FMT_UINT, FLG_MODIFY },
68 { "edid", &edid_info, FMT_EDID, FLG_RDONLY|FLG_NOAUTO },
69 { "msg.default.attrs", &msg_default_attrs, FMT_ATTRS, 0 },
70 { "msg.default.bg", &msg_default_bg, FMT_COLOR, 0 },
71 { "msg.default.fg", &msg_default_fg, FMT_COLOR, 0 },
72 { "msg.kernel.attrs", &msg_kernel_attrs, FMT_ATTRS, 0 },
73 { "msg.kernel.bg", &msg_kernel_bg, FMT_COLOR, 0 },
74 { "msg.kernel.fg", &msg_kernel_fg, FMT_COLOR, 0 },
75 { "splash.enable", &splash_enable, FMT_UINT, FLG_WRONLY },
76 { "splash.progress", &splash_progress, FMT_UINT, FLG_WRONLY },
77 };
78
79 int display_field_tab_len = sizeof(display_field_tab) /
80 sizeof(display_field_tab[0]);
81
82 static int
83 display_get_edid(int fd, struct wsdisplayio_edid_info *ei)
84 {
85 size_t sz = 256;
86 int res;
87
88 while (sz <= 65536) {
89 ei->buffer_size = sz;
90 ei->edid_data = malloc(sz);
91 if (ei->edid_data == NULL) {
92 res = -1;
93 break;
94 }
95
96 res = ioctl(fd, WSDISPLAYIO_GET_EDID, ei);
97 if (res == 0 || errno != EAGAIN)
98 break;
99
100 free(ei->edid_data);
101 ei->edid_data = NULL;
102 sz *= 2;
103 }
104
105 return res;
106 }
107
108 void
109 display_get_values(int fd)
110 {
111 if (field_by_value(&font.name)->flags & FLG_GET) {
112 gfont.gf_name = fontname_buf;
113 gfont.gf_size = sizeof(fontname_buf);
114 font.name = gfont.gf_name;
115 if (ioctl(fd, WSDISPLAYIO_GFONT, &gfont) < 0)
116 field_disable_by_value(&font.name);
117 }
118
119 if (field_by_value(&dpytype)->flags & FLG_GET)
120 if (ioctl(fd, WSDISPLAYIO_GTYPE, &dpytype) < 0)
121 err(EXIT_FAILURE, "WSDISPLAYIO_GTYPE");
122
123 if (field_by_value(&border)->flags & FLG_GET)
124 if (ioctl(fd, WSDISPLAYIO_GBORDER, &border) < 0)
125 field_disable_by_value(&border);
126
127 if (field_by_value(&backlight.curval)->flags & FLG_GET) {
128 backlight.param = WSDISPLAYIO_PARAM_BACKLIGHT;
129 if (ioctl(fd, WSDISPLAYIO_GETPARAM, &backlight) < 0)
130 field_disable_by_value(&backlight.curval);
131 }
132
133 if (field_by_value(&brightness.curval)->flags & FLG_GET) {
134 brightness.param = WSDISPLAYIO_PARAM_BRIGHTNESS;
135 if (ioctl(fd, WSDISPLAYIO_GETPARAM, &brightness))
136 field_disable_by_value(&brightness.curval);
137 }
138
139 if (field_by_value(&contrast.curval)->flags & FLG_GET) {
140 contrast.param = WSDISPLAYIO_PARAM_CONTRAST;
141 if (ioctl(fd, WSDISPLAYIO_GETPARAM, &contrast))
142 field_disable_by_value(&contrast.curval);
143 }
144
145 if (field_by_value(&msg_default_attrs)->flags & FLG_GET ||
146 field_by_value(&msg_default_bg)->flags & FLG_GET ||
147 field_by_value(&msg_default_fg)->flags & FLG_GET ||
148 field_by_value(&msg_kernel_attrs)->flags & FLG_GET ||
149 field_by_value(&msg_kernel_bg)->flags & FLG_GET ||
150 field_by_value(&msg_kernel_fg)->flags & FLG_GET) {
151 struct wsdisplay_msgattrs ma;
152
153 if (ioctl(fd, WSDISPLAYIO_GMSGATTRS, &ma) < 0) {
154 field_disable_by_value(&msg_default_attrs);
155 field_disable_by_value(&msg_default_bg);
156 field_disable_by_value(&msg_default_fg);
157 field_disable_by_value(&msg_kernel_attrs);
158 field_disable_by_value(&msg_kernel_bg);
159 field_disable_by_value(&msg_kernel_fg);
160 } else {
161 msg_default_attrs = ma.default_attrs;
162 if (ma.default_attrs & WSATTR_WSCOLORS) {
163 msg_default_bg = ma.default_bg;
164 msg_default_fg = ma.default_fg;
165 } else
166 msg_default_bg = msg_default_fg = -1;
167
168 msg_kernel_attrs = ma.kernel_attrs;
169 if (ma.kernel_attrs & WSATTR_WSCOLORS) {
170 msg_kernel_bg = ma.kernel_bg;
171 msg_kernel_fg = ma.kernel_fg;
172 } else
173 msg_kernel_bg = msg_kernel_fg = -1;
174 }
175 }
176
177 if (field_by_value(&scroll_l.fastlines)->flags & FLG_GET ||
178 field_by_value(&scroll_l.slowlines)->flags & FLG_GET) {
179 if (ioctl(fd, WSDISPLAYIO_DGSCROLL, &scroll_l) < 0) {
180 field_disable_by_value(&scroll_l.fastlines);
181 field_disable_by_value(&scroll_l.slowlines);
182 }
183 }
184
185 if (field_by_value(&edid_info)->flags & FLG_GET) {
186 if (display_get_edid(fd, &edid_info) < 0)
187 field_disable_by_value(&edid_info);
188 }
189 }
190
191 void
192 display_put_values(int fd)
193 {
194
195 if (field_by_value(&font.name)->flags & FLG_SET) {
196 if (ioctl(fd, WSDISPLAYIO_SFONT, &font) < 0)
197 err(EXIT_FAILURE, "WSDISPLAYIO_SFONT");
198 pr_field(field_by_value(&font.name), " -> ");
199 }
200
201 if (field_by_value(&border)->flags & FLG_SET) {
202 if (ioctl(fd, WSDISPLAYIO_SBORDER, &border) < 0)
203 err(EXIT_FAILURE, "WSDISPLAYIO_SBORDER");
204 pr_field(field_by_value(&border), " -> ");
205 }
206
207 if (field_by_value(&backlight.curval)->flags & FLG_SET) {
208 backlight.param = WSDISPLAYIO_PARAM_BACKLIGHT;
209 if (ioctl(fd, WSDISPLAYIO_SETPARAM, &backlight) < 0)
210 err(EXIT_FAILURE, "WSDISPLAYIO_PARAM_BACKLIGHT");
211 pr_field(field_by_value(&backlight.curval), " -> ");
212 }
213
214 if (field_by_value(&brightness.curval)->flags & FLG_SET) {
215 brightness.param = WSDISPLAYIO_PARAM_BRIGHTNESS;
216 if (ioctl(fd, WSDISPLAYIO_SETPARAM, &brightness) < 0)
217 err(EXIT_FAILURE, "WSDISPLAYIO_PARAM_BRIGHTNESS");
218 pr_field(field_by_value(&brightness.curval), " -> ");
219 }
220
221 if (field_by_value(&contrast.curval)->flags & FLG_SET) {
222 contrast.param = WSDISPLAYIO_PARAM_CONTRAST;
223 if (ioctl(fd, WSDISPLAYIO_SETPARAM, &contrast) < 0)
224 err(EXIT_FAILURE, "WSDISPLAYIO_PARAM_CONTRAST");
225 pr_field(field_by_value(&contrast.curval), " -> ");
226 }
227
228 if (field_by_value(&splash_enable)->flags & FLG_SET) {
229 if (ioctl(fd, WSDISPLAYIO_SSPLASH, &splash_enable) < 0)
230 err(EXIT_FAILURE, "WSDISPLAYIO_SSPLASH");
231 pr_field(field_by_value(&splash_enable), " -> ");
232 }
233
234 if (field_by_value(&splash_progress)->flags & FLG_SET) {
235 if (ioctl(fd, WSDISPLAYIO_SPROGRESS, &splash_progress) < 0)
236 err(EXIT_FAILURE, "WSDISPLAYIO_SPROGRESS");
237 pr_field(field_by_value(&splash_progress), " -> ");
238 }
239
240 if (field_by_value(&msg_default_attrs)->flags & FLG_SET ||
241 field_by_value(&msg_default_bg)->flags & FLG_SET ||
242 field_by_value(&msg_default_fg)->flags & FLG_SET ||
243 field_by_value(&msg_kernel_attrs)->flags & FLG_SET ||
244 field_by_value(&msg_kernel_bg)->flags & FLG_SET ||
245 field_by_value(&msg_kernel_fg)->flags & FLG_SET) {
246 struct wsdisplay_msgattrs ma;
247
248 if (ioctl(fd, WSDISPLAYIO_GMSGATTRS, &ma) < 0)
249 err(EXIT_FAILURE, "WSDISPLAYIO_GMSGATTRS");
250
251 if (field_by_value(&msg_default_attrs)->flags & FLG_SET) {
252 ma.default_attrs = msg_default_attrs;
253 pr_field(field_by_value(&msg_default_attrs), " -> ");
254 }
255 if (ma.default_attrs & WSATTR_WSCOLORS) {
256 if (field_by_value(&msg_default_bg)->flags & FLG_SET) {
257 ma.default_bg = msg_default_bg;
258 pr_field(field_by_value(&msg_default_bg),
259 " -> ");
260 }
261 if (field_by_value(&msg_default_fg)->flags & FLG_SET) {
262 ma.default_fg = msg_default_fg;
263 pr_field(field_by_value(&msg_default_fg),
264 " -> ");
265 }
266 }
267
268 if (field_by_value(&msg_kernel_attrs)->flags & FLG_SET) {
269 ma.kernel_attrs = msg_kernel_attrs;
270 pr_field(field_by_value(&msg_kernel_attrs), " -> ");
271 }
272 if (ma.default_attrs & WSATTR_WSCOLORS) {
273 if (field_by_value(&msg_kernel_bg)->flags & FLG_SET) {
274 ma.kernel_bg = msg_kernel_bg;
275 pr_field(field_by_value(&msg_kernel_bg),
276 " -> ");
277 }
278 if (field_by_value(&msg_kernel_fg)->flags & FLG_SET) {
279 ma.kernel_fg = msg_kernel_fg;
280 pr_field(field_by_value(&msg_kernel_fg),
281 " -> ");
282 }
283 }
284
285 if (ioctl(fd, WSDISPLAYIO_SMSGATTRS, &ma) < 0)
286 err(EXIT_FAILURE, "WSDISPLAYIO_SMSGATTRS");
287 }
288
289 scroll_l.which = 0;
290 if (field_by_value(&scroll_l.fastlines)->flags & FLG_SET)
291 scroll_l.which |= WSDISPLAY_SCROLL_DOFASTLINES;
292 if (field_by_value(&scroll_l.slowlines)->flags & FLG_SET)
293 scroll_l.which |= WSDISPLAY_SCROLL_DOSLOWLINES;
294 if (scroll_l.which != 0 &&
295 ioctl(fd, WSDISPLAYIO_DSSCROLL, &scroll_l) < 0)
296 err(EXIT_FAILURE, "WSDISPLAYIO_DSSCROLL");
297 if (scroll_l.which & WSDISPLAY_SCROLL_DOFASTLINES)
298 pr_field(field_by_value(&scroll_l.fastlines), " -> ");
299 if (scroll_l.which & WSDISPLAY_SCROLL_DOSLOWLINES)
300 pr_field(field_by_value(&scroll_l.slowlines), " -> ");
301 }
302