Команды преобразования данных
К этой группе можно отнести множество команд микропроцессора, но большинство из них имеют те или иные особенности, которые требуют отнести их к другим функциональным группам.
Поэтому из всей совокупности команд микропроцессора непосредственно к командам преобразования данных можно отнести только одну команду:
[адрес_таблицы_перекодировки]
Это очень интересная и полезная команда. Ее действие заключается в том, что она замещает значение в регистре al другим байтом из таблицы в памяти, расположенной по адресу, указанному операндом адрес_таблицы_перекодировки.
Слово “таблица” весьма условно — по сути это просто строка байт. Адрес байта в строке, которым будет производиться замещение содержимого регистра al, определяется суммой (bx) + (al), то есть содержимое al выполняет роль индекса в байтовом массиве.
При работе с командой xlat обратите внимание на следующий тонкий момент. Несмотря на то, что в команде указывается адрес строки байт, из которой должно быть извлечено новое значение, этот адрес должен быть предварительно загружен (например, с помощью команды lea) в регистр bx. Таким образом, операнд адрес_таблицы_перекодировки на самом деле не нужен (необязательность операнда показана заключением его в квадратные скобки). Что касается строки байт (таблицы перекодировки), то она представляет собой область памяти размером от 1 до 255 байт (диапазон числа без знака в 8-битном регистре).
В качестве иллюстрации работы данной команды мы рассмотрим программу, которая преобразует двузначное шестнадцатеричное число, вводимое с клавиатуры (то есть в символьном виде), в эквивалентное двоичное представление в регистре al. Ниже (листинг 2) приведен вариант этой программы с использованием команды xlat.
Листинг 2. Использование таблицы перекодировки <1>;---------Prg_7_3.asm---------------------- <2>;Программа преобразования двузначного шестнадцатеричного числа <3>;в двоичное представление с использованием команды xlat. <4>;Вход: исходное шестнадцатеричное число; вводится с клавиатуры. <5>;Выход: результат преобразования в регистре al. <6>.data ;сегмент данных <7> message db ‘Введите две шестнадцатеричные цифры,$’ <8> tabl db 48 dup (0),0,1,2,3,4,5,6,7,8,9, 8 dup (0), <9> db 0ah,0bh,0ch,odh,0eh,0fh,27 dup (0) <10> db 0ah,0bh,0ch,odh,0eh,0fh, 153 dup (0) <11> .stack 256 ;сегмент стека <12> .code <13> ;начало сегмента кода <14> proc main ;начало процедуры main <15> mov ax,@data ;физический адрес сегмента данных в регистр ax <16> mov ds,ax ;ax записываем в ds <17> lea bx,tabl ;загрузка адреса строки байт в регистр bx <18> mov ah,9 <19> mov dx,offset message <20> int 21h ;вывести приглашение к вводу <21> xor ax,ax ;очистить регистр ax <22> mov ah,1h ;значение 1h в регистр ah <23> int 21h ;вводим первую цифру в al <24> xlat ;перекодировка первого введенного символа в al <25> mov dl,al <26> shl dl,4 ;сдвиг dl влево для освобождения места для младшей цифры <27> int 21h ;ввод второго символа в al <28> xlat ;перекодировка второго введенного символа в al <29> add al,dl ;складываем для получения результата <30> mov ax,4c00h ;пересылка 4c00h в регистр ax <31> int 21h ;завершение программы <32> endp main ;конец процедуры main <33> code ends ;конец сегмента кода <34> endmain ;конец программы с точкой входа main |
Сама по себе программа проста; сложность вызывает обычно формирование таблицы перекодировки. Обсудим этот момент подробнее.
Прежде всего нужно определиться с значениями тех байтов, которые вы будете изменять. В нашем случае это символы шестнадцатеричных цифр. Сконструируем в сегменте данных таблицу, в которой на места байтов, соответствующих символам шестнадцатеричных цифр, помещаем их новые значения, то есть двоичные эквиваленты шестнадцатеричных цифр. Строки 8-10 листинга 2 демонстрируют, как это сделать. Байты этой таблицы, смещения которых не совпадают со значением кодов шестнадцатеричных цифр, нулевые. Таковыми являются первые 48 байт таблицы, промежуточные байты и часть в конце таблицы.
Желательно определить все 256 байт таблицы. Дело в том, что если мы ошибочно поместим в al код символа, отличный от символа шестнадцатеричной цифры, то после выполнения команды xlat получим непредсказуемый результат. В случае листинга 2 это будет ноль, что не совсем корректно, так как непонятно, что же в действительности было в al — код символа “0” или что-то другое.
Поэтому, наверное, есть смысл здесь поставить “защиту от дурака”, поместив в неиспользуемые байты таблицы какой-нибудь определенный символ. После каждого выполнения xlat нужно будет просто контролировать значение в al на предмет совпадения с этим символом, и если оно произошло, выдавать сообщение об ошибке.
После того как таблица составлена, с ней можно работать. В сегменте команд строка 18 инициализирует регистр bx значением адреса таблицы tabl. Далее все очень просто. Поочередно вводятся символы двух шестнадцатеричных цифр, и производится их перекодировка в соответствующие двоичные эквиваленты.