2 * GRUB -- GRand Unified Bootloader
3 * Copyright (C) 2005,2006,2007,2008,2009,2010 Free Software Foundation, Inc.
5 * GRUB is free software: you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation, either version 3 of the License, or
8 * (at your option) any later version.
10 * GRUB 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.
15 * You should have received a copy of the GNU General Public License
16 * along with GRUB. If not, see <http://www.gnu.org/licenses/>.
19 #define grub_video_render_target grub_video_fbrender_target
22 #include <grub/types.h>
24 #include <grub/misc.h>
26 #include <grub/video.h>
27 #include <grub/video_fb.h>
30 #include <grub/cache.h>
32 #define GRUB_SIS315PRO_PCIID 0x03251039
33 #define GRUB_SIS315PRO_TOTAL_MEMORY_SPACE 0x800000
34 #define GRUB_SIS315PRO_MMIO_SPACE 0x1000
38 struct grub_video_mode_info mode_info
;
41 volatile grub_uint8_t
*direct_ptr
;
44 grub_uint32_t mmiobase
;
45 volatile grub_uint32_t
*mmioptr
;
46 grub_pci_device_t dev
;
51 read_sis_cmd (grub_uint8_t addr
)
53 grub_outb (addr
, framebuffer
.io
+ 0x44);
54 return grub_inb (framebuffer
.io
+ 0x45);
58 write_sis_cmd (grub_uint8_t val
, grub_uint8_t addr
)
60 grub_outb (addr
, framebuffer
.io
+ 0x44);
61 grub_outb (val
, framebuffer
.io
+ 0x45);
66 grub_video_sis315pro_video_init (void)
68 /* Reset frame buffer. */
69 grub_memset (&framebuffer
, 0, sizeof(framebuffer
));
71 return grub_video_fb_init ();
75 grub_video_sis315pro_video_fini (void)
77 if (framebuffer
.mapped
)
79 grub_pci_device_unmap_range (framebuffer
.dev
, framebuffer
.ptr
,
80 GRUB_SIS315PRO_TOTAL_MEMORY_SPACE
);
81 grub_pci_device_unmap_range (framebuffer
.dev
, framebuffer
.direct_ptr
,
82 GRUB_SIS315PRO_TOTAL_MEMORY_SPACE
);
85 return grub_video_fb_fini ();
89 #include "sis315_init.c"
92 /* Helper for grub_video_sis315pro_setup. */
94 find_card (grub_pci_device_t dev
, grub_pci_id_t pciid
, void *data
)
97 grub_pci_address_t addr
;
100 addr
= grub_pci_make_address (dev
, GRUB_PCI_REG_CLASS
);
101 class = grub_pci_read (addr
);
103 if (((class >> 16) & 0xffff) != GRUB_PCI_CLASS_SUBCLASS_VGA
104 || pciid
!= GRUB_SIS315PRO_PCIID
)
109 addr
= grub_pci_make_address (dev
, GRUB_PCI_REG_ADDRESS_REG0
);
110 framebuffer
.base
= grub_pci_read (addr
) & GRUB_PCI_ADDR_MEM_MASK
;
111 addr
= grub_pci_make_address (dev
, GRUB_PCI_REG_ADDRESS_REG1
);
112 framebuffer
.mmiobase
= grub_pci_read (addr
) & GRUB_PCI_ADDR_MEM_MASK
;
113 addr
= grub_pci_make_address (dev
, GRUB_PCI_REG_ADDRESS_REG2
);
114 framebuffer
.io
= (grub_pci_read (addr
) & GRUB_PCI_ADDR_IO_MASK
)
115 + GRUB_MACHINE_PCI_IO_BASE
;
116 framebuffer
.dev
= dev
;
123 grub_video_sis315pro_setup (unsigned int width
, unsigned int height
,
124 unsigned int mode_type
,
125 unsigned int mode_mask
__attribute__ ((unused
)))
133 /* Decode depth from mode_type. If it is zero, then autodetect. */
134 depth
= (mode_type
& GRUB_VIDEO_MODE_TYPE_DEPTH_MASK
)
135 >> GRUB_VIDEO_MODE_TYPE_DEPTH_POS
;
137 if ((width
!= 640 && width
!= 0) || (height
!= 480 && height
!= 0)
138 || (depth
!= 8 && depth
!= 0))
139 return grub_error (GRUB_ERR_NOT_IMPLEMENTED_YET
,
140 "Only 640x480x8 is supported");
142 grub_pci_iterate (find_card
, &found
);
144 return grub_error (GRUB_ERR_IO
, "Couldn't find graphics card");
146 /* Fill mode info details. */
147 framebuffer
.mode_info
.width
= 640;
148 framebuffer
.mode_info
.height
= 480;
149 framebuffer
.mode_info
.mode_type
= (GRUB_VIDEO_MODE_TYPE_INDEX_COLOR
150 | GRUB_VIDEO_MODE_TYPE_DOUBLE_BUFFERED
151 | GRUB_VIDEO_MODE_TYPE_UPDATING_SWAP
);
152 framebuffer
.mode_info
.bpp
= 8;
153 framebuffer
.mode_info
.bytes_per_pixel
= 1;
154 framebuffer
.mode_info
.pitch
= 640 * 1;
155 framebuffer
.mode_info
.number_of_colors
= 16;
156 framebuffer
.mode_info
.red_mask_size
= 0;
157 framebuffer
.mode_info
.red_field_pos
= 0;
158 framebuffer
.mode_info
.green_mask_size
= 0;
159 framebuffer
.mode_info
.green_field_pos
= 0;
160 framebuffer
.mode_info
.blue_mask_size
= 0;
161 framebuffer
.mode_info
.blue_field_pos
= 0;
162 framebuffer
.mode_info
.reserved_mask_size
= 0;
163 framebuffer
.mode_info
.reserved_field_pos
= 0;
165 framebuffer
.mode_info
.blit_format
166 = grub_video_get_blit_format (&framebuffer
.mode_info
);
170 if (found
&& (framebuffer
.base
== 0 || framebuffer
.mmiobase
== 0))
172 grub_pci_address_t addr
;
173 /* FIXME: choose address dynamically if needed. */
174 framebuffer
.base
= 0x40000000;
175 framebuffer
.mmiobase
= 0x04000000;
176 framebuffer
.io
= 0xb300;
178 addr
= grub_pci_make_address (framebuffer
.dev
, GRUB_PCI_REG_ADDRESS_REG0
);
179 grub_pci_write (addr
, framebuffer
.base
| GRUB_PCI_ADDR_MEM_PREFETCH
);
181 addr
= grub_pci_make_address (framebuffer
.dev
, GRUB_PCI_REG_ADDRESS_REG1
);
182 grub_pci_write (addr
, framebuffer
.mmiobase
);
184 addr
= grub_pci_make_address (framebuffer
.dev
, GRUB_PCI_REG_ADDRESS_REG2
);
185 grub_pci_write (addr
, framebuffer
.io
| GRUB_PCI_ADDR_SPACE_IO
);
188 addr
= grub_pci_make_address (framebuffer
.dev
, GRUB_PCI_REG_CACHELINE
);
189 grub_pci_write (addr
, 0x80004700);
191 /* Enable address spaces. */
192 addr
= grub_pci_make_address (framebuffer
.dev
, GRUB_PCI_REG_COMMAND
);
193 grub_pci_write (addr
, 0x7);
195 addr
= grub_pci_make_address (framebuffer
.dev
, 0x30);
196 grub_pci_write (addr
, 0x04060001);
198 framebuffer
.io
+= GRUB_MACHINE_PCI_IO_BASE
;
203 /* We can safely discard volatile attribute. */
206 = grub_pci_device_map_range_cached (framebuffer
.dev
,
208 GRUB_SIS315PRO_TOTAL_MEMORY_SPACE
);
209 framebuffer
.direct_ptr
210 = grub_pci_device_map_range (framebuffer
.dev
,
212 GRUB_SIS315PRO_TOTAL_MEMORY_SPACE
);
213 framebuffer
.mmioptr
= grub_pci_device_map_range (framebuffer
.dev
,
214 framebuffer
.mmiobase
,
215 GRUB_SIS315PRO_MMIO_SPACE
);
217 framebuffer
.mapped
= 1;
220 /* Prevent garbage from appearing on the screen. */
221 grub_memset (framebuffer
.ptr
, 0,
222 framebuffer
.mode_info
.height
* framebuffer
.mode_info
.pitch
);
223 grub_arch_sync_dma_caches (framebuffer
.ptr
,
224 framebuffer
.mode_info
.height
225 * framebuffer
.mode_info
.pitch
);
228 grub_outb (GRUB_VGA_IO_MISC_NEGATIVE_VERT_POLARITY
229 | GRUB_VGA_IO_MISC_NEGATIVE_HORIZ_POLARITY
230 | GRUB_VGA_IO_MISC_UPPER_64K
231 | GRUB_VGA_IO_MISC_EXTERNAL_CLOCK_0
232 | GRUB_VGA_IO_MISC_28MHZ
233 | GRUB_VGA_IO_MISC_ENABLE_VRAM_ACCESS
234 | GRUB_VGA_IO_MISC_COLOR
,
235 GRUB_VGA_IO_MISC_WRITE
+ GRUB_MACHINE_PCI_IO_BASE
);
237 grub_vga_sr_write (0x86, 5);
238 for (i
= 6; i
<= 0x27; i
++)
239 grub_vga_sr_write (0, i
);
241 for (i
= 0x31; i
<= 0x3d; i
++)
242 grub_vga_sr_write (0, i
);
244 for (i
= 0; i
< ARRAY_SIZE (sr_dump
); i
++)
245 grub_vga_sr_write (sr_dump
[i
].val
, sr_dump
[i
].reg
);
247 for (i
= 0x30; i
< 0x40; i
++)
248 grub_vga_cr_write (0, i
);
250 grub_vga_cr_write (0x77, 0x40);
251 grub_vga_cr_write (0x77, 0x41);
252 grub_vga_cr_write (0x00, 0x42);
253 grub_vga_cr_write (0x5b, 0x43);
254 grub_vga_cr_write (0x00, 0x44);
255 grub_vga_cr_write (0x23, 0x48);
256 grub_vga_cr_write (0xaa, 0x49);
257 grub_vga_cr_write (0x02, 0x37);
258 grub_vga_cr_write (0x20, 0x5b);
259 grub_vga_cr_write (0x00, 0x83);
260 grub_vga_cr_write (0x80, 0x63);
262 grub_vga_cr_write (0x0c, GRUB_VGA_CR_VSYNC_END
);
263 grub_vga_cr_write (0x5f, GRUB_VGA_CR_HTOTAL
);
264 grub_vga_cr_write (0x4f, GRUB_VGA_CR_HORIZ_END
);
265 grub_vga_cr_write (0x50, GRUB_VGA_CR_HBLANK_START
);
266 grub_vga_cr_write (0x82, GRUB_VGA_CR_HBLANK_END
);
267 grub_vga_cr_write (0x54, GRUB_VGA_CR_HORIZ_SYNC_PULSE_START
);
268 grub_vga_cr_write (0x80, GRUB_VGA_CR_HORIZ_SYNC_PULSE_END
);
269 grub_vga_cr_write (0x0b, GRUB_VGA_CR_VERT_TOTAL
);
270 grub_vga_cr_write (0x3e, GRUB_VGA_CR_OVERFLOW
);
271 grub_vga_cr_write (0x00, GRUB_VGA_CR_BYTE_PANNING
);
272 grub_vga_cr_write (0x40, GRUB_VGA_CR_CELL_HEIGHT
);
273 grub_vga_cr_write (0x00, GRUB_VGA_CR_CURSOR_START
);
274 grub_vga_cr_write (0x00, GRUB_VGA_CR_CURSOR_END
);
275 grub_vga_cr_write (0x00, GRUB_VGA_CR_START_ADDR_HIGH_REGISTER
);
276 grub_vga_cr_write (0x00, GRUB_VGA_CR_START_ADDR_LOW_REGISTER
);
277 grub_vga_cr_write (0x00, GRUB_VGA_CR_CURSOR_ADDR_HIGH
);
278 grub_vga_cr_write (0x00, GRUB_VGA_CR_CURSOR_ADDR_LOW
);
279 grub_vga_cr_write (0xea, GRUB_VGA_CR_VSYNC_START
);
280 grub_vga_cr_write (0x8c, GRUB_VGA_CR_VSYNC_END
);
281 grub_vga_cr_write (0xdf, GRUB_VGA_CR_VDISPLAY_END
);
282 grub_vga_cr_write (0x28, GRUB_VGA_CR_PITCH
);
283 grub_vga_cr_write (0x40, GRUB_VGA_CR_UNDERLINE_LOCATION
);
284 grub_vga_cr_write (0xe7, GRUB_VGA_CR_VERTICAL_BLANK_START
);
285 grub_vga_cr_write (0x04, GRUB_VGA_CR_VERTICAL_BLANK_END
);
286 grub_vga_cr_write (0xa3, GRUB_VGA_CR_MODE
);
287 grub_vga_cr_write (0xff, GRUB_VGA_CR_LINE_COMPARE
);
289 grub_vga_cr_write (0x0c, GRUB_VGA_CR_VSYNC_END
);
290 grub_vga_cr_write (0x5f, GRUB_VGA_CR_HTOTAL
);
291 grub_vga_cr_write (0x4f, GRUB_VGA_CR_HORIZ_END
);
292 grub_vga_cr_write (0x50, GRUB_VGA_CR_HBLANK_START
);
293 grub_vga_cr_write (0x82, GRUB_VGA_CR_HBLANK_END
);
294 grub_vga_cr_write (0x55, GRUB_VGA_CR_HORIZ_SYNC_PULSE_START
);
295 grub_vga_cr_write (0x81, GRUB_VGA_CR_HORIZ_SYNC_PULSE_END
);
296 grub_vga_cr_write (0x0b, GRUB_VGA_CR_VERT_TOTAL
);
297 grub_vga_cr_write (0x3e, GRUB_VGA_CR_OVERFLOW
);
298 grub_vga_cr_write (0xe9, GRUB_VGA_CR_VSYNC_START
);
299 grub_vga_cr_write (0x8b, GRUB_VGA_CR_VSYNC_END
);
300 grub_vga_cr_write (0xdf, GRUB_VGA_CR_VDISPLAY_END
);
301 grub_vga_cr_write (0xe7, GRUB_VGA_CR_VERTICAL_BLANK_START
);
302 grub_vga_cr_write (0x04, GRUB_VGA_CR_VERTICAL_BLANK_END
);
303 grub_vga_cr_write (0x40, GRUB_VGA_CR_CELL_HEIGHT
);
304 grub_vga_cr_write (0x50, GRUB_VGA_CR_PITCH
);
306 grub_vga_cr_write (0x00, 0x19);
307 grub_vga_cr_write (0x00, 0x1a);
308 grub_vga_cr_write (0x6c, 0x52);
309 grub_vga_cr_write (0x2e, 0x34);
310 grub_vga_cr_write (0x00, 0x31);
313 grub_vga_cr_write (0, GRUB_VGA_CR_START_ADDR_HIGH_REGISTER
);
314 grub_vga_cr_write (0, GRUB_VGA_CR_START_ADDR_LOW_REGISTER
);
316 for (i
= 0; i
< 16; i
++)
317 grub_vga_write_arx (i
, i
);
318 grub_vga_write_arx (1, GRUB_VGA_ARX_MODE
);
319 grub_vga_write_arx (0, GRUB_VGA_ARX_OVERSCAN
);
320 grub_vga_write_arx (0, GRUB_VGA_ARX_COLOR_PLANE_ENABLE
);
321 grub_vga_write_arx (0, GRUB_VGA_ARX_HORIZONTAL_PANNING
);
322 grub_vga_write_arx (0, GRUB_VGA_ARX_COLOR_SELECT
);
324 grub_outb (0xff, GRUB_VGA_IO_PIXEL_MASK
+ GRUB_MACHINE_PCI_IO_BASE
);
326 for (i
= 0; i
< ARRAY_SIZE (gr
); i
++)
327 grub_vga_gr_write (gr
[i
], i
);
329 for (i
= 0; i
< GRUB_VIDEO_FBSTD_NUMCOLORS
; i
++)
330 grub_vga_palette_write (i
, grub_video_fbstd_colors
[i
].r
,
331 grub_video_fbstd_colors
[i
].g
,
332 grub_video_fbstd_colors
[i
].b
);
336 if (read_sis_cmd (0x5) != 0xa1)
337 write_sis_cmd (0x86, 0x5);
339 write_sis_cmd (read_sis_cmd (0x20) | 0xa1, 0x20);
340 write_sis_cmd (read_sis_cmd (0x1e) | 0xda, 0x1e);
342 #define IND_SIS_CMDQUEUE_SET 0x26
343 #define IND_SIS_CMDQUEUE_THRESHOLD 0x27
345 #define COMMAND_QUEUE_THRESHOLD 0x1F
346 #define SIS_CMD_QUEUE_RESET 0x01
348 #define SIS_AGP_CMDQUEUE_ENABLE 0x80 /* 315/330/340 series SR26 */
349 #define SIS_VRAM_CMDQUEUE_ENABLE 0x40
350 #define SIS_MMIO_CMD_ENABLE 0x20
351 #define SIS_CMD_QUEUE_SIZE_512k 0x00
352 #define SIS_CMD_QUEUE_SIZE_1M 0x04
353 #define SIS_CMD_QUEUE_SIZE_2M 0x08
354 #define SIS_CMD_QUEUE_SIZE_4M 0x0C
355 #define SIS_CMD_QUEUE_RESET 0x01
356 #define SIS_CMD_AUTO_CORR 0x02
359 write_sis_cmd (COMMAND_QUEUE_THRESHOLD
, IND_SIS_CMDQUEUE_THRESHOLD
);
360 write_sis_cmd (SIS_CMD_QUEUE_RESET
, IND_SIS_CMDQUEUE_SET
);
361 framebuffer
.mmioptr
[0x85C4 / 4] = framebuffer
.mmioptr
[0x85C8 / 4];
362 write_sis_cmd (SIS_MMIO_CMD_ENABLE
| SIS_CMD_AUTO_CORR
, IND_SIS_CMDQUEUE_SET
);
363 framebuffer
.mmioptr
[0x85C0 / 4] = (0x1000000 - (512 * 1024));
368 err
= grub_video_fb_setup (mode_type
, mode_mask
,
369 &framebuffer
.mode_info
,
370 framebuffer
.ptr
, NULL
, NULL
);
374 /* Copy default palette to initialize emulated palette. */
375 err
= grub_video_fb_set_palette (0, GRUB_VIDEO_FBSTD_EXT_NUMCOLORS
,
376 grub_video_fbstd_colors
);
384 grub_video_sis315pro_swap_buffers (void)
387 s
= (framebuffer
.mode_info
.height
388 * framebuffer
.mode_info
.pitch
389 * framebuffer
.mode_info
.bytes_per_pixel
);
390 grub_video_fb_swap_buffers ();
391 grub_arch_sync_dma_caches (framebuffer
.ptr
, s
);
392 return GRUB_ERR_NONE
;
396 grub_video_sis315pro_get_info_and_fini (struct grub_video_mode_info
*mode_info
,
399 grub_memcpy (mode_info
, &(framebuffer
.mode_info
), sizeof (*mode_info
));
400 *framebuf
= (void *) framebuffer
.direct_ptr
;
402 grub_video_fb_fini ();
404 return GRUB_ERR_NONE
;
407 static struct grub_video_adapter grub_video_sis315pro_adapter
=
409 .name
= "SIS315PRO Video Driver",
410 .id
= GRUB_VIDEO_DRIVER_SIS315PRO
,
412 .prio
= GRUB_VIDEO_ADAPTER_PRIO_NATIVE
,
414 .init
= grub_video_sis315pro_video_init
,
415 .fini
= grub_video_sis315pro_video_fini
,
416 .setup
= grub_video_sis315pro_setup
,
417 .get_info
= grub_video_fb_get_info
,
418 .get_info_and_fini
= grub_video_sis315pro_get_info_and_fini
,
419 .set_palette
= grub_video_fb_set_palette
,
420 .get_palette
= grub_video_fb_get_palette
,
421 .set_viewport
= grub_video_fb_set_viewport
,
422 .get_viewport
= grub_video_fb_get_viewport
,
423 .set_region
= grub_video_fb_set_region
,
424 .get_region
= grub_video_fb_get_region
,
425 .set_area_status
= grub_video_fb_set_area_status
,
426 .get_area_status
= grub_video_fb_get_area_status
,
427 .map_color
= grub_video_fb_map_color
,
428 .map_rgb
= grub_video_fb_map_rgb
,
429 .map_rgba
= grub_video_fb_map_rgba
,
430 .unmap_color
= grub_video_fb_unmap_color
,
431 .fill_rect
= grub_video_fb_fill_rect
,
432 .blit_bitmap
= grub_video_fb_blit_bitmap
,
433 .blit_render_target
= grub_video_fb_blit_render_target
,
434 .scroll
= grub_video_fb_scroll
,
435 .swap_buffers
= grub_video_sis315pro_swap_buffers
,
436 .create_render_target
= grub_video_fb_create_render_target
,
437 .delete_render_target
= grub_video_fb_delete_render_target
,
438 .set_active_render_target
= grub_video_fb_set_active_render_target
,
439 .get_active_render_target
= grub_video_fb_get_active_render_target
,
444 GRUB_MOD_INIT(video_sis315pro
)
446 grub_video_register (&grub_video_sis315pro_adapter
);
449 GRUB_MOD_FINI(video_sis315pro
)
451 grub_video_unregister (&grub_video_sis315pro_adapter
);
457 grub_video_sis315pro_setup (640, 400, 0, 0);