2018-05-06 21:58:06 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
2015-02-03 11:32:31 +00:00
|
|
|
/*
|
|
|
|
* Simplefb device tree support
|
|
|
|
*
|
|
|
|
* (C) Copyright 2015
|
|
|
|
* Stephen Warren <swarren@wwwdotorg.org>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <common.h>
|
2017-04-05 22:23:43 +00:00
|
|
|
#include <dm.h>
|
2015-02-03 11:32:31 +00:00
|
|
|
#include <fdt_support.h>
|
2020-10-31 03:38:53 +00:00
|
|
|
#include <asm/global_data.h>
|
2018-03-04 16:20:11 +00:00
|
|
|
#include <linux/libfdt.h>
|
2017-04-05 22:23:43 +00:00
|
|
|
#include <video.h>
|
2015-02-03 11:32:31 +00:00
|
|
|
|
|
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
|
|
|
2021-11-15 15:32:19 +00:00
|
|
|
static int fdt_simplefb_configure_node(void *blob, int off)
|
2015-02-03 11:32:31 +00:00
|
|
|
{
|
2017-04-05 22:23:42 +00:00
|
|
|
int xsize, ysize;
|
|
|
|
int bpix; /* log2 of bits per pixel */
|
|
|
|
const char *name;
|
|
|
|
ulong fb_base;
|
2020-12-03 23:55:23 +00:00
|
|
|
struct video_uc_plat *plat;
|
2017-04-05 22:23:43 +00:00
|
|
|
struct video_priv *uc_priv;
|
|
|
|
struct udevice *dev;
|
|
|
|
int ret;
|
2017-04-05 22:23:42 +00:00
|
|
|
|
2017-04-05 22:23:43 +00:00
|
|
|
ret = uclass_first_device_err(UCLASS_VIDEO, &dev);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
uc_priv = dev_get_uclass_priv(dev);
|
2020-12-03 23:55:18 +00:00
|
|
|
plat = dev_get_uclass_plat(dev);
|
2017-04-05 22:23:43 +00:00
|
|
|
xsize = uc_priv->xsize;
|
|
|
|
ysize = uc_priv->ysize;
|
|
|
|
bpix = uc_priv->bpix;
|
|
|
|
fb_base = plat->base;
|
2017-04-05 22:23:42 +00:00
|
|
|
switch (bpix) {
|
|
|
|
case 4: /* VIDEO_BPP16 */
|
|
|
|
name = "r5g6b5";
|
|
|
|
break;
|
|
|
|
case 5: /* VIDEO_BPP32 */
|
|
|
|
name = "a8r8g8b8";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return fdt_setup_simplefb_node(blob, off, fb_base, xsize, ysize,
|
|
|
|
xsize * (1 << bpix) / 8, name);
|
2015-02-03 11:32:31 +00:00
|
|
|
}
|
|
|
|
|
2021-11-15 15:32:19 +00:00
|
|
|
int fdt_simplefb_add_node(void *blob)
|
2015-02-03 11:32:31 +00:00
|
|
|
{
|
|
|
|
static const char compat[] = "simple-framebuffer";
|
|
|
|
static const char disabled[] = "disabled";
|
|
|
|
int off, ret;
|
|
|
|
|
|
|
|
off = fdt_add_subnode(blob, 0, "framebuffer");
|
|
|
|
if (off < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
ret = fdt_setprop(blob, off, "status", disabled, sizeof(disabled));
|
|
|
|
if (ret < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
ret = fdt_setprop(blob, off, "compatible", compat, sizeof(compat));
|
|
|
|
if (ret < 0)
|
|
|
|
return -1;
|
|
|
|
|
2021-11-15 15:32:19 +00:00
|
|
|
return fdt_simplefb_configure_node(blob, off);
|
2015-02-03 11:32:31 +00:00
|
|
|
}
|
|
|
|
|
2021-11-15 15:32:19 +00:00
|
|
|
int fdt_simplefb_enable_existing_node(void *blob)
|
2015-02-03 11:32:31 +00:00
|
|
|
{
|
|
|
|
int off;
|
|
|
|
|
|
|
|
off = fdt_node_offset_by_compatible(blob, -1, "simple-framebuffer");
|
|
|
|
if (off < 0)
|
|
|
|
return -1;
|
|
|
|
|
2021-11-15 15:32:19 +00:00
|
|
|
return fdt_simplefb_configure_node(blob, off);
|
2015-02-03 11:32:31 +00:00
|
|
|
}
|
2021-11-15 15:32:21 +00:00
|
|
|
|
2023-02-05 22:44:27 +00:00
|
|
|
#if IS_ENABLED(CONFIG_VIDEO)
|
2021-11-15 15:32:21 +00:00
|
|
|
int fdt_simplefb_enable_and_mem_rsv(void *blob)
|
|
|
|
{
|
|
|
|
struct fdt_memory mem;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* nothing to do when video is not active */
|
|
|
|
if (!video_is_active())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ret = fdt_simplefb_enable_existing_node(blob);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* nothing to do when the frame buffer is not defined */
|
|
|
|
if (gd->video_bottom == gd->video_top)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* reserved with no-map tag the video buffer */
|
|
|
|
mem.start = gd->video_bottom;
|
|
|
|
mem.end = gd->video_top - 1;
|
|
|
|
|
|
|
|
return fdtdec_add_reserved_memory(blob, "framebuffer", &mem, NULL, 0, NULL,
|
|
|
|
FDTDEC_RESERVED_MEMORY_NO_MAP);
|
|
|
|
}
|
|
|
|
#endif
|