Версия для печати темы

Нажмите сюда для просмотра этой темы в обычном формате

Форум «Всё о Паскале» _ Теоретические вопросы _ Буфер клавиатуры

Автор: Bokul 12.12.2006 4:51

Решил даром не флудить.
Что представляет с себя буфер клавиатуры? Где он хранится и каковой его размер?
И как вот этот код

Цитата

while keypressed do readkey;


или


repeat
readkey
until not keypressed;


его очищает?

Автор: Lapp 12.12.2006 5:11

Цитата(Bokul @ 12.12.2006 1:51) *

Что представляет с себя буфер клавиатуры? Где он хранится и каковой его размер?
И как вот этот код .. его очищает?

Обычный буфер типа FIFO (первым вошел - первым вышел), размер, кажется, 16 байт... могу ошибаться, давно было..
Находится где-то в области досовской памяти.. Тебе нужно точно?
Как очищает? да просто считывает все символы из него! smile.gif поскольку ввод с клавы гораздо медленнее, чем работа ReadKey, то можно быть практически уверенным, что этот цикл очистит буфер. Хотя, есть принципиальная возможность, что сразу после него и до работы того, для чего ты его очищал, в него что-то все же ляжет. Но вероятность невелика smile.gif

Автор: Bokul 12.12.2006 5:20

Спасибо. good.gif

Цитата
Тебе нужно точно?

Если знаешь - давай...
Цитата
Находится где-то в области досовской памяти..

Что такое досовская память?

Автор: Lapp 12.12.2006 6:37

> Если знаешь - давай...
Когда-то знал, мог наизусть ответить.. Сейчас - надо рыть Интернет, бумага уже не сохранилась sad.gif У меня была брошюрка в страниц 10, которую я замусолил до безобразия, таская с собой - там было все абсолютно. Извиняюсь за флуд, но давай подождем xds - я уверен, он скажет адрес smile.gif.

> Что такое досовская память?
Ну, область памяти, где DOS хранит свои данные. Там все имеет определенные адреса, которые, слава Богу, не менялись от версии к версии. Это позволяло достаточно надежно использовать сами данные в обход системных вызовов. Само по себе это нехорошо, но иногда давало определенные преимущества - типа скорость, а также контроль самого низкого уровня.

Автор: Bokul 12.12.2006 6:47

Цитата
давай подождем xds - я уверен, он скажет адрес smile.gif.

lol.gif
Согласен, будем ждать ходячего справочникаsmile.gif
Цитата
Ну, область памяти, где DOS хранит свои данные. Там все имеет определенные адреса, которые, слава Богу, не менялись от версии к версии. Это позволяло достаточно надежно использовать сами данные в обход системных вызовов. Само по себе это нехорошо, но иногда давало определенные преимущества - типа скорость, а также контроль самого низкого уровня.

Конечно не по теме, но что поделаешь, все-равно ждать приходится, smile.gif : позволяет ли Windows проделывать тоже самое? И что происходит, когда я запускаю прогу, скомпилированную под Dos, на винде (я имею ввиду на уровне памяти)?

Автор: Lapp 12.12.2006 7:04

Цитата(Bokul @ 12.12.2006 3:47) *

Конечно не по теме, но что поделаешь, все-равно ждать приходится, smile.gif : позволяет ли Windows проделывать тоже самое? И что происходит, когда я запускаю прогу, скомпилированную под Dos, на винде (я имею ввиду на уровне памяти)?

Да, за тему пришлось выйти, к теории Паскаля это имеет весьма малое отношение. Но обнаруживает разницу между задачами, скомпилированными в TP и в FPC.
Для запуска досовской программы Винда запускает виртуальную DOS-машину, которая довольно точно копирует настоящий DOS, включая распределение памяти. Так что твой вопрос (или мой ответ?..) имеет отношение только к тому, что скомпилированно в TP. Но если говорить об очистке буфера клавиатуры, то принцип тот же - тебе надо читать из буфера, пока он не очистится.

Автор: xds 12.12.2006 10:03

Разбирайся wink.gif

program Code;

uses
Crt;

var
KStart: Word absolute $40:$1A;
{ адрес начала (откуда считывается) очереди данных клавиатуры }
KEnd: Word absolute $40:$1C;
{ адрес конца (куда записывается) очереди данных клавиатуры }
KBuf: array[0..15] of Word absolute $40:$1E;
{ собственно, буфер - 16 записей по 2 байта (ASCII-, Scan-код) }

{ - если очередь пуста, то KStart = KEnd }
{ - если очередь переполнена, то KEnd указывает на запись перед KStart }

var
i, is, ie: Integer;

begin
ClrScr;
repeat
is := (KStart - Ofs(KBuf)) shr 1;
ie := (KEnd - Ofs(KBuf)) shr 1;
GotoXY(1, 1);
Writeln('ASCII':14, ' Scan');
Writeln;
for i := 0 to 15 do
begin
if i = ie then
Write('KEnd -> ')
else if i = is then
Write('KStart ->')
else
Write(' ');
Writeln(Lo(KBuf[i]):4, Hi(KBuf[i]):4);
end;
until Lo(KBuf[(ie - 1) and $F]) = 27;
end.

Автор: SKVOZNJAK 13.12.2006 6:24

Блин, как просто это в досе\виндОсе. Уже даже и не верится. Стабильный размер буфера, никаких событий и загрузки порциями, даже автоповтор эмулировать не надо smile.gif

Автор: Bokul 13.12.2006 6:33

Цитата
Стабильный размер буфера, никаких событий и загрузки порциями, даже автоповтор эмулировать не надо

Чтобы не расценивать твое сообщение как флуд smile.gif , дай, если есть конечно, пару ссылочок по теме для Windows. rolleyes.gif smile.gif

Автор: SKVOZNJAK 14.12.2006 16:17

А кто сказал что это виндовс ;) Речь идёт о библиотеке сдл и о настальгии по таким простым методам доступа к клаве в любое время суток smile.gif http://sdl4fp.sourceforge.net/index.html Главная её особенность - она работает в линуксе! Вот с чтением клавы в видеорежиме там очень непросто. И к портам только суперпользователь напрямую обратиться может и ось реально многозадачная. Теоретически SDL можно скомпилить и под виндовс. А практически, почти всё в ней импортируется, а под виндой импорт не работает, чего-то не хватает. Возможно исходники скомпилируются, если в каталог добавить файл SDL.DLL скачанный из ослика. Но это непроверено. Вот работающий пример использования клавы в видеорежиме. Так же получается доступ к мыши, джойстику. Читаются события keydown и keyup, а для клавиш модификаторов ещё и маски.
program demo;


uses
SDL, SDL_Video, SDL_Events;

const
width = 720;
height = 400;
colordepth = 32;

var
screen: PSDL_Surface;
image: PSDL_Surface;
event: PSDL_Event;
done: Boolean;
filename: String;

label 1;


begin

SDL_Init(SDL_INIT_VIDEO);
event := new(PSDL_Event);
screen := SDL_SetVideoMode(width, height, colordepth, (SDL_FULLSCREEN + SDL_SWSURFACE));
if screen = nil then
begin
writeLn ('Couldn''t initialize video mode at ', width, 'x', height, 'x', colordepth, 'bpp');
halt(1);
end;
filename := '0.bmp';
image := SDL_loadBMP(@filename[1]);
if (image = nil) then
begin
writeLn('Couldn''t load ' + filename);
halt(1);
end;
if (SDL_BlitSurface(image, nil, screen, nil) < 0) then
begin
writeLn('BlitSurface error:' {+ SDL_GetError()});
end;
SDL_updateRect(screen, 0, 0, image^.w, image^.h);
SDL_freeSurface(image);
done := FALSE;
while (done = FALSE) do
begin
if (SDL_PollEvent(event) > 0) then
begin
CASE event^.eventtype OF
SDL_KEYDOWN, SDL_MOUSEBUTTONDOWN :done := TRUE;
end;
end;
end;


SDL_freeSurface(screen);
SDL_Quit ;
end.

Автор: TarasBer 9.01.2009 19:15

Цитата(xds @ 12.12.2006 6:03) *

Код

var
  KStart: Word absolute $40:$1A;
    { адрес начала (откуда считывается) очереди данных клавиатуры }
  KEnd: Word absolute $40:$1C;
    { адрес конца (куда записывается) очереди данных клавиатуры }
  KBuf: array[0..15] of Word absolute $40:$1E;
    { собственно, буфер - 16 записей по 2 байта (ASCII-, Scan-код) }

  { - если очередь пуста, то KStart = KEnd }
  { - если очередь переполнена, то KEnd указывает на запись перед KStart }



Ура, одним стандартным модулем меньше в моём списке uses.
Спасибо.