00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030 #include <isa/atkbd.h>
00031 #include <isa/8259.h>
00032 #include <sys/video.h>
00033 #include <sys/idt.h>
00034 #include <sys/gdt.h>
00035 #include <sys/io.h>
00036 #include <lib/kmalloc.h>
00037 #include <lib/kprintf.h>
00038 #include <ubixos/types.h>
00039 #include <ubixos/sched.h>
00040 #include <ubixos/endtask.h>
00041 #include <ubixos/tty.h>
00042 #include <ubixos/spinlock.h>
00043
00044 static int atkbd_scan();
00045
00046 static unsigned int keyMap = 0x0;
00047 static unsigned int ledStatus = 0x0;
00048 static char stdinBuffer[512];
00049 static uInt16 stdinSize;
00050 static uInt32 controlKeys = 0x0;
00051
00052 static spinLock_t atkbdSpinLock = SPIN_LOCK_INITIALIZER;
00053
00054 static unsigned int keyboardMap[255][8] = {
00055
00056 { 0, 0, 0, 0, 0, 0, 0, 0},
00057 { 0x1B, 0x1B, 0x1B, 0x1B, 0x1B, 0x1B, 0x1B, 0x1B},
00058 { 0x31, 0x21, 0, 0, 0x31, 0x31, 0x21, 0x21},
00059 { 0x32, 0x40, 0, 0, 0x32, 0x32, 0x40, 0x40},
00060 { 0x33, 0x23, 0, 0, 0x33, 0x33, 0x23, 0x23},
00061 { 0x34, 0x24, 0, 0, 0x34, 0x34, 0x24, 0x24},
00062 { 0x35, 0x25, 0, 0, 0x35, 0x35, 0x25, 0x25},
00063 { 0x36, 0x5E, 0, 0, 0x36, 0x36, 0x5E, 0x5E},
00064 { 0x37, 0x26, 0, 0, 0x37, 0x37, 0x26, 0x26},
00065 { 0x38, 0x2A, 0, 0, 0x38, 0x38, 0x2A, 0x2A},
00066 { 0x39, 0x28, 0, 0, 0x39, 0x39, 0x28, 0x28},
00067 { 0x30, 0x29, 0, 0, 0x30, 0x30, 0x29, 0x29},
00068 { 0x2D, 0x5F, 0, 0, 0x2D, 0x2D, 0x5F, 0x5F},
00069 { 0x3D, 0x2B, 0, 0, 0x3D, 0x3D, 0x2B, 0x2B},
00070 { 0x08, 0x08, 0x8, 0x8, 0x08, 0x08, 0x08, 0x08},
00071 { 0x09, 0, 0, 0, 0, 0, 0, 0},
00072 { 0x71, 0x51, 0, 0, 0, 0, 0, 0},
00073 { 0x77, 0x57, 0, 0, 0, 0, 0, 0},
00074 { 0x65, 0x45, 0, 0, 0, 0, 0, 0},
00075 { 0x72, 0x52, 0, 0, 0, 0, 0, 0},
00076 { 0x74, 0x54, 0, 0, 0, 0, 0, 0},
00077 { 0x79, 0x59, 0, 0, 0, 0, 0, 0},
00078 { 0x75, 0x55, 0, 0, 0, 0, 0, 0},
00079 { 0x69, 0x49, 0, 0, 0, 0, 0, 0},
00080 { 0x6F, 0x4F, 0, 0, 0, 0, 0, 0},
00081 { 0x70, 0x50, 0, 0, 0, 0, 0, 0},
00082 { 0x5B, 0x7B, 0, 0, 0, 0, 0, 0},
00083 { 0x5D, 0x7D, 0, 0, 0, 0, 0, 0},
00084 { 0x0A, 0, 0, 0, 0, 0, 0, 0},
00085 { 0, 0, 0, 0, 0, 0, 0, 0},
00086 { 0x61, 0x41, 0x41, 0, 0, 0, 0, 0},
00087 { 0x73, 0x53, 0, 0, 0, 0, 0, 0},
00088 { 0x64, 0x44, 0, 0, 0, 0, 0, 0},
00089 { 0x66, 0x46, 0, 0, 0, 0, 0, 0},
00090 { 0x67, 0x47, 0, 0, 0, 0, 0, 0},
00091 { 0x68, 0x48, 0, 0, 0, 0, 0, 0},
00092 { 0x6A, 0x4A, 0, 0, 0, 0, 0, 0},
00093 { 0x6B, 0x4B, 0, 0, 0, 0, 0, 0},
00094 { 0x6C, 0x4C, 0, 0, 0, 0, 0, 0},
00095 { 0x3B, 0x3A, 0, 0, 0, 0, 0, 0},
00096 { 0x27, 0x22, 0, 0, 0, 0, 0, 0},
00097 { 0x60, 0x7E, 0, 0, 0, 0, 0, 0},
00098 { 0x2A, 0x0, 0, 0, 0, 0, 0, 0},
00099 { 0x5C, 0x3C, 0, 0, 0, 0, 0, 0},
00100 { 0x7A, 0x5A, 0, 0, 0, 0, 0, 0},
00101 { 0x78, 0x58, 0, 0, 0, 0, 0, 0},
00102 { 0x63, 0x43, 0x3, 0x9, 0, 0, 0, 0},
00103 { 0x76, 0x56, 0, 0, 0, 0, 0, 0},
00104 { 0x62, 0x42, 0, 0, 0, 0, 0, 0},
00105 { 0x6E, 0x4E, 0, 0, 0, 0, 0, 0},
00106 { 0x6D, 0x4D, 0, 0, 0, 0, 0, 0},
00107 { 0x2C, 0x3C, 0, 0, 0, 0, 0, 0},
00108 { 0x2E, 0x3E, 0, 0, 0, 0, 0, 0},
00109 { 0x2F, 0x3F, 0, 0, 0, 0, 0, 0},
00110 { 0, 0, 0, 0, 0, 0, 0, 0},
00111 { 0, 0, 0, 0, 0, 0, 0, 0},
00112 { 0, 0, 0, 0, 0, 0, 0, 0},
00113 { 0x20, 0, 0, 0, 0, 0, 0, 0},
00114 { 0, 0, 0, 0, 0, 0, 0, 0},
00115 { 0x3000, 0, 0, 0x3000, 0, 0, 0, 0},
00116 { 0x3001, 0, 0, 0x3001, 0, 0, 0, 0},
00117 { 0x3002, 0, 0, 0x3002, 0, 0, 0, 0},
00118 { 0x3003, 0, 0, 0x3003, 0, 0, 0, 0},
00119 { 0x3004, 0, 0, 0x3004, 0, 0, 0, 0},
00120 { 0x4000, 0, 0, 0, 0, 0, 0, 0},
00121 { 0x4100, 0, 0, 0, 0, 0, 0, 0},
00122 { 0x4200, 0, 0, 0, 0, 0, 0, 0},
00123 { 0x4300, 0, 0, 0, 0, 0, 0, 0},
00124 { 0x4400, 0, 0, 0, 0, 0, 0, 0},
00125 { 0, 0, 0, 0, 0, 0, 0, 0},
00126 { 0, 0, 0, 0, 0, 0, 0, 0},
00127 { 0x4700, 0, 0, 0, 0, 0, 0, 0},
00128 { 0x4800, 0, 0, 0, 0, 0, 0, 0},
00129 { 0x4900, 0, 0, 0, 0, 0, 0, 0},
00130 { 0x2D, 0, 0, 0, 0, 0, 0, 0},
00131 { 0x4B00, 0, 0, 0, 0, 0, 0, 0},
00132 { 0x4C00, 0, 0, 0, 0, 0, 0, 0},
00133 { 0x4D00, 0, 0, 0, 0, 0, 0, 0},
00134 { 0x2B, 0, 0, 0, 0, 0, 0, 0},
00135 { 0x4F00, 0, 0, 0, 0, 0, 0, 0},
00136 { 0x5000, 0, 0, 0, 0, 0, 0, 0},
00137 { 0x5100, 0, 0, 0, 0, 0, 0, 0},
00138 { 0x5200, 0, 0, 0, 0, 0, 0, 0},
00139 { 0x5300, 0, 0, 0, 0, 0, 0, 0},
00140 { 0, 0, 0, 0, 0, 0, 0, 0},
00141 { 0, 0, 0, 0, 0, 0, 0, 0}
00142 };
00143
00144
00145
00146
00147
00148
00149
00150
00151
00152
00153
00154
00155 int atkbd_init() {
00156
00157 setVector(&atkbd_isr, mVec+0x1, dPresent + dInt + dDpl0);
00158
00159
00160 setLED();
00161
00162
00163 atkbd_scan();
00164
00165
00166 irqEnable(0x1);
00167
00168
00169 kprintf("atkbd0 - Address: [0x%X], Keyboard Buffer: [0x%X], Buffer Size [%i]\n",&atkbd_isr,&stdinBuffer,512);
00170
00171
00172 return(0x0);
00173 }
00174
00175
00176
00177
00178
00179 asm(
00180 ".globl atkbd_isr \n"
00181 "atkbd_isr: \n"
00182 " pusha \n"
00183 " push %ss \n"
00184 " push %ds \n"
00185 " push %es \n"
00186 " push %fs \n"
00187 " push %gs \n"
00188 " call keyboardHandler \n"
00189 " mov $0x20,%dx \n"
00190 " mov $0x20,%ax \n"
00191 " outb %al,%dx \n"
00192 " pop %gs \n"
00193 " pop %fs \n"
00194 " pop %es \n"
00195 " pop %ds \n"
00196 " pop %ss \n"
00197 " popa \n"
00198 " iret \n"
00199 );
00200
00201 static int atkbd_scan() {
00202 int code = 0x0;
00203 int val = 0x0;
00204
00205 code = inportByte(0x60);
00206 val = inportByte(0x61);
00207
00208 outportByte(0x61,val | 0x80);
00209 outportByte(0x61,val);
00210
00211 return(code);
00212 }
00213
00214 void keyboardHandler() {
00215 int key = 0x0;
00216
00217 if (!spinTryLock(&atkbdSpinLock))
00218 return;
00219
00220 key = atkbd_scan();
00221
00222 if (key > 255)
00223 return;
00224
00225
00226 if (key == 0x1D && !(controlKeys & controlKey)) {
00227 controlKeys |= controlKey;
00228 }
00229 if (key == 0x80 + 0x1D) {
00230 controlKeys &= (0xFF - controlKey);
00231 }
00232
00233 if (key == 0x38 && !(controlKeys & altKey)) {
00234 controlKeys |= altKey;
00235 }
00236 if (key == 0x80 + 0x38) {
00237 controlKeys &= (0xFF - altKey);
00238 }
00239
00240 if ((key == 0x2A || key == 0x36) && !(controlKeys & shiftKey)) {
00241 controlKeys |= shiftKey;
00242 }
00243 if ((key == 0x80 + 0x2A) || (key == 0x80 + 0x36)) {
00244 controlKeys &= (0xFF - shiftKey);
00245 }
00246
00247 if (key == 0x3A) {
00248 ledStatus ^= ledCapslock;
00249 setLED();
00250 }
00251
00252 if (key == 0x45) {
00253 ledStatus ^= ledNumlock;
00254 setLED();
00255 }
00256
00257 if (key == 0x46) {
00258 ledStatus ^= ledScrolllock;
00259 setLED();
00260 }
00261
00262 if (controlKeys == 0) { keyMap = 0; }
00263 else if (controlKeys == 1) { keyMap = 1; }
00264 else if (controlKeys == 2) { keyMap = 2; }
00265 else if (controlKeys == 4) { keyMap = 3; }
00266
00267 if (((uInt)(keyboardMap[key][keyMap]) > 0) && ((uInt32)(keyboardMap[key][keyMap]) < 0xFF)) {
00268 switch ((uInt32)keyboardMap[key][keyMap]) {
00269 case 8:
00270 backSpace();
00271 if (tty_foreground == 0x0) {
00272 stdinBuffer[stdinSize] = keyboardMap[key][keyMap];
00273 stdinSize++;
00274 }
00275 else {
00276 tty_foreground->stdin[tty_foreground->stdinSize] = keyboardMap[key][keyMap];
00277 tty_foreground->stdinSize++;
00278 }
00279 break;
00280 case 0x3:
00281
00282
00283 kprintf("CTRL-C pressed\n");
00284
00285 break;
00286 case 0x9:
00287 kprintf("REBOOTING");
00288 while(inportByte(0x64) & 0x02);
00289 outportByte(0x64, 0xFE);
00290 break;
00291 default:
00292 if (tty_foreground == 0x0) {
00293 stdinBuffer[stdinSize] = keyboardMap[key][keyMap];
00294 stdinSize++;
00295 }
00296 else {
00297 tty_foreground->stdin[tty_foreground->stdinSize] = keyboardMap[key][keyMap];
00298 tty_foreground->stdinSize++;
00299 }
00300 break;
00301 }
00302 }
00303 else {
00304 switch ((keyboardMap[key][keyMap] >> 8)) {
00305 case 0x30:
00306 tty_change(keyboardMap[key][keyMap] & 0xFF);
00307
00308 break;
00309 default:
00310 break;
00311 }
00312 }
00313
00314
00315 spinUnlock(&atkbdSpinLock);
00316 return;
00317 }
00318
00319 void setLED() {
00320 outportByte(0x60, 0xED);
00321 while(inportByte(0x64) & 2);
00322 outportByte(0x60, ledStatus);
00323 while(inportByte(0x64) & 2);
00324 }
00325
00326
00327 unsigned char getch() {
00328 uInt8 retKey = 0x0;
00329 uInt32 i = 0x0;
00330
00331
00332
00333
00334
00335
00336
00337
00338
00339
00340
00341
00342
00343
00344
00345 if (tty_foreground == 0x0) {
00346 if (stdinSize == 0x0) {
00347
00348 return(0x0);
00349 }
00350
00351 retKey = stdinBuffer[0];
00352 stdinSize--;
00353
00354 for (i=0x0;i<stdinSize;i++) {
00355 stdinBuffer[i] = stdinBuffer[i+0x1];
00356 }
00357 }
00358 else {
00359 if (tty_foreground->stdinSize == 0x0) {
00360
00361 return(0x0);
00362 }
00363
00364 retKey = tty_foreground->stdin[0];
00365 tty_foreground->stdinSize--;
00366
00367 for (i=0x0;i<tty_foreground->stdinSize;i++) {
00368 tty_foreground->stdin[i] = tty_foreground->stdin[i+0x1];
00369 }
00370 }
00371
00372 return(retKey);
00373 }
00374
00375
00376
00377
00378
00379
00380
00381
00382
00383
00384
00385
00386
00387
00388
00389
00390
00391
00392
00393
00394
00395
00396
00397
00398
00399
00400
00401
00402
00403
00404
00405
00406
00407
00408
00409
00410
00411
00412
00413
00414
00415
00416
00417
00418
00419
00420
00421
00422
00423
00424
00425
00426
00427
00428
00429
00430
00431
00432
00433
00434
00435
00436
00437
00438
00439
00440
00441
00442
00443
00444
00445
00446
00447
00448
00449
00450
00451
00452
00453
00454
00455
00456
00457
00458
00459
00460
00461
00462
00463
00464
00465
00466
00467
00468
00469
00470
00471
00472
00473
00474
00475
00476
00477
00478
00479
00480
00481
00482
00483
00484
00485
00486
00487
00488