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/vitals.h>
00041 #include <ubixos/endtask.h>
00042 #include <ubixos/tty.h>
00043 #include <ubixos/spinlock.h>
00044
00045 static int atkbd_scan();
00046
00047 static unsigned int keyMap = 0x0;
00048 static unsigned int ledStatus = 0x0;
00049 static char stdinBuffer[512];
00050 static uInt16 stdinSize;
00051 static uInt32 controlKeys = 0x0;
00052
00053 static spinLock_t atkbdSpinLock = SPIN_LOCK_INITIALIZER;
00054
00055 static unsigned int keyboardMap[255][8] = {
00056
00057 { 0, 0, 0, 0, 0, 0, 0, 0},
00058 { 0x1B, 0x1B, 0x1B, 0x1B, 0x1B, 0x1B, 0x1B, 0x1B},
00059 { 0x31, 0x21, 0, 0, 0x31, 0x31, 0x21, 0x21},
00060 { 0x32, 0x40, 0, 0, 0x32, 0x32, 0x40, 0x40},
00061 { 0x33, 0x23, 0, 0, 0x33, 0x33, 0x23, 0x23},
00062 { 0x34, 0x24, 0, 0, 0x34, 0x34, 0x24, 0x24},
00063 { 0x35, 0x25, 0, 0, 0x35, 0x35, 0x25, 0x25},
00064 { 0x36, 0x5E, 0, 0, 0x36, 0x36, 0x5E, 0x5E},
00065 { 0x37, 0x26, 0, 0, 0x37, 0x37, 0x26, 0x26},
00066 { 0x38, 0x2A, 0, 0, 0x38, 0x38, 0x2A, 0x2A},
00067 { 0x39, 0x28, 0, 0, 0x39, 0x39, 0x28, 0x28},
00068 { 0x30, 0x29, 0, 0, 0x30, 0x30, 0x29, 0x29},
00069 { 0x2D, 0x5F, 0, 0, 0x2D, 0x2D, 0x5F, 0x5F},
00070 { 0x3D, 0x2B, 0, 0, 0x3D, 0x3D, 0x2B, 0x2B},
00071 { 0x08, 0x08, 0x8, 0x8, 0x08, 0x08, 0x08, 0x08},
00072 { 0x09, 0, 0, 0, 0, 0, 0, 0},
00073 { 0x71, 0x51, 0, 0, 0, 0, 0, 0},
00074 { 0x77, 0x57, 0, 0, 0, 0, 0, 0},
00075 { 0x65, 0x45, 0, 0, 0, 0, 0, 0},
00076 { 0x72, 0x52, 0, 0, 0, 0, 0, 0},
00077 { 0x74, 0x54, 0, 0, 0, 0, 0, 0},
00078 { 0x79, 0x59, 0, 0, 0, 0, 0, 0},
00079 { 0x75, 0x55, 0, 0, 0, 0, 0, 0},
00080 { 0x69, 0x49, 0, 0, 0, 0, 0, 0},
00081 { 0x6F, 0x4F, 0, 0, 0, 0, 0, 0},
00082 { 0x70, 0x50, 0, 0, 0, 0, 0, 0},
00083 { 0x5B, 0x7B, 0, 0, 0, 0, 0, 0},
00084 { 0x5D, 0x7D, 0, 0, 0, 0, 0, 0},
00085 { 0x0A, 0, 0, 0, 0, 0, 0, 0},
00086 { 0, 0, 0, 0, 0, 0, 0, 0},
00087 { 0x61, 0x41, 0x41, 0, 0, 0, 0, 0},
00088 { 0x73, 0x53, 0, 0, 0, 0, 0, 0},
00089 { 0x64, 0x44, 0, 0, 0, 0, 0, 0},
00090 { 0x66, 0x46, 0, 0, 0, 0, 0, 0},
00091 { 0x67, 0x47, 0, 0, 0, 0, 0, 0},
00092 { 0x68, 0x48, 0, 0, 0, 0, 0, 0},
00093 { 0x6A, 0x4A, 0, 0, 0, 0, 0, 0},
00094 { 0x6B, 0x4B, 0, 0, 0, 0, 0, 0},
00095 { 0x6C, 0x4C, 0, 0, 0, 0, 0, 0},
00096 { 0x3B, 0x3A, 0, 0, 0, 0, 0, 0},
00097 { 0x27, 0x22, 0, 0, 0, 0, 0, 0},
00098 { 0x60, 0x7E, 0, 0, 0, 0, 0, 0},
00099 { 0x2A, 0x0, 0, 0, 0, 0, 0, 0},
00100 { 0x5C, 0x3C, 0, 0, 0, 0, 0, 0},
00101 { 0x7A, 0x5A, 0, 0, 0, 0, 0, 0},
00102 { 0x78, 0x58, 0, 0, 0, 0, 0, 0},
00103 { 0x63, 0x43, 0x3, 0x9, 0, 0, 0, 0},
00104 { 0x76, 0x56, 0, 0, 0, 0, 0, 0},
00105 { 0x62, 0x42, 0, 0, 0, 0, 0, 0},
00106 { 0x6E, 0x4E, 0, 0, 0, 0, 0, 0},
00107 { 0x6D, 0x4D, 0, 0, 0, 0, 0, 0},
00108 { 0x2C, 0x3C, 0, 0, 0, 0, 0, 0},
00109 { 0x2E, 0x3E, 0, 0, 0, 0, 0, 0},
00110 { 0x2F, 0x3F, 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 { 0, 0, 0, 0, 0, 0, 0, 0},
00114 { 0x20, 0, 0, 0, 0, 0, 0, 0},
00115 { 0, 0, 0, 0, 0, 0, 0, 0},
00116 { 0x3000, 0, 0, 0x3000, 0, 0, 0, 0},
00117 { 0x3001, 0, 0, 0x3001, 0, 0, 0, 0},
00118 { 0x3002, 0, 0, 0x3002, 0, 0, 0, 0},
00119 { 0x3003, 0, 0, 0x3003, 0, 0, 0, 0},
00120 { 0x3004, 0, 0, 0x3004, 0, 0, 0, 0},
00121 { 0x4000, 0, 0, 0, 0, 0, 0, 0},
00122 { 0x4100, 0, 0, 0, 0, 0, 0, 0},
00123 { 0x4200, 0, 0, 0, 0, 0, 0, 0},
00124 { 0x4300, 0, 0, 0, 0, 0, 0, 0},
00125 { 0x4400, 0, 0, 0, 0, 0, 0, 0},
00126 { 0, 0, 0, 0, 0, 0, 0, 0},
00127 { 0, 0, 0, 0, 0, 0, 0, 0},
00128 { 0x4700, 0, 0, 0, 0, 0, 0, 0},
00129 { 0x4800, 0, 0, 0, 0, 0, 0, 0},
00130 { 0x4900, 0, 0, 0, 0, 0, 0, 0},
00131 { 0x2D, 0, 0, 0, 0, 0, 0, 0},
00132 { 0x4B00, 0, 0, 0, 0, 0, 0, 0},
00133 { 0x4C00, 0, 0, 0, 0, 0, 0, 0},
00134 { 0x4D00, 0, 0, 0, 0, 0, 0, 0},
00135 { 0x2B, 0, 0, 0, 0, 0, 0, 0},
00136 { 0x4F00, 0, 0, 0, 0, 0, 0, 0},
00137 { 0x5000, 0, 0, 0, 0, 0, 0, 0},
00138 { 0x5100, 0, 0, 0, 0, 0, 0, 0},
00139 { 0x5200, 0, 0, 0, 0, 0, 0, 0},
00140 { 0x5300, 0, 0, 0, 0, 0, 0, 0},
00141 { 0, 0, 0, 0, 0, 0, 0, 0},
00142 { 0, 0, 0, 0, 0, 0, 0, 0}
00143 };
00144
00145
00146
00147
00148
00149
00150
00151
00152
00153
00154
00155
00156 int atkbd_init() {
00157
00158 setVector(&atkbd_isr, mVec+0x1, dPresent + dInt + dDpl0);
00159
00160
00161 setLED();
00162
00163
00164 atkbd_scan();
00165
00166
00167 irqEnable(0x1);
00168
00169
00170 kprintf("atkbd0 - Address: [0x%X], Keyboard Buffer: [0x%X], Buffer Size [%i]\n",&atkbd_isr,&stdinBuffer,512);
00171
00172
00173 return(0x0);
00174 }
00175
00176
00177
00178
00179
00180 asm(
00181 ".globl atkbd_isr \n"
00182 "atkbd_isr: \n"
00183 " pusha \n"
00184 " push %ss \n"
00185 " push %ds \n"
00186 " push %es \n"
00187 " push %fs \n"
00188 " push %gs \n"
00189 " call keyboardHandler \n"
00190 " mov $0x20,%dx \n"
00191 " mov $0x20,%ax \n"
00192 " outb %al,%dx \n"
00193 " pop %gs \n"
00194 " pop %fs \n"
00195 " pop %es \n"
00196 " pop %ds \n"
00197 " pop %ss \n"
00198 " popa \n"
00199 " iret \n"
00200 );
00201
00202 static int atkbd_scan() {
00203 int code = 0x0;
00204 int val = 0x0;
00205
00206 code = inportByte(0x60);
00207 val = inportByte(0x61);
00208
00209 outportByte(0x61,val | 0x80);
00210 outportByte(0x61,val);
00211
00212 return(code);
00213 }
00214
00215 void keyboardHandler() {
00216 int key = 0x0;
00217
00218 if (!spinTryLock(&atkbdSpinLock))
00219 return;
00220
00221 key = atkbd_scan();
00222
00223 if (key > 255)
00224 return;
00225
00226
00227 if (key == 0x1D && !(controlKeys & controlKey)) {
00228 controlKeys |= controlKey;
00229 }
00230 if (key == 0x80 + 0x1D) {
00231 controlKeys &= (0xFF - controlKey);
00232 }
00233
00234 if (key == 0x38 && !(controlKeys & altKey)) {
00235 controlKeys |= altKey;
00236 }
00237 if (key == 0x80 + 0x38) {
00238 controlKeys &= (0xFF - altKey);
00239 }
00240
00241 if ((key == 0x2A || key == 0x36) && !(controlKeys & shiftKey)) {
00242 controlKeys |= shiftKey;
00243 }
00244 if ((key == 0x80 + 0x2A) || (key == 0x80 + 0x36)) {
00245 controlKeys &= (0xFF - shiftKey);
00246 }
00247
00248 if (key == 0x3A) {
00249 ledStatus ^= ledCapslock;
00250 setLED();
00251 }
00252
00253 if (key == 0x45) {
00254 ledStatus ^= ledNumlock;
00255 setLED();
00256 }
00257
00258 if (key == 0x46) {
00259 ledStatus ^= ledScrolllock;
00260 setLED();
00261 }
00262
00263 if (controlKeys == 0) { keyMap = 0; }
00264 else if (controlKeys == 1) { keyMap = 1; }
00265 else if (controlKeys == 2) { keyMap = 2; }
00266 else if (controlKeys == 4) { keyMap = 3; }
00267
00268 if (((uInt)(keyboardMap[key][keyMap]) > 0) && ((uInt32)(keyboardMap[key][keyMap]) < 0xFF)) {
00269 switch ((uInt32)keyboardMap[key][keyMap]) {
00270 case 8:
00271 backSpace();
00272 if (tty_foreground == 0x0) {
00273 stdinBuffer[stdinSize] = keyboardMap[key][keyMap];
00274 stdinSize++;
00275 }
00276 else {
00277 tty_foreground->stdin[tty_foreground->stdinSize] = keyboardMap[key][keyMap];
00278 tty_foreground->stdinSize++;
00279 }
00280 break;
00281 case 0x3:
00282
00283
00284
00285 kprintf("FreePages: [0x%X]\n",systemVitals->freePages);
00286 break;
00287 case 0x9:
00288 kprintf("REBOOTING");
00289 while(inportByte(0x64) & 0x02);
00290 outportByte(0x64, 0xFE);
00291 break;
00292 default:
00293 if (tty_foreground == 0x0) {
00294 stdinBuffer[stdinSize] = keyboardMap[key][keyMap];
00295 stdinSize++;
00296 }
00297 else {
00298 tty_foreground->stdin[tty_foreground->stdinSize] = keyboardMap[key][keyMap];
00299 tty_foreground->stdinSize++;
00300 }
00301 break;
00302 }
00303 }
00304 else {
00305 switch ((keyboardMap[key][keyMap] >> 8)) {
00306 case 0x30:
00307 tty_change(keyboardMap[key][keyMap] & 0xFF);
00308
00309 break;
00310 default:
00311 break;
00312 }
00313 }
00314
00315
00316 spinUnlock(&atkbdSpinLock);
00317 return;
00318 }
00319
00320 void setLED() {
00321 outportByte(0x60, 0xED);
00322 while(inportByte(0x64) & 2);
00323 outportByte(0x60, ledStatus);
00324 while(inportByte(0x64) & 2);
00325 }
00326
00327
00328 unsigned char getch() {
00329 uInt8 retKey = 0x0;
00330 uInt32 i = 0x0;
00331
00332
00333
00334
00335
00336
00337
00338
00339
00340
00341
00342
00343
00344
00345
00346 if (tty_foreground == 0x0) {
00347 if (stdinSize == 0x0) {
00348
00349 return(0x0);
00350 }
00351
00352 retKey = stdinBuffer[0];
00353 stdinSize--;
00354
00355 for (i=0x0;i<stdinSize;i++) {
00356 stdinBuffer[i] = stdinBuffer[i+0x1];
00357 }
00358 }
00359 else {
00360 if (tty_foreground->stdinSize == 0x0) {
00361
00362 return(0x0);
00363 }
00364
00365 retKey = tty_foreground->stdin[0];
00366 tty_foreground->stdinSize--;
00367
00368 for (i=0x0;i<tty_foreground->stdinSize;i++) {
00369 tty_foreground->stdin[i] = tty_foreground->stdin[i+0x1];
00370 }
00371 }
00372
00373 return(retKey);
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