Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
СПО_Лекции_2011_01.docx
Скачиваний:
8
Добавлен:
31.07.2019
Размер:
521.09 Кб
Скачать

Пример: команды ror (циклический сдвиг вправо)

бит 7

бит 6

бит 5

бит бит бит 4 3 2

бит 1

бит 0

выдвигается

Операнд

1

0

0

1 1 0

1

1

ror операнд,3

1 0 0

1

1

011

Результат

0

1

1

1 0 0

1

1

Как видно из рисунка выше, биты вращаются, то есть каждый бит, который выталкивается снова вставляется с другой стороны. Флаг переноса cf содержит значение последнего выдвинутого бита.

RCL и RCR сдвигают все биты операнда влево (для RCL) или вправо (для RCR) на один разряд, при этом старший(для RCL) или младший(для RCR) бит становится значением флага переноса cf; одновременно старое значение флага переноса cf вдвигается в операнд справа(для RCL) или слева(для RCR) и становится значением младшего(для RCL) или старшего(для RCR) бита операнда. Указанные действия повторяются количество раз, равное значению второго операнда.

ROL и ROR сдвигают все биты операнда влево(для ROL) или вправо(для ROR) на один разряд, при этом старший(для ROL) или младший(для ROR) бит операнда вдвигается в операнд справа(для ROL) или слева(для ROR) и становится значением младшего(для ROL) или старшего(для ROR) бита операнда; одновременно выдвигаемый бит становится значением флага переноса cf. Указанные действия повторяются количество раз, равное значению второго операнда.

Стековые операции.

Иногда при вызове некоторых подпрограмм нужно сохранить все регистры для корректной работы программы в дальнейшем. Для этого вы будете последовательно сохранять каждый регистр в стеке и извлекать точно также:

Push eax Push ebx Push ecx

[также все остальные регистры]

[вытаскиваем оттуда остальные регистры] Pop ecx Pop ebx Pop eax

Такой подход очень громоздок именно для этого и создана команда pushad. Она сохраняет все расширенные регистры в стеке. Соответственно команда popad нужна для извлечения из стека всех регистров.

Pushad

Popad

Также есть команды для сохранения младших регистров pushaw и popaw. Они равносильны командам PUSH AX, CX, DX, BX, SP, BP, SI, DI и POP DI,SI,BP,BX,DX,CX,AX.

Иногда надо сохранять значения флагов после выполнения некоторой операции. Для этого нужны команды pushf и popf. Эти команды сохраняют в стеке значение регистра флагов или восстанавливают его оттуда. Изменение флагов процессора.

CLC установка CF=0 CLD установка DF=0 CLI установка IF=0

LAHF загрузка регистра AH из регистра флагов: устанавливает биты 7, 6, 4, 2 и 0 по значениям флагов SF ZF AF PF CF.

SAHF запись регистра AH в регистр флагов: устанавливает флаги SF ZF AF PF CF по битам 7, 6, 4, 2 и 0 регистра AH. STC установка CF=1 STD установка DF=1 STI установка IF=1 Обработка блоков данных Загрузка и выгрузка данных.

Команды lodsb, lodsw, lodsd нужны, для того чтобы загрузить байт (слово, двойное слово) из памяти, на которую указывает регистр ESI в регистр al,ax или eax. После присваивания значения регистру значение регистра ESI увеличивается на 1,2 или 4. Вот эквиваленты этих команд

Lodsb Mov al, byte ptr [esi]

Inc esi /dec esi Lodsw Mov ax, word ptr [esi] Add esi, 2h /sub esi, 2h Lodsd Mov eax, dword ptr [esi] Add esi, 4h / sub esi, 4h

Следовательно, эти команды облегчают нам обработку данных, сразу увеличивая значение регистра esi для следующей порции данных. Регистр может также уменьшаться в зависимости от флага направления DF. Если он выставлен то регистр будет уменьшаться, а если он сброшен, то регистр esi будет увеличиваться.

Абсолютно, противоположное делают команды stosb, stows и stosd. Они загружают регистр al, ax, eax в память, на которую указывает регистр edi. Изменение регистра edi зависит от того же самого флага направления DF.

movsb Mov al, byte ptr [esi]

Mov byte ptr [edi], al Inc edi /dec edi Inc esi /dec esi stosw Mov ax, word ptr [esi]

Mov word ptr [edi], ax Add edi, 2h /sub edi, 2h Add esi, 2h /sub esi, 2h stosd Mov eax, dword ptr [esi]

Mov dword ptr [edi], eax Add edi, 4h / sub edi, 4h Add esi, 4h / sub esi, 4h

Сравнение данных.

Команды scasb, scasw и scasd сравнивают содержимое регистра al, ax, eax с содержимым памяти на которую указывает регистр edi. Разумеется с увеличением регистра edi. Более продвинутые команды cmpsb, cmpsw, cmpsd сравнивают значения памяти, на которые указывают регистры esi и edi соответственно. С соответствующим изменением регистра флагов.

Префикс rep.

Все вышеописанные команды предназначены удобно использовать в циклах, т.е. при повторяющихся операциях. Префикс rep нужен для повторения команды обработки блоков количество раз, которое указано в регистре ecx. Пример:

Mov ecx, 045h Mov edi, 0422310h Mov al, 088h Rep stosb

После этого память в диапазоне 0422310h-0422355h будет заполнена значениями 88h. Для команд cmps* и scas* префикс повторяет команду только до тех пор, пока флаг ZF установлен. Префиксы repe, repz это тоже самое. Если после очередного выполнения команды флаг ZF сбросился, то повторы прекращаются.

Не менее полезен префикс repne, он будет повторять команду пока флаг ZF сброшен. Если после очередного выполнения команды флаг ZF установился, то повторы прекращаются. То же самое делает префикс repnz.

Str1 db 'qwerty' Str2 db '111111'

Mov ecx, 06h Mov esi, offset str1 Mov edi, offset str2 Rep movsb

После всех этих манипуляций данные будут такими:

Str1 db 'qwerty' Str2 db 'qwerty'

Ещё пример:

Str1 db 'qwerty' Str2 db 'qwe111'

Mov ecx, 06h Mov esi, offset str1 Mov edi, offset str2 Rep cmpsb

Это повторение закончится на 4 шаге при сравнении 'r' и '1' т.е. при сравнении 'r' и '1' флаг ZF сбросится и повторение прекратится. Тем самым можно сравнивать две строки а после Rep cmpsb проверять равен ли регистр ECX нулю, если он равен нулю, то строки равны.

Str1 db 'qwerty' Str2 db '1111t1'

Mov ecx, 06h Mov esi, offset str1 Mov edi, offset str2 Repne cmpsb

Это повторение прекратится на 5 шаге при сравнении 't' и 't' т.е. при их сравнении флаг ZF установится и повторение прекратится.