Пишем свою операционную систему. Работа с клавиатурой
Доброго времени суток!
До предыдущего выпуска пользователь не как не мог взаимодействовать с нашей операционной системой, лишь наблюдать её работу. Теперь она уже умеет как-то реагировать на нажатия клавиш, отображая текстовые сообщения, но этого мало для нормального ввода данных. Однако сначала приведу Makefile для предыдущего выпуска:
Обычно драйвер клавиатуры создаёт программный буфер нажатых клавиш. Если приложение хочет получить ввод с клавиатуры, оно читает содержимое буфера или ждёт пока в нём появится код клавиши. Преобразование из скан-кода в код символа также возложим на функцию чтения символа из буфера, чтобы упростить обработчик прерывания. Начнём с описания данных:
В буфер помещаются 16 последних нажатых клавиш, если их не считывать самые старые будут теряться, всё так и должно быть. Для организации списка используется указатель на "голову" и на "хвост" списка кодов клавиш. При помещении очередного символа в буфер "хвост" увеличивается на 1. При чтении символа из буфера "голова" тоже сдвигается на 1. Если голова или хвост достигают конца массива, они становятся равными нулю. Таким образом
получается так называемый кольцевой буфер. BIOS и DOS используют такой же, и в нашем случае нет смысла придумывать что-то сложнее. Помимо прочего такая структура устойчива к той ситуации, когда запись в буфер по прерыванию происходит в середине функции чтения.
Изменим обработчик прерывания IRQ1, убрав вывод сообщений и добавив запись символа в буфер:
Первая функция позволяет читать одиночный символ с клавиатуры, вторая - считать целую строку символов. Замечу, что вторая функция имеет защиту от переполнение буфера с помощью ограничения длины вводимой строки.
Чтобы отделить преобразование символов от работы с буфером вынесем чтение скан-кода в отдельную функцию, которую добавим в конце tty.c:
uint8 in_scancode() {
uint8 result;
if (key_buffer_head != key_buffer_tail) {
if (key_buffer_head >= KEY_BUFFER_SIZE) {
key_buffer_head = 0;
}
result = key_buffer[key_buffer_head];
key_buffer_head++;
} else {
result = 0;
}
return result;
}
Эта функция возвращает скан-код нажатой клавиши или 0, если буфер пуст.
Преобразовывать скан-код в символ придётся по специальной таблице, которую опишем в дополнительном файле scancodes.h:
Эту таблицу с небольшими изменениями я позаимствовал отсюда: http://sysbin.com/files/lowlevel/osdev11.htm#keyboard. По сути у нас две таблицы - одна обычная, другая для символов, при вводе которых был зажат Shift. Теперь мы можем написать и in_char:
char in_char(bool wait) {
static bool shift = false;
uint8 chr;
do {
chr = in_scancode();
switch (chr) {
case 0x2A:
case 0x36:
shift = true;
break;
case 0x2A + 0x80:
case 0x36 + 0x80:
shift = false;
break;
}
if (chr & 0x80) {
chr = 0;
}
if (shift) {
chr = scancodes_shifted[chr];
} else {
chr = scancodes[chr];
}
} while (wait && (!chr));
return chr;
}
Эта функция уже умеет ждать нажатий при необходимости, а также преобразовывать скан-коды в символы с учётом нажатости клавиши Shift.
Сделать функцию in_string не составит особого труда:
void in_string(char *buffer, size_t buffer_size) {
char chr;
size_t position = 0;
do {
chr = in_char(true);
switch (chr) {
case 0:
break;
case 8:
if (position > 0) {
position--;
out_char(8);
}
break;
case '\n':
out_char('\n');
break;
default:
if (position < buffer_size - 1) {
buffer[position] = chr;
position++;
out_char(chr);
}
}
} while (chr != '\n');
buffer[position] = 0;
}
Ну вот и готова поддержка ввода текста, теперь можно написать в kernel_main что-нибудь вроде этого:
При условии, что в stdlib.c у нас есть strcmp, ничто не мешает реализовать что-нибудь вроде командного интерпретатора и сделать ОС немного интерактивной и полезной.
Заключение
Теперь у нас есть достаточные возможности для поддержания диалога с пользователем, во всяком случае вы наглядно видите результат своей работы. Далее пойдут более теоретические части. В первую очередь - менеджер памяти. Кстати, дальше того, что мы сделали сейчас, ни одна рассылка не продвинулась (разве что
в одной была сделана на скорую руку многозадачность), так что теперь мы не будем никого повторять и идти по своему собственному уникальному пути. До встречи!
P.S.: Пожалуйста, пишите свои замечания и предложения по поводу постройки архитектуры системы. Также было бы не плохо выбрать более красивое название. Мой адрес, как обычно, kiv.apple@gmail.com.