summaryrefslogtreecommitdiff
path: root/Kernel/Keyboard.cpp
blob: cd41f2723c5ca18e06d3d8234c4915d55f1a09d8 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
#include "types.h"
#include "i386.h"
#include "IO.h"
#include "VGA.h"
#include "PIC.h"
#include "Keyboard.h"
#include <AK/Assertions.h>

#define IRQ_KEYBOARD             1

#define I8042_BUFFER             0x60
#define I8042_STATUS             0x64

#define SET_LEDS                 0xED
#define DATA_AVAILABLE           0x01
#define I8042_ACK                0xFA

#define MOD_ALT     1
#define MOD_CTRL    2
#define MOD_SHIFT   4

static char map[0x100] =
{
    0, 0, '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '-', '=', 0x08, 0,
    'q', 'w', 'e', 'r', 't', 'y', 'u', 'i', 'o', 'p', '[', ']', 0, 0,
    'a', 's', 'd', 'f', 'g', 'h', 'j', 'k', 'l', ';', '\'', '`', 0, '\\',
    'z', 'x', 'c', 'v', 'b', 'n', 'm', ',', '.', '/',
    0, 0, 0, ' '
};

static char shift_map[0x100] =
{
    0, 0, '!', '@', '#', '$', '%', '^', '&', '*', '(', ')', '_', '+', 0, 0,
    'Q', 'W', 'E', 'R', 'T', 'Y', 'U', 'I', 'O', 'P', '{', '}', 0, 0,
    'A', 'S', 'D', 'F', 'G', 'H', 'J', 'K', 'L', ':', '"', '~', 0, '|',
    'Z', 'X', 'C', 'V', 'B', 'N', 'M', '<', '>', '?',
    0, 0, 0, ' '
};


void Keyboard::handleIRQ()
{
    while (IO::in8(0x64) & 1) {
        BYTE ch = IO::in8(0x60);
        switch (ch) {
        case 0x38: m_modifiers |= MOD_ALT; break;
        case 0xB8: m_modifiers &= ~MOD_ALT; break;
        case 0x1D: m_modifiers |= MOD_CTRL; break;
        case 0x9D: m_modifiers &= ~MOD_CTRL; break;
        case 0x2A: m_modifiers |= MOD_SHIFT; break;
        case 0xAA: m_modifiers &= ~MOD_SHIFT; break;
        case 0x1C: /* enter */ m_queue.enqueue('\n'); break;
        case 0xFA: /* i8042 ack */ break;
        default:
            if (ch & 0x80) {
                // key has been depressed
                break;
            }
            if (!m_modifiers)
                m_queue.enqueue(map[ch]);
            else if (m_modifiers & MOD_SHIFT)
                m_queue.enqueue(shift_map[ch]);
            else if (m_modifiers & MOD_CTRL) {
                // FIXME: This is obviously not a good enough way to process ctrl+whatever.
                m_queue.enqueue('^');
                m_queue.enqueue(shift_map[ch]);
            }
        }
        //break;
    }
}

Keyboard::Keyboard()
    : IRQHandler(IRQ_KEYBOARD)
{
    // Empty the buffer of any pending data.
    // I don't care what you've been pressing until now!
    while (IO::in8(I8042_STATUS ) & DATA_AVAILABLE)
        IO::in8(I8042_BUFFER);

    enableIRQ();
}

Keyboard::~Keyboard()
{
    ASSERT_NOT_REACHED();
}

bool Keyboard::hasDataAvailableForRead() const
{
    return !m_queue.isEmpty();
}

ssize_t Keyboard::read(byte* buffer, size_t size)
{
    ssize_t nread = 0;
    while (nread < size) {
        if (m_queue.isEmpty())
            break;
        buffer[nread++] = m_queue.dequeue();
    }
    return nread;
}

ssize_t Keyboard::write(const byte* data, size_t size)
{
    return 0;
}