1
2
3
4
5
6
7
8
9
10
11
12
13
14
15#include "libbb.h"
16
17#define DEFAULTFBDEV FB_0
18#define DEFAULTFBMODE "/etc/fb.modes"
19
20enum {
21 CMD_FB = 1,
22 CMD_DB = 2,
23 CMD_GEOMETRY = 3,
24 CMD_TIMING = 4,
25 CMD_ACCEL = 5,
26 CMD_HSYNC = 6,
27 CMD_VSYNC = 7,
28 CMD_LACED = 8,
29 CMD_DOUBLE = 9,
30
31 CMD_ALL = 11,
32 CMD_INFO = 12,
33 CMD_CHANGE = 13,
34
35#if ENABLE_FEATURE_FBSET_FANCY
36 CMD_XRES = 100,
37 CMD_YRES = 101,
38 CMD_VXRES = 102,
39 CMD_VYRES = 103,
40 CMD_DEPTH = 104,
41 CMD_MATCH = 105,
42 CMD_PIXCLOCK = 106,
43 CMD_LEFT = 107,
44 CMD_RIGHT = 108,
45 CMD_UPPER = 109,
46 CMD_LOWER = 110,
47 CMD_HSLEN = 111,
48 CMD_VSLEN = 112,
49 CMD_CSYNC = 113,
50 CMD_GSYNC = 114,
51 CMD_EXTSYNC = 115,
52 CMD_BCAST = 116,
53 CMD_RGBA = 117,
54 CMD_STEP = 118,
55 CMD_MOVE = 119,
56#endif
57};
58
59
60#define FB_ACTIVATE_ALL 64
61enum {
62 FBIOGET_VSCREENINFO = 0x4600,
63 FBIOPUT_VSCREENINFO = 0x4601
64};
65struct fb_bitfield {
66 uint32_t offset;
67 uint32_t length;
68 uint32_t msb_right;
69};
70struct fb_var_screeninfo {
71 uint32_t xres;
72 uint32_t yres;
73 uint32_t xres_virtual;
74 uint32_t yres_virtual;
75 uint32_t xoffset;
76 uint32_t yoffset;
77
78 uint32_t bits_per_pixel;
79 uint32_t grayscale;
80
81 struct fb_bitfield red;
82 struct fb_bitfield green;
83 struct fb_bitfield blue;
84 struct fb_bitfield transp;
85
86 uint32_t nonstd;
87
88 uint32_t activate;
89
90 uint32_t height;
91 uint32_t width;
92
93 uint32_t accel_flags;
94
95
96 uint32_t pixclock;
97 uint32_t left_margin;
98 uint32_t right_margin;
99 uint32_t upper_margin;
100 uint32_t lower_margin;
101 uint32_t hsync_len;
102 uint32_t vsync_len;
103 uint32_t sync;
104 uint32_t vmode;
105 uint32_t reserved[6];
106};
107
108
109static const struct cmdoptions_t {
110 const char name[9];
111 const unsigned char param_count;
112 const unsigned char code;
113} g_cmdoptions[] = {
114
115 { "fb" , 1, CMD_FB },
116 { "db" , 1, CMD_DB },
117 { "a" , 0, CMD_ALL },
118 { "i" , 0, CMD_INFO },
119 { "g" , 5, CMD_GEOMETRY },
120 { "t" , 7, CMD_TIMING },
121 { "accel" , 1, CMD_ACCEL },
122 { "hsync" , 1, CMD_HSYNC },
123 { "vsync" , 1, CMD_VSYNC },
124 { "laced" , 1, CMD_LACED },
125 { "double" , 1, CMD_DOUBLE },
126 { "n" , 0, CMD_CHANGE },
127#if ENABLE_FEATURE_FBSET_FANCY
128 { "all" , 0, CMD_ALL },
129 { "xres" , 1, CMD_XRES },
130 { "yres" , 1, CMD_YRES },
131 { "vxres" , 1, CMD_VXRES },
132 { "vyres" , 1, CMD_VYRES },
133 { "depth" , 1, CMD_DEPTH },
134 { "match" , 0, CMD_MATCH },
135 { "geometry", 5, CMD_GEOMETRY },
136 { "pixclock", 1, CMD_PIXCLOCK },
137 { "left" , 1, CMD_LEFT },
138 { "right" , 1, CMD_RIGHT },
139 { "upper" , 1, CMD_UPPER },
140 { "lower" , 1, CMD_LOWER },
141 { "hslen" , 1, CMD_HSLEN },
142 { "vslen" , 1, CMD_VSLEN },
143 { "timings" , 7, CMD_TIMING },
144 { "csync" , 1, CMD_CSYNC },
145 { "gsync" , 1, CMD_GSYNC },
146 { "extsync" , 1, CMD_EXTSYNC },
147 { "bcast" , 1, CMD_BCAST },
148 { "rgba" , 1, CMD_RGBA },
149 { "step" , 1, CMD_STEP },
150 { "move" , 1, CMD_MOVE },
151#endif
152};
153
154#if ENABLE_FEATURE_FBSET_READMODE
155
156enum {
157 FB_VMODE_INTERLACED = 1,
158 FB_VMODE_DOUBLE = 2,
159 FB_SYNC_HOR_HIGH_ACT = 1,
160 FB_SYNC_VERT_HIGH_ACT = 2,
161 FB_SYNC_EXT = 4,
162 FB_SYNC_COMP_HIGH_ACT = 8,
163};
164#endif
165
166#if ENABLE_FEATURE_FBSET_READMODE
167static void ss(uint32_t *x, uint32_t flag, char *buf, const char *what)
168{
169 if (strcmp(buf, what) == 0)
170 *x &= ~flag;
171 else
172 *x |= flag;
173}
174
175static int readmode(struct fb_var_screeninfo *base, const char *fn,
176 const char *mode)
177{
178 char *token[2], *p, *s;
179 parser_t *parser = config_open(fn);
180
181 while (config_read(parser, token, 2, 1, "# \t\r", PARSE_NORMAL)) {
182 if (strcmp(token[0], "mode") != 0 || !token[1])
183 continue;
184 p = strstr(token[1], mode);
185 if (!p)
186 continue;
187 s = p + strlen(mode);
188
189
190 if (((!*s || isspace(*s)) && '"' != s[-1])
191 || ('"' == *s && '"' == p[-1])
192 ) {
193
194 break;
195 }
196 }
197
198 if (!token[0])
199 return 0;
200
201 while (config_read(parser, token, 2, 1, "# \t", PARSE_NORMAL)) {
202 int i;
203
204
205 if (strcmp(token[0], "endmode") == 0) {
206
207 return 1;
208 }
209 p = token[1];
210 i = index_in_strings(
211 "geometry\0timings\0interlaced\0double\0vsync\0hsync\0csync\0extsync\0",
212 token[0]);
213 switch (i) {
214 case 0:
215
216 sscanf(p, "%d %d %d %d %d",
217 &(base->xres), &(base->yres),
218 &(base->xres_virtual), &(base->yres_virtual),
219 &(base->bits_per_pixel));
220
221 break;
222 case 1:
223 sscanf(p, "%d %d %d %d %d %d %d",
224 &(base->pixclock),
225 &(base->left_margin), &(base->right_margin),
226 &(base->upper_margin), &(base->lower_margin),
227 &(base->hsync_len), &(base->vsync_len));
228
229 break;
230 case 2:
231 case 3: {
232 static const uint32_t syncs[] = {FB_VMODE_INTERLACED, FB_VMODE_DOUBLE};
233 ss(&base->vmode, syncs[i-2], p, "false");
234
235 break;
236 }
237 case 4:
238 case 5:
239 case 6: {
240 static const uint32_t syncs[] = {FB_SYNC_VERT_HIGH_ACT, FB_SYNC_HOR_HIGH_ACT, FB_SYNC_COMP_HIGH_ACT};
241 ss(&base->sync, syncs[i-4], p, "low");
242
243 break;
244 }
245 case 7:
246 ss(&base->sync, FB_SYNC_EXT, p, "false");
247
248 break;
249 }
250 }
251 return 0;
252}
253#endif
254
255static void setmode(struct fb_var_screeninfo *base,
256 struct fb_var_screeninfo *set)
257{
258 if ((int32_t) set->xres > 0)
259 base->xres = set->xres;
260 if ((int32_t) set->yres > 0)
261 base->yres = set->yres;
262 if ((int32_t) set->xres_virtual > 0)
263 base->xres_virtual = set->xres_virtual;
264 if ((int32_t) set->yres_virtual > 0)
265 base->yres_virtual = set->yres_virtual;
266 if ((int32_t) set->bits_per_pixel > 0)
267 base->bits_per_pixel = set->bits_per_pixel;
268}
269
270static void showmode(struct fb_var_screeninfo *v)
271{
272 double drate = 0, hrate = 0, vrate = 0;
273
274 if (v->pixclock) {
275 drate = 1e12 / v->pixclock;
276 hrate = drate / (v->left_margin + v->xres + v->right_margin + v->hsync_len);
277 vrate = hrate / (v->upper_margin + v->yres + v->lower_margin + v->vsync_len);
278 }
279 printf("\nmode \"%ux%u-%u\"\n"
280#if ENABLE_FEATURE_FBSET_FANCY
281 "\t# D: %.3f MHz, H: %.3f kHz, V: %.3f Hz\n"
282#endif
283 "\tgeometry %u %u %u %u %u\n"
284 "\ttimings %u %u %u %u %u %u %u\n"
285 "\taccel %s\n"
286 "\trgba %u/%u,%u/%u,%u/%u,%u/%u\n"
287 "endmode\n\n",
288 v->xres, v->yres, (int) (vrate + 0.5),
289#if ENABLE_FEATURE_FBSET_FANCY
290 drate / 1e6, hrate / 1e3, vrate,
291#endif
292 v->xres, v->yres, v->xres_virtual, v->yres_virtual, v->bits_per_pixel,
293 v->pixclock, v->left_margin, v->right_margin, v->upper_margin, v->lower_margin,
294 v->hsync_len, v->vsync_len,
295 (v->accel_flags > 0 ? "true" : "false"),
296 v->red.length, v->red.offset, v->green.length, v->green.offset,
297 v->blue.length, v->blue.offset, v->transp.length, v->transp.offset);
298}
299
300int fbset_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
301int fbset_main(int argc, char **argv)
302{
303 enum {
304 OPT_CHANGE = (1 << 0),
305
306 OPT_READMODE = (1 << 2),
307 OPT_ALL = (1 << 9),
308 };
309 struct fb_var_screeninfo var, varset;
310 int fh, i;
311 unsigned options = 0;
312
313 const char *fbdev = DEFAULTFBDEV;
314 const char *modefile = DEFAULTFBMODE;
315 char *thisarg, *mode = NULL;
316
317 memset(&varset, 0xff, sizeof(varset));
318
319
320 argv++;
321 argc--;
322 for (; argc > 0 && (thisarg = *argv) != NULL; argc--, argv++) {
323 if (thisarg[0] == '-') for (i = 0; i < ARRAY_SIZE(g_cmdoptions); i++) {
324 if (strcmp(thisarg + 1, g_cmdoptions[i].name) != 0)
325 continue;
326 if (argc <= g_cmdoptions[i].param_count)
327 bb_show_usage();
328
329 switch (g_cmdoptions[i].code) {
330 case CMD_FB:
331 fbdev = argv[1];
332 break;
333 case CMD_DB:
334 modefile = argv[1];
335 break;
336 case CMD_GEOMETRY:
337 varset.xres = xatou32(argv[1]);
338 varset.yres = xatou32(argv[2]);
339 varset.xres_virtual = xatou32(argv[3]);
340 varset.yres_virtual = xatou32(argv[4]);
341 varset.bits_per_pixel = xatou32(argv[5]);
342 break;
343 case CMD_TIMING:
344 varset.pixclock = xatou32(argv[1]);
345 varset.left_margin = xatou32(argv[2]);
346 varset.right_margin = xatou32(argv[3]);
347 varset.upper_margin = xatou32(argv[4]);
348 varset.lower_margin = xatou32(argv[5]);
349 varset.hsync_len = xatou32(argv[6]);
350 varset.vsync_len = xatou32(argv[7]);
351 break;
352 case CMD_ALL:
353 options |= OPT_ALL;
354 break;
355 case CMD_CHANGE:
356 options |= OPT_CHANGE;
357 break;
358#if ENABLE_FEATURE_FBSET_FANCY
359 case CMD_XRES:
360 varset.xres = xatou32(argv[1]);
361 break;
362 case CMD_YRES:
363 varset.yres = xatou32(argv[1]);
364 break;
365 case CMD_DEPTH:
366 varset.bits_per_pixel = xatou32(argv[1]);
367 break;
368#endif
369 }
370 argc -= g_cmdoptions[i].param_count;
371 argv += g_cmdoptions[i].param_count;
372 goto contin;
373 }
374 if (argc != 1)
375 bb_show_usage();
376 mode = *argv;
377 options |= OPT_READMODE;
378 contin: ;
379 }
380
381 fh = xopen(fbdev, O_RDONLY);
382 xioctl(fh, FBIOGET_VSCREENINFO, &var);
383 if (options & OPT_READMODE) {
384#if !ENABLE_FEATURE_FBSET_READMODE
385 bb_show_usage();
386#else
387 if (!readmode(&var, modefile, mode)) {
388 bb_error_msg_and_die("unknown video mode '%s'", mode);
389 }
390#endif
391 }
392
393 setmode(&var, &varset);
394 if (options & OPT_CHANGE) {
395 if (options & OPT_ALL)
396 var.activate = FB_ACTIVATE_ALL;
397 xioctl(fh, FBIOPUT_VSCREENINFO, &var);
398 }
399 showmode(&var);
400
401
402
403 return EXIT_SUCCESS;
404}
405