/* * vim:tabstop=4 * vim:noexpandtab * * Bochs VBE / QEMU vga=std Graphics Driver */ #include #include uint16_t bochs_resolution_x = 1024; uint16_t bochs_resolution_y = 768; uint16_t bochs_resolution_b = 32; uint16_t bochs_current_bank = 0; #define BOCHS_BANK_SIZE 16384 #define BOCHS_VID_MEMORY ((uint32_t *)0xA0000) #define BOCHS_BANKS (bochs_resolution_x * bochs_resolution_y * bochs_resolution_b / (BOCHS_BANK_SIZE * 32)) void graphics_install_bochs() { outports(0x1CE, 0x00); uint16_t i = inports(0x1CF); if (i < 0xB0C0 || i > 0xB0C6) { kprintf("[bochs] You are not a Bochs VBE pseudo-card!\n"); kprintf("[bochs] 0x%x is totally wrong!\n", (unsigned int)i); return; } kprintf("[bochs] Successfully detected a Bochs VBE setup!\n"); kprintf("[bochs] You are using QEMU or Bochs and I love you.\n"); outports(0x1CF, 0xB0C4); i = inports(0x1CF); kprintf("[bochs] Enabling 1024x768x32 graphics mode!\n"); /* Disable VBE */ outports(0x1CE, 0x04); outports(0x1CF, 0x00); /* Set X resolution to 1024 */ outports(0x1CE, 0x01); outports(0x1CF, 1024); bochs_resolution_x = 1024; /* Set Y resolution to 768 */ outports(0x1CE, 0x02); outports(0x1CF, 768); bochs_resolution_y = 768; /* Set bpp to 32 */ outports(0x1CE, 0x03); outports(0x1CF, 0x20); bochs_resolution_b = 32; /* Re-enable VBE */ outports(0x1CE, 0x04); outports(0x1CF, 0x01); } void bochs_set_bank( uint16_t bank ) { if (bank == bochs_current_bank) { /* We are already in this bank, stop wasting cycles */ return; } outports(0x1CE, 0x05); /* Bank */ outports(0x1CF, bank); bochs_current_bank = bank; } void bochs_set_coord( uint16_t x, uint16_t y, uint32_t color ) { uint32_t location = y * bochs_resolution_x + x; bochs_set_bank(location / BOCHS_BANK_SIZE); uint32_t offset = location % BOCHS_BANK_SIZE; BOCHS_VID_MEMORY[offset] = color; } void bochs_scroll() { __asm__ __volatile__ ("cli"); uint32_t * bank_store = malloc(sizeof(uint32_t) * BOCHS_BANK_SIZE); for (uint32_t i = 1; i < BOCHS_BANKS; ++i) { bochs_set_bank(i); memcpy(bank_store, BOCHS_VID_MEMORY, sizeof(uint32_t) * BOCHS_BANK_SIZE); bochs_set_bank(i - 1); memcpy(BOCHS_VID_MEMORY, bank_store, sizeof(uint32_t) * BOCHS_BANK_SIZE); } free(bank_store); __asm__ __volatile__ ("sti"); } void bochs_draw_logo() { /* This is slow and ineffecient, but it's also dead simple. */ fs_node_t * file = kopen("/bs.bmp",0); char *bufferb = malloc(file->length); /* Read the boot logo */ size_t bytes_read = read_fs(file, 0, file->length, (uint8_t *)bufferb); size_t i; uint16_t x = 0; /* -> 212 */ uint16_t y = 0; /* -> 68 */ /* Get the width / height of the image */ signed int *bufferi = (signed int *)((uintptr_t)bufferb + 2); uint32_t width = bufferi[4]; uint32_t height = bufferi[5]; /* Skip right to the important part */ for (i = 54; i < bytes_read; i += 3) { /* Extract the color */ uint32_t color = bufferb[i] + bufferb[i+1] * 0x100 + bufferb[i+2] * 0x10000; /* Set our point */ bochs_set_coord((bochs_resolution_x - width) / 2 + x, (bochs_resolution_y - height) / 2 + (height - y), color); ++x; /* If we hit the end of the line, move on */ if (x == width) { x = 0; ++y; } } free(bufferb); }