Patrick Georgi | ac95903 | 2020-05-05 22:49:26 +0200 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0-or-later */ |
Alexandru Gagniuc | fb22ff4 | 2016-02-25 14:22:03 -0800 | [diff] [blame] | 2 | |
Elyes HAOUAS | 0edf6a5 | 2019-10-26 18:41:47 +0200 | [diff] [blame] | 3 | #include <boot/coreboot_tables.h> |
Alexandru Gagniuc | fb22ff4 | 2016-02-25 14:22:03 -0800 | [diff] [blame] | 4 | #include <console/console.h> |
Patrick Rudolph | 92106b1 | 2020-02-19 12:54:06 +0100 | [diff] [blame] | 5 | #include <fsp/graphics.h> |
Alexandru Gagniuc | fb22ff4 | 2016-02-25 14:22:03 -0800 | [diff] [blame] | 6 | #include <fsp/util.h> |
Patrick Georgi | 8269096 | 2017-09-27 15:24:58 +0200 | [diff] [blame] | 7 | #include <soc/intel/common/vbt.h> |
Elyes HAOUAS | bd1683d | 2019-05-15 21:05:37 +0200 | [diff] [blame] | 8 | #include <types.h> |
Patrick Rudolph | 92106b1 | 2020-02-19 12:54:06 +0100 | [diff] [blame] | 9 | #include <framebuffer_info.h> |
Alexandru Gagniuc | fb22ff4 | 2016-02-25 14:22:03 -0800 | [diff] [blame] | 10 | |
| 11 | enum pixel_format { |
| 12 | pixel_rgbx_8bpc = 0, |
| 13 | pixel_bgrx_8bpc = 1, |
| 14 | pixel_bitmask = 2, /* defined by <rgb>_mask values */ |
| 15 | }; |
| 16 | |
Lee Leahy | 5a9ca4d | 2016-09-28 17:15:00 -0700 | [diff] [blame] | 17 | static const uint8_t fsp_graphics_info_guid[16] = { |
Alexandru Gagniuc | fb22ff4 | 2016-02-25 14:22:03 -0800 | [diff] [blame] | 18 | 0xce, 0x2c, 0xf6, 0x39, 0x25, 0x68, 0x69, 0x46, |
| 19 | 0xbb, 0x56, 0x54, 0x1a, 0xba, 0x75, 0x3a, 0x07 |
| 20 | }; |
| 21 | |
| 22 | struct hob_graphics_info { |
| 23 | uint64_t framebuffer_base; |
| 24 | uint32_t framebuffer_size; |
| 25 | uint32_t version; |
| 26 | uint32_t horizontal_resolution; |
| 27 | uint32_t vertical_resolution; |
| 28 | uint32_t pixel_format; /* See enum pixel_format */ |
| 29 | uint32_t red_mask; |
| 30 | uint32_t green_mask; |
| 31 | uint32_t blue_mask; |
| 32 | uint32_t reserved_mask; |
| 33 | uint32_t pixels_per_scanline; |
Stefan Reinauer | 6a00113 | 2017-07-13 02:20:27 +0200 | [diff] [blame] | 34 | } __packed; |
Alexandru Gagniuc | fb22ff4 | 2016-02-25 14:22:03 -0800 | [diff] [blame] | 35 | |
| 36 | struct pixel { |
| 37 | uint8_t pos; |
| 38 | uint8_t size; |
| 39 | }; |
| 40 | |
| 41 | static const struct fsp_framebuffer { |
| 42 | struct pixel red; |
| 43 | struct pixel green; |
| 44 | struct pixel blue; |
| 45 | struct pixel rsvd; |
| 46 | } fsp_framebuffer_format_map[] = { |
| 47 | [pixel_rgbx_8bpc] = { {0, 8}, {8, 8}, {16, 8}, {24, 8} }, |
| 48 | [pixel_bgrx_8bpc] = { {16, 8}, {8, 8}, {0, 8}, {24, 8} }, |
| 49 | }; |
| 50 | |
Patrick Rudolph | 92106b1 | 2020-02-19 12:54:06 +0100 | [diff] [blame] | 51 | |
Tim Wawrzynczak | 84428f7 | 2021-09-14 13:59:33 -0600 | [diff] [blame] | 52 | void fsp_report_framebuffer_info(const uintptr_t framebuffer_bar, |
| 53 | enum lb_fb_orientation orientation) |
Alexandru Gagniuc | fb22ff4 | 2016-02-25 14:22:03 -0800 | [diff] [blame] | 54 | { |
| 55 | size_t size; |
| 56 | const struct hob_graphics_info *ginfo; |
| 57 | const struct fsp_framebuffer *fbinfo; |
| 58 | |
Patrick Rudolph | 92106b1 | 2020-02-19 12:54:06 +0100 | [diff] [blame] | 59 | /* |
| 60 | * Pci enumeration happens after silicon init. |
| 61 | * After enumeration graphic framebuffer base may be relocated. |
| 62 | */ |
| 63 | if (!framebuffer_bar) { |
| 64 | printk(BIOS_ALERT, "Framebuffer BAR invalid\n"); |
| 65 | return; |
| 66 | } |
| 67 | |
Lee Leahy | ac3b0a6 | 2016-07-27 07:40:25 -0700 | [diff] [blame] | 68 | ginfo = fsp_find_extension_hob_by_guid(fsp_graphics_info_guid, &size); |
Alexandru Gagniuc | fb22ff4 | 2016-02-25 14:22:03 -0800 | [diff] [blame] | 69 | |
| 70 | if (!ginfo) { |
| 71 | printk(BIOS_ALERT, "Graphics hand-off block not found\n"); |
Patrick Rudolph | 92106b1 | 2020-02-19 12:54:06 +0100 | [diff] [blame] | 72 | return; |
Alexandru Gagniuc | fb22ff4 | 2016-02-25 14:22:03 -0800 | [diff] [blame] | 73 | } |
| 74 | |
Lee Leahy | 016d8f7 | 2016-05-17 08:40:02 -0700 | [diff] [blame] | 75 | if (ginfo->pixel_format >= ARRAY_SIZE(fsp_framebuffer_format_map)) { |
Alexandru Gagniuc | fb22ff4 | 2016-02-25 14:22:03 -0800 | [diff] [blame] | 76 | printk(BIOS_ALERT, "FSP set unknown framebuffer format: %d\n", |
| 77 | ginfo->pixel_format); |
Patrick Rudolph | 92106b1 | 2020-02-19 12:54:06 +0100 | [diff] [blame] | 78 | return; |
Alexandru Gagniuc | fb22ff4 | 2016-02-25 14:22:03 -0800 | [diff] [blame] | 79 | } |
| 80 | |
| 81 | fbinfo = fsp_framebuffer_format_map + ginfo->pixel_format; |
| 82 | |
Patrick Rudolph | 92106b1 | 2020-02-19 12:54:06 +0100 | [diff] [blame] | 83 | const struct lb_framebuffer fb = { |
| 84 | .physical_address = framebuffer_bar, |
| 85 | .x_resolution = ginfo->horizontal_resolution, |
| 86 | .y_resolution = ginfo->vertical_resolution, |
| 87 | .bytes_per_line = ginfo->pixels_per_scanline * 4, |
| 88 | .bits_per_pixel = fbinfo->rsvd.size + fbinfo->red.size + |
| 89 | fbinfo->green.size + fbinfo->blue.size, |
| 90 | .red_mask_pos = fbinfo->red.pos, |
| 91 | .red_mask_size = fbinfo->red.size, |
| 92 | .green_mask_pos = fbinfo->green.pos, |
| 93 | .green_mask_size = fbinfo->green.size, |
| 94 | .blue_mask_pos = fbinfo->blue.pos, |
| 95 | .blue_mask_size = fbinfo->blue.size, |
| 96 | .reserved_mask_pos = fbinfo->rsvd.pos, |
| 97 | .reserved_mask_size = fbinfo->rsvd.size, |
Tim Wawrzynczak | 84428f7 | 2021-09-14 13:59:33 -0600 | [diff] [blame] | 98 | .orientation = orientation, |
Patrick Rudolph | 92106b1 | 2020-02-19 12:54:06 +0100 | [diff] [blame] | 99 | }; |
Aaron Durbin | bdb5c8f | 2017-05-16 21:39:50 -0500 | [diff] [blame] | 100 | |
Patrick Rudolph | 92106b1 | 2020-02-19 12:54:06 +0100 | [diff] [blame] | 101 | fb_add_framebuffer_info_ex(&fb); |
Lee Leahy | 5a9ca4d | 2016-09-28 17:15:00 -0700 | [diff] [blame] | 102 | } |