Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
Скляров И. Изучаем Assembler за 7 дней (2010).pdf
Скачиваний:
1335
Добавлен:
23.02.2015
Размер:
2.11 Mб
Скачать

http://www.sklyaroff.ru

79

соответствует умножению операнда на степени двойки, а вправо, соответственно, делению операнда на степени двойки. То есть сдвиг влево на один бит соответствует умножению операнда на 2, сдвиг влево на два бита соответствует умножению на 4, на три бита — умножению на 8 и т. д.

Аналогично с делением. Сдвиг вправо на один бит соответствует делению на 2, сдвиг вправо на два бита соответствует делению на 4, на три бита — делению на 8 и т. д. Можете проверить это самостоятельно.

Команды арифметического сдвига выполняют умножение и деление на степень двойки намного быстрее, чем команды MUL, DIV и IDIV, поэтому программисты на ассемблере и используют их.

4.6.2. Команды циклического сдвига

Команды из второй группы имеют следующий синтаксис:

ROL операнд,счетчик ; циклический сдвиг влево

ROR операнд,счетчик ; циклический сдвиг вправо

RCL операнд,счетчик ; циклический сдвиг влево через флаг переноса RCR операнд,счетчик ; циклический сдвиг вправо через флаг переноса

Команды циклического сдвига выполняют циклический сдвиг битов (ротацию) влево или вправо.

Команды ROL и ROR сдвигают все биты операнда соответственно влево или вправо на столько бит, сколько указано в счетчике, но в отличие от команд линейного (нециклического) сдвига не обнуляют освобожденные разряды, а снова вдвигают их с другой стороны, т. е. старшие биты попадают в младшие или наоборот.

Примеры:

mov al,C8h ; al= 11001000b ror al,4

; Теперь al = 10001100b. Флаг переноса cf=1, т. к. последний выдвинутый слева бит был 1.

mov bh,D6h ; bh= 11010110b rol bh,3

; Теперь bh = 10110110b. Флаг переноса cf=0, т. к. последний выдвинутый слева бит был 0.

Команды RCL и RCR выполняют действие аналогичное командам ROL и ROR, но включают флаг переноса CF в цикл, как если бы он был дополнительным битом в операнде. Последний выдвигаемый бит заносится в флаг CF, а значение CF при этом поступает в освободившуюся позицию Должно быть понятно, что команда RCL помещает значение CF в младший разряд, а команда RCR в старший.

Примеры:

mov al,C8h ; al= 11001000b rcr al,4

; Теперь al = 11000110b. В старший разряд был вдвинут флаг переноса cf=1, т. к. последний выдвинутый слева бит был 1.

mov bh,D6h ; bh= 11010110b rcl bh,3

; Теперь bh = 01101100b. В младший разряд был вдвинут флаг переноса cf=0, т. к. последний выдвинутый справа бит был 0.

4.7. Команды обработки строк/цепочечные команды

Команды обработки строк или цепочечные команды работают с последовательностями элементов. Один элемент может быть размером в байт (8 бит), слово (16 бит) или двойное слово (32 бита). Обычно последовательностями

http://www.sklyaroff.ru

80

элементов являются обычные строки символов, поэтому эти команды и называют командами обработки строк. Во всех цепочечных командах строка-источник должна находиться стандартно по адресу DS:SI (или DS:ESI), то есть в сегменте данных, определяемом регистром DS со смещением SI (или ESI), а строка-приемник по адресу в ES:DI (или ES:EDI) иначе говоря в сегменте данных, адресуемом сегментным регистром ES.

Все цепочечные команды за один раз обрабатывают только один элемент цепочки (строки). Для того чтобы команда выполнялась над всей строкой, используется один из префиксов повторения операций:

REP — повторять;

REPE — повторять, пока равно;

REPZ — повторять, пока ноль; REPNE — повторять, пока не равно; REPNZ — повторять, пока не ноль.

Эти префиксы указываются перед нужной цепочечной командой в поле метки и заставляют команду выполняться в цикле. Без префикса цепочечная команда выполняется один раз. Все префиксы анализируют значение регистра ECX/CX, а также флаг нуля ZF (кроме префикса REP).

Префикс повторения REP (от англ. REPeat — повторять) обычно используется совместно с командами MOVS и STOS и заставляет данные команды выполняться, пока содержимое в регистрах ECX/CX не станет равным 0. При этом цепочечная команда, перед которой стоит префикс, автоматически уменьшает содержимое ECX/CX на единицу. Та же команда, но без префикса этого не делает.

Префиксы повторения REPE (REPeat while Equal — повторять пока равно) и REPZ (REPeat while Zero — повторять пока ноль) являются абсолютно идентичными и обычно применяются с командами CMPS и SCAS для поиска отличающихся элементов цепочек. Эти префиксы заставляют цепочечные команды циклически выполняться, пока содержимое ECX/CX не равно нулю или пока флаг ZF не будет сброшен в 0, если это произойдет, управление будет передано следующей команде программы.

Префиксы повторения REPNE (REPeat while Not Equal — повторять пока не равно) или REPNZ (REPeat while Not Zero — повторять пока не ноль) также являются абсолютно идентичными и используются обычно с командами CMPS и SCAS, но для поиска совпадающих элементов. Эти префиксы заставляют цепочечные команды циклически выполняться до тех пор, пока содержимое ECX/CX не равно нулю или пока флаг ZF не будет установлен в 1, если это произойдет, управление будет передано следующей команде программы.

Строки в командах обработки строк могут обрабатываться либо от начала к концу, либо, наоборот, от конца к началу. Направление обработки задает флаг направления DF. Флаг направления анализируют цепочечные команды и если DF=0, то обрабатывают элементы в направлении возрастания адресов, если DF=1, то в направлении убывания адресов. Состоянием флага DF можно управлять с помощью команд CLD и STD, не имеющих операндов

CLD (Clear Dierction Flag) — сбрасывает флаг направления DF в 0, в результате чего при последующих строковых операциях значение в регистрах DI и SI будет увеличиваться.

STD (Set Direction Flag) — устанавливает флаг направления DF в 1, в результате чего при последующих строковых операциях значение в регистрах DI и SI будет уменьшаться.

Обратите внимание, что в каждой группе цепочечных команд (ниже) имеется одна команда с двумя операндами и несколько команд без операндов. На самом деле в процессоре существуют только цепочечные команды без операндов. При использовании команды с операндами транслятор ассемблера сам определяет по типу указанных операндов, какую из трех форм команды без операндов сгенерировать.

4.7.1. Команды пересылки цепочек

В эту группу входят следующие команды:

http://www.sklyaroff.ru

81

movs приемник,источник (MOVe String) — переслать цепочку; movsb (MOVe String Byte) — переслать цепочку байт;

movsw (MOVe String Word) — переслать цепочку слов;

movsd (MOVe String Double word) — переслать цепочку двойных слов.

Команда MOVSB копирует байт из ячейки памяти, адресуемой парой регистров DS:SI (DS:ESI), в ячейку памяти, адресуемой парой регистров ES:DI (ES:EDI). Действие команды MOVSW аналогично, только копируется слово. Соответственно команда MOVSD копирует двойное слово. Команда MOVS с операндами транслируется в одну из трех команд без операндов: MOVSB, MOVSW, MOVSD.

Без префикса команда пересылки цепочек копирует только один элемент цепочки. С префиксом REP можно переслать до 64 Кбайт данных в 16-разрядной программе или до 4 Гбайт данных в 32-разрядной программе. Число пересылаемых элементов должно быть указано в регистре CX/ECX.

Алгоритм пересылки элементов цепочки может выглядеть так:

5.Загрузить адрес источника в регистр DS:SI (DS:ESI), а адрес приемника в ES:DI (ES:EDI).

6.Задать с помощью команды CLD или STD направление обработки цепочки (от начала к концу или от конца к началу).

7.Задать в регистре ECX/CX число обрабатываемых элементов.

8.Выдать команду MOVS (MOVSB, MOVSW или MOVSD) с префиксом REP. Например, следующий блок кода копирует 10 байт строки text1 в строку text2:

lea SI,text1 lea DI,text2 cld

mov CX,10 rep movsb

При этом предполагается, что строка text1 находится в сегменте данных, на который указывает регистр DS, а строка text2 — в сегменте, на который указывает регистр ES (для программы типа .COM значение регистров DS и ES совпадает).

4.7.2. Команды сравнения цепочек

В эту группу входят следующие команды:

cmps приемник,источник (CoMPare String) — сравнить строки; cmpsb (CoMPare String Word) — сравнить строку байт;

cmpsw (CoMPare String Word) — сравнить строку слов;

cmpsd (CoMPare String Double word) — сравнить строку двойных слов.

Команды этой группы сравнивают элементы цепочки-приемника (адресуемой регистрами ES:DI (ES:EDI)) с элементами цепочки-источника (адресуемой регистрами DS:SI (DS:ESI)). В зависимости от флага DF команда сравнения цепочек увеличивает или уменьшает адреса в регистрах SI и DI на один байт (CMPSB), слово

(CMPSW) или двойное слово (CMPSD).

Принцип работы команд сравнения цепочек аналогичен команде сравнения CMP. Они так же, как и CMP, производят вычитание элементов, не записывая при этом результата, и устанавливают флаги ZF, SF и OF.

Если использовать команду сравнения цепочек с префиксами повторения REPNE/REPNZ или REPE/REPZ, то сравниваться будут столько элементов, сколько указано в регистре CX/ECX. При этом в случае префиксов REPNE/REPNZ сравнение прекратится при первом совпадении в цепочке, а в случае префиксов REPE/REPZ – при первом несовпадении.

Для определения причины, которая привела к выходу из цикла сравнения цепочек, обычно используется команда условного перехода JCXZ. Эта команда анализирует содержимое регистра CX/ECX, и если оно равно нулю передает управление на метку, указанную в качестве операнда в JCXZ. Если значение в CX/ECX не равно нулю, то это означает, что выход произошел по причине совпадения или несовпадения очередных элементов в цепочке.

http://www.sklyaroff.ru

82

В листинге 4.1 в качестве примера показана программа сравнения двух строк. Если строки text1 и text2 равны, то осуществляется переход на метку Equality.

Листинг 4.1. Программа сравнения двух строк (cmpstr.asm)

.model small

.stack 100h

.code

start:

mov

ax,@data

 

mov

ds,ax

 

mov

es,ax

 

lea

si,text1

; в SI адрес начала строки text1

lea

di,text2

; в DI адрес начала строки text2

cld

 

 

; в CX длина строки text1 (число символов для сравнения) mov cx,len_text1

repe cmpsb

; сравнивать, пока элементы равны

je

Equality

 

mov

dx,offset mes_no

 

mov

ah,9

 

 

int

21h

 

 

jmp

exit

 

 

Equality:

 

 

 

mov

dx,offset mes_yes

 

mov

ah,9

 

 

int

21h

 

 

exit:

 

; выход из программы

mov

ax,4C00h

 

int

21h

 

 

.data

 

 

 

text1

db

"Ivan Sklyaroff"

; первая строка

len_text1=$-text1

 

; длина первой строки

text2

db

"Ivan Sklyaroff"

; вторая строка

mes_yes

db

"Строки одинаковы",'$'

mes_no

db

"Строки различны",'$'

 

end start

4.7.3. Команды сканирования цепочек

В эту группу входят следующие команды:

scas приемник (SCAning String) — сканировать цепочку; scasb (SCAning String Byte) — сканировать цепочку байт; scasw (SCAning String Word) — сканировать цепочку слов;

scasd (SCAning String Double Word) — сканировать цепочку двойных слов.

http://www.sklyaroff.ru

83

Команды сканирования цепочек осуществляют поиск заданного значения в строке. Значение, которое требуется найти, необходимо предварительно поместить в регистр AL (если ищется байт), в AX (если ищется слово) или в EAX (если осуществляется поиск двойного слова), а адрес строки должен быть сформирован в регистре ES:DI (ES:EDI). Принцип поиска тот же, что и в команде сравнения CMPS, то есть выполняется последовательное вычитание (из содержимого регистра аккумулятора содержимое очередного элемента цепочки) и в зависимости от результатов вычитания производится установка флагов, при этом ни один из операндов не изменяется.

Если использовать команду сканирования цепочки с префиксом REPE или REPZ, то команда будет выполняться в цикле до тех пор, пока не будет достигнут конец цепочки (содержимое CX/ECX равно 0) или пока не встретится элемент, отличный от элемента в регистре AL/AX/EAX.

Если применить префикс REPNE или REPNZ, то цикл будет выполняться пока в цепочке не встретится элемент, совпадающий с элементом в регистре AL/AX/EAX или пока не будет достигнут конец цепочки (содержимое CX/ECX равно 0).

То есть команда сканирования цепочки с префиксом REPE или REPZ позволяет найти элемент цепочки, отличающийся от заданного в аккумуляторе, а с префиксом REPNE или REPNZ – совпадающий по значению с элементом в аккумуляторе.

В листинге 4.2 в качестве примера показана программа, которая ищет символ 't' в строке. Если символ найден, то осуществляется переход на метку found.

Листинг 4.2 Поиск символа в строке (scastr.asm)

.model small

.stack 100h

.code

start:

 

 

mov

ax,@data

 

mov

ds,ax

 

mov

es,ax

 

cld

 

 

lea

di,text1

; в DI адрес начала строки text1

mov

cx,len_text1

; в CX длину строки text1

mov

ah,0

 

mov

al,'t'

; ищем символ 't' в строке

repne

scasb

 

je

found

 

mov

dx,offset mes_no

mov

ah,9

 

int

21h

 

jmp

exit

 

found:

 

 

mov

dx,offset mes_yes

mov

ah,9

 

int

21h

 

exit:

 

 

mov

ax,4C00h

 

int

21h

 

.data