blob: 4771dbe37cf60f79b52009e6cc992072f8798fe2 [file] [log] [blame]
Nico Huber83693c82016-10-08 22:17:55 +02001--
Nico Huber3d06de82018-05-29 01:35:04 +02002-- Copyright (C) 2015-2018 secunet Security Networks AG
Nico Huber194e57e2017-07-15 21:15:46 +02003-- Copyright (C) 2017 Nico Huber <nico.h@gmx.de>
Nico Huber83693c82016-10-08 22:17:55 +02004--
5-- This program is free software; you can redistribute it and/or modify
6-- it under the terms of the GNU General Public License as published by
Nico Huber125a29e2016-10-18 00:23:54 +02007-- the Free Software Foundation; either version 2 of the License, or
8-- (at your option) any later version.
Nico Huber83693c82016-10-08 22:17:55 +02009--
10-- This program is distributed in the hope that it will be useful,
11-- but WITHOUT ANY WARRANTY; without even the implied warranty of
12-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13-- GNU General Public License for more details.
14--
15
Nico Huber194e57e2017-07-15 21:15:46 +020016with HW.Config;
Nico Huber83693c82016-10-08 22:17:55 +020017with HW.Time;
Nico Huberc5c66ec2019-09-28 23:59:45 +020018with HW.PCI;
Nico Huber83693c82016-10-08 22:17:55 +020019with HW.Port_IO;
Nico Hubercf88f3d2018-06-05 13:27:34 +020020with HW.GFX.Framebuffer_Filler;
Nico Huber83693c82016-10-08 22:17:55 +020021
Elyes Haouas066e52e2022-09-26 12:48:47 +020022use type HW.Word32;
23
Nico Huber83693c82016-10-08 22:17:55 +020024package HW.GFX.GMA
25with
26 Abstract_State =>
27 (State,
28 Init_State,
Nico Huber83693c82016-10-08 22:17:55 +020029 (Device_State with External)),
Nico Huber27088aa2018-06-10 13:28:05 +020030 Initializes => Init_State
Nico Huber83693c82016-10-08 22:17:55 +020031is
32
Nico Hubera02b2c62018-01-09 15:58:34 +010033 GTT_Page_Size : constant := 4096;
34 type GTT_Address_Type is mod 2 ** 39;
35 subtype GTT_Range is Natural range 0 .. 16#8_0000# - 1;
36 GTT_Rotation_Offset : constant GTT_Range := GTT_Range'Last / 2 + 1;
37
Arthur Heymans960e2392026-03-03 19:45:24 +010038 type Generation is (I945, G45, Ironlake, Haswell, Broxton, Skylake, Tigerlake);
Nico Huber6621a142018-06-07 23:56:54 +020039
Nico Huber83693c82016-10-08 22:17:55 +020040 type CPU_Type is
Arthur Heymans960e2392026-03-03 19:45:24 +010041 (I945G,
42 I945GM,
43 G45,
Nico Huber7f3e2802019-09-28 20:40:55 +020044 GM45,
Arthur Heymans73ea0322018-03-28 17:17:07 +020045 Ironlake,
Nico Huber83693c82016-10-08 22:17:55 +020046 Sandybridge,
47 Ivybridge,
48 Haswell,
49 Broadwell,
Nico Huber21da5742017-01-20 14:00:53 +010050 Broxton,
Nico Huber88badbe2018-09-27 16:36:47 +020051 Skylake,
Tim Wawrzynczak605660b2022-06-08 12:48:19 -060052 Kabylake,
Tim Wawrzynczake9631d82022-09-09 12:26:32 -060053 Tigerlake,
54 Alderlake);
Nico Huber83693c82016-10-08 22:17:55 +020055
Nico Huber25fdb152019-02-17 15:54:39 +010056 type CPU_Variant is (Normal, ULT, ULX);
Nico Huber83693c82016-10-08 22:17:55 +020057
Nico Hubere79babd2020-12-20 01:33:26 +010058 type PCH_Type is
59 (No_PCH,
60 Ibex_Peak,
61 Cougar_Point, -- Panther Point compatible
62 Lynx_Point, -- Wildcat Point compatible
Nico Huberdde06302020-12-20 02:18:30 +010063 Sunrise_Point, -- Union Point compatible
Tim Wawrzynczak605660b2022-06-08 12:48:19 -060064 Cannon_Point,
Tim Wawrzynczake9631d82022-09-09 12:26:32 -060065 Tiger_Point,
66 Alder_Point);
Nico Hubere79babd2020-12-20 01:33:26 +010067
Nico Huber83693c82016-10-08 22:17:55 +020068 type Port_Type is
69 (Disabled,
Nico Huber8beafd72020-01-07 14:59:44 +010070 LVDS,
71 eDP,
Nico Huber83693c82016-10-08 22:17:55 +020072 DP1,
73 DP2,
74 DP3,
Nico Huberad096092024-07-02 18:45:44 +020075 DP_TC1, -- DP_TC ports use Type-C lanes to
76 DP_TC2, -- implement physical DP or DP++ ports.
77 DP_TC3,
78 DP_TC4,
79 HDMI1, -- HDMI ports share lanes with their DP siblings.
80 HDMI2, -- They can also be implemented as DVI.
81 HDMI3,
82 HDMI_TC1, -- HDMI_TC use Type-C lanes to implement
83 HDMI_TC2, -- physical HDMI, DVI or the HDMI part of DP++.
84 HDMI_TC3,
85 HDMI_TC4,
Tim Wawrzynczak605660b2022-06-08 12:48:19 -060086 Analog,
Nico Huberad096092024-07-02 18:45:44 +020087 USBC1, -- Type-C lanes used to implement
88 USBC2, -- physical USB-C ports with DP Alt Mode.
89 USBC3,
90 USBC4);
Nico Huber040d9b62019-02-18 00:52:00 +010091 subtype Active_Port_Type is Port_Type
92 range Port_Type'Succ (Disabled) .. Port_Type'Last;
Nico Huber8beafd72020-01-07 14:59:44 +010093 subtype Internal_Port_Type is Port_Type range LVDS .. eDP;
Nico Huberad096092024-07-02 18:45:44 +020094 subtype Physical_DP_Ports is Port_Type range DP1 .. DP_TC4;
95 subtype Physical_HDMI_Ports is Port_Type range HDMI1 .. HDMI_TC4;
96 subtype Physical_USBC_Ports is Port_Type range USBC1 .. USBC4;
Nico Huber83693c82016-10-08 22:17:55 +020097
Nico Hubera02b2c62018-01-09 15:58:34 +010098 type Cursor_Mode is (No_Cursor, ARGB_Cursor);
99 type Cursor_Size is (Cursor_64x64, Cursor_128x128, Cursor_256x256);
100 Cursor_Width : constant array (Cursor_Size) of Width_Type := (64, 128, 256);
101
102 subtype Cursor_Pos is Int32 range Int32'First / 2 .. Int32'Last / 2;
103
104 type Cursor_Type is record
105 Mode : Cursor_Mode;
106 Size : Cursor_Size;
107 Center_X : Cursor_Pos;
108 Center_Y : Cursor_Pos;
109 GTT_Offset : GTT_Range;
110 end record;
111 Default_Cursor : constant Cursor_Type :=
112 (Mode => No_Cursor,
113 Size => Cursor_Size'First,
114 Center_X => 0,
115 Center_Y => 0,
116 GTT_Offset => 0);
117
Nico Huber99f10f32016-11-20 00:34:05 +0100118 type Pipe_Config is record
Nico Huber83693c82016-10-08 22:17:55 +0200119 Port : Port_Type;
120 Framebuffer : Framebuffer_Type;
Nico Hubera02b2c62018-01-09 15:58:34 +0100121 Cursor : Cursor_Type;
Nico Huber83693c82016-10-08 22:17:55 +0200122 Mode : Mode_Type;
123 end record;
Nico Huber99f10f32016-11-20 00:34:05 +0100124 type Pipe_Index is (Primary, Secondary, Tertiary);
125 type Pipe_Configs is array (Pipe_Index) of Pipe_Config;
Nico Huber83693c82016-10-08 22:17:55 +0200126
Nico Huber3675db52016-11-04 16:27:29 +0100127 -- Special framebuffer offset to indicate legacy VGA plane.
128 -- Only valid on primary pipe.
129 VGA_PLANE_FRAMEBUFFER_OFFSET : constant := 16#ffff_ffff#;
130
Nico Huberbebca132017-06-12 23:04:46 +0200131 pragma Warnings (GNATprove, Off, "unused variable ""Write_Delay""",
132 Reason => "Write_Delay is used for debugging only");
Nico Huber83693c82016-10-08 22:17:55 +0200133 procedure Initialize
Nico Huber2b6f6992017-07-09 18:11:34 +0200134 (Write_Delay : in Word64 := 0;
Nico Huber793a8d42016-11-21 18:57:03 +0100135 Clean_State : in Boolean := False;
Nico Huber83693c82016-10-08 22:17:55 +0200136 Success : out Boolean)
137 with
138 Global =>
Nico Huber27088aa2018-06-10 13:28:05 +0200139 (In_Out => (Device_State, Port_IO.State),
Nico Hubere317e9c2019-09-29 03:03:18 +0200140 Output => (State, Init_State),
Nico Huber83693c82016-10-08 22:17:55 +0200141 Input => (Time.State)),
142 Post => Success = Is_Initialized;
143 function Is_Initialized return Boolean
144 with
145 Global => (Input => Init_State);
Nico Huberbebca132017-06-12 23:04:46 +0200146 pragma Warnings (GNATprove, On, "unused variable ""Write_Delay""");
Nico Huber83693c82016-10-08 22:17:55 +0200147
Nico Huber42fb2d02017-09-01 17:01:51 +0200148 procedure Power_Up_VGA
149 with
Nico Hubercf88f3d2018-06-05 13:27:34 +0200150 Global =>
Nico Hubere317e9c2019-09-29 03:03:18 +0200151 (Input => (State, Time.State),
Nico Hubercf88f3d2018-06-05 13:27:34 +0200152 In_Out => (Device_State),
153 Proof_In => (Init_State)),
Nico Huber42fb2d02017-09-01 17:01:51 +0200154 Pre => Is_Initialized;
155
Nico Hubercf88f3d2018-06-05 13:27:34 +0200156 ----------------------------------------------------------------------------
Nico Huber83693c82016-10-08 22:17:55 +0200157
Nico Hubercf88f3d2018-06-05 13:27:34 +0200158 procedure Update_Outputs (Configs : Pipe_Configs)
159 with
160 Global =>
Nico Hubere317e9c2019-09-29 03:03:18 +0200161 (Input => (Time.State),
Nico Hubercf88f3d2018-06-05 13:27:34 +0200162 In_Out => (State, Device_State, Port_IO.State),
163 Proof_In => (Init_State)),
164 Pre => Is_Initialized;
165
166 procedure Update_Cursor (Pipe : Pipe_Index; Cursor : Cursor_Type)
167 with
168 Global =>
Nico Hubere317e9c2019-09-29 03:03:18 +0200169 (In_Out => (State, Device_State),
Nico Hubercf88f3d2018-06-05 13:27:34 +0200170 Proof_In => (Init_State)),
Nico Hubere317e9c2019-09-29 03:03:18 +0200171 Pre => Is_Initialized;
Nico Hubercf88f3d2018-06-05 13:27:34 +0200172
Nico Huber15ffc4f2018-01-11 14:44:43 +0100173 procedure Place_Cursor
174 (Pipe : Pipe_Index;
175 X : Cursor_Pos;
Nico Hubercf88f3d2018-06-05 13:27:34 +0200176 Y : Cursor_Pos)
177 with
178 Global =>
Nico Hubere317e9c2019-09-29 03:03:18 +0200179 (In_Out => (State, Device_State),
Nico Hubercf88f3d2018-06-05 13:27:34 +0200180 Proof_In => (Init_State)),
Nico Hubere317e9c2019-09-29 03:03:18 +0200181 Pre => Is_Initialized;
Nico Hubercf88f3d2018-06-05 13:27:34 +0200182
Nico Huber15ffc4f2018-01-11 14:44:43 +0100183 procedure Move_Cursor
184 (Pipe : Pipe_Index;
185 X : Cursor_Pos;
Nico Hubercf88f3d2018-06-05 13:27:34 +0200186 Y : Cursor_Pos)
187 with
188 Global =>
Nico Hubere317e9c2019-09-29 03:03:18 +0200189 (In_Out => (State, Device_State),
Nico Hubercf88f3d2018-06-05 13:27:34 +0200190 Proof_In => (Init_State)),
Nico Hubere317e9c2019-09-29 03:03:18 +0200191 Pre => Is_Initialized;
Nico Huber15ffc4f2018-01-11 14:44:43 +0100192
Nico Hubercf88f3d2018-06-05 13:27:34 +0200193 ----------------------------------------------------------------------------
Nico Huber83693c82016-10-08 22:17:55 +0200194
Nico Huberbc0588e2020-07-21 12:17:19 +0200195 procedure Backlight_On (Port : Active_Port_Type)
196 with
197 Global => (In_Out => Device_State);
198
199 procedure Backlight_Off (Port : Active_Port_Type)
200 with
201 Global => (In_Out => Device_State);
202
203 procedure Set_Brightness (Port : Active_Port_Type; Level : Word32)
204 with
205 Global => (In_Out => Device_State);
206
207 procedure Get_Max_Brightness (Port : Active_Port_Type; Level : out Word32)
208 with
209 Global => (In_Out => Device_State);
210
211 ----------------------------------------------------------------------------
212
Nico Huber83693c82016-10-08 22:17:55 +0200213 procedure Write_GTT
214 (GTT_Page : GTT_Range;
215 Device_Address : GTT_Address_Type;
Nico Hubercf88f3d2018-06-05 13:27:34 +0200216 Valid : Boolean)
217 with
Nico Huberadfe11f2018-06-10 14:59:04 +0200218 Global =>
Nico Hubere317e9c2019-09-29 03:03:18 +0200219 (Input => State,
Nico Huberadfe11f2018-06-10 14:59:04 +0200220 In_Out => Device_State, Proof_In => Init_State),
Nico Hubercf88f3d2018-06-05 13:27:34 +0200221 Pre => Is_Initialized;
Nico Huber83693c82016-10-08 22:17:55 +0200222
Nico Huberceda17d2018-06-09 22:00:29 +0200223 procedure Read_GTT
224 (Device_Address : out GTT_Address_Type;
225 Valid : out Boolean;
226 GTT_Page : in GTT_Range)
227 with
Nico Huberadfe11f2018-06-10 14:59:04 +0200228 Global =>
Nico Hubere317e9c2019-09-29 03:03:18 +0200229 (Input => State,
Nico Huberadfe11f2018-06-10 14:59:04 +0200230 In_Out => Device_State, Proof_In => Init_State),
Nico Huberceda17d2018-06-09 22:00:29 +0200231 Pre => Is_Initialized;
232
Arthur Heymans960e2392026-03-03 19:45:24 +0100233 procedure GTT_Entry_Count (Count : out Natural)
234 with
235 Global =>
236 (Input => State,
237 In_Out => Device_State, Proof_In => Init_State),
238 Pre => Is_Initialized and HW.Config.Dynamic_MMIO;
239
Nico Huber5374c3a2017-07-15 21:48:06 +0200240 procedure Setup_Default_FB
241 (FB : in Framebuffer_Type;
242 Clear : in Boolean := True;
243 Success : out Boolean)
Nico Huber194e57e2017-07-15 21:15:46 +0200244 with
Nico Hubercf88f3d2018-06-05 13:27:34 +0200245 Global =>
Nico Hubere317e9c2019-09-29 03:03:18 +0200246 (In_Out =>
Nico Hubercf88f3d2018-06-05 13:27:34 +0200247 (State, Device_State,
248 Framebuffer_Filler.State, Framebuffer_Filler.Base_Address),
249 Proof_In => (Init_State)),
Nico Huber5374c3a2017-07-15 21:48:06 +0200250 Pre => Is_Initialized and HW.Config.Dynamic_MMIO;
Nico Huber83693c82016-10-08 22:17:55 +0200251
Nico Huberc3f66f62017-07-16 21:39:54 +0200252 procedure Map_Linear_FB (Linear_FB : out Word64; FB : in Framebuffer_Type)
253 with
Nico Hubercf88f3d2018-06-05 13:27:34 +0200254 Global =>
Nico Hubere317e9c2019-09-29 03:03:18 +0200255 (In_Out => (State, Device_State),
Nico Hubercf88f3d2018-06-05 13:27:34 +0200256 Proof_In => (Init_State)),
Nico Hubere317e9c2019-09-29 03:03:18 +0200257 Pre => Is_Initialized and HW.Config.Dynamic_MMIO;
Nico Huberc3f66f62017-07-16 21:39:54 +0200258
Nico Hubercf88f3d2018-06-05 13:27:34 +0200259 ----------------------------------------------------------------------------
260
261 pragma Warnings (GNATprove, Off, "subprogram ""Dump_Configs"" has no effect",
262 Reason => "It's only used for debugging");
263 procedure Dump_Configs (Configs : Pipe_Configs);
264
Nico Huber83693c82016-10-08 22:17:55 +0200265private
266
Nico Huberc5c66ec2019-09-28 23:59:45 +0200267 PCI_Usable : Boolean with Part_Of => State;
268 use type HW.PCI.Index;
269 procedure PCI_Read16 (Value : out Word16; Offset : HW.PCI.Index)
270 with
271 Pre => PCI_Usable and Offset mod 2 = 0;
272
273 ----------------------------------------------------------------------------
274
Nico Huber3299ad52018-06-02 16:53:39 +0200275 -- For the default framebuffer setup (see below) with 90 degree rotations,
276 -- we expect the offset which is used for the final scanout to be above
277 -- `GTT_Rotation_Offset`. So we can use `Offset - GTT_Rotation_Offset` for
278 -- the physical memory location and aperture mapping.
279 function Phys_Offset (FB : Framebuffer_Type) return Word32 is
280 (if Rotation_90 (FB)
281 then FB.Offset - Word32 (GTT_Rotation_Offset) * GTT_Page_Size
282 else FB.Offset);
283
Nico Huber8c45bcf2016-11-20 17:30:57 +0100284 ----------------------------------------------------------------------------
285 -- State tracking for the currently configured pipes
286
287 Cur_Configs : Pipe_Configs with Part_Of => State;
288
Nico Huber3d06de82018-05-29 01:35:04 +0200289 function Requires_Scaling (Pipe_Cfg : Pipe_Config) return Boolean is
290 (Requires_Scaling (Pipe_Cfg.Framebuffer, Pipe_Cfg.Mode));
291
Nico Huberb217ece2018-06-02 16:56:35 +0200292 function Scaling_Type (Pipe_Cfg : Pipe_Config) return Scaling_Aspect is
293 (Scaling_Type (Pipe_Cfg.Framebuffer, Pipe_Cfg.Mode));
294
Nico Huber8c45bcf2016-11-20 17:30:57 +0100295 ----------------------------------------------------------------------------
296 -- Internal representation of a single pipe's configuration
297
Tim Wawrzynczak605660b2022-06-08 12:48:19 -0600298 type GPU_Port is
299 (DIGI_A, DIGI_B, DIGI_C, DIGI_D, DIGI_E,
300 DDI_TC1, DDI_TC2, DDI_TC3, DDI_TC4, DDI_TC5, DDI_TC6,
301 LVDS, VGA);
Nico Huber83693c82016-10-08 22:17:55 +0200302
303 subtype Digital_Port is GPU_Port range DIGI_A .. DIGI_E;
Arthur Heymans5d08a932018-03-28 17:00:18 +0200304 subtype GMCH_DP_Port is GPU_Port range DIGI_B .. DIGI_D;
305 subtype GMCH_HDMI_Port is GPU_Port range DIGI_B .. DIGI_C;
Tim Wawrzynczak605660b2022-06-08 12:48:19 -0600306 subtype Combo_Port is GPU_Port range DIGI_A .. DIGI_C;
307 subtype USBC_Port is GPU_Port range DDI_TC1 .. DDI_TC6;
308 subtype TGL_Digital_Port is GPU_Port range DIGI_A .. DDI_TC6;
309
310 function Is_Digital_Port (Port : GPU_Port) return Boolean is
311 (Port in Digital_Port or Port in TGL_Digital_Port);
Nico Huber83693c82016-10-08 22:17:55 +0200312
313 type PCH_Port is
314 (PCH_DAC, PCH_LVDS,
Tim Wawrzynczak605660b2022-06-08 12:48:19 -0600315 PCH_HDMI_A, PCH_HDMI_B, PCH_HDMI_C, PCH_HDMI_D,
316 PCH_DP_B, PCH_DP_C, PCH_DP_D,
317 PCH_TC1, PCH_TC2, PCH_TC3, PCH_TC4, PCH_TC5, PCH_TC6);
Nico Huber83693c82016-10-08 22:17:55 +0200318
Tim Wawrzynczak605660b2022-06-08 12:48:19 -0600319 subtype PCH_HDMI_Port is PCH_Port range PCH_HDMI_A .. PCH_HDMI_D;
Nico Huber83693c82016-10-08 22:17:55 +0200320 subtype PCH_DP_Port is PCH_Port range PCH_DP_B .. PCH_DP_D;
Tim Wawrzynczak605660b2022-06-08 12:48:19 -0600321 subtype PCH_TC_Port is PCH_Port range PCH_TC1 .. PCH_TC6;
Nico Huber83693c82016-10-08 22:17:55 +0200322
Nico Huber5dbaf4b2020-01-08 17:24:58 +0100323 type Panel_Control is (No_Panel, Panel_1, Panel_2);
324 subtype Valid_Panels is Panel_Control range Panel_1 .. Panel_2;
Nico Huber2bbd6e72020-01-07 18:22:59 +0100325
Nico Huber83693c82016-10-08 22:17:55 +0200326 type Port_Config is
327 record
328 Port : GPU_Port;
329 PCH_Port : GMA.PCH_Port;
330 Display : Display_Type;
Nico Huber2bbd6e72020-01-07 18:22:59 +0100331 Panel : Panel_Control;
Nico Huber83693c82016-10-08 22:17:55 +0200332 Mode : Mode_Type;
333 Is_FDI : Boolean;
334 FDI : DP_Link;
335 DP : DP_Link;
336 end record;
337
338 type FDI_Training_Type is (Simple_Training, Full_Training, Auto_Training);
339
340 ----------------------------------------------------------------------------
341
Tim Wawrzynczak24748f32022-09-09 10:54:05 -0600342 type DP_Port is (DP_A, DP_B, DP_C, DP_D, DP_E, DP_F, DP_G, DP_H, DP_I);
Nico Huber83693c82016-10-08 22:17:55 +0200343
Nico Huber247adf32017-06-12 14:39:11 +0200344 ----------------------------------------------------------------------------
345
346 subtype DDI_HDMI_Buf_Trans_Range is Integer range 0 .. 11;
347
Nico Huber0164b022017-08-24 15:12:51 +0200348 ----------------------------------------------------------------------------
349
Nico Huberc5c767a2018-06-03 01:09:04 +0200350 Tile_Width : constant array (Tiling_Type) of Width_Type :=
Nico Huber0164b022017-08-24 15:12:51 +0200351 (Linear => 16, X_Tiled => 128, Y_Tiled => 32);
Nico Huberc5c767a2018-06-03 01:09:04 +0200352 Tile_Rows : constant array (Tiling_Type) of Height_Type :=
Nico Huber9b479412017-08-27 11:55:56 +0200353 (Linear => 1, X_Tiled => 8, Y_Tiled => 32);
Nico Huber0164b022017-08-24 15:12:51 +0200354
Nico Huber5ef4d602017-12-13 13:56:47 +0100355 function FB_Pitch (Px : Pos_Pixel_Type; FB : Framebuffer_Type) return Natural
356 is (Natural (Div_Round_Up
357 (Pixel_To_Bytes (Px, FB), Tile_Width (FB.Tiling) * 4)));
Nico Huber0164b022017-08-24 15:12:51 +0200358
359 function Valid_Stride (FB : Framebuffer_Type) return Boolean is
Nico Huber5ef4d602017-12-13 13:56:47 +0100360 (FB.Width + FB.Start_X <= FB.Stride and
Nico Huber9b479412017-08-27 11:55:56 +0200361 Pixel_To_Bytes (FB.Stride, FB) mod (Tile_Width (FB.Tiling) * 4) = 0 and
Nico Huber5ef4d602017-12-13 13:56:47 +0100362 FB.Height + FB.Start_Y <= FB.V_Stride and
Nico Huber9b479412017-08-27 11:55:56 +0200363 FB.V_Stride mod Tile_Rows (FB.Tiling) = 0);
Nico Huber0164b022017-08-24 15:12:51 +0200364
Nico Huber83693c82016-10-08 22:17:55 +0200365end HW.GFX.GMA;