mirror of
https://github.com/torvalds/linux.git
synced 2025-11-03 01:59:51 +02:00
Convert drm_sysfb_get_format_si() to lookup the screen_info color format as struct pixel_format with screen_info_pixel_format(). Then search the list of given formats for the screen_info format with pixel_format_equal(). Replaces custom code with helpers. The pixel-compare helper pixel_format_equal() also handles indexed color formats. Prepares for sysfb drivers to support color palettes. Signed-off-by: Thomas Zimmermann <tzimmermann@suse.de> Reviewed-by: Javier Martinez Canillas <javierm@redhat.com> Link: https://lore.kernel.org/r/20250714151513.309475-4-tzimmermann@suse.de
104 lines
2.7 KiB
C
104 lines
2.7 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
|
|
#include <linux/export.h>
|
|
#include <linux/limits.h>
|
|
#include <linux/minmax.h>
|
|
#include <linux/screen_info.h>
|
|
|
|
#include <drm/drm_fourcc.h>
|
|
#include <drm/drm_print.h>
|
|
|
|
#include "drm_sysfb_helper.h"
|
|
|
|
static s64 drm_sysfb_get_validated_size0(struct drm_device *dev, const char *name,
|
|
u64 value, u64 max)
|
|
{
|
|
if (!value) {
|
|
drm_warn(dev, "%s of 0 not allowed\n", name);
|
|
return -EINVAL;
|
|
} else if (value > min(max, S64_MAX)) {
|
|
drm_warn(dev, "%s of %llu exceeds maximum of %llu\n", name, value, max);
|
|
return -EINVAL;
|
|
}
|
|
return value;
|
|
}
|
|
|
|
int drm_sysfb_get_width_si(struct drm_device *dev, const struct screen_info *si)
|
|
{
|
|
return drm_sysfb_get_validated_int0(dev, "width", si->lfb_width, U16_MAX);
|
|
}
|
|
EXPORT_SYMBOL(drm_sysfb_get_width_si);
|
|
|
|
int drm_sysfb_get_height_si(struct drm_device *dev, const struct screen_info *si)
|
|
{
|
|
return drm_sysfb_get_validated_int0(dev, "height", si->lfb_height, U16_MAX);
|
|
}
|
|
EXPORT_SYMBOL(drm_sysfb_get_height_si);
|
|
|
|
struct resource *drm_sysfb_get_memory_si(struct drm_device *dev,
|
|
const struct screen_info *si,
|
|
struct resource *res)
|
|
{
|
|
ssize_t num;
|
|
|
|
num = screen_info_resources(si, res, 1);
|
|
if (!num) {
|
|
drm_warn(dev, "memory resource not found\n");
|
|
return NULL;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
EXPORT_SYMBOL(drm_sysfb_get_memory_si);
|
|
|
|
int drm_sysfb_get_stride_si(struct drm_device *dev, const struct screen_info *si,
|
|
const struct drm_format_info *format,
|
|
unsigned int width, unsigned int height, u64 size)
|
|
{
|
|
u64 lfb_linelength = si->lfb_linelength;
|
|
|
|
if (!lfb_linelength)
|
|
lfb_linelength = drm_format_info_min_pitch(format, 0, width);
|
|
|
|
return drm_sysfb_get_validated_int0(dev, "stride", lfb_linelength, div64_u64(size, height));
|
|
}
|
|
EXPORT_SYMBOL(drm_sysfb_get_stride_si);
|
|
|
|
u64 drm_sysfb_get_visible_size_si(struct drm_device *dev, const struct screen_info *si,
|
|
unsigned int height, unsigned int stride, u64 size)
|
|
{
|
|
u64 vsize = PAGE_ALIGN(height * stride);
|
|
|
|
return drm_sysfb_get_validated_size0(dev, "visible size", vsize, size);
|
|
}
|
|
EXPORT_SYMBOL(drm_sysfb_get_visible_size_si);
|
|
|
|
const struct drm_format_info *drm_sysfb_get_format_si(struct drm_device *dev,
|
|
const struct drm_sysfb_format *formats,
|
|
size_t nformats,
|
|
const struct screen_info *si)
|
|
{
|
|
const struct drm_format_info *format = NULL;
|
|
struct pixel_format pixel;
|
|
size_t i;
|
|
int ret;
|
|
|
|
ret = screen_info_pixel_format(si, &pixel);
|
|
if (ret)
|
|
return NULL;
|
|
|
|
for (i = 0; i < nformats; ++i) {
|
|
const struct drm_sysfb_format *f = &formats[i];
|
|
|
|
if (pixel_format_equal(&pixel, &f->pixel)) {
|
|
format = drm_format_info(f->fourcc);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!format)
|
|
drm_warn(dev, "No compatible color format found\n");
|
|
|
|
return format;
|
|
}
|
|
EXPORT_SYMBOL(drm_sysfb_get_format_si);
|