Files
rt-thread/components/drivers/graphic/graphic.c
GuEe-GUI 5abecc1fd0 [dm][graphic] support dm mode
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>
2025-12-15 16:54:23 +08:00

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;
}