mirror of
https://github.com/AsahiLinux/u-boot
synced 2024-12-23 11:33:32 +00:00
4eea531859
Add sandbox driver and tests for the new OSD uclass. Signed-off-by: Mario Six <mario.six@gdsys.cc> Reviewed-by: Simon Glass <sjg@chromium.org>
161 lines
3.1 KiB
C
161 lines
3.1 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* (C) Copyright 2018
|
|
* Mario Six, Guntermann & Drunck GmbH, mario.six@gdsys.cc
|
|
*/
|
|
#include <common.h>
|
|
#include <display.h>
|
|
#include <dm.h>
|
|
#include <video_osd.h>
|
|
|
|
#include "sandbox_osd.h"
|
|
|
|
struct sandbox_osd_priv {
|
|
uint width;
|
|
uint height;
|
|
u16 *buf;
|
|
};
|
|
|
|
static const struct udevice_id sandbox_osd_ids[] = {
|
|
{ .compatible = "sandbox,sandbox_osd" },
|
|
{ }
|
|
};
|
|
|
|
inline u16 make_memval(u8 chr, u8 color)
|
|
{
|
|
return chr * 0x100 + color;
|
|
}
|
|
|
|
int sandbox_osd_get_info(struct udevice *dev, struct video_osd_info *info)
|
|
{
|
|
struct sandbox_osd_priv *priv = dev_get_priv(dev);
|
|
|
|
info->width = priv->width;
|
|
info->height = priv->height;
|
|
info->major_version = 1;
|
|
info->minor_version = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int sandbox_osd_set_mem(struct udevice *dev, uint col, uint row, u8 *buf,
|
|
size_t buflen, uint count)
|
|
{
|
|
struct sandbox_osd_priv *priv = dev_get_priv(dev);
|
|
int pos;
|
|
u8 *mem = (u8 *)priv->buf;
|
|
int i;
|
|
|
|
pos = 2 * (row * priv->width + col);
|
|
|
|
if (pos >= 2 * (priv->width * priv->height))
|
|
return -EINVAL;
|
|
|
|
for (i = 0; i < count; i++)
|
|
memcpy(mem + pos + (i * buflen), buf, buflen);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int _sandbox_osd_set_size(struct udevice *dev, uint col, uint row)
|
|
{
|
|
struct sandbox_osd_priv *priv = dev_get_priv(dev);
|
|
int i;
|
|
uint size;
|
|
|
|
priv->width = col;
|
|
priv->height = row;
|
|
size = priv->width * priv->height;
|
|
if (!priv->buf)
|
|
priv->buf = calloc(size, sizeof(u16));
|
|
else
|
|
priv->buf = realloc(priv->buf, size * sizeof(u16));
|
|
|
|
if (!priv->buf)
|
|
return -ENOMEM;
|
|
|
|
/* Fill OSD with black spaces */
|
|
for (i = 0; i < size; i++)
|
|
priv->buf[i] = make_memval(' ', 'k');
|
|
|
|
return 0;
|
|
}
|
|
|
|
int sandbox_osd_set_size(struct udevice *dev, uint col, uint row)
|
|
{
|
|
return _sandbox_osd_set_size(dev, col, row);
|
|
}
|
|
|
|
int sandbox_osd_print(struct udevice *dev, uint col, uint row, ulong color,
|
|
char *text)
|
|
{
|
|
struct sandbox_osd_priv *priv = dev_get_priv(dev);
|
|
char cval;
|
|
char *p;
|
|
int pos;
|
|
|
|
if (col >= priv->width || row >= priv->height)
|
|
return -EINVAL;
|
|
|
|
switch (color) {
|
|
case COLOR_BLACK:
|
|
cval = 'k';
|
|
break;
|
|
case COLOR_WHITE:
|
|
cval = 'w';
|
|
break;
|
|
case COLOR_RED:
|
|
cval = 'r';
|
|
break;
|
|
case COLOR_GREEN:
|
|
cval = 'g';
|
|
break;
|
|
case COLOR_BLUE:
|
|
cval = 'b';
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
p = text;
|
|
pos = row * priv->width + col;
|
|
|
|
while (*p)
|
|
priv->buf[pos++] = make_memval(*(p++), cval);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int sandbox_osd_get_mem(struct udevice *dev, u8 *buf, size_t buflen)
|
|
{
|
|
struct sandbox_osd_priv *priv = dev_get_priv(dev);
|
|
uint memsize = 2 * (priv->width * priv->height);
|
|
|
|
if (buflen < memsize)
|
|
return -EINVAL;
|
|
|
|
memcpy(buf, priv->buf, memsize);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct video_osd_ops sandbox_osd_ops = {
|
|
.get_info = sandbox_osd_get_info,
|
|
.set_mem = sandbox_osd_set_mem,
|
|
.set_size = sandbox_osd_set_size,
|
|
.print = sandbox_osd_print,
|
|
};
|
|
|
|
int sandbox_osd_probe(struct udevice *dev)
|
|
{
|
|
return _sandbox_osd_set_size(dev, 10, 10);
|
|
}
|
|
|
|
U_BOOT_DRIVER(sandbox_osd_drv) = {
|
|
.name = "sandbox_osd_drv",
|
|
.id = UCLASS_VIDEO_OSD,
|
|
.ops = &sandbox_osd_ops,
|
|
.of_match = sandbox_osd_ids,
|
|
.probe = sandbox_osd_probe,
|
|
.priv_auto_alloc_size = sizeof(struct sandbox_osd_priv),
|
|
};
|