Часть команд процессор может обработать только со значениями одной разрядности. Процессор без колебаний перемножит байт на байт и слово на слово. Но никак не байт на слово, или двойное слово на слово. Чтобы складывать, перемножать и выполнять другие операции со значениями разных размеров, программист должен озаботиться преобразованием типов в своей программе. Под преобразованием типов я […]
DIV
Синтаксис | DIV делитель | ||||||||||||||||||||
Машинный код | F6 /6 — DIV r/m8 F7 /6 — DIV r/m16 F7 /6 — DIV r/m32 | ||||||||||||||||||||
Описание | Деление без знака. | ||||||||||||||||||||
Действие | DIV используется при делении без учета знака. Команда делит 8-, 16-, 32- битные значение регистра (делимое) на хранимые в регистре или памяти байт, слово или двойное слово (делитель). Частное от деления помещается в AL, AX или EAX регистр соответственно. Остаток помещается в AH, DX или EDX. | ||||||||||||||||||||
Размер делимого (8-, 16- или 32-битный регистр), а также выделяемое место под результат определяются размером регистра, используемого в качестве делителя:
| |||||||||||||||||||||
Флаги | Флаги OF, SF, ZF, AF, PF и CF не установлены. |
Пример использования команды div:
1 2 3 | mov ax,240 mov cl,86 div cl ;AL = AX / CL {2}, остаток в AH {68} |
MUL
Синтаксис | MUL множитель | ||||||||||||||||
Машинный код | F6 /4 — MUL r/m8 F7 /4 — MUL r/m16 F7 /4 — MUL r/m32 | ||||||||||||||||
Описание | Умножение без знака. | ||||||||||||||||
Действие | Команда получает единственный явный аргумент, размер которого определяет регистры, выделеяемые командой под результат операции умножения (подробности приведены в таблице ниже). Другой множитель предварительно помещается в AL/AX/EAX регистр. | ||||||||||||||||
Размеры множителей и регистры, отводимые под результат исполнения команды:
| |||||||||||||||||
Флаги | Если старшая половина результата нулевая: OF=CF=0 SF=? ZF=? AF=? PF=?. Если старшая половина результата ненулевая: OF=CF=1 SF=? ZF=? AF=? PF=?. |
Пример использования команды mul:
1 2 3 | mov ax,5280 mov dx,3256 mul dx ;DX:AX = 1065300h |
Поход на рынок, как правило, заканчивается покупкой какого-либо товара. Если мы хотим купить несколько единиц товара одного наименования, то продавец посчитает нам итоговую сумму с использованием простейшего компьютера – микрокалькулятора. В аналогии с ассемблером при умножении товаров программа микрокалькулятора будет оперировать с числами без знака, поскольку нельзя купить минус три пури или минус пять круассанов. […]
CMP
Синтаксис | CMP приёмник,источник |
Машинный код | 3C ib — CMP AL,imm8 3D iw — CMP AX,imm16 3D id — CMP EAX,imm32 80 /7 ib — CMP r/m8,imm8 81 /7 iw — CMP r/m16,imm16 81 /7 id — CMP r/m32,imm32 83 /7 ib — CMP r/m16,imm8 83 /7 ib — CMP r/m32,imm8 38 /r — CMP r/m8,r8 39 /r — CMP r/m16,r16 39 /r — CMP r/m32,r32 3A /r — CMP r8,r/m8 3B /r — CMP r16,r/m16 3B /r — CMP r32,r/m32 |
Описание | Сравнение. |
Действие | Инструкция CMP сравнивает 2 операнда. Основное применение инструкции – проверка условий в циклах. CMP вычитает один операнд от другого, затем сравнивает, являются операнды равными или нет (см. команда SUB). Исходные операнды не меняются. Также используется для условных переходов по меткам. |
Флаги | OF=r SF=r ZF=r AF=r PF=r CF=r |
Пример использования инструкции CMP:
1 2 3 4 5 | CMP AX,0 ; Сравнить AX с 0 JE proc ; Если ноль, переход на метку proc ... ... proc: ... |
CMP часто используется для сверки счетчика, то есть сравнения числа повторений цикла с условием. Рассмотрим типичный случай:
1 2 | cmp cx,0xA ; Проверка, достиг ли счетчик 16 Jle lp1 ; Если счетчик равен или меньше 16, перейти на метку lp1 |
SBB
Синтаксис | SBB приемник,источник |
Машинный код | 1C ib — SBB AL,imm8 1D iw — SBB AX,imm16 1D id — SBB EAX,imm32 80 /3 ib — SBB r/m8,imm8 81 /3 iw — SBB r/m16,imm16 81 /3 id — SBB r/m32,imm32 83 /3 ib — SBB r/m16,imm8 83 /3 ib — SBB r/m32,imm8 18 /r — SBB r/m8,r8 19 /r — SBB r/m16,r16 19 /r — SBB r/m32,r32 1A /r — SBB r8,r/m8 1B /r — SBB r16,r/m16 1B /r — SBB r32,r/m32 |
Описание | Вычитание с заёмом. |
Действие | Вычитание двух целочисленных двоичных операндов с учётом флага CF. |
Флаги | OF=r SF=r ZF=r AF=r PF=r CF=r |
Команда SBB дополняет команду SUB при выполнени вычитания чисел частями. Состояние флага CF представляет собой заём от предыдущего вычитания. Приёмник = приёмник — (источник + CF)
Пример использования команды SBB для вычитания операнда размером в слово (2 байта) из двойного слова (4 байта):
1 2 3 4 5 6 7 8 9 | mov ax, word[a] ;ax = 0x680 mov bx, word[a+2] ;bx = 0x7700 sub ax,[b] ;0x680 - 0x43F2 = 0xC28E sbb bx,0 ;0x7700 - (0 + 1) = 0x76FF ;ax:bx = 0xC28E76FF ;------ Переменные ------ a dd 0x77000680 b dw 0x43F2 |
Регистры процессоров ограничены в разрядности архитектурой процессора (как правило, x86/64), поэтому сложение чисел, как из реального мира, так и из виртуального в ассемблере имеет дополнительные нюансы. В предыдущем Уроке 9. Сложение и вычитание операции сложения и вычитания выполнялись только в границах размеров регистров. Сложение операндов превышающих размеры регистров приводит к переполнению и установке флага CF. […]
NEG
Синтаксис | NEG операнд |
Машинный код | F6 /3 — NEG r/m8 F7 /3 — NEG r/m16 F7 /3 — NEG r/m32 |
Описание | Изменение знака. |
Действие | Команда вычисляет двоичное дополнение операнда. |
Флаги | Если операнд равен нулю: CF=0 OF=r SF=r ZF=r AF=r PF=r. Если операнд не равен нулю: CF=1 OF=r SF=r ZF=r AF=r PF=r. |
Команда NEG меняет знак операнда, преобразуя его значение в дополнительный код. Инструкция имеет смысл только для чисел со знаком, поскольку преобразование теряет смысл, если число 1 преобразуется в интерпретируемое беззнаковым число 255 (FF).
По выполнению инструкции устанавливаются флаги, которые характеризуют преобразование. Если произошло преобразование положительного числа в отрицательное, то SF=1, для всех остальных случаев SF=0.
Пример использования команды NEG:
1 2 3 4 5 6 7 8 | ; f = (a + b - 1) * (-a) mov ah,100 ; a = 100 mov al,85 ; b = 85 mov bl,ah ; bl = a neg bl ; bl = -a add ah,al ; a + b dec ah ; a + b - 1 mul bl ; ax = (a + b - 1) * (-a) |
SUB
Синтаксис | SUB приемник,источник |
Машинный код | 2C ib — SUB AL,imm8 2D iw — SUB AX,imm16 2D id — SUB EAX,imm32 80 /5 ib — SUB r/m8,imm8 81 /5 iw — SUB r/m16,imm16 81 /5 id — SUB r/m32,imm32 83 /5 ib — SUB r/m16,imm8 83 /5 ib — SUB r/m32,imm8 28 /r — SUB r/m8,r8 29 /r — SUB r/m16,r16 29 /r — SUB r/m32,r32 2A /r — SUB r8,r/m8 2B /r — SUB r16,r/m16 2B /r — SUB r32,r/m32 |
Описание | Вычитание. |
Действие | Вычитание двух целочисленных двоичных операндов. приёмник = приёмник — источник |
Флаги | OF=r SF=r ZF=r AF=r PF=r CF=r |
Команда SUB вычитает из первого операнда (приёмника) второй операнд. Результат вычитания переписывает значение операнда — приёмника.
Флаг SF имеет значение только для чисел со знаком. SF=1 указывает на отрицательный знак результата. Пример:
1 2 3 | mov ah,100 mov al,127 sub ah,al ; AH = 0xE5 = -27, SF=1 |
Если числа равноразмерные и без знака, то CF=1 будет указывать на некорректный результат (результат ушел в отрицательное значение). Например:
1 2 3 | mov ah,250 mov al,251 sub ah,al ;AH = 0xFF = -1 (число со знаком) и 0xFF = 255 (число без знака). Т.е. результат некорректный |
Для чисел со знаком и при вычитании меньшего по размеру числа от большего (пример, вычитание слова от двойного слова) CF=1 означает, что вычитание выполнилось с заёмом. Инструкция SBB продолжит вычитание с учетом флага CF:
1 2 3 4 5 6 7 8 | mov ax,0x37DD ; Старшая часть уменьшаемого числа mov bx,0x0077 ; Младшая часть уменьшаемого числа mov dx,0x3DDE ; Вычитаемое число sub bx,dx ; Сперва вычитаем от младшей части ; BX = BX - DX = 0xC299. SF=1, CF=1 sbb ax,0 ; Затем вычитаем от старшей части с учетом CF ; AX = AX - 1 = 0x37DC |
После того, как мы изучили механизм представления и обработки чисел процессором, самое время начать знакомство с инструкциями. Ну а к самым простым инструкциям в ассемблере безусловно относятся мнемоники для сложения и вычитания чисел — ADD и SUB. Сложение — команда ADD Инструкция ADD выполняет сложение двух операндов. ADD можно «скармливать» числа со знаком и без […]