2 #include <minix/chardriver.h>
3 #include <minix/drivers.h>
5 #include <minix/sysutil.h>
6 #include <minix/type.h>
8 #include <sys/ioc_fb.h>
10 #include <sys/ioctl.h>
15 #include <dev/videomode/videomode.h>
16 #include <dev/videomode/edidvar.h>
17 #include <dev/videomode/edidreg.h>
24 * Function prototypes for the fb driver.
26 static int fb_open(devminor_t minor
, int access
, endpoint_t user_endpt
);
27 static int fb_close(devminor_t minor
);
28 static ssize_t
fb_read(devminor_t minor
, u64_t pos
, endpoint_t ep
,
29 cp_grant_id_t gid
, size_t size
, int flags
, cdev_id_t id
);
30 static ssize_t
fb_write(devminor_t minor
, u64_t pos
, endpoint_t ep
,
31 cp_grant_id_t gid
, size_t size
, int flags
, cdev_id_t id
);
32 static int fb_ioctl(devminor_t minor
, unsigned long request
, endpoint_t ep
,
33 cp_grant_id_t gid
, int flags
, endpoint_t user_ep
, cdev_id_t id
);
34 static void paint_bootlogo(int minor
);
35 static void paint_restartlogo(int minor
);
36 static void paint_centered(int minor
, char *data
, int width
, int height
);
37 static int do_get_varscreeninfo(int minor
, endpoint_t ep
, cp_grant_id_t gid
);
38 static int do_put_varscreeninfo(int minor
, endpoint_t ep
, cp_grant_id_t gid
);
39 static int do_get_fixscreeninfo(int minor
, endpoint_t ep
, cp_grant_id_t gid
);
40 static int do_pan_display(int minor
, endpoint_t ep
, cp_grant_id_t gid
);
41 static int keep_displaying_restarted(void);
43 /* Entry points to the fb driver. */
44 static struct chardriver fb_tab
=
47 .cdr_close
= fb_close
,
49 .cdr_write
= fb_write
,
53 /** Represents the /dev/fb device. */
54 static int has_restarted
= 0;
55 static u64_t has_restarted_t1
, has_restarted_t2
;
57 static int open_counter
[FB_DEV_NR
]; /* Open count */
60 fb_open(devminor_t minor
, int UNUSED(access
), endpoint_t
UNUSED(user_endpt
))
63 static int initialized
= 0;
64 static struct edid_info info
;
65 static struct edid_info
*infop
= NULL
;
67 if (minor
< 0 || minor
>= FB_DEV_NR
) return ENXIO
;
70 r
= fb_edid_read(minor
, &info
);
71 infop
= (r
== 0) ? &info
: NULL
;
74 if (arch_fb_init(minor
, infop
) == OK
) {
75 open_counter
[minor
]++;
78 read_frclock_64(&has_restarted_t1
);
79 paint_restartlogo(minor
);
81 paint_bootlogo(minor
);
91 fb_close(devminor_t minor
)
93 if (minor
< 0 || minor
>= FB_DEV_NR
) return ENXIO
;
94 assert(open_counter
[minor
] > 0);
95 open_counter
[minor
]--;
100 fb_read(devminor_t minor
, u64_t pos
, endpoint_t ep
, cp_grant_id_t gid
,
101 size_t size
, int UNUSED(flags
), cdev_id_t
UNUSED(id
))
106 if (minor
< 0 || minor
>= FB_DEV_NR
) return ENXIO
;
107 assert(open_counter
[minor
] > 0);
109 arch_get_device(minor
, &dev
);
111 if (size
== 0 || pos
>= dev
.dv_size
) return 0;
112 if (pos
+ size
> dev
.dv_size
)
113 size
= (size_t)(dev
.dv_size
- pos
);
115 r
= sys_safecopyto(ep
, gid
, 0, (vir_bytes
)(dev
.dv_base
+ (size_t)pos
),
118 return (r
!= OK
) ? r
: size
;
122 fb_ioctl(devminor_t minor
, unsigned long request
, endpoint_t ep
,
123 cp_grant_id_t gid
, int UNUSED(flags
), endpoint_t
UNUSED(user_ep
),
124 cdev_id_t
UNUSED(id
))
126 /* Process I/O control requests */
129 if (minor
< 0 || minor
>= FB_DEV_NR
) return ENXIO
;
132 case FBIOGET_VSCREENINFO
:
133 r
= do_get_varscreeninfo(minor
, ep
, gid
);
135 case FBIOPUT_VSCREENINFO
:
136 r
= do_put_varscreeninfo(minor
, ep
, gid
);
138 case FBIOGET_FSCREENINFO
:
139 r
= do_get_fixscreeninfo(minor
, ep
, gid
);
141 case FBIOPAN_DISPLAY
:
142 r
= do_pan_display(minor
, ep
, gid
);
150 do_get_varscreeninfo(int minor
, endpoint_t ep
, cp_grant_id_t gid
)
153 struct fb_var_screeninfo fbvs
;
155 if ((r
= arch_get_varscreeninfo(minor
, &fbvs
)) == OK
) {
156 r
= sys_safecopyto(ep
, gid
, 0, (vir_bytes
) &fbvs
, sizeof(fbvs
));
163 do_put_varscreeninfo(int minor
, endpoint_t ep
, cp_grant_id_t gid
)
166 struct fb_var_screeninfo fbvs_copy
;
168 if (has_restarted
&& keep_displaying_restarted()) {
172 if ((r
= sys_safecopyfrom(ep
, gid
, 0, (vir_bytes
) &fbvs_copy
,
173 sizeof(fbvs_copy
))) != OK
) {
177 return arch_put_varscreeninfo(minor
, &fbvs_copy
);
181 do_pan_display(int minor
, endpoint_t ep
, cp_grant_id_t gid
)
184 struct fb_var_screeninfo fbvs_copy
;
186 if (has_restarted
&& keep_displaying_restarted()) {
190 if ((r
= sys_safecopyfrom(ep
, gid
, 0, (vir_bytes
) &fbvs_copy
,
191 sizeof(fbvs_copy
))) != OK
) {
195 return arch_pan_display(minor
, &fbvs_copy
);
199 do_get_fixscreeninfo(int minor
, endpoint_t ep
, cp_grant_id_t gid
)
202 struct fb_fix_screeninfo fbfs
;
204 if ((r
= arch_get_fixscreeninfo(minor
, &fbfs
)) == OK
) {
205 r
= sys_safecopyto(ep
, gid
, 0, (vir_bytes
) &fbfs
, sizeof(fbfs
));
212 fb_write(devminor_t minor
, u64_t pos
, endpoint_t ep
, cp_grant_id_t gid
,
213 size_t size
, int UNUSED(flags
), cdev_id_t
UNUSED(id
))
218 if (minor
< 0 || minor
>= FB_DEV_NR
) return ENXIO
;
219 assert(open_counter
[minor
] > 0);
221 if (has_restarted
&& keep_displaying_restarted())
224 arch_get_device(minor
, &dev
);
226 if (size
== 0 || pos
>= dev
.dv_size
) return 0;
227 if (pos
+ size
> dev
.dv_size
)
228 size
= (size_t)(dev
.dv_size
- pos
);
230 r
= sys_safecopyfrom(ep
, gid
, 0,
231 (vir_bytes
)(dev
.dv_base
+ (size_t)pos
), size
);
233 return (r
!= OK
) ? r
: size
;
237 sef_cb_lu_state_save(int UNUSED(result
), int UNUSED(flags
))
239 /* Save the state. */
240 ds_publish_u32("open_counter", open_counter
[0], DSF_OVERWRITE
);
247 /* Restore the state. */
250 ds_retrieve_u32("open_counter", &value
);
251 ds_delete_u32("open_counter");
252 open_counter
[0] = (int) value
;
258 sef_cb_init(int type
, sef_init_info_t
*UNUSED(info
))
260 /* Initialize the fb driver. */
261 int do_announce_driver
= TRUE
;
266 printf("framebuffer fresh: pid %d\n", getpid());
270 /* Restore the state. */
272 do_announce_driver
= FALSE
;
274 printf("framebuffer: I'm a new version!\n");
277 case SEF_INIT_RESTART
:
278 printf("framebuffer restarted: pid %d\n", getpid());
283 /* Announce we are up when necessary. */
284 if (do_announce_driver
) {
285 chardriver_announce();
288 /* Initialization completed successfully. */
295 /* Register init callbacks. Use the same function for all event types */
296 sef_setcb_init_fresh(sef_cb_init
);
297 sef_setcb_init_lu(sef_cb_init
);
298 sef_setcb_init_restart(sef_cb_init
);
300 /* Register live update callbacks */
301 sef_setcb_lu_state_save(sef_cb_lu_state_save
);
303 /* Let SEF perform startup. */
308 main(int argc
, char *argv
[])
310 env_setargs(argc
, argv
);
311 fb_edid_args_parse();
314 chardriver_task(&fb_tab
);
319 keep_displaying_restarted()
324 read_frclock_64(&has_restarted_t2
);
325 delta
= delta_frclock_64(has_restarted_t1
, has_restarted_t2
);
326 micro_delta
= frclock_64_to_micros(delta
);
328 #define DISPLAY_1SEC 1000000 /* 1 second in microseconds */
329 if (micro_delta
< DISPLAY_1SEC
) {
338 paint_bootlogo(int minor
)
340 paint_centered(minor
, bootlogo_data
, bootlogo_width
, bootlogo_height
);
344 paint_restartlogo(int minor
)
346 paint_centered(minor
, restartlogo_data
, restartlogo_width
,
351 paint_centered(int minor
, char *data
, int width
, int height
)
354 u32_t i
, min_x
, min_y
, max_x
, max_y
, x_painted
= 0, rows
= 0;
357 struct fb_var_screeninfo fbvs
;
359 /* Put display in a known state to simplify positioning code below */
360 if ((r
= arch_get_varscreeninfo(minor
, &fbvs
)) != OK
) {
361 printf("fb: unable to get screen info: %d\n", r
);
364 if ((r
= arch_pan_display(minor
, &fbvs
)) != OK
) {
365 printf("fb: unable to pan display: %d\n", r
);
368 arch_get_device(minor
, &dev
);
370 /* Paint on a white canvas */
371 bytespp
= fbvs
.bits_per_pixel
/ 8;
372 for (i
= 0; i
< fbvs
.xres
* fbvs
.yres
* bytespp
; i
+= bytespp
)
373 *((u32_t
*)((u32_t
) dev
.dv_base
+ i
)) = 0x00FFFFFF;
375 /* First seek to start */
376 min_x
= fbvs
.xres
/ 2 - width
/ 2;
377 max_x
= fbvs
.xres
/ 2 + width
/ 2;
378 min_y
= fbvs
.yres
/ 2 - height
/ 2;
379 max_y
= fbvs
.yres
/ 2 + height
/ 2;
380 i
= min_x
* fbvs
.xres
+ min_y
;
382 /* Add the image data */
383 for (i
= ((min_y
* fbvs
.xres
) + min_x
) * bytespp
; rows
< height
;) {
384 GET_PIXEL(data
, pixel
);
386 ((unsigned char *)((u32_t
) dev
.dv_base
+ i
))[0] = pixel
[2];
387 ((unsigned char *)((u32_t
) dev
.dv_base
+ i
))[1] = pixel
[1];
388 ((unsigned char *)((u32_t
) dev
.dv_base
+ i
))[2] = pixel
[0];
389 ((unsigned char *)((u32_t
) dev
.dv_base
+ i
))[3] = 0;
391 x_painted
++;/* Keep tab of how many row pixels we've painted */
392 if (x_painted
== width
) {
393 /* We've reached the end of the row, carriage return
394 * and go to next line.
398 i
= (((min_y
+ rows
) * fbvs
.xres
) + min_x
) * 4;