blob: 2a033a028b63c0e44dac3cfbebc82853157a85f9 [file] [log] [blame]
Patrick Georgi593124d2020-05-10 19:44:08 +02001/* SPDX-License-Identifier: MIT */
Patrick Georgi16849bb2020-05-10 17:52:40 +02002/*
3 * Copied from Linux drivers/gpu/drm/ast/ast_mode.c
4 */
Patrick Rudolphf1a4ae02019-09-30 11:02:04 +02005#include <edid.h>
6
7#include "ast_drv.h"
8
9/*
10 * Set framebuffer MMIO address, which must fall into BAR0 MMIO window.
11 *
12 * Complete reimplementation as the original expects multiple kernel internal
13 * subsystems to be present.
14 */
15int ast_crtc_do_set_base(struct drm_crtc *crtc)
16{
17 struct ast_private *ast = crtc->dev->dev_private;
18 struct drm_framebuffer *fb = crtc->primary->fb;
19
20 /* PCI BAR 0 */
21 struct resource *res = find_resource(crtc->dev->pdev, 0x10);
22 if (!res) {
23 printk(BIOS_ERR, "BAR0 resource not found.\n");
24 return -EIO;
25 }
26
27 if (res->size < fb->pitches[0] * crtc->mode.vdisplay) {
28 dev_err(dev->pdev, "Framebuffer doesn't fit into BAR0 MMIO window\n");
29 return -ENOMEM;
30 }
31
32 fb->mmio_addr = (u32)res2mmio(res, 4095, 4095);
33
34 ast_set_offset_reg(crtc);
35 ast_set_start_address_crt1(ast, fb->mmio_addr);
36
37 return 0;
38}
39
40static void ast_edid_to_drmmode(struct edid *edid, struct drm_display_mode *mode)
41{
42 memset(mode, 0, sizeof(*mode));
43
44 mode->hdisplay = edid->mode.ha;
45 mode->vdisplay = edid->mode.va;
46 mode->crtc_hdisplay = edid->mode.ha;
47 mode->crtc_vdisplay = edid->mode.va;
48
49 /* EDID clock is in 10kHz, but drm clock is in KHz */
50 mode->clock = edid->mode.pixel_clock * 10;
51 mode->vrefresh = edid->mode.refresh;
52
53 mode->crtc_hblank_start = edid->mode.ha;
54 mode->crtc_hblank_end = edid->mode.ha + edid->mode.hbl;
55 mode->crtc_hsync_start = edid->mode.ha + edid->mode.hso;
56 mode->crtc_hsync_end = edid->mode.ha + edid->mode.hso + edid->mode.hspw;
57 mode->crtc_htotal = mode->crtc_hblank_end;
58
59 mode->crtc_vblank_start = edid->mode.va;
60 mode->crtc_vblank_end = edid->mode.va + edid->mode.vbl;
61 mode->crtc_vsync_start = edid->mode.va + edid->mode.vso;
62 mode->crtc_vsync_end = edid->mode.va + edid->mode.vso + edid->mode.vspw;
63 mode->crtc_vtotal = mode->crtc_vblank_end;
64
65 mode->flags = 0;
66 if (edid->mode.phsync == '+')
67 mode->flags |= DRM_MODE_FLAG_PHSYNC;
68 else
69 mode->flags |= DRM_MODE_FLAG_NHSYNC;
70
71 if (edid->mode.pvsync == '+')
72 mode->flags |= DRM_MODE_FLAG_PVSYNC;
73 else
74 mode->flags |= DRM_MODE_FLAG_NVSYNC;
75}
76
77static int ast_select_mode(struct drm_connector *connector,
78 struct edid *edid)
79{
80 struct ast_private *ast = connector->dev->dev_private;
81 bool widescreen;
82 u8 raw[128];
83 bool flags = false;
84
85 if (ast->tx_chip_type == AST_TX_DP501) {
86 ast->dp501_maxclk = 0xff;
87 flags = ast_dp501_read_edid(connector->dev, (u8 *)raw);
88 if (flags)
89 ast->dp501_maxclk = ast_get_dp501_max_clk(connector->dev);
90 else
91 dev_err(dev->pdev, "I2C transmission error\n");
92 }
93
94 if (!flags)
95 ast_software_i2c_read(ast, raw);
96
97 if (decode_edid(raw, sizeof(raw), edid) != EDID_CONFORMANT) {
98 dev_err(dev->pdev, "Failed to decode EDID\n");
99 printk(BIOS_DEBUG, "Assuming VGA for KVM\n");
100
101 memset(edid, 0, sizeof(*edid));
102
103 edid->mode.pixel_clock = 6411;
104 edid->mode.refresh = 60;
105 edid->mode.ha = 1024;
106 edid->mode.hspw = 4;
107 edid->mode.hso = 56;
108 edid->mode.hbl = 264;
109 edid->mode.phsync = '-';
110
111 edid->mode.va = 768;
112 edid->mode.vspw = 3;
113 edid->mode.vso = 1;
114 edid->mode.vbl = 26;
115 edid->mode.pvsync = '+';
116 }
117
118 printk(BIOS_DEBUG, "AST: Display has %dpx x %dpx\n", edid->mode.ha, edid->mode.va);
119
120 widescreen = !!(((edid->mode.ha * 4) % (edid->mode.va * 3)));
121
122 while (ast_mode_valid(connector, edid->mode.ha, edid->mode.va) != MODE_OK) {
123 /* Select a compatible smaller mode */
124 if (edid->mode.ha > 1920 && widescreen) {
125 edid->mode.ha = 1920;
126 edid->mode.va = 1080;
127 } else if (edid->mode.ha >= 1920 && widescreen) {
128 edid->mode.ha = 1680;
129 edid->mode.va = 1050;
130 } else if (edid->mode.ha >= 1680 && widescreen) {
131 edid->mode.ha = 1600;
132 edid->mode.va = 900;
133 } else if (edid->mode.ha >= 1680 && !widescreen) {
134 edid->mode.ha = 1600;
135 edid->mode.va = 1200;
136 } else if (edid->mode.ha >= 1600 && widescreen) {
137 edid->mode.ha = 1440;
138 edid->mode.va = 900;
139 } else if (edid->mode.ha >= 1440 && widescreen) {
140 edid->mode.ha = 1360;
141 edid->mode.va = 768;
142 } else if (edid->mode.ha >= 1360 && widescreen) {
143 edid->mode.ha = 1280;
144 edid->mode.va = 800;
145 } else if (edid->mode.ha >= 1360 && !widescreen) {
146 edid->mode.ha = 1280;
147 edid->mode.va = 1024;
148 } else if (edid->mode.ha >= 1280) {
149 edid->mode.ha = 1024;
150 edid->mode.va = 768;
151 } else if (edid->mode.ha >= 1024) {
152 edid->mode.ha = 800;
153 edid->mode.va = 600;
154 } else if (edid->mode.ha >= 800) {
155 edid->mode.ha = 640;
156 edid->mode.va = 480;
157 } else {
158 dev_err(dev->pdev, "No compatible mode found.\n");
159
160 return -EIO;
161 }
162 };
163
164 return 0;
165}
166
167int ast_driver_framebuffer_init(struct drm_device *dev, int flags)
168{
169 struct drm_display_mode adjusted_mode;
170 struct drm_crtc crtc;
171 struct drm_format format;
172 struct drm_primary primary;
173 struct drm_framebuffer fb;
174 struct drm_connector connector;
175 struct edid edid;
176 int ret;
177
178 /* Init wrapper structs */
179 connector.dev = dev;
180
181 format.cpp[0] = 4; /* 32 BPP */
182 fb.format = &format;
183
184 primary.fb = &fb;
185
186 crtc.dev = dev;
187 crtc.primary = &primary;
188
189 /* Read EDID and find mode */
190 ret = ast_select_mode(&connector, &edid);
191 if (ret) {
192 dev_err(dev->pdev, "Failed to select mode.\n");
193 return ret;
194 }
195
196 /* Updated edid for set_vbe_mode_info_valid */
197 edid.x_resolution = edid.mode.ha;
198 edid.y_resolution = edid.mode.va;
199 edid.framebuffer_bits_per_pixel = format.cpp[0] * 8;
200 edid.bytes_per_line = ALIGN_UP(edid.x_resolution * format.cpp[0], 8);
201
202 /* Updated framebuffer info for ast_crtc_mode_set */
203 fb.pitches[0] = edid.bytes_per_line;
204
205 printk(BIOS_DEBUG, "Using framebuffer %dpx x %dpx pitch %d @ %d BPP\n",
206 edid.x_resolution, edid.y_resolution, edid.bytes_per_line,
207 edid.framebuffer_bits_per_pixel);
208
209 /* Convert EDID to AST DRM mode */
210 ast_edid_to_drmmode(&edid, &crtc.mode);
211
212 memcpy(&adjusted_mode, &crtc.mode, sizeof(crtc.mode));
213
214 ret = ast_crtc_mode_set(&crtc, &crtc.mode, &adjusted_mode);
215 if (ret) {
216 dev_err(dev->pdev, "Failed to set mode.\n");
217 return ret;
218 }
219
220 ast_hide_cursor(&crtc);
221
222 /* Advertise new mode */
223 set_vbe_mode_info_valid(&edid, fb.mmio_addr);
224
225 /* Clear display */
226 memset((void *)fb.mmio_addr, 0, edid.bytes_per_line * edid.y_resolution);
227
228 return 0;
229}