/* * 86Box A hypervisor and IBM PC system emulator that specializes in * running old operating systems and software designed for IBM * PC systems and compatibles from 1981 through fairly recent * system designs based on the PCI bus. * * This file is part of the 86Box distribution. * * Hercules emulation. * * * * Authors: Sarah Walker, * Miran Grca, * * Copyright 2008-2019 Sarah Walker. * Copyright 2016-2019 Miran Grca. */ #include #include #include #include #include #include <86box/86box.h> #include "cpu.h" #include <86box/mem.h> #include <86box/rom.h> #include <86box/io.h> #include <86box/timer.h> #include <86box/lpt.h> #include <86box/pit.h> #include <86box/device.h> #include <86box/video.h> typedef struct { mem_mapping_t mapping; uint8_t crtc[32]; int crtcreg; uint8_t ctrl, ctrl2, stat; uint64_t dispontime, dispofftime; pc_timer_t timer; int firstline, lastline; int linepos, displine; int vc, sc; uint16_t ma, maback; int con, coff, cursoron; int dispon, blink; int vsynctime; int vadj; int cols[256][2][2]; uint8_t *vram; } hercules_t; static video_timings_t timing_hercules = {VIDEO_ISA, 8, 16, 32, 8, 16, 32}; static void recalc_timings(hercules_t *dev) { double disptime; double _dispontime, _dispofftime; disptime = dev->crtc[0] + 1; _dispontime = dev->crtc[1]; _dispofftime = disptime - _dispontime; _dispontime *= HERCCONST; _dispofftime *= HERCCONST; dev->dispontime = (uint64_t)(_dispontime); dev->dispofftime = (uint64_t)(_dispofftime); } static uint8_t crtcmask[32] = { 0xff, 0xff, 0xff, 0xff, 0x7f, 0x1f, 0x7f, 0x7f, 0xf3, 0x1f, 0x7f, 0x1f, 0x3f, 0xff, 0x3f, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }; static void hercules_out(uint16_t addr, uint8_t val, void *priv) { hercules_t *dev = (hercules_t *)priv; uint8_t old; switch (addr) { case 0x03b0: case 0x03b2: case 0x03b4: case 0x03b6: dev->crtcreg = val & 31; break; case 0x03b1: case 0x03b3: case 0x03b5: case 0x03b7: old = dev->crtc[dev->crtcreg]; dev->crtc[dev->crtcreg] = val & crtcmask[dev->crtcreg]; /* * Fix for Generic Turbo XT BIOS, which * sets up cursor registers wrong. */ if (dev->crtc[10] == 6 && dev->crtc[11] == 7) { dev->crtc[10] = 0xb; dev->crtc[11] = 0xc; } #if 0 if (old ^ val) recalc_timings(dev); #else if (old != val) { if ((dev->crtcreg < 0xe) || (dev->crtcreg > 0x10)) { fullchange = changeframecount; recalc_timings(dev); } } #endif break; case 0x03b8: old = dev->ctrl; if (!(dev->ctrl2 & 0x01) && !(val & 0x02)) val = (val & 0xfd) | (dev->ctrl & 0x02); if (!(dev->ctrl2 & 0x02) && !(val & 0x80)) val = (val & 0x7f) | (dev->ctrl & 0x80); dev->ctrl = val; if (old ^ val) recalc_timings(dev); break; case 0x03bf: dev->ctrl2 = val; if (val & 0x02) mem_mapping_set_addr(&dev->mapping, 0xb0000, 0x10000); else mem_mapping_set_addr(&dev->mapping, 0xb0000, 0x08000); break; default: break; } } static uint8_t hercules_in(uint16_t addr, void *priv) { hercules_t *dev = (hercules_t *)priv; uint8_t ret = 0xff; switch (addr) { case 0x03b0: case 0x03b2: case 0x03b4: case 0x03b6: ret = dev->crtcreg; break; case 0x03b1: case 0x03b3: case 0x03b5: case 0x03b7: ret = dev->crtc[dev->crtcreg]; break; case 0x03ba: ret = 0x72; /* Hercules ident */ #if 0 if (dev->stat & 0x08) ret |= 0x88; #else if (dev->stat & 0x08) ret |= 0x80; #endif if ((dev->stat & 0x09) == 0x01) ret |= (dev->stat & 0x01); if ((ret & 0x81) == 0x80) ret |= 0x08; break; default: break; } return(ret); } static void hercules_waitstates(void *p) { int ws_array[16] = {3, 4, 5, 6, 7, 8, 4, 5, 6, 7, 8, 4, 5, 6, 7, 8}; int ws; ws = ws_array[cycles & 0xf]; cycles -= ws; } static void hercules_write(uint32_t addr, uint8_t val, void *priv) { hercules_t *dev = (hercules_t *)priv; if (dev->ctrl2 & 0x01) dev->vram[addr & 0xffff] = val; else dev->vram[addr & 0x0fff] = val; hercules_waitstates(dev); } static uint8_t hercules_read(uint32_t addr, void *priv) { hercules_t *dev = (hercules_t *)priv; if (dev->ctrl2 & 0x01) return(dev->vram[addr & 0xffff]); else return(dev->vram[addr & 0x0fff]); hercules_waitstates(dev); } static void hercules_poll(void *priv) { hercules_t *dev = (hercules_t *)priv; uint8_t chr, attr; uint16_t ca, dat; int oldsc, blink; int x, c, oldvc; int drawcursor; ca = (dev->crtc[15] | (dev->crtc[14] << 8)) & 0x3fff; if (! dev->linepos) { timer_advance_u64(&dev->timer, dev->dispofftime); dev->stat |= 1; dev->linepos = 1; oldsc = dev->sc; if ((dev->crtc[8] & 3) == 3) dev->sc = (dev->sc << 1) & 7; if (dev->dispon) { if (dev->displine < dev->firstline) { dev->firstline = dev->displine; video_wait_for_buffer(); } dev->lastline = dev->displine; // if ((dev->ctrl & 2) && (dev->ctrl2 & 1)) { if (dev->ctrl & 2) { ca = (dev->sc & 3) * 0x2000; // if ((dev->ctrl & 0x80) && (dev->ctrl2 & 2)) if (dev->ctrl & 0x80) ca += 0x8000; for (x = 0; x < dev->crtc[1]; x++) { if (dev->ctrl & 8) // dat = (dev->vram[((dev->ma << 1) & 0x1fff) + ca] << 8) | dev->vram[((dev->ma << 1) & 0x1fff) + ca + 1]; dat = (dev->vram[((dev->ma << 1) & 0x1fff) + ca] << 8) | dev->vram[((dev->ma << 1) & 0x1fff) + ca + 1]; else dat = 0; dev->ma++; for (c = 0; c < 16; c++) buffer32->line[dev->displine][(x << 4) + c] = (dat & (32768 >> c)) ? 7 : 0; for (c = 0; c < 16; c += 8) video_blend((x << 4) + c, dev->displine); } } else { for (x = 0; x < dev->crtc[1]; x++) { if (dev->ctrl & 8) { chr = dev->vram[(dev->ma << 1) & 0xfff]; attr = dev->vram[((dev->ma << 1) + 1) & 0xfff]; } else chr = attr = 0; drawcursor = ((dev->ma == ca) && dev->con && dev->cursoron); blink = ((dev->blink & 16) && (dev->ctrl & 0x20) && (attr & 0x80) && !drawcursor); if (dev->sc == 12 && ((attr & 7) == 1)) { for (c = 0; c < 9; c++) buffer32->line[dev->displine][(x * 9) + c] = dev->cols[attr][blink][1]; } else { for (c = 0; c < 8; c++) buffer32->line[dev->displine][(x * 9) + c] = dev->cols[attr][blink][(fontdatm[chr][dev->sc] & (1 << (c ^ 7))) ? 1 : 0]; if ((chr & ~0x1f) == 0xc0) buffer32->line[dev->displine][(x * 9) + 8] = dev->cols[attr][blink][fontdatm[chr][dev->sc] & 1]; else buffer32->line[dev->displine][(x * 9) + 8] = dev->cols[attr][blink][0]; } dev->ma++; if (drawcursor) { for (c = 0; c < 9; c++) buffer32->line[dev->displine][(x * 9) + c] ^= dev->cols[attr][0][1]; } } } } dev->sc = oldsc; if (dev->vc == dev->crtc[7] && !dev->sc) dev->stat |= 8; dev->displine++; if (dev->displine >= 500) dev->displine = 0; } else { timer_advance_u64(&dev->timer, dev->dispontime); dev->linepos = 0; if (dev->vsynctime) { dev->vsynctime--; if (! dev->vsynctime) dev->stat &= ~8; } if (dev->sc == (dev->crtc[11] & 31) || ((dev->crtc[8] & 3)==3 && dev->sc == ((dev->crtc[11] & 31) >> 1))) { dev->con = 0; dev->coff = 1; } if (dev->vadj) { dev->sc++; dev->sc &= 31; dev->ma = dev->maback; dev->vadj--; if (! dev->vadj) { dev->dispon = 1; dev->ma = dev->maback = (dev->crtc[13] | (dev->crtc[12] << 8)) & 0x3fff; dev->sc = 0; } } else if (dev->sc == dev->crtc[9] || ((dev->crtc[8] & 3) == 3 && dev->sc == (dev->crtc[9] >> 1))) { dev->maback = dev->ma; dev->sc = 0; oldvc = dev->vc; dev->vc++; dev->vc &= 127; if (dev->vc == dev->crtc[6]) dev->dispon = 0; if (oldvc == dev->crtc[4]) { dev->vc = 0; dev->vadj = dev->crtc[5]; if (! dev->vadj) dev->dispon = 1; if (! dev->vadj) dev->ma = dev->maback = (dev->crtc[13] | (dev->crtc[12] << 8)) & 0x3fff; if ((dev->crtc[10] & 0x60) == 0x20) dev->cursoron = 0; else dev->cursoron = dev->blink & 16; } if (dev->vc == dev->crtc[7]) { dev->dispon = 0; dev->displine = 0; dev->vsynctime = 16;//(crtcm[3]>>4)+1; if (dev->crtc[7]) { // if ((dev->ctrl & 2) && (dev->ctrl2 & 1)) if (dev->ctrl & 2) x = dev->crtc[1] << 4; else x = dev->crtc[1] * 9; dev->lastline++; if ((dev->ctrl & 8) && x && (dev->lastline - dev->firstline) && ((x != xsize) || ((dev->lastline - dev->firstline) != ysize) || video_force_resize_get())) { xsize = x; ysize = dev->lastline - dev->firstline; if (xsize < 64) xsize = 656; if (ysize < 32) ysize = 200; set_screen_size(xsize, ysize); if (video_force_resize_get()) video_force_resize_set(0); } video_blit_memtoscreen_8(0, dev->firstline, 0, ysize, xsize, ysize); frames++; if ((dev->ctrl & 2) && (dev->ctrl2 & 1)) { video_res_x = dev->crtc[1] * 16; video_res_y = dev->crtc[6] * 4; video_bpp = 1; } else { video_res_x = dev->crtc[1]; video_res_y = dev->crtc[6]; video_bpp = 0; } } dev->firstline = 1000; dev->lastline = 0; dev->blink++; } } else { dev->sc++; dev->sc &= 31; dev->ma = dev->maback; } if (dev->dispon) dev->stat &= ~1; if ((dev->sc == (dev->crtc[10] & 31) || ((dev->crtc[8] & 3)==3 && dev->sc == ((dev->crtc[10] & 31) >> 1)))) dev->con = 1; } } static void * hercules_init(const device_t *info) { hercules_t *dev; int c; dev = (hercules_t *)malloc(sizeof(hercules_t)); memset(dev, 0x00, sizeof(hercules_t)); dev->vram = (uint8_t *)malloc(0x10000); timer_add(&dev->timer, hercules_poll, dev, 1); mem_mapping_add(&dev->mapping, 0xb0000, 0x10000, hercules_read,NULL,NULL, hercules_write,NULL,NULL, dev->vram, MEM_MAPPING_EXTERNAL, dev); io_sethandler(0x03b0, 16, hercules_in,NULL,NULL, hercules_out,NULL,NULL, dev); for (c = 0; c < 256; c++) { dev->cols[c][0][0] = dev->cols[c][1][0] = dev->cols[c][1][1] = 16; if (c & 0x08) dev->cols[c][0][1] = 15 + 16; else dev->cols[c][0][1] = 7 + 16; } dev->cols[0x70][0][1] = 16; dev->cols[0x70][0][0] = dev->cols[0x70][1][0] = dev->cols[0x70][1][1] = 16 + 15; dev->cols[0xF0][0][1] = 16; dev->cols[0xF0][0][0] = dev->cols[0xF0][1][0] = dev->cols[0xF0][1][1] = 16 + 15; dev->cols[0x78][0][1] = 16 + 7; dev->cols[0x78][0][0] = dev->cols[0x78][1][0] = dev->cols[0x78][1][1] = 16 + 15; dev->cols[0xF8][0][1] = 16 + 7; dev->cols[0xF8][0][0] = dev->cols[0xF8][1][0] = dev->cols[0xF8][1][1] = 16 + 15; dev->cols[0x00][0][1] = dev->cols[0x00][1][1] = 16; dev->cols[0x08][0][1] = dev->cols[0x08][1][1] = 16; dev->cols[0x80][0][1] = dev->cols[0x80][1][1] = 16; dev->cols[0x88][0][1] = dev->cols[0x88][1][1] = 16; overscan_x = overscan_y = 0; cga_palette = device_get_config_int("rgb_type") << 1; if (cga_palette > 6) cga_palette = 0; cgapal_rebuild(); herc_blend = device_get_config_int("blend"); video_inform(VIDEO_FLAG_TYPE_MDA, &timing_hercules); /* Force the LPT3 port to be enabled. */ lpt3_init(0x3BC); return(dev); } static void hercules_close(void *priv) { hercules_t *dev = (hercules_t *)priv; if (!dev) return; if (dev->vram) free(dev->vram); free(dev); } static void speed_changed(void *priv) { hercules_t *dev = (hercules_t *)priv; recalc_timings(dev); } static const device_config_t hercules_config[] = { { "rgb_type", "Display type", CONFIG_SELECTION, "", 0, "", { 0 }, { { "Default", 0 }, { "Green", 1 }, { "Amber", 2 }, { "Gray", 3 }, { "" } } }, { "blend", "Blend", CONFIG_BINARY, "", 1 }, { "", "", -1 } }; const device_t hercules_device = { "Hercules", DEVICE_ISA, 0, hercules_init, hercules_close, NULL, { NULL }, speed_changed, NULL, hercules_config };