mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2026-02-06 00:45:22 +08:00
1. Add backlight framework for graphic. 2. Add framebuffer and plane, power, EDID for graphic framework 3. Add boot logo render for graphic 4. Update lcd.h Signed-off-by: GuEe-GUI <2991707448@qq.com>
1496 lines
38 KiB
C
1496 lines
38 KiB
C
/*
|
|
* Copyright (c) 2006-2023, RT-Thread Development Team
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Change Logs:
|
|
* Date Author Notes
|
|
* 2023-02-25 GuEe-GUI the first version
|
|
*/
|
|
|
|
#include <rthw.h>
|
|
#include <rtthread.h>
|
|
#include <rtdevice.h>
|
|
|
|
#define DBG_TAG "rtdm.graphic"
|
|
#define DBG_LVL DBG_INFO
|
|
#include <rtdbg.h>
|
|
|
|
#define raw_to_graphic(dev) rt_container_of(dev, struct rt_graphic_device, parent)
|
|
|
|
struct fb_format
|
|
{
|
|
rt_uint32_t mode;
|
|
rt_uint32_t bits_per_pixel;
|
|
struct fb_bitfield red;
|
|
struct fb_bitfield green;
|
|
struct fb_bitfield blue;
|
|
struct fb_bitfield transp;
|
|
};
|
|
|
|
static const struct fb_format graphic_formats[] =
|
|
{
|
|
{ RTGRAPHIC_PIXEL_FORMAT_GRAY4, 4, },
|
|
{ RTGRAPHIC_PIXEL_FORMAT_GRAY16, 16, },
|
|
{ RTGRAPHIC_PIXEL_FORMAT_RGB332, 8, { 5, 3 }, { 2, 3 }, { 0, 2 }, },
|
|
{ RTGRAPHIC_PIXEL_FORMAT_RGB444, 12, { 8, 4 }, { 4, 4 }, { 0, 4 }, },
|
|
{ RTGRAPHIC_PIXEL_FORMAT_RGB565, 16, { 11, 5 }, { 5, 6 }, { 0, 5 }, },
|
|
{ RTGRAPHIC_PIXEL_FORMAT_RGB565P, 16, { 0, 5 }, { 5, 6 }, { 11, 5 }, },
|
|
{ RTGRAPHIC_PIXEL_FORMAT_BGR565, 16, { 0, 5 }, { 5, 6 }, { 11, 5 }, },
|
|
{ RTGRAPHIC_PIXEL_FORMAT_RGB666, 18, { 12, 6 }, { 6, 6 }, { 0, 6 }, },
|
|
{ RTGRAPHIC_PIXEL_FORMAT_RGB888, 24, { 16, 8 }, { 8, 8 }, { 0, 8 }, },
|
|
{ RTGRAPHIC_PIXEL_FORMAT_BGR888, 24, { 0, 8 }, { 8, 8 }, { 16, 8 }, },
|
|
{ RTGRAPHIC_PIXEL_FORMAT_ARGB888, 32, { 16, 8 }, { 8, 8 }, { 0, 8 }, { 24, 8 }, },
|
|
{ RTGRAPHIC_PIXEL_FORMAT_ABGR888, 32, { 0, 8 }, { 8, 8 }, { 16, 8 }, { 24, 8 }, },
|
|
};
|
|
|
|
/* RT-Thread device max id is 255 */
|
|
static rt_uint8_t fbcon_map[256] = {};
|
|
|
|
static struct rt_dm_ida graphic_ida = RT_DM_IDA_INIT(GRAPHIC_FRAMEBUFFER);
|
|
|
|
rt_inline void spin_lock(struct rt_spinlock *spinlock)
|
|
{
|
|
rt_hw_spin_lock(&spinlock->lock);
|
|
}
|
|
|
|
rt_inline void spin_unlock(struct rt_spinlock *spinlock)
|
|
{
|
|
rt_hw_spin_unlock(&spinlock->lock);
|
|
}
|
|
|
|
static rt_bool_t plane_need_update(struct rt_graphic_plane *plane)
|
|
{
|
|
if (plane->ops->update)
|
|
{
|
|
if (!plane->graphic->update_timer)
|
|
{
|
|
return RT_TRUE;
|
|
}
|
|
}
|
|
|
|
return RT_FALSE;
|
|
}
|
|
|
|
static struct rt_graphic_plane *plane_get_current(struct rt_graphic_device *gdev)
|
|
{
|
|
if (gdev->ops->current_plane)
|
|
{
|
|
return gdev->ops->current_plane(gdev);
|
|
}
|
|
|
|
return gdev->primary_plane;
|
|
}
|
|
|
|
static rt_err_t plane_fb_remap(struct rt_graphic_plane *plane,
|
|
rt_uint32_t mode, struct rt_device_rect_info *rect)
|
|
{
|
|
rt_err_t err;
|
|
|
|
if (plane->ops->fb_cleanup && (err = plane->ops->fb_cleanup(plane)))
|
|
{
|
|
return err;
|
|
}
|
|
|
|
if (!(err = plane->ops->fb_remap(plane, mode, rect)))
|
|
{
|
|
plane->mode = mode;
|
|
plane->x = rect->x;
|
|
plane->y = rect->y;
|
|
plane->width = rect->width;
|
|
plane->height = rect->height;
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
static rt_err_t plane_fb_pan_display(struct rt_graphic_plane *plane,
|
|
struct rt_device_rect_info *rect)
|
|
{
|
|
void *framebuffer_end = plane->framebuffer;
|
|
rt_size_t byte_per_pixel = plane->bits_per_pixel / 8;
|
|
|
|
framebuffer_end += rect->x * byte_per_pixel;
|
|
framebuffer_end += rect->y * plane->line_length;
|
|
framebuffer_end += rect->width * rect->height * byte_per_pixel;
|
|
|
|
if (framebuffer_end < plane->framebuffer + plane->framebuffer_len)
|
|
{
|
|
return plane->ops->fb_pan_display(plane, rect);
|
|
}
|
|
|
|
return -RT_EINVAL;
|
|
}
|
|
|
|
static rt_err_t graphic_dpms_switch(struct rt_graphic_device *gdev, rt_uint32_t dpms)
|
|
{
|
|
rt_err_t err;
|
|
|
|
if (!(err = gdev->ops->dpms_switch(gdev, dpms)))
|
|
{
|
|
gdev->dpms = dpms;
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
static rt_err_t _graphic_open(rt_device_t dev, rt_uint16_t oflag)
|
|
{
|
|
struct rt_graphic_device *gdev = raw_to_graphic(dev);
|
|
struct rt_graphic_plane *plane = gdev->primary_plane;
|
|
|
|
if (dev->ref_count > 0 && (oflag & RT_DEVICE_OFLAG_WRONLY))
|
|
{
|
|
return -RT_EBUSY;
|
|
}
|
|
|
|
if (plane->ops->fb_pan_display)
|
|
{
|
|
struct rt_device_rect_info rect;
|
|
|
|
rect.x = 0;
|
|
rect.y = 0;
|
|
rect.width = plane->width;
|
|
rect.height = plane->height;
|
|
|
|
return plane->ops->fb_pan_display(plane, &rect);
|
|
}
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
static rt_ssize_t _graphic_read(rt_device_t dev, rt_off_t pos, void *buffer, rt_size_t size)
|
|
{
|
|
rt_ssize_t res;
|
|
struct rt_graphic_device *gdev = raw_to_graphic(dev);
|
|
struct rt_graphic_plane *plane = gdev->primary_plane;
|
|
|
|
res = rt_min_t(rt_ssize_t, plane->framebuffer_len - pos, size);
|
|
|
|
if (res > 0)
|
|
{
|
|
rt_memcpy(buffer, plane->framebuffer + pos, res);
|
|
}
|
|
else
|
|
{
|
|
res = 0;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static rt_ssize_t _graphic_write(rt_device_t dev, rt_off_t pos, const void *buffer, rt_size_t size)
|
|
{
|
|
rt_ssize_t res;
|
|
struct rt_graphic_device *gdev = raw_to_graphic(dev);
|
|
struct rt_graphic_plane *plane = gdev->primary_plane;
|
|
|
|
res = rt_min_t(rt_ssize_t, plane->framebuffer_len - pos, size);
|
|
|
|
if (res > 0)
|
|
{
|
|
rt_memcpy(plane->framebuffer + pos, buffer, res);
|
|
}
|
|
else
|
|
{
|
|
res = 0;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static rt_err_t _graphic_control(rt_device_t dev, int cmd, void *args)
|
|
{
|
|
rt_err_t err = RT_EOK;
|
|
rt_bool_t need_schedule = RT_FALSE;
|
|
struct rt_graphic_device *gdev = raw_to_graphic(dev);
|
|
|
|
_retry:
|
|
if (need_schedule)
|
|
{
|
|
rt_thread_yield();
|
|
}
|
|
|
|
spin_lock(&gdev->lock);
|
|
|
|
switch (cmd)
|
|
{
|
|
case RT_DEVICE_CTRL_CURSOR_SET_POSITION:
|
|
if (args)
|
|
{
|
|
struct rt_graphic_plane *plane = gdev->cursor_plane;
|
|
|
|
if (plane)
|
|
{
|
|
rt_uint32_t position = (rt_uint32_t)(rt_ubase_t)args;
|
|
|
|
plane->x = position >> 16;
|
|
plane->y = position & 0xffff;
|
|
|
|
if (plane_need_update(plane))
|
|
{
|
|
struct rt_device_rect_info rect;
|
|
|
|
rect.x = plane->x;
|
|
rect.y = plane->y;
|
|
/* Ask driver to update position only */
|
|
rect.width = 0;
|
|
rect.height = 0;
|
|
|
|
err = plane->ops->update(plane, &rect);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_ENOSYS;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case RT_DEVICE_CTRL_CURSOR_SET_TYPE:
|
|
if (args)
|
|
{
|
|
struct rt_graphic_plane *plane = gdev->cursor_plane;
|
|
|
|
if (plane)
|
|
{
|
|
struct rt_device_rect_info rect =
|
|
{
|
|
.x = 0,
|
|
.y = 0,
|
|
.width = plane->width,
|
|
.height = plane->height,
|
|
};
|
|
|
|
if (!plane->framebuffer)
|
|
{
|
|
rt_uint32_t mode;
|
|
|
|
if (plane->mode == RTGRAPHIC_PIXEL_FORMAT_MONO)
|
|
{
|
|
mode = plane->modes[0];
|
|
}
|
|
else
|
|
{
|
|
mode = plane->mode;
|
|
}
|
|
|
|
if ((err = plane_fb_remap(plane, mode, &rect)))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
rt_memcpy(plane->framebuffer, args, plane->screen_len);
|
|
|
|
/* Force to update */
|
|
err = plane->ops->update(plane, &rect);
|
|
}
|
|
else
|
|
{
|
|
err = -RT_ENOSYS;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_RECT_UPDATE:
|
|
if (args)
|
|
{
|
|
struct rt_graphic_plane *plane = plane_get_current(gdev);
|
|
|
|
if (plane_need_update(plane))
|
|
{
|
|
err = plane->ops->update(plane, args);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_POWERON:
|
|
if (gdev->ops->dpms_switch)
|
|
{
|
|
err = graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_ON);
|
|
}
|
|
#ifdef RT_GRAPHIC_BACKLIGHT
|
|
if (!err && gdev->backlight)
|
|
{
|
|
spin_unlock(&gdev->lock);
|
|
err = rt_backlight_set_power(gdev->backlight, RT_BACKLIGHT_POWER_NORMAL);
|
|
spin_lock(&gdev->lock);
|
|
}
|
|
#endif /* RT_GRAPHIC_BACKLIGHT */
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_POWEROFF:
|
|
if (gdev->ops->dpms_switch)
|
|
{
|
|
err = graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_OFF);
|
|
}
|
|
#ifdef RT_GRAPHIC_BACKLIGHT
|
|
if (!err && gdev->backlight)
|
|
{
|
|
spin_unlock(&gdev->lock);
|
|
err = rt_backlight_set_power(gdev->backlight, RT_BACKLIGHT_POWER_POWERDOWN);
|
|
spin_lock(&gdev->lock);
|
|
}
|
|
#endif /* RT_GRAPHIC_BACKLIGHT */
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_GET_INFO:
|
|
if (args)
|
|
{
|
|
struct rt_device_graphic_info *info = args;
|
|
struct rt_graphic_plane *plane = plane_get_current(gdev);
|
|
|
|
info->pixel_format = plane->mode;
|
|
info->bits_per_pixel = plane->bits_per_pixel;
|
|
info->pitch = plane->line_length;
|
|
info->width = plane->width;
|
|
info->height = plane->height;
|
|
info->framebuffer = plane->framebuffer;
|
|
info->smem_len = plane->framebuffer_len;
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_SET_MODE:
|
|
if (args)
|
|
{
|
|
rt_uint32_t mode = (rt_uint32_t)(rt_ubase_t)args;
|
|
struct rt_graphic_plane *plane = plane_get_current(gdev);
|
|
|
|
if (mode != plane->mode)
|
|
{
|
|
err = -RT_ENOSYS;
|
|
|
|
if (plane->modes_nr > 1)
|
|
{
|
|
for (int i = 0; i < plane->modes_nr; ++i)
|
|
{
|
|
if (mode == plane->modes[i])
|
|
{
|
|
struct rt_device_rect_info rect =
|
|
{
|
|
.x = plane->x,
|
|
.y = plane->y,
|
|
.width = plane->width,
|
|
.height = plane->height,
|
|
};
|
|
|
|
err = plane_fb_remap(plane, mode, &rect);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_GET_EXT:
|
|
if (args)
|
|
{
|
|
rt_memcpy(args, &gdev->edid, sizeof(gdev->edid));
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_SET_BRIGHTNESS:
|
|
if (gdev->ops->set_brightness)
|
|
{
|
|
err = gdev->ops->set_brightness(gdev, (rt_uint32_t)(rt_ubase_t)args);
|
|
}
|
|
#ifdef RT_GRAPHIC_BACKLIGHT
|
|
if (!err && gdev->backlight)
|
|
{
|
|
spin_unlock(&gdev->lock);
|
|
|
|
return rt_backlight_set_brightness(gdev->backlight, (rt_uint32_t)(rt_ubase_t)args);
|
|
}
|
|
#endif /* RT_GRAPHIC_BACKLIGHT */
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_GET_BRIGHTNESS:
|
|
if (args)
|
|
{
|
|
if (gdev->ops->get_brightness)
|
|
{
|
|
err = gdev->ops->get_brightness(gdev, args);
|
|
}
|
|
else
|
|
{
|
|
*(rt_uint32_t *)args = RT_UINT32_MAX >> 1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
#ifdef RT_GRAPHIC_BACKLIGHT
|
|
if (!err && gdev->backlight)
|
|
{
|
|
spin_unlock(&gdev->lock);
|
|
|
|
return rt_backlight_get_brightness(gdev->backlight, args);
|
|
}
|
|
#endif /* RT_GRAPHIC_BACKLIGHT */
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_GET_MODE:
|
|
if (args)
|
|
{
|
|
struct rt_graphic_plane *plane = plane_get_current(gdev);
|
|
|
|
*(rt_uint32_t *)args = plane->mode;
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_GET_STATUS:
|
|
if (args)
|
|
{
|
|
if (gdev->ops->get_status)
|
|
{
|
|
err = gdev->ops->get_status(gdev, args);
|
|
}
|
|
else
|
|
{
|
|
err = -RT_ENOSYS;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_PAN_DISPLAY:
|
|
if (args)
|
|
{
|
|
struct rt_graphic_plane *plane = plane_get_current(gdev);
|
|
|
|
if (plane->ops->fb_pan_display)
|
|
{
|
|
rt_size_t offset;
|
|
struct rt_device_rect_info rect;
|
|
|
|
offset = (rt_size_t)(args - plane->framebuffer);
|
|
|
|
rect.x = (offset % plane->line_length) / (plane->bits_per_pixel / 8);
|
|
rect.y = offset / plane->line_length;
|
|
rect.width = plane->width;
|
|
rect.height = plane->height;
|
|
|
|
err = plane_fb_pan_display(plane, &rect);
|
|
}
|
|
else
|
|
{
|
|
err = -RT_ENOSYS;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case RTGRAPHIC_CTRL_WAIT_VSYNC:
|
|
if (gdev->ops->wait_vsync)
|
|
{
|
|
err = gdev->ops->wait_vsync(gdev);
|
|
}
|
|
break;
|
|
|
|
case RT_DEVICE_CTRL_NOTIFY_SET:
|
|
if (args)
|
|
{
|
|
if (rt_atomic_load(&gdev->event_notifying) == RT_TRUE)
|
|
{
|
|
need_schedule = RT_TRUE;
|
|
|
|
spin_unlock(&gdev->lock);
|
|
goto _retry;
|
|
}
|
|
|
|
rt_memcpy(&gdev->event_notify, args, sizeof(gdev->event_notify));
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case FBIOGET_VSCREENINFO:
|
|
if (args)
|
|
{
|
|
struct fb_var_screeninfo *var = args;
|
|
struct rt_graphic_plane *plane = plane_get_current(gdev);
|
|
|
|
rt_memset(var, 0, sizeof(*var));
|
|
|
|
var->xres = plane->width;
|
|
var->yres = plane->height;
|
|
var->xres_virtual = plane->width;
|
|
var->yres_virtual = plane->height * (plane->framebuffer_len / plane->screen_len);
|
|
var->bits_per_pixel = plane->bits_per_pixel;
|
|
|
|
if (plane == gdev->primary_plane)
|
|
{
|
|
var->width = gdev->edid.width_cm ? gdev->edid.width_cm * 10 : -1;
|
|
var->height = gdev->edid.height_cm ? gdev->edid.height_cm * 10 : -1;
|
|
}
|
|
else
|
|
{
|
|
var->width = -1;
|
|
var->height = -1;
|
|
}
|
|
|
|
if (plane->mode == RTGRAPHIC_PIXEL_FORMAT_GRAY4 ||
|
|
plane->mode == RTGRAPHIC_PIXEL_FORMAT_GRAY16)
|
|
{
|
|
var->grayscale = 1;
|
|
}
|
|
else
|
|
{
|
|
const struct fb_format *fmt = &graphic_formats[0];
|
|
|
|
for (int i = 0; i < RT_ARRAY_SIZE(graphic_formats); ++i, ++fmt)
|
|
{
|
|
if (fmt->mode == plane->mode)
|
|
{
|
|
rt_memcpy(&var->red, &fmt->red, sizeof(fmt->red));
|
|
rt_memcpy(&var->green, &fmt->green, sizeof(fmt->green));
|
|
rt_memcpy(&var->blue, &fmt->blue, sizeof(fmt->blue));
|
|
rt_memcpy(&var->transp, &fmt->transp, sizeof(fmt->transp));
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (gdev->update_timer)
|
|
{
|
|
rt_uint64_t update_ps;
|
|
rt_tick_t update_tick;
|
|
|
|
rt_timer_control(gdev->update_timer, RT_TIMER_CTRL_GET_TIME, &update_tick);
|
|
/*
|
|
* 1s update_ms * 1000
|
|
* -------------------- = ----------------- (second/tick)
|
|
* RT_TICK_PER_SECOND update_tick
|
|
*
|
|
* 1000000ps = 1ms
|
|
*/
|
|
update_ps = (update_tick * 1000000) / (RT_TICK_PER_SECOND * 1000);
|
|
|
|
var->pixclock = update_ps / (var->xres * var->yres);
|
|
}
|
|
else
|
|
{
|
|
var->pixclock = (RT_GRAPHIC_UPDATE_MS * 1000000) / (var->xres * var->yres);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case FBIOPUT_VSCREENINFO:
|
|
if (args)
|
|
{
|
|
rt_uint32_t mode;
|
|
struct fb_format fmt;
|
|
struct rt_device_rect_info rect;
|
|
struct fb_var_screeninfo *var = args;
|
|
struct rt_graphic_plane *plane = plane_get_current(gdev);
|
|
const rt_size_t cmp_offset = rt_offsetof(struct fb_format, bits_per_pixel);
|
|
const rt_size_t cmp_size = sizeof(struct fb_format) - cmp_offset;
|
|
|
|
if (!plane->ops->fb_pan_display)
|
|
{
|
|
if (var->xres != plane->width || var->yres != plane->height ||
|
|
var->xoffset != plane->x || var->yoffset != plane->y)
|
|
{
|
|
err = -RT_ENOSYS;
|
|
break;
|
|
}
|
|
}
|
|
|
|
mode = plane->mode;
|
|
rect.x = plane->x;
|
|
rect.y = plane->y;
|
|
rect.width = plane->width;
|
|
rect.height = plane->height;
|
|
|
|
fmt.bits_per_pixel = var->bits_per_pixel;
|
|
rt_memcpy(&fmt.red, &var->red, sizeof(fmt.red));
|
|
rt_memcpy(&fmt.green, &var->green, sizeof(fmt.green));
|
|
rt_memcpy(&fmt.blue, &var->blue, sizeof(fmt.blue));
|
|
rt_memcpy(&fmt.transp, &var->transp, sizeof(fmt.transp));
|
|
|
|
for (int i = 0; i < RT_ARRAY_SIZE(graphic_formats); ++i)
|
|
{
|
|
void *cmp_to = ((void *)&fmt) + cmp_offset;
|
|
void *cmp_from = ((void *)&graphic_formats[i]) + cmp_offset;
|
|
|
|
if (!rt_memcmp(cmp_to, cmp_from, cmp_size))
|
|
{
|
|
mode = graphic_formats[i].mode;
|
|
break;
|
|
}
|
|
}
|
|
|
|
err = -RT_ENOSYS;
|
|
|
|
for (int i = 0; i < plane->modes_nr; ++i)
|
|
{
|
|
/* Check supported and commit */
|
|
if (plane->modes[i] == mode && plane->mode != mode)
|
|
{
|
|
err = plane_fb_remap(plane, mode, &rect);
|
|
}
|
|
}
|
|
|
|
if (!err && plane->ops->fb_pan_display)
|
|
{
|
|
rect.x = var->xoffset;
|
|
rect.y = var->yoffset;
|
|
rect.width = var->xres;
|
|
rect.height = var->yres;
|
|
|
|
err = plane_fb_pan_display(plane, &rect);
|
|
}
|
|
|
|
if (!err && var->rotate && plane->ops->prop_set)
|
|
{
|
|
err = plane->ops->prop_set(plane, RT_GRAPHIC_PLANE_PROP_ROTATE,
|
|
(void *)(rt_ubase_t)((var->rotate % 360) / 90));
|
|
}
|
|
|
|
if (!err && plane == gdev->primary_plane && plane->ops->update)
|
|
{
|
|
rt_uint32_t update_ms = 0;
|
|
|
|
if (var->pixclock)
|
|
{
|
|
rt_uint64_t clock_cycles;
|
|
|
|
clock_cycles = var->pixclock;
|
|
clock_cycles *= rect.width;
|
|
clock_cycles *= rect.height;
|
|
/* Seconds in pico seconds */
|
|
clock_cycles /= 1000000000000ULL;
|
|
|
|
update_ms = (rt_uint32_t)clock_cycles * 1000;
|
|
}
|
|
|
|
err = rt_graphic_device_update_auto(gdev, update_ms);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case FBIOGET_FSCREENINFO:
|
|
if (args)
|
|
{
|
|
struct fb_fix_screeninfo *fix = args;
|
|
struct rt_graphic_plane *plane = plane_get_current(gdev);
|
|
|
|
rt_memset(fix, 0, sizeof(*fix));
|
|
|
|
rt_snprintf(fix->id, rt_min_t(int, sizeof(fix->id), RT_NAME_MAX),
|
|
"%s", rt_dm_dev_get_name(&gdev->parent));
|
|
|
|
fix->smem_start = (unsigned long)rt_kmem_v2p(plane->framebuffer);
|
|
fix->smem_len = plane->framebuffer_len;
|
|
fix->mmio_start = fix->smem_start;
|
|
fix->mmio_len = plane->screen_len;
|
|
fix->line_length = plane->line_length;
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case FBIOGET_PIXELINFO:
|
|
if (args)
|
|
{
|
|
struct rt_graphic_plane *plane = plane_get_current(gdev);
|
|
|
|
*(rt_uint32_t *)args = plane->mode;
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case FBIOPAN_DISPLAY:
|
|
if (args)
|
|
{
|
|
struct fb_var_screeninfo *var = args;
|
|
struct rt_graphic_plane *plane = plane_get_current(gdev);
|
|
|
|
if (plane->ops->fb_pan_display)
|
|
{
|
|
struct rt_device_rect_info rect;
|
|
|
|
rect.x = var->xoffset;
|
|
rect.y = var->yoffset;
|
|
rect.width = var->xres;
|
|
rect.height = var->yres;
|
|
|
|
err = plane_fb_pan_display(plane, &rect);
|
|
}
|
|
else
|
|
{
|
|
err = -RT_ENOSYS;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case FBIO_CURSOR:
|
|
err = -RT_EINVAL;
|
|
break;
|
|
|
|
case FBIOGET_CON2FBMAP:
|
|
if (args)
|
|
{
|
|
struct fb_con2fbmap *con2fbmap = args;
|
|
|
|
if (con2fbmap->console < RT_ARRAY_SIZE(fbcon_map))
|
|
{
|
|
con2fbmap->framebuffer = fbcon_map[con2fbmap->console];
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EFULL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case FBIOPUT_CON2FBMAP:
|
|
if (args)
|
|
{
|
|
struct fb_con2fbmap *con2fbmap = args;
|
|
|
|
if (con2fbmap->console < RT_ARRAY_SIZE(fbcon_map) &&
|
|
con2fbmap->framebuffer < RT_ARRAY_SIZE(fbcon_map))
|
|
{
|
|
struct rt_device *vt;
|
|
|
|
vt = rt_dm_device_find(MASTER_ID_TTY, con2fbmap->console);
|
|
|
|
#ifdef RT_SERIAL_VIRTUAL
|
|
if (!vt)
|
|
{
|
|
vt = rt_device_find("vuart");
|
|
}
|
|
#endif
|
|
|
|
if (vt)
|
|
{
|
|
if (!(err = rt_device_open(vt, RT_DEVICE_OFLAG_RDWR)))
|
|
{
|
|
err = rt_device_control(vt, FBIOPUT_CON2FBMAP, con2fbmap);
|
|
|
|
if (!err)
|
|
{
|
|
fbcon_map[con2fbmap->console] = con2fbmap->framebuffer;
|
|
}
|
|
|
|
rt_device_close(vt);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EEMPTY;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EFULL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
err = -RT_EINVAL;
|
|
}
|
|
break;
|
|
|
|
case FBIOBLANK:
|
|
if (gdev->ops->dpms_switch)
|
|
{
|
|
rt_uint32_t dpms;
|
|
|
|
switch ((rt_uint32_t)(rt_ubase_t)args)
|
|
{
|
|
case FB_BLANK_UNBLANK:
|
|
/* Display: On, HSync: On, VSync: On */
|
|
dpms = RT_GRAPHIC_DPMS_ON;
|
|
break;
|
|
|
|
case FB_BLANK_NORMAL:
|
|
/* Display: Off, HSync: On, VSync: On */
|
|
dpms = RT_GRAPHIC_DPMS_STANDBY;
|
|
break;
|
|
|
|
case FB_BLANK_HSYNC_SUSPEND:
|
|
/* Display: Off, HSync: Off, VSync: On */
|
|
dpms = RT_GRAPHIC_DPMS_STANDBY;
|
|
break;
|
|
|
|
case FB_BLANK_VSYNC_SUSPEND:
|
|
/* Display: Off, HSync: On, VSync: Off */
|
|
dpms = RT_GRAPHIC_DPMS_SUSPEND;
|
|
break;
|
|
|
|
case FB_BLANK_POWERDOWN:
|
|
/* Display: Off, HSync: Off, VSync: Off */
|
|
dpms = RT_GRAPHIC_DPMS_OFF;
|
|
break;
|
|
|
|
default:
|
|
err = -RT_EINVAL;
|
|
break;
|
|
}
|
|
|
|
if (!err)
|
|
{
|
|
graphic_dpms_switch(gdev, dpms);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case FBIO_WAITFORVSYNC:
|
|
if (gdev->ops->wait_vsync)
|
|
{
|
|
err = gdev->ops->wait_vsync(gdev);
|
|
}
|
|
break;
|
|
|
|
case FBIO_ALLOC:
|
|
case FBIO_FREE:
|
|
case FBIOGET_GLYPH:
|
|
case FBIOGET_HWCINFO:
|
|
case FBIOPUT_MODEINFO:
|
|
case FBIOGET_DISPINFO:
|
|
LOG_D("FB IOCTL (%x) only used for SiS 300/630/730/540/315/550/650/740 frame buffer device", cmd);
|
|
/* Fall through */
|
|
case FBIOGET_VBLANK:
|
|
case FBIOGETCMAP: /* fb_cmap */
|
|
case FBIOPUTCMAP: /* fb_cmap */
|
|
/* Fall through */
|
|
default:
|
|
if (gdev->ops->control)
|
|
{
|
|
err = gdev->ops->control(gdev, cmd, args);
|
|
}
|
|
else
|
|
{
|
|
err = -RT_ENOSYS;
|
|
}
|
|
}
|
|
|
|
spin_unlock(&gdev->lock);
|
|
|
|
return err;
|
|
}
|
|
|
|
#ifdef RT_USING_DEVICE_OPS
|
|
const static struct rt_device_ops _graphic_ops =
|
|
{
|
|
.open = _graphic_open,
|
|
.read = _graphic_read,
|
|
.write = _graphic_write,
|
|
.control = _graphic_control,
|
|
};
|
|
#endif
|
|
|
|
static void graphic_ofw_init(struct rt_graphic_device *gdev)
|
|
{
|
|
#ifdef RT_USING_OFW
|
|
struct rt_ofw_node *np = gdev->parent.ofw_node;
|
|
|
|
#ifdef RT_GRAPHIC_BACKLIGHT
|
|
if (!gdev->backlight)
|
|
{
|
|
struct rt_ofw_node *bl_np = rt_ofw_parse_phandle(np, "backlight", 0);
|
|
|
|
if (bl_np && (gdev->backlight = rt_ofw_data(bl_np)))
|
|
{
|
|
rt_device_open(&gdev->backlight->parent, RT_DEVICE_OFLAG_RDWR);
|
|
}
|
|
rt_ofw_node_put(bl_np);
|
|
}
|
|
#endif /* RT_GRAPHIC_BACKLIGHT */
|
|
|
|
(void)np;
|
|
#endif /* RT_USING_OFW */
|
|
}
|
|
|
|
static void graphic_edid_res(struct rt_graphic_device *gdev,
|
|
rt_uint32_t *out_width, rt_uint32_t *out_height)
|
|
{
|
|
struct edid *edid = &gdev->edid;
|
|
struct detailed_timing *dt = &edid->detailed_timings[0];
|
|
struct detailed_pixel_timing *dpt = &dt->data.pixel_data;
|
|
|
|
*out_width = dpt->hactive_lo + ((dpt->hactive_hblank_hi & 0xf0) << 4);
|
|
*out_height = dpt->vactive_lo + ((dpt->vactive_vblank_hi & 0xf0) << 4);
|
|
}
|
|
|
|
#ifdef RT_USING_PM
|
|
static rt_err_t _graphic_pm_dpms_switch(struct rt_graphic_device *gdev, rt_uint8_t mode)
|
|
{
|
|
rt_err_t err;
|
|
|
|
spin_lock(&gdev->lock);
|
|
|
|
switch (mode)
|
|
{
|
|
case PM_SLEEP_MODE_NONE:
|
|
case PM_SLEEP_MODE_IDLE:
|
|
case PM_SLEEP_MODE_LIGHT:
|
|
case PM_SLEEP_MODE_DEEP:
|
|
err = graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_ON);
|
|
break;
|
|
|
|
case PM_SLEEP_MODE_STANDBY:
|
|
err = graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_STANDBY);
|
|
break;
|
|
|
|
case PM_SLEEP_MODE_SHUTDOWN:
|
|
err = graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_OFF);
|
|
break;
|
|
|
|
default:
|
|
err = -RT_EINVAL;
|
|
break;
|
|
}
|
|
|
|
spin_unlock(&gdev->lock);
|
|
|
|
return err;
|
|
}
|
|
|
|
static rt_err_t _graphic_pm_suspend(const struct rt_device *device, rt_uint8_t mode)
|
|
{
|
|
struct rt_graphic_device *gdev = raw_to_graphic(device);
|
|
|
|
return _graphic_pm_dpms_switch(gdev, mode);
|
|
}
|
|
|
|
static void _graphic_pm_resume(const struct rt_device *device, rt_uint8_t mode)
|
|
{
|
|
struct rt_graphic_device *gdev = raw_to_graphic(device);
|
|
|
|
_graphic_pm_dpms_switch(gdev, mode);
|
|
}
|
|
|
|
static const struct rt_device_pm_ops _graphic_pm_ops =
|
|
{
|
|
.suspend = _graphic_pm_suspend,
|
|
.resume = _graphic_pm_resume,
|
|
};
|
|
#endif /* RT_USING_PM */
|
|
|
|
rt_err_t rt_graphic_device_register(struct rt_graphic_device *gdev)
|
|
{
|
|
rt_err_t err;
|
|
int device_id;
|
|
const char *dev_name;
|
|
|
|
if (!gdev || !gdev->ops)
|
|
{
|
|
return -RT_EINVAL;
|
|
}
|
|
|
|
if (!gdev->primary_plane)
|
|
{
|
|
LOG_E("%s: Not %s found", rt_dm_dev_get_name(&gdev->parent), "primary plane");
|
|
|
|
return -RT_EINVAL;
|
|
}
|
|
|
|
if ((device_id = rt_dm_ida_alloc(&graphic_ida)) < 0)
|
|
{
|
|
return -RT_EFULL;
|
|
}
|
|
|
|
rt_dm_dev_set_name(&gdev->parent, "fb%u", device_id);
|
|
dev_name = rt_dm_dev_get_name(&gdev->parent);
|
|
|
|
rt_list_init(&gdev->overlay_nodes);
|
|
rt_dm_ida_init(&gdev->plane_ida, CUSTOM);
|
|
rt_spin_lock_init(&gdev->lock);
|
|
|
|
graphic_ofw_init(gdev);
|
|
|
|
if (!gdev->primary_plane->width || !gdev->primary_plane->height)
|
|
{
|
|
rt_uint32_t mode, width, height;
|
|
struct rt_device_rect_info rect;
|
|
|
|
graphic_edid_res(gdev, &width, &height);
|
|
rect.x = 0;
|
|
rect.y = 0;
|
|
rect.width = width;
|
|
rect.height = height;
|
|
|
|
if (gdev->primary_plane->mode == RTGRAPHIC_PIXEL_FORMAT_MONO)
|
|
{
|
|
mode = gdev->primary_plane->modes[0];
|
|
}
|
|
else
|
|
{
|
|
mode = gdev->primary_plane->mode;
|
|
}
|
|
|
|
err = plane_fb_remap(gdev->primary_plane, mode, &rect);
|
|
|
|
if (err)
|
|
{
|
|
LOG_E("%s: Set %s error = %s", rt_dm_dev_get_name(&gdev->parent),
|
|
"primary plane", rt_strerror(err));
|
|
|
|
goto _fail;
|
|
}
|
|
}
|
|
|
|
gdev->parent.type = RT_Device_Class_Graphic;
|
|
#ifdef RT_USING_DEVICE_OPS
|
|
gdev->parent.ops = &_graphic_ops;
|
|
#else
|
|
gdev->parent.open = _graphic_open;
|
|
gdev->parent.read = _graphic_read;
|
|
gdev->parent.write = _graphic_write;
|
|
gdev->parent.control = _graphic_control;
|
|
#endif
|
|
gdev->parent.master_id = graphic_ida.master_id;
|
|
gdev->parent.device_id = device_id;
|
|
|
|
if ((err = rt_device_register(&gdev->parent, dev_name, RT_DEVICE_FLAG_RDWR)))
|
|
{
|
|
goto _fail;
|
|
}
|
|
|
|
#ifdef RT_USING_PM
|
|
rt_pm_device_register(&gdev->parent, &_graphic_pm_ops);
|
|
#endif
|
|
|
|
if ((err = rt_graphic_logo_render(gdev)))
|
|
{
|
|
LOG_D("Logo render error = %s", rt_strerror(err));
|
|
}
|
|
|
|
return RT_EOK;
|
|
|
|
_fail:
|
|
rt_dm_ida_free(&graphic_ida, device_id);
|
|
|
|
return err;
|
|
}
|
|
|
|
rt_err_t rt_graphic_device_unregister(struct rt_graphic_device *gdev)
|
|
{
|
|
const char *dev_name;
|
|
struct rt_graphic_plane *plane, *plane_next;
|
|
|
|
if (!gdev)
|
|
{
|
|
return -RT_EINVAL;
|
|
}
|
|
|
|
dev_name = rt_dm_dev_get_name(&gdev->parent);
|
|
|
|
if (gdev->parent.ref_count)
|
|
{
|
|
LOG_E("%s: there is %u user", dev_name, gdev->parent.ref_count);
|
|
return -RT_EINVAL;
|
|
}
|
|
|
|
#ifdef RT_USING_PM
|
|
rt_pm_device_unregister(&gdev->parent);
|
|
#endif
|
|
|
|
rt_graphic_device_update_auto(gdev, 0);
|
|
|
|
if (gdev->ops->dpms_switch)
|
|
{
|
|
graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_OFF);
|
|
}
|
|
#ifdef RT_GRAPHIC_BACKLIGHT
|
|
if (gdev->backlight)
|
|
{
|
|
rt_backlight_set_power(gdev->backlight, RT_BACKLIGHT_POWER_POWERDOWN);
|
|
rt_device_close(&gdev->backlight->parent);
|
|
}
|
|
#endif
|
|
|
|
rt_list_for_each_entry_safe(plane, plane_next, &gdev->overlay_nodes, list)
|
|
{
|
|
rt_graphic_device_del_plane(gdev, plane);
|
|
rt_graphic_device_free_plane(plane);
|
|
}
|
|
|
|
rt_graphic_device_del_plane(gdev, gdev->primary_plane);
|
|
rt_graphic_device_free_plane(gdev->primary_plane);
|
|
|
|
if (gdev->cursor_plane)
|
|
{
|
|
rt_graphic_device_del_plane(gdev, gdev->cursor_plane);
|
|
rt_graphic_device_free_plane(gdev->cursor_plane);
|
|
}
|
|
|
|
rt_dm_ida_free(&graphic_ida, gdev->parent.device_id);
|
|
|
|
rt_device_unregister(&gdev->parent);
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
struct rt_graphic_plane *rt_graphic_device_alloc_plane(struct rt_graphic_device *gdev,
|
|
rt_size_t priv_size, const struct rt_graphic_plane_ops *ops,
|
|
const rt_uint32_t *modes, rt_uint32_t modes_nr, rt_uint8_t type)
|
|
{
|
|
struct rt_graphic_plane *plane = RT_NULL;
|
|
|
|
if (!gdev || !ops || !modes || !modes_nr || type > RT_GRAPHIC_PLANE_TYPE_CURSOR)
|
|
{
|
|
return RT_NULL;
|
|
}
|
|
|
|
plane = rt_calloc(1, sizeof(*plane) + priv_size);
|
|
|
|
if (plane)
|
|
{
|
|
rt_list_init(&plane->list);
|
|
plane->type = type;
|
|
plane->modes_nr = modes_nr;
|
|
plane->modes = modes;
|
|
plane->mode = RTGRAPHIC_PIXEL_FORMAT_MONO;
|
|
plane->graphic = gdev;
|
|
plane->ops = ops;
|
|
}
|
|
|
|
return plane;
|
|
}
|
|
|
|
void rt_graphic_device_free_plane(struct rt_graphic_plane *plane)
|
|
{
|
|
if (!plane)
|
|
{
|
|
return;
|
|
}
|
|
|
|
rt_free(plane);
|
|
}
|
|
|
|
rt_err_t rt_graphic_device_add_plane(struct rt_graphic_device *gdev,
|
|
struct rt_graphic_plane *plane)
|
|
{
|
|
rt_err_t err = RT_EOK;
|
|
|
|
if (!gdev || !plane)
|
|
{
|
|
return -RT_EINVAL;
|
|
}
|
|
|
|
if (!plane->ops)
|
|
{
|
|
LOG_E("%s: %s have no plane ops",
|
|
rt_dm_dev_get_name(&gdev->parent), plane->name);
|
|
return -RT_EINVAL;
|
|
}
|
|
|
|
plane->id = rt_dm_ida_alloc(&gdev->plane_ida);
|
|
|
|
if (plane->id == RT_DM_IDA_NUM)
|
|
{
|
|
LOG_E("%s: %s is out of plane max(%d)",
|
|
rt_dm_dev_get_name(&gdev->parent), plane->name, RT_DM_IDA_NUM - 1);
|
|
return -RT_EFULL;
|
|
}
|
|
|
|
if (plane->type == RT_GRAPHIC_PLANE_TYPE_PRIMARY)
|
|
{
|
|
if (gdev->primary_plane)
|
|
{
|
|
err = -RT_EINVAL;
|
|
goto _free_ida;
|
|
}
|
|
if (!plane->name[0])
|
|
{
|
|
rt_strncpy(plane->name, "primary", sizeof(plane->name));
|
|
}
|
|
gdev->primary_plane = plane;
|
|
}
|
|
else if (plane->type == RT_GRAPHIC_PLANE_TYPE_CURSOR)
|
|
{
|
|
if (gdev->cursor_plane)
|
|
{
|
|
err = -RT_EINVAL;
|
|
goto _free_ida;
|
|
}
|
|
if (!plane->name[0])
|
|
{
|
|
rt_strncpy(plane->name, "cursor", sizeof(plane->name));
|
|
}
|
|
gdev->cursor_plane = plane;
|
|
}
|
|
else if (plane->type == RT_GRAPHIC_PLANE_TYPE_OVERLAY)
|
|
{
|
|
if (!plane->name[0])
|
|
{
|
|
rt_snprintf(plane->name, sizeof(plane->name), "overlay-%u", plane->id);
|
|
}
|
|
spin_lock(&gdev->lock);
|
|
rt_list_insert_before(&gdev->overlay_nodes, &plane->list);
|
|
rt_graphic_device_leave(gdev);
|
|
}
|
|
else
|
|
{
|
|
LOG_E("What the fuck plane type(%u)", plane->type);
|
|
RT_ASSERT(0);
|
|
}
|
|
|
|
_free_ida:
|
|
if (err)
|
|
{
|
|
rt_dm_ida_free(&gdev->plane_ida, plane->id);
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
rt_err_t rt_graphic_device_del_plane(struct rt_graphic_device *gdev,
|
|
struct rt_graphic_plane *plane)
|
|
{
|
|
if (!gdev || !plane)
|
|
{
|
|
return -RT_EINVAL;
|
|
}
|
|
|
|
if (plane->ops->fb_cleanup)
|
|
{
|
|
/* Ignore error */
|
|
plane->ops->fb_cleanup(plane);
|
|
}
|
|
|
|
if (plane->type == RT_GRAPHIC_PLANE_TYPE_PRIMARY)
|
|
{
|
|
gdev->primary_plane = RT_NULL;
|
|
}
|
|
else if (plane->type == RT_GRAPHIC_PLANE_TYPE_CURSOR)
|
|
{
|
|
gdev->cursor_plane = RT_NULL;
|
|
}
|
|
else if (plane->type == RT_GRAPHIC_PLANE_TYPE_OVERLAY)
|
|
{
|
|
spin_lock(&gdev->lock);
|
|
rt_list_remove(&plane->list);
|
|
spin_unlock(&gdev->lock);
|
|
}
|
|
|
|
rt_dm_ida_free(&gdev->plane_ida, plane->id);
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
void rt_graphic_device_hotplug_event(struct rt_graphic_device *gdev)
|
|
{
|
|
rt_err_t err;
|
|
rt_uint32_t width, height;
|
|
struct rt_device_rect_info rect;
|
|
|
|
RT_ASSERT(gdev != RT_NULL);
|
|
|
|
rt_graphic_device_enter(gdev);
|
|
|
|
graphic_edid_res(gdev, &width, &height);
|
|
rect.x = 0;
|
|
rect.y = 0;
|
|
rect.width = width;
|
|
rect.height = height;
|
|
|
|
err = plane_fb_remap(gdev->primary_plane, gdev->primary_plane->mode, &rect);
|
|
|
|
if (err)
|
|
{
|
|
/* What the fuck? */
|
|
LOG_E("%s: hotplug event process error = %s",
|
|
rt_dm_dev_get_name(&gdev->parent), rt_strerror(err));
|
|
goto _out_lock;
|
|
}
|
|
|
|
_out_lock:
|
|
rt_graphic_device_leave(gdev);
|
|
|
|
rt_atomic_store(&gdev->event_notifying, RT_TRUE);
|
|
|
|
if (gdev->event_notify.notify)
|
|
{
|
|
gdev->event_notify.notify(gdev->event_notify.dev);
|
|
}
|
|
|
|
rt_atomic_store(&gdev->event_notifying, RT_FALSE);
|
|
}
|
|
|
|
static void graphic_device_plane_update(struct rt_graphic_plane *plane,
|
|
struct rt_device_rect_info *rect)
|
|
{
|
|
if (plane->ops->update)
|
|
{
|
|
plane->ops->update(plane, rect);
|
|
}
|
|
}
|
|
|
|
static void graphic_device_update(void *param)
|
|
{
|
|
struct rt_device_rect_info rect;
|
|
struct rt_graphic_plane *plane;
|
|
struct rt_graphic_device *gdev = param;
|
|
|
|
rect.x = 0;
|
|
rect.y = 0;
|
|
|
|
spin_lock(&gdev->lock);
|
|
|
|
rect.width = gdev->primary_plane->width;
|
|
rect.height = gdev->primary_plane->height;
|
|
graphic_device_plane_update(gdev->primary_plane, &rect);
|
|
|
|
rt_list_for_each_entry(plane, &gdev->overlay_nodes, list)
|
|
{
|
|
rect.width = plane->width;
|
|
rect.height = plane->height;
|
|
|
|
graphic_device_plane_update(plane, &rect);
|
|
}
|
|
|
|
if ((plane = gdev->cursor_plane))
|
|
{
|
|
rect.x = plane->x;
|
|
rect.y = plane->y;
|
|
/* Ask driver to update position only */
|
|
rect.width = 0;
|
|
rect.height = 0;
|
|
|
|
graphic_device_plane_update(plane, &rect);
|
|
}
|
|
|
|
spin_unlock(&gdev->lock);
|
|
}
|
|
|
|
rt_err_t rt_graphic_device_update_auto(struct rt_graphic_device *gdev, rt_uint32_t update_ms)
|
|
{
|
|
if (!gdev)
|
|
{
|
|
return -RT_EINVAL;
|
|
}
|
|
|
|
if (update_ms)
|
|
{
|
|
if (!gdev->update_timer)
|
|
{
|
|
char name[RT_NAME_MAX];
|
|
|
|
rt_snprintf(name, sizeof(name), "update-%s", rt_dm_dev_get_name(&gdev->parent));
|
|
|
|
gdev->update_timer = rt_timer_create(name, &graphic_device_update, gdev,
|
|
rt_tick_from_millisecond(update_ms),
|
|
RT_TIMER_FLAG_PERIODIC);
|
|
|
|
if (!gdev->update_timer)
|
|
{
|
|
return -RT_ENOMEM;
|
|
}
|
|
}
|
|
|
|
rt_timer_start(gdev->update_timer);
|
|
}
|
|
else if (gdev->update_timer)
|
|
{
|
|
rt_timer_stop(gdev->update_timer);
|
|
rt_timer_delete(gdev->update_timer);
|
|
|
|
gdev->update_timer = RT_NULL;
|
|
}
|
|
|
|
return RT_EOK;
|
|
}
|
|
|
|
void rt_graphic_device_enter(struct rt_graphic_device *gdev)
|
|
{
|
|
RT_ASSERT(gdev != RT_NULL);
|
|
|
|
spin_lock(&gdev->lock);
|
|
|
|
if (gdev->update_timer)
|
|
{
|
|
rt_timer_stop(gdev->update_timer);
|
|
}
|
|
}
|
|
|
|
void rt_graphic_device_leave(struct rt_graphic_device *gdev)
|
|
{
|
|
RT_ASSERT(gdev != RT_NULL);
|
|
|
|
if (gdev->update_timer)
|
|
{
|
|
rt_timer_start(gdev->update_timer);
|
|
}
|
|
|
|
spin_unlock(&gdev->lock);
|
|
}
|
|
|
|
rt_uint32_t rt_graphic_mode_bpp(rt_uint32_t mode)
|
|
{
|
|
for (int i = 0; i < RT_ARRAY_SIZE(graphic_formats); ++i)
|
|
{
|
|
if (graphic_formats[i].mode == mode)
|
|
{
|
|
return graphic_formats[i].bits_per_pixel;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|