[e16e8f2] | 1 | /* |
---|
| 2 | * writechr: Write a single character in AL to the console without |
---|
| 3 | * mangling any registers. This does raw console writes, |
---|
| 4 | * since some PXE BIOSes seem to interfere regular console I/O. |
---|
| 5 | */ |
---|
| 6 | #include <sys/io.h> |
---|
| 7 | #include <fs.h> |
---|
| 8 | #include <com32.h> |
---|
| 9 | |
---|
| 10 | #include "bios.h" |
---|
| 11 | #include "graphics.h" |
---|
| 12 | #include <syslinux/video.h> |
---|
| 13 | |
---|
| 14 | __export void writechr(char data) |
---|
| 15 | { |
---|
| 16 | if (UsingVGA & 0x08) |
---|
| 17 | syslinux_force_text_mode(); |
---|
| 18 | |
---|
| 19 | write_serial(data); /* write to serial port if needed */ |
---|
| 20 | |
---|
| 21 | /* Write to screen? */ |
---|
| 22 | if (DisplayCon & 0x01) { |
---|
| 23 | com32sys_t ireg, oreg; |
---|
| 24 | bool curxyok = false; |
---|
| 25 | uint16_t dx; |
---|
| 26 | |
---|
| 27 | memset(&ireg, 0, sizeof ireg); |
---|
| 28 | ireg.ebx.b[1] = *(uint8_t *)BIOS_page; |
---|
| 29 | ireg.eax.b[1] = 0x03; /* Read cursor position */ |
---|
| 30 | __intcall(0x10, &ireg, &oreg); |
---|
| 31 | ireg.edx.l = oreg.edx.l; |
---|
| 32 | |
---|
| 33 | switch (data) { |
---|
| 34 | case 8: |
---|
| 35 | if (ireg.edx.b[0]--) { |
---|
| 36 | curxyok = true; |
---|
| 37 | break; |
---|
| 38 | } |
---|
| 39 | |
---|
| 40 | ireg.edx.b[0] = VidCols; |
---|
| 41 | if (ireg.edx.b[1]--) { |
---|
| 42 | curxyok = true; |
---|
| 43 | break; |
---|
| 44 | } |
---|
| 45 | |
---|
| 46 | ireg.edx.b[1] = 0; |
---|
| 47 | curxyok = true; |
---|
| 48 | break; |
---|
| 49 | case 13: |
---|
| 50 | ireg.edx.b[0] = 0; |
---|
| 51 | curxyok = true; |
---|
| 52 | break; |
---|
| 53 | case 10: |
---|
| 54 | break; |
---|
| 55 | default: |
---|
| 56 | dx = ireg.edx.w[0]; |
---|
| 57 | |
---|
| 58 | ireg.ebx.b[1] = *(uint8_t *)BIOS_page; |
---|
| 59 | ireg.ebx.b[0] = 0x07; /* White on black */ |
---|
| 60 | ireg.ecx.w[0] = 1; /* One only */ |
---|
| 61 | ireg.eax.b[0] = data; |
---|
| 62 | ireg.eax.b[1] = 0x09; /* Write char and attribute */ |
---|
| 63 | __intcall(0x10, &ireg, NULL); |
---|
| 64 | |
---|
| 65 | ireg.edx.w[0] = dx; |
---|
| 66 | if (++ireg.edx.b[0] <= VidCols) |
---|
| 67 | curxyok = true; |
---|
| 68 | else |
---|
| 69 | ireg.edx.b[0] = 0; |
---|
| 70 | } |
---|
| 71 | |
---|
| 72 | if (!curxyok && ++ireg.edx.b[1] > VidRows) { |
---|
| 73 | /* Scroll */ |
---|
| 74 | ireg.edx.b[1]--; |
---|
| 75 | ireg.ebx.b[1] = *(uint8_t *)BIOS_page; |
---|
| 76 | ireg.eax.b[1] = 0x02; |
---|
| 77 | __intcall(0x10, &ireg, NULL); |
---|
| 78 | |
---|
| 79 | ireg.eax.w[0] = 0x0601; /* Scroll up one line */ |
---|
| 80 | ireg.ebx.b[1] = ScrollAttribute; |
---|
| 81 | ireg.ecx.w[0] = 0; |
---|
| 82 | ireg.edx.w[0] = ScreenSize; /* The whole screen */ |
---|
| 83 | __intcall(0x10, &ireg, NULL); |
---|
| 84 | } else { |
---|
| 85 | ireg.ebx.b[1] = *(uint8_t *)BIOS_page; |
---|
| 86 | ireg.eax.b[1] = 0x02; /* Set cursor position */ |
---|
| 87 | __intcall(0x10, &ireg, NULL); |
---|
| 88 | } |
---|
| 89 | } |
---|
| 90 | } |
---|
| 91 | |
---|
| 92 | void pm_writechr(com32sys_t *regs) |
---|
| 93 | { |
---|
| 94 | writechr(regs->eax.b[0]); |
---|
| 95 | } |
---|