efigop.c revision 1.1 1 1.1 jmcneill /* $NetBSD: efigop.c,v 1.1 2021/09/28 11:38:07 jmcneill Exp $ */
2 1.1 jmcneill
3 1.1 jmcneill /*-
4 1.1 jmcneill * Copyright (c) 2021 Jared McNeill <jmcneill (at) invisible.ca>
5 1.1 jmcneill * All rights reserved.
6 1.1 jmcneill *
7 1.1 jmcneill * Redistribution and use in source and binary forms, with or without
8 1.1 jmcneill * modification, are permitted provided that the following conditions
9 1.1 jmcneill * are met:
10 1.1 jmcneill * 1. Redistributions of source code must retain the above copyright
11 1.1 jmcneill * notice, this list of conditions and the following disclaimer.
12 1.1 jmcneill * 2. Redistributions in binary form must reproduce the above copyright
13 1.1 jmcneill * notice, this list of conditions and the following disclaimer in the
14 1.1 jmcneill * documentation and/or other materials provided with the distribution.
15 1.1 jmcneill *
16 1.1 jmcneill * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
17 1.1 jmcneill * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 1.1 jmcneill * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 1.1 jmcneill * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
20 1.1 jmcneill * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 1.1 jmcneill * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22 1.1 jmcneill * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 1.1 jmcneill * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 1.1 jmcneill * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 1.1 jmcneill * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 1.1 jmcneill * SUCH DAMAGE.
27 1.1 jmcneill */
28 1.1 jmcneill
29 1.1 jmcneill #include "efiboot.h"
30 1.1 jmcneill
31 1.1 jmcneill #include <libfdt.h>
32 1.1 jmcneill
33 1.1 jmcneill static EFI_GRAPHICS_OUTPUT_PROTOCOL *gop = NULL;
34 1.1 jmcneill
35 1.1 jmcneill void
36 1.1 jmcneill efi_gop_probe(void)
37 1.1 jmcneill {
38 1.1 jmcneill EFI_HANDLE *gop_handle;
39 1.1 jmcneill UINTN ngop_handle;
40 1.1 jmcneill EFI_STATUS status;
41 1.1 jmcneill
42 1.1 jmcneill status = LibLocateHandle(ByProtocol, &GraphicsOutputProtocol, NULL,
43 1.1 jmcneill &ngop_handle, &gop_handle);
44 1.1 jmcneill if (EFI_ERROR(status) || ngop_handle == 0) {
45 1.1 jmcneill return;
46 1.1 jmcneill }
47 1.1 jmcneill
48 1.1 jmcneill for (size_t n = 0; n < ngop_handle; n++) {
49 1.1 jmcneill status = uefi_call_wrapper(BS->HandleProtocol, 3,
50 1.1 jmcneill gop_handle[n], &GraphicsOutputProtocol, (void **)&gop);
51 1.1 jmcneill if (EFI_ERROR(status) || gop->Mode == NULL) {
52 1.1 jmcneill gop = NULL;
53 1.1 jmcneill continue;
54 1.1 jmcneill } else {
55 1.1 jmcneill break;
56 1.1 jmcneill }
57 1.1 jmcneill }
58 1.1 jmcneill }
59 1.1 jmcneill
60 1.1 jmcneill static uint32_t
61 1.1 jmcneill efi_gop_bpp(EFI_GRAPHICS_OUTPUT_MODE_INFORMATION *info)
62 1.1 jmcneill {
63 1.1 jmcneill if (info->PixelFormat == PixelRedGreenBlueReserved8BitPerColor ||
64 1.1 jmcneill info->PixelFormat == PixelBlueGreenRedReserved8BitPerColor) {
65 1.1 jmcneill return 24;
66 1.1 jmcneill }
67 1.1 jmcneill
68 1.1 jmcneill return popcount32(info->PixelInformation.RedMask) +
69 1.1 jmcneill popcount32(info->PixelInformation.GreenMask) +
70 1.1 jmcneill popcount32(info->PixelInformation.BlueMask);
71 1.1 jmcneill }
72 1.1 jmcneill
73 1.1 jmcneill static void
74 1.1 jmcneill efi_gop_printmode(UINT32 mode, EFI_GRAPHICS_OUTPUT_MODE_INFORMATION *info)
75 1.1 jmcneill {
76 1.1 jmcneill char buf[sizeof("NNNNN: HHHHHxVVVVV BB")];
77 1.1 jmcneill
78 1.1 jmcneill snprintf(buf, sizeof(buf), "%-5u: %ux%u %u", mode,
79 1.1 jmcneill info->HorizontalResolution, info->VerticalResolution,
80 1.1 jmcneill efi_gop_bpp(info));
81 1.1 jmcneill
82 1.1 jmcneill printf("%-21s", buf);
83 1.1 jmcneill }
84 1.1 jmcneill
85 1.1 jmcneill void
86 1.1 jmcneill efi_gop_show(void)
87 1.1 jmcneill {
88 1.1 jmcneill if (gop == NULL) {
89 1.1 jmcneill return;
90 1.1 jmcneill }
91 1.1 jmcneill
92 1.1 jmcneill printf("GOP: ");
93 1.1 jmcneill efi_gop_printmode(gop->Mode->Mode, gop->Mode->Info);
94 1.1 jmcneill printf("\n");
95 1.1 jmcneill }
96 1.1 jmcneill
97 1.1 jmcneill void
98 1.1 jmcneill efi_gop_dump(void)
99 1.1 jmcneill {
100 1.1 jmcneill EFI_STATUS status;
101 1.1 jmcneill EFI_GRAPHICS_OUTPUT_MODE_INFORMATION *info;
102 1.1 jmcneill UINTN size;
103 1.1 jmcneill
104 1.1 jmcneill if (gop == NULL) {
105 1.1 jmcneill return;
106 1.1 jmcneill }
107 1.1 jmcneill
108 1.1 jmcneill for (UINT32 mode = 0; mode < gop->Mode->MaxMode; mode++) {
109 1.1 jmcneill status = uefi_call_wrapper(gop->QueryMode, 4, gop, mode,
110 1.1 jmcneill &size, &info);
111 1.1 jmcneill if (EFI_ERROR(status)) {
112 1.1 jmcneill continue;
113 1.1 jmcneill }
114 1.1 jmcneill if (mode == gop->Mode->Mode) {
115 1.1 jmcneill printf(" -> ");
116 1.1 jmcneill } else {
117 1.1 jmcneill printf(" ");
118 1.1 jmcneill }
119 1.1 jmcneill efi_gop_printmode(mode, info);
120 1.1 jmcneill if (mode != gop->Mode->MaxMode - 1 &&
121 1.1 jmcneill mode % 3 == 2) {
122 1.1 jmcneill printf("\n");
123 1.1 jmcneill }
124 1.1 jmcneill }
125 1.1 jmcneill printf("\n");
126 1.1 jmcneill }
127 1.1 jmcneill
128 1.1 jmcneill void
129 1.1 jmcneill efi_gop_setmode(UINT32 mode)
130 1.1 jmcneill {
131 1.1 jmcneill EFI_STATUS status;
132 1.1 jmcneill
133 1.1 jmcneill if (gop == NULL) {
134 1.1 jmcneill return;
135 1.1 jmcneill }
136 1.1 jmcneill
137 1.1 jmcneill status = uefi_call_wrapper(gop->SetMode, 2, gop, mode);
138 1.1 jmcneill if (EFI_ERROR(status)) {
139 1.1 jmcneill printf("Failed to set video mode: %ld\n", (long)status);
140 1.1 jmcneill }
141 1.1 jmcneill }
142