| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237 |
- // SPDX-License-Identifier: GPL-2.0
- #include <linux/export.h>
- #include <linux/ioport.h>
- #include <linux/screen_info.h>
- #include <linux/string.h>
- #include <video/pixel_format.h>
- static void resource_init_named(struct resource *r,
- resource_size_t start, resource_size_t size,
- const char *name, unsigned int flags)
- {
- memset(r, 0, sizeof(*r));
- r->start = start;
- r->end = start + size - 1;
- r->name = name;
- r->flags = flags;
- }
- static void resource_init_io_named(struct resource *r,
- resource_size_t start, resource_size_t size,
- const char *name)
- {
- resource_init_named(r, start, size, name, IORESOURCE_IO);
- }
- static void resource_init_mem_named(struct resource *r,
- resource_size_t start, resource_size_t size,
- const char *name)
- {
- resource_init_named(r, start, size, name, IORESOURCE_MEM);
- }
- static inline bool __screen_info_has_ega_gfx(unsigned int mode)
- {
- switch (mode) {
- case 0x0d: /* 320x200-4 */
- case 0x0e: /* 640x200-4 */
- case 0x0f: /* 640x350-1 */
- case 0x10: /* 640x350-4 */
- return true;
- default:
- return false;
- }
- }
- static inline bool __screen_info_has_vga_gfx(unsigned int mode)
- {
- switch (mode) {
- case 0x10: /* 640x480-1 */
- case 0x12: /* 640x480-4 */
- case 0x13: /* 320-200-8 */
- case 0x6a: /* 800x600-4 (VESA) */
- return true;
- default:
- return __screen_info_has_ega_gfx(mode);
- }
- }
- /**
- * screen_info_resources() - Get resources from screen_info structure
- * @si: the screen_info
- * @r: pointer to an array of resource structures
- * @num: number of elements in @r:
- *
- * Returns:
- * The number of resources stored in @r on success, or a negative errno code otherwise.
- *
- * A call to screen_info_resources() returns the resources consumed by the
- * screen_info's device or framebuffer. The result is stored in the caller-supplied
- * array @r with up to @num elements. The function returns the number of
- * initialized elements.
- */
- ssize_t screen_info_resources(const struct screen_info *si, struct resource *r, size_t num)
- {
- struct resource *pos = r;
- unsigned int type = screen_info_video_type(si);
- u64 base, size;
- switch (type) {
- case VIDEO_TYPE_MDA:
- if (num > 0)
- resource_init_io_named(pos++, 0x3b0, 12, "mda");
- if (num > 1)
- resource_init_io_named(pos++, 0x3bf, 0x01, "mda");
- if (num > 2)
- resource_init_mem_named(pos++, 0xb0000, 0x2000, "mda");
- break;
- case VIDEO_TYPE_CGA:
- if (num > 0)
- resource_init_io_named(pos++, 0x3d4, 0x02, "cga");
- if (num > 1)
- resource_init_mem_named(pos++, 0xb8000, 0x2000, "cga");
- break;
- case VIDEO_TYPE_EGAM:
- if (num > 0)
- resource_init_io_named(pos++, 0x3bf, 0x10, "ega");
- if (num > 1)
- resource_init_mem_named(pos++, 0xb0000, 0x8000, "ega");
- break;
- case VIDEO_TYPE_EGAC:
- if (num > 0)
- resource_init_io_named(pos++, 0x3c0, 0x20, "ega");
- if (num > 1) {
- if (__screen_info_has_ega_gfx(si->orig_video_mode))
- resource_init_mem_named(pos++, 0xa0000, 0x10000, "ega");
- else
- resource_init_mem_named(pos++, 0xb8000, 0x8000, "ega");
- }
- break;
- case VIDEO_TYPE_VGAC:
- if (num > 0)
- resource_init_io_named(pos++, 0x3c0, 0x20, "vga+");
- if (num > 1) {
- if (__screen_info_has_vga_gfx(si->orig_video_mode))
- resource_init_mem_named(pos++, 0xa0000, 0x10000, "vga+");
- else
- resource_init_mem_named(pos++, 0xb8000, 0x8000, "vga+");
- }
- break;
- case VIDEO_TYPE_VLFB:
- case VIDEO_TYPE_EFI:
- base = __screen_info_lfb_base(si);
- if (!base)
- break;
- size = __screen_info_lfb_size(si, type);
- if (!size)
- break;
- if (num > 0)
- resource_init_mem_named(pos++, base, size, "lfb");
- break;
- case VIDEO_TYPE_PICA_S3:
- case VIDEO_TYPE_MIPS_G364:
- case VIDEO_TYPE_SGI:
- case VIDEO_TYPE_TGAC:
- case VIDEO_TYPE_SUN:
- case VIDEO_TYPE_SUNPCI:
- case VIDEO_TYPE_PMAC:
- default:
- /* not supported */
- return -EINVAL;
- }
- return pos - r;
- }
- EXPORT_SYMBOL(screen_info_resources);
- /*
- * The meaning of depth and bpp for direct-color formats is
- * inconsistent:
- *
- * - DRM format info specifies depth as the number of color
- * bits; including alpha, but not including filler bits.
- * - Linux' EFI platform code computes lfb_depth from the
- * individual color channels, including the reserved bits.
- * - VBE 1.1 defines lfb_depth for XRGB1555 as 16, but later
- * versions use 15.
- * - On the kernel command line, 'bpp' of 32 is usually
- * XRGB8888 including the filler bits, but 15 is XRGB1555
- * not including the filler bit.
- *
- * It is not easily possible to fix this in struct screen_info,
- * as this could break UAPI. The best solution is to compute
- * bits_per_pixel from the color bits, reserved bits and
- * reported lfb_depth, whichever is highest.
- */
- u32 __screen_info_lfb_bits_per_pixel(const struct screen_info *si)
- {
- u32 bits_per_pixel = si->lfb_depth;
- if (bits_per_pixel > 8) {
- bits_per_pixel = max(max3(si->red_size + si->red_pos,
- si->green_size + si->green_pos,
- si->blue_size + si->blue_pos),
- si->rsvd_size + si->rsvd_pos);
- bits_per_pixel = max_t(u32, bits_per_pixel, si->lfb_depth);
- }
- return bits_per_pixel;
- }
- EXPORT_SYMBOL(__screen_info_lfb_bits_per_pixel);
- static int __screen_info_lfb_pixel_format(const struct screen_info *si, struct pixel_format *f)
- {
- u32 bits_per_pixel = __screen_info_lfb_bits_per_pixel(si);
- if (bits_per_pixel > U8_MAX)
- return -EINVAL;
- f->bits_per_pixel = bits_per_pixel;
- if (si->lfb_depth > 8) {
- f->indexed = false;
- f->alpha.offset = 0;
- f->alpha.length = 0;
- f->red.offset = si->red_pos;
- f->red.length = si->red_size;
- f->green.offset = si->green_pos;
- f->green.length = si->green_size;
- f->blue.offset = si->blue_pos;
- f->blue.length = si->blue_size;
- } else {
- f->indexed = true;
- f->index.offset = 0;
- f->index.length = si->lfb_depth;
- }
- return 0;
- }
- /**
- * screen_info_pixel_format - Returns the screen-info format as pixel-format description
- *
- * @si: the screen_info
- * @f: pointer to return pixel-format description
- *
- * Returns:
- * 0 on success, or a negative errno code otherwise.
- */
- int screen_info_pixel_format(const struct screen_info *si, struct pixel_format *f)
- {
- unsigned int type = screen_info_video_type(si);
- /* TODO: Add support for additional types as needed. */
- switch (type) {
- case VIDEO_TYPE_VLFB:
- case VIDEO_TYPE_EFI:
- return __screen_info_lfb_pixel_format(si, f);
- }
- /* not supported */
- return -EINVAL;
- }
- EXPORT_SYMBOL(screen_info_pixel_format);
|