|
|
1.1 ! root 1: // 16bit code to handle serial and printer services. ! 2: // ! 3: // Copyright (C) 2008,2009 Kevin O'Connor <[email protected]> ! 4: // Copyright (C) 2002 MandrakeSoft S.A. ! 5: // ! 6: // This file may be distributed under the terms of the GNU LGPLv3 license. ! 7: ! 8: #include "biosvar.h" // SET_BDA ! 9: #include "util.h" // debug_enter ! 10: #include "bregs.h" // struct bregs ! 11: ! 12: // Timers based on 18.2Hz clock irq. ! 13: struct tick_timer_s { ! 14: u16 last_tick, remaining; ! 15: }; ! 16: ! 17: struct tick_timer_s ! 18: initTickTimer(u16 count) ! 19: { ! 20: struct tick_timer_s tt = {GET_BDA(timer_counter), count}; ! 21: return tt; ! 22: } ! 23: ! 24: int ! 25: checkTickTimer(struct tick_timer_s *tt) ! 26: { ! 27: u16 timer = GET_BDA(timer_counter); ! 28: if (tt->last_tick != timer) { ! 29: tt->last_tick = timer; ! 30: tt->last_tick--; ! 31: if (!tt->last_tick) ! 32: return 1; ! 33: } ! 34: return 0; ! 35: } ! 36: ! 37: ! 38: /**************************************************************** ! 39: * COM ports ! 40: ****************************************************************/ ! 41: ! 42: static u16 ! 43: detect_serial(u16 port, u8 timeout, u8 count) ! 44: { ! 45: outb(0x02, port+SEROFF_IER); ! 46: u8 ier = inb(port+SEROFF_IER); ! 47: if (ier != 0x02) ! 48: return 0; ! 49: u8 iir = inb(port+SEROFF_IIR); ! 50: if ((iir & 0x3f) != 0x02) ! 51: return 0; ! 52: ! 53: outb(0x00, port+SEROFF_IER); ! 54: SET_BDA(port_com[count], port); ! 55: SET_BDA(com_timeout[count], timeout); ! 56: return 1; ! 57: } ! 58: ! 59: void ! 60: serial_setup() ! 61: { ! 62: if (! CONFIG_SERIAL) ! 63: return; ! 64: dprintf(3, "init serial\n"); ! 65: ! 66: u16 count = 0; ! 67: count += detect_serial(PORT_SERIAL1, 0x0a, count); ! 68: count += detect_serial(PORT_SERIAL2, 0x0a, count); ! 69: count += detect_serial(PORT_SERIAL3, 0x0a, count); ! 70: count += detect_serial(PORT_SERIAL4, 0x0a, count); ! 71: dprintf(1, "Found %d serial ports\n", count); ! 72: ! 73: // Equipment word bits 9..11 determing # serial ports ! 74: u16 eqb = GET_BDA(equipment_list_flags); ! 75: SET_BDA(equipment_list_flags, (eqb & 0xf1ff) | (count << 9)); ! 76: } ! 77: ! 78: static u16 ! 79: getComAddr(struct bregs *regs) ! 80: { ! 81: if (regs->dx >= 4) { ! 82: set_invalid(regs); ! 83: return 0; ! 84: } ! 85: u16 addr = GET_BDA(port_com[regs->dx]); ! 86: if (! addr) ! 87: set_invalid(regs); ! 88: return addr; ! 89: } ! 90: ! 91: // SERIAL - INITIALIZE PORT ! 92: static void ! 93: handle_1400(struct bregs *regs) ! 94: { ! 95: u16 addr = getComAddr(regs); ! 96: if (!addr) ! 97: return; ! 98: outb(inb(addr+SEROFF_LCR) | 0x80, addr+SEROFF_LCR); ! 99: if ((regs->al & 0xE0) == 0) { ! 100: outb(0x17, addr+SEROFF_DLL); ! 101: outb(0x04, addr+SEROFF_DLH); ! 102: } else { ! 103: u16 val16 = 0x600 >> ((regs->al & 0xE0) >> 5); ! 104: outb(val16 & 0xFF, addr+SEROFF_DLL); ! 105: outb(val16 >> 8, addr+SEROFF_DLH); ! 106: } ! 107: outb(regs->al & 0x1F, addr+SEROFF_LCR); ! 108: regs->ah = inb(addr+SEROFF_LSR); ! 109: regs->al = inb(addr+SEROFF_MSR); ! 110: set_success(regs); ! 111: } ! 112: ! 113: // SERIAL - WRITE CHARACTER TO PORT ! 114: static void ! 115: handle_1401(struct bregs *regs) ! 116: { ! 117: u16 addr = getComAddr(regs); ! 118: if (!addr) ! 119: return; ! 120: struct tick_timer_s tt = initTickTimer(GET_BDA(com_timeout[regs->dx])); ! 121: for (;;) { ! 122: u8 lsr = inb(addr+SEROFF_LSR); ! 123: if ((lsr & 0x60) == 0x60) { ! 124: // Success - can write data ! 125: outb(regs->al, addr+SEROFF_DATA); ! 126: // XXX - reread lsr? ! 127: regs->ah = lsr; ! 128: break; ! 129: } ! 130: if (checkTickTimer(&tt)) { ! 131: // Timed out - can't write data. ! 132: regs->ah = lsr | 0x80; ! 133: break; ! 134: } ! 135: yield(); ! 136: } ! 137: set_success(regs); ! 138: } ! 139: ! 140: // SERIAL - READ CHARACTER FROM PORT ! 141: static void ! 142: handle_1402(struct bregs *regs) ! 143: { ! 144: u16 addr = getComAddr(regs); ! 145: if (!addr) ! 146: return; ! 147: struct tick_timer_s tt = initTickTimer(GET_BDA(com_timeout[regs->dx])); ! 148: for (;;) { ! 149: u8 lsr = inb(addr+SEROFF_LSR); ! 150: if (lsr & 0x01) { ! 151: // Success - can read data ! 152: regs->al = inb(addr+SEROFF_DATA); ! 153: regs->ah = lsr; ! 154: break; ! 155: } ! 156: if (checkTickTimer(&tt)) { ! 157: // Timed out - can't read data. ! 158: regs->ah = lsr | 0x80; ! 159: break; ! 160: } ! 161: yield(); ! 162: } ! 163: set_success(regs); ! 164: } ! 165: ! 166: // SERIAL - GET PORT STATUS ! 167: static void ! 168: handle_1403(struct bregs *regs) ! 169: { ! 170: u16 addr = getComAddr(regs); ! 171: if (!addr) ! 172: return; ! 173: regs->ah = inb(addr+SEROFF_LSR); ! 174: regs->al = inb(addr+SEROFF_MSR); ! 175: set_success(regs); ! 176: } ! 177: ! 178: static void ! 179: handle_14XX(struct bregs *regs) ! 180: { ! 181: set_unimplemented(regs); ! 182: } ! 183: ! 184: // INT 14h Serial Communications Service Entry Point ! 185: void VISIBLE16 ! 186: handle_14(struct bregs *regs) ! 187: { ! 188: debug_enter(regs, DEBUG_HDL_14); ! 189: if (! CONFIG_SERIAL) { ! 190: handle_14XX(regs); ! 191: return; ! 192: } ! 193: ! 194: switch (regs->ah) { ! 195: case 0x00: handle_1400(regs); break; ! 196: case 0x01: handle_1401(regs); break; ! 197: case 0x02: handle_1402(regs); break; ! 198: case 0x03: handle_1403(regs); break; ! 199: default: handle_14XX(regs); break; ! 200: } ! 201: } ! 202: ! 203: // XXX - Baud Rate Generator Table ! 204: u8 BaudTable[16] VAR16FIXED(0xe729); ! 205: ! 206: ! 207: /**************************************************************** ! 208: * LPT ports ! 209: ****************************************************************/ ! 210: ! 211: static u16 ! 212: detect_parport(u16 port, u8 timeout, u8 count) ! 213: { ! 214: // clear input mode ! 215: outb(inb(port+2) & 0xdf, port+2); ! 216: ! 217: outb(0xaa, port); ! 218: if (inb(port) != 0xaa) ! 219: // Not present ! 220: return 0; ! 221: SET_BDA(port_lpt[count], port); ! 222: SET_BDA(lpt_timeout[count], timeout); ! 223: return 1; ! 224: } ! 225: ! 226: void ! 227: lpt_setup() ! 228: { ! 229: if (! CONFIG_LPT) ! 230: return; ! 231: dprintf(3, "init lpt\n"); ! 232: ! 233: u16 count = 0; ! 234: count += detect_parport(PORT_LPT1, 0x14, count); ! 235: count += detect_parport(PORT_LPT2, 0x14, count); ! 236: dprintf(1, "Found %d lpt ports\n", count); ! 237: ! 238: // Equipment word bits 14..15 determing # parallel ports ! 239: u16 eqb = GET_BDA(equipment_list_flags); ! 240: SET_BDA(equipment_list_flags, (eqb & 0x3fff) | (count << 14)); ! 241: } ! 242: ! 243: static u16 ! 244: getLptAddr(struct bregs *regs) ! 245: { ! 246: if (regs->dx >= 3) { ! 247: set_invalid(regs); ! 248: return 0; ! 249: } ! 250: u16 addr = GET_BDA(port_lpt[regs->dx]); ! 251: if (! addr) ! 252: set_invalid(regs); ! 253: return addr; ! 254: } ! 255: ! 256: // INT 17 - PRINTER - WRITE CHARACTER ! 257: static void ! 258: handle_1700(struct bregs *regs) ! 259: { ! 260: u16 addr = getLptAddr(regs); ! 261: if (!addr) ! 262: return; ! 263: ! 264: struct tick_timer_s tt = initTickTimer(GET_BDA(lpt_timeout[regs->dx])); ! 265: ! 266: outb(regs->al, addr); ! 267: u8 val8 = inb(addr+2); ! 268: outb(val8 | 0x01, addr+2); // send strobe ! 269: udelay(5); ! 270: outb(val8 & ~0x01, addr+2); ! 271: ! 272: for (;;) { ! 273: u8 v = inb(addr+1); ! 274: if (!(v & 0x40)) { ! 275: // Success ! 276: regs->ah = v ^ 0x48; ! 277: break; ! 278: } ! 279: if (checkTickTimer(&tt)) { ! 280: // Timeout ! 281: regs->ah = (v ^ 0x48) | 0x01; ! 282: break; ! 283: } ! 284: yield(); ! 285: } ! 286: ! 287: set_success(regs); ! 288: } ! 289: ! 290: // INT 17 - PRINTER - INITIALIZE PORT ! 291: static void ! 292: handle_1701(struct bregs *regs) ! 293: { ! 294: u16 addr = getLptAddr(regs); ! 295: if (!addr) ! 296: return; ! 297: ! 298: u8 val8 = inb(addr+2); ! 299: outb(val8 & ~0x04, addr+2); // send init ! 300: udelay(5); ! 301: outb(val8 | 0x04, addr+2); ! 302: ! 303: regs->ah = inb(addr+1) ^ 0x48; ! 304: set_success(regs); ! 305: } ! 306: ! 307: // INT 17 - PRINTER - GET STATUS ! 308: static void ! 309: handle_1702(struct bregs *regs) ! 310: { ! 311: u16 addr = getLptAddr(regs); ! 312: if (!addr) ! 313: return; ! 314: regs->ah = inb(addr+1) ^ 0x48; ! 315: set_success(regs); ! 316: } ! 317: ! 318: static void ! 319: handle_17XX(struct bregs *regs) ! 320: { ! 321: set_unimplemented(regs); ! 322: } ! 323: ! 324: // INT17h : Printer Service Entry Point ! 325: void VISIBLE16 ! 326: handle_17(struct bregs *regs) ! 327: { ! 328: debug_enter(regs, DEBUG_HDL_17); ! 329: if (! CONFIG_LPT) { ! 330: handle_17XX(regs); ! 331: return; ! 332: } ! 333: ! 334: switch (regs->ah) { ! 335: case 0x00: handle_1700(regs); break; ! 336: case 0x01: handle_1701(regs); break; ! 337: case 0x02: handle_1702(regs); break; ! 338: default: handle_17XX(regs); break; ! 339: } ! 340: }
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.