В этом поле драйверу передается
Таблица 12
(0) 13 | header | Заголовок запроса. |
(+13) 1 | media | В этом поле драйверу передается байт-описатель среды носителя данных. |
(+14) 4 | buf_adr | Адрес буфера для передачи данных. |
(+18) 2 | count | Количество передаваемых байтов для символьных устройств или секторов для блочных устройств. |
(+20) 2 | sector | Номер начального сектора, если драйвер использует 16-битовую адресацию секторов или -1 для 32-битовой адресации. Это поле не используется символьными драйверами. |
(+22) 4 | vol_id | Указатель на метку тома в формате ASCIIZ. Возвращается блочным драйвером, если он выставляет ошибку 15 (неправильная смена диска). Это поле должно содержать ссылку на метку требуемого диска. |
(+26) 4 | sect32 | Номер начального сектора, если содержимое поля sector равно -1. Первым идет старшее слово номера сектора. Если обнаружена ошибка с номером 15, в это поле записывается указатель на метку тома. |
Для команды 9 (запись с проверкой) драйвер должен после выполнения записи проверить записанные данные. Если с консоли введена команда DOS VERIFY ON, то для дисковых устройств DOS вместо команды записи 8 использует команду записи с проверкой 9.
Драйвер может возвратить количество переданных байтов меньше запрошенного, это само по себе не является ошибкой. Иногда может получиться так, что надо выполнить запись 64 Кбайтов. Такая операция может вызвать переход за границу сегмента в буфере передачи данных. В этом случае драйвер должен проигнорировать лишние байты. Например, надо записать 10000h байтов, распределенных по секторам из буфера с адресом ХХХХ:0001. В этом случае драйвер должен проигнорировать последние два байта. Они будут записаны в следующий раз.
IOCTL чтение/запись - это обмен с устройством управляющей информацией. Мы будем подробно говорить об этом ниже.
Команда с кодом 16 - "вывод, пока не занято", предназначена для работы с такими устройствами ввода/вывода, которые имеют свой собственный буфер, например, принтеры.
Приведем исходный текст программы драйвера, использующего команды ввода и вывода. Для ввода драйвер использует клавиатуру, вывод осуществляется на экран дисплея. .MODEL tiny .CODE ; Драйвер состоит из одного ; сегмента кода org 0 ; Эта строка может отсутствовать include sysp.inc ;======================================================== iodrv PROC far ;драйвер - это FAR-процедура ;======================================================== ; Заголовок драйвера dd 0ffffffffh ;адрес следующего драйвера dw 8000h ;байт атрибутов dw dev_strategy ;адрес процедуры стратегии dw dev_interrupt ;адрес процедуры прерывания db 'IODRIVER' ;имя устройства (дополненное ; пробелами) ;======================================================== ; Программа стратегии dev_strategy: mov cs:req_seg,es mov cs:req_off,bx ret ; Здесь запоминается адрес заголовка запроса req_seg dw ? req_off dw ? ;======================================================== ;Обработчик прерывания dev_interrupt: push es ;сохраняем регистры push ds push ax push bx push cx push dx push si push di push bp ; Устанавливаем ES:BX на заголовок запроса mov ax,cs:req_seg mov es,ax mov bx,cs:req_off ; Получаем код команды из заголовка запроса и умножаем ; его на два, чтобы использовать в качестве индекса ; таблицы адресов обработчиков команд mov al,es:[bx]+2 shl al,1 sub ah,ah ; Обнуляем AH lea di,functions ; DI указывает на смещение ; таблицы add di,ax ; Добавляем смещение в таблице jmp word ptr [di] ; Переходим на адрес из таблицы functions LABEL WORD ; Таблица функций dw initialize dw check_media dw make_bpb dw ioctl_in dw input_data dw nondestruct_in dw input_status dw clear_input dw output_data dw output_verify dw output_status dw clear_output dw ioctl_out dw Device_open dw Device_close dw Removable_media ; Выход из драйвера, если функция не поддерживается check_media: make_bpb: ioctl_in: nondestruct_in: input_status: clear_input: output_verify: output_status: clear_output: ioctl_out: Removable_media: Device_open: Device_close: or es:word ptr [bx]+3,8103h jmp quit ;======================================================== ; Обработчик команды вывода данных output_data: ; Записываем в регистр CL количество ; выводимых символов mov cl,es:[bx]+18 push cx ; Выводим сообщение о начале вывода mov ax,cs mov ds,ax mov si,offset outmsg call dpc pop cx ; Загружаем в DS:SI адрес буфера данных mov ax,es:[bx]+16 mov ds,ax mov si,es:[bx]+14 ; Выводим на экран символы из буфера out_loop: mov al,ds:byte ptr [si] @@out_ch al inc si loop out_loop jmp quit ;======================================================== ; Обработчик команды ввода данных input_data: ; Записываем в регистр CL количество ; вводимых символов mov cl,es:[bx]+18 push cx ; Выводим сообщение о начале ввода mov ax,cs mov ds,ax mov si,offset inpmsg call dpc ; Загружаем в DS:SI адрес буфера данных pop cx mov ax,es:[bx]+16 mov ds,ax mov di,es:[bx]+14 ; Вводим символы с клавиатуры и записываем в буфер inp_loop: mov ax,0 int 16h mov ds:byte ptr [di],al @@out_ch al inc di loop inp_loop jmp quit ;======================================================== quit: or es:word ptr [bx]+3,100h pop bp pop di pop si pop dx pop cx pop bx pop ax pop ds pop es ret ;======================================================== ; Процедура выводит на экран строку ; символов в формате ASCIIZ dpc proc near push si dpc_loop: cmp ds:byte ptr [si],0 jz end_dpc mov al,ds:byte ptr [si] @@out_ch al inc si jmp dpc_loop end_dpc: pop si ret dpc endp ;======================================================== hello db 13,10,'++' db 13,10,'¦ *IODRV* (C)Frolov A., 1990 ¦' db 13,10,'++' db 13,10,0 outmsg DB 13,10,'___ Вывод на устройство IODRIVER ___',0 inpmsg DB 13,10,'___ Ввод с устройства IODRIVER ___',0 ;======================================================== E_O_P: ;Метка конца программы initialize: lea ax,E_O_P ;смещение конца программы в AX mov es:word ptr [bx]+14,ax ;помещаем его в заголовок mov es:word ptr [bx]+16,cs ; ; Стираем экран mov dh,18h mov dl,80h xor cx,cx mov bh,7 xor al,al mov ah,6 int 10h ; Устанавливаем курсор в левый верхний угол экрана mov bh,0 xor dx,dx mov ah,2 int 10h ; Выводим сообщение mov ax,cs mov ds,ax mov si,offset hello call dpc jmp quit iodrv ENDP END iodrv
Для работы с этим драйвером можно использовать приводимую ниже программу, составленную на языке Си. Программа открывает устройство, вводит из него восемь символов, печатает введенные символы на экране и выводит их обратно на устройство: #include <io.h> #include <conio.h> #include <stdio.h> #include <fcntl.h> #include <sys\types.h> #include <sys\stat.h> #include <malloc.h> #include <errno.h> #include <dos.h> int main(void); int main(void) { char buf[100]; int io_handle; unsigned count; // Открываем устройство с именем IODRIVER if( (io_handle = open("IODRIVER", O_RDWR)) == - 1 ) { // Если открыть не удалось, выводим // код ошибки printf("Ошибка при открытии устройства %d",errno); return errno; } // Читаем 8 байт из устройства в буфер buf if( (count = read(io_handle, buf, 8)) == -1 ) { // Если при чтении произошла ошибка, // выводим ее код printf("Ошибка чтения %d",errno); return errno; } // Закрываем прочитанную строку нулем // для последующего вывода функцией printf buf[8]=0; printf("\n___ Введена строка: %s ___",buf); // Выводим только что прочитанные данные // обратно на то же устройство if( (count = write(io_handle, buf, 8)) == -1 ) { // Если при записи произошла ошибка, // выводим ее код printf("Ошибка записи %d",errno); return errno; } // Закрываем устройство close(io_handle); exit(0); }
Эта программа служит примером того, как можно организовать взаимодействие драйвера и прикладной программы, работающей с драйвером. Позже мы приведем пример более сложного драйвера символьного устройства.