Share efidrm's and vesadrm's validation of struct screen_info in shared helpers. Update the drivers.
Most validation helpers test individual values against limits and can be shared as they are. For color formats, a common helper looks up the correct DRM format info from a driver-provided list of color formats. These screen_info helpers are only available if CONFIG_SCREEN_INFO has been selected, as done by efidrm and vesadrm. Signed-off-by: Thomas Zimmermann <tzimmerm...@suse.de> Reviewed-by: Javier Martinez Canillas <javi...@redhat.com> --- drivers/gpu/drm/sysfb/Makefile | 1 + drivers/gpu/drm/sysfb/drm_sysfb_helper.h | 34 ++++++ drivers/gpu/drm/sysfb/drm_sysfb_screen_info.c | 107 ++++++++++++++++++ drivers/gpu/drm/sysfb/efidrm.c | 105 ++--------------- drivers/gpu/drm/sysfb/vesadrm.c | 105 ++--------------- 5 files changed, 160 insertions(+), 192 deletions(-) create mode 100644 drivers/gpu/drm/sysfb/drm_sysfb_screen_info.c diff --git a/drivers/gpu/drm/sysfb/Makefile b/drivers/gpu/drm/sysfb/Makefile index 861b4026f4a6e..a156c496413d3 100644 --- a/drivers/gpu/drm/sysfb/Makefile +++ b/drivers/gpu/drm/sysfb/Makefile @@ -3,6 +3,7 @@ drm_sysfb_helper-y := \ drm_sysfb.o \ drm_sysfb_modeset.o +drm_sysfb_helper-$(CONFIG_SCREEN_INFO) += drm_sysfb_screen_info.o obj-$(CONFIG_DRM_SYSFB_HELPER) += drm_sysfb_helper.o obj-$(CONFIG_DRM_EFIDRM) += efidrm.o diff --git a/drivers/gpu/drm/sysfb/drm_sysfb_helper.h b/drivers/gpu/drm/sysfb/drm_sysfb_helper.h index 1697cf7ace973..cb08a88242cc1 100644 --- a/drivers/gpu/drm/sysfb/drm_sysfb_helper.h +++ b/drivers/gpu/drm/sysfb/drm_sysfb_helper.h @@ -6,12 +6,46 @@ #include <linux/container_of.h> #include <linux/iosys-map.h> +#include <video/pixel_format.h> + #include <drm/drm_crtc.h> #include <drm/drm_device.h> #include <drm/drm_modes.h> struct drm_format_info; struct drm_scanout_buffer; +struct screen_info; + +/* + * Input parsing + */ + +struct drm_sysfb_format { + struct pixel_format pixel; + u32 fourcc; +}; + +int drm_sysfb_get_validated_int(struct drm_device *dev, const char *name, + u64 value, u32 max); +int drm_sysfb_get_validated_int0(struct drm_device *dev, const char *name, + u64 value, u32 max); + +#if defined(CONFIG_SCREEN_INFO) +int drm_sysfb_get_width_si(struct drm_device *dev, const struct screen_info *si); +int drm_sysfb_get_height_si(struct drm_device *dev, const struct screen_info *si); +struct resource *drm_sysfb_get_memory_si(struct drm_device *dev, + const struct screen_info *si, + struct resource *res); +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 drm_sysfb_get_visible_size_si(struct drm_device *dev, const struct screen_info *si, + unsigned int height, unsigned int stride, u64 size); +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); +#endif /* * Input parsing diff --git a/drivers/gpu/drm/sysfb/drm_sysfb_screen_info.c b/drivers/gpu/drm/sysfb/drm_sysfb_screen_info.c new file mode 100644 index 0000000000000..0b3fb874a51f2 --- /dev/null +++ b/drivers/gpu/drm/sysfb/drm_sysfb_screen_info.c @@ -0,0 +1,107 @@ +// 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; + u32 bits_per_pixel; + size_t i; + + bits_per_pixel = __screen_info_lfb_bits_per_pixel(si); + + for (i = 0; i < nformats; ++i) { + const struct pixel_format *f = &formats[i].pixel; + + if (bits_per_pixel == f->bits_per_pixel && + si->red_size == f->red.length && + si->red_pos == f->red.offset && + si->green_size == f->green.length && + si->green_pos == f->green.offset && + si->blue_size == f->blue.length && + si->blue_pos == f->blue.offset) { + format = drm_format_info(formats[i].fourcc); + break; + } + } + + if (!format) + drm_warn(dev, "No compatible color format found\n"); + + return format; +} +EXPORT_SYMBOL(drm_sysfb_get_format_si); diff --git a/drivers/gpu/drm/sysfb/efidrm.c b/drivers/gpu/drm/sysfb/efidrm.c index a77ea5285cc1d..07eb494203a25 100644 --- a/drivers/gpu/drm/sysfb/efidrm.c +++ b/drivers/gpu/drm/sysfb/efidrm.c @@ -33,72 +33,10 @@ #define DRIVER_MAJOR 1 #define DRIVER_MINOR 0 -static s64 efidrm_get_validated_size0(struct drm_device *dev, const char *name, - u64 value, u64 max) -{ - if (!value) { - drm_err(dev, "%s of 0 not allowed\n", name); - return -EINVAL; - } else if (value > max) { - drm_err(dev, "%s of %llu exceeds maximum of %llu\n", name, value, max); - return -EINVAL; - } - return value; -} - -static int efidrm_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); -} - -static int efidrm_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); -} - -static struct resource *efidrm_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_err(dev, "memory resource not found\n"); - return NULL; - } - - return res; -} - -static int efidrm_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)); -} - -static u64 efidrm_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 efidrm_get_validated_size0(dev, "visible size", vsize, size); -} - static const struct drm_format_info *efidrm_get_format_si(struct drm_device *dev, const struct screen_info *si) { - static const struct { - struct pixel_format pixel; - u32 fourcc; - } efi_formats[] = { + static const struct drm_sysfb_format formats[] = { { PIXEL_FORMAT_XRGB1555, DRM_FORMAT_XRGB1555, }, { PIXEL_FORMAT_RGB565, DRM_FORMAT_RGB565, }, { PIXEL_FORMAT_RGB888, DRM_FORMAT_RGB888, }, @@ -106,33 +44,8 @@ static const struct drm_format_info *efidrm_get_format_si(struct drm_device *dev { PIXEL_FORMAT_XBGR8888, DRM_FORMAT_XBGR8888, }, { PIXEL_FORMAT_XRGB2101010, DRM_FORMAT_XRGB2101010, }, }; - const struct drm_format_info *format = NULL; - u32 bits_per_pixel; - size_t i; - - bits_per_pixel = __screen_info_lfb_bits_per_pixel(si); - - for (i = 0; i < ARRAY_SIZE(efi_formats); ++i) { - const struct pixel_format *f = &efi_formats[i].pixel; - - if (bits_per_pixel == f->bits_per_pixel && - si->red_size == f->red.length && - si->red_pos == f->red.offset && - si->green_size == f->green.length && - si->green_pos == f->green.offset && - si->blue_size == f->blue.length && - si->blue_pos == f->blue.offset) { - format = drm_format_info(efi_formats[i].fourcc); - break; - } - } - - if (!format) - return ERR_PTR(-EINVAL); - if (format->is_color_indexed) - return ERR_PTR(-EINVAL); - return format; + return drm_sysfb_get_format_si(dev, formats, ARRAY_SIZE(formats), si); } static u64 efidrm_get_mem_flags(struct drm_device *dev, resource_size_t start, @@ -268,21 +181,21 @@ static struct efidrm_device *efidrm_device_create(struct drm_driver *drv, */ format = efidrm_get_format_si(dev, si); - if (IS_ERR(format)) - return ERR_CAST(format); - width = efidrm_get_width_si(dev, si); + if (!format) + return ERR_PTR(-EINVAL); + width = drm_sysfb_get_width_si(dev, si); if (width < 0) return ERR_PTR(width); - height = efidrm_get_height_si(dev, si); + height = drm_sysfb_get_height_si(dev, si); if (height < 0) return ERR_PTR(height); - res = efidrm_get_memory_si(dev, si, &resbuf); + res = drm_sysfb_get_memory_si(dev, si, &resbuf); if (!res) return ERR_PTR(-EINVAL); - stride = efidrm_get_stride_si(dev, si, format, width, height, resource_size(res)); + stride = drm_sysfb_get_stride_si(dev, si, format, width, height, resource_size(res)); if (stride < 0) return ERR_PTR(stride); - vsize = efidrm_get_visible_size_si(dev, si, height, stride, resource_size(res)); + vsize = drm_sysfb_get_visible_size_si(dev, si, height, stride, resource_size(res)); if (!vsize) return ERR_PTR(-EINVAL); diff --git a/drivers/gpu/drm/sysfb/vesadrm.c b/drivers/gpu/drm/sysfb/vesadrm.c index d87ff77be20de..4d62c78e7d1e0 100644 --- a/drivers/gpu/drm/sysfb/vesadrm.c +++ b/drivers/gpu/drm/sysfb/vesadrm.c @@ -36,105 +36,18 @@ #define VESADRM_GAMMA_LUT_SIZE 256 -static s64 vesadrm_get_validated_size0(struct drm_device *dev, const char *name, - u64 value, u64 max) -{ - if (!value) { - drm_err(dev, "vesadrm: %s of 0 not allowed\n", name); - return -EINVAL; - } else if (value > max) { - drm_err(dev, "vesadrm: %s of %llu exceeds maximum of %llu\n", name, value, max); - return -EINVAL; - } - return value; -} - -static int vesadrm_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); -} - -static int vesadrm_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); -} - -static struct resource *vesadrm_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_err(dev, "vesadrm: memory resource not found\n"); - return NULL; - } - - return res; -} - -static int vesadrm_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)); -} - -static u64 vesadrm_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 vesadrm_get_validated_size0(dev, "visible size", vsize, size); -} - static const struct drm_format_info *vesadrm_get_format_si(struct drm_device *dev, const struct screen_info *si) { - static const struct { - struct pixel_format pixel; - u32 fourcc; - } vesa_formats[] = { + static const struct drm_sysfb_format formats[] = { { PIXEL_FORMAT_XRGB1555, DRM_FORMAT_XRGB1555, }, { PIXEL_FORMAT_RGB565, DRM_FORMAT_RGB565, }, { PIXEL_FORMAT_RGB888, DRM_FORMAT_RGB888, }, { PIXEL_FORMAT_XRGB8888, DRM_FORMAT_XRGB8888, }, { PIXEL_FORMAT_XBGR8888, DRM_FORMAT_XBGR8888, }, }; - const struct drm_format_info *format = NULL; - u32 bits_per_pixel; - size_t i; - - bits_per_pixel = __screen_info_lfb_bits_per_pixel(si); - - for (i = 0; i < ARRAY_SIZE(vesa_formats); ++i) { - const struct pixel_format *f = &vesa_formats[i].pixel; - if (bits_per_pixel == f->bits_per_pixel && - si->red_size == f->red.length && - si->red_pos == f->red.offset && - si->green_size == f->green.length && - si->green_pos == f->green.offset && - si->blue_size == f->blue.length && - si->blue_pos == f->blue.offset) { - format = drm_format_info(vesa_formats[i].fourcc); - break; - } - } - - if (!format) - return ERR_PTR(-EINVAL); - if (format->is_color_indexed) - return ERR_PTR(-EINVAL); - - return format; + return drm_sysfb_get_format_si(dev, formats, ARRAY_SIZE(formats), si); } /* @@ -426,21 +339,21 @@ static struct vesadrm_device *vesadrm_device_create(struct drm_driver *drv, */ format = vesadrm_get_format_si(dev, si); - if (IS_ERR(format)) - return ERR_CAST(format); - width = vesadrm_get_width_si(dev, si); + if (!format) + return ERR_PTR(-EINVAL); + width = drm_sysfb_get_width_si(dev, si); if (width < 0) return ERR_PTR(width); - height = vesadrm_get_height_si(dev, si); + height = drm_sysfb_get_height_si(dev, si); if (height < 0) return ERR_PTR(height); - res = vesadrm_get_memory_si(dev, si, &resbuf); + res = drm_sysfb_get_memory_si(dev, si, &resbuf); if (!res) return ERR_PTR(-EINVAL); - stride = vesadrm_get_stride_si(dev, si, format, width, height, resource_size(res)); + stride = drm_sysfb_get_stride_si(dev, si, format, width, height, resource_size(res)); if (stride < 0) return ERR_PTR(stride); - vsize = vesadrm_get_visible_size_si(dev, si, height, stride, resource_size(res)); + vsize = drm_sysfb_get_visible_size_si(dev, si, height, stride, resource_size(res)); if (!vsize) return ERR_PTR(-EINVAL); -- 2.49.0