Студопедия

КАТЕГОРИИ:

АвтоАвтоматизацияАрхитектураАстрономияАудитБиологияБухгалтерияВоенное делоГенетикаГеографияГеологияГосударствоДомЖурналистика и СМИИзобретательствоИностранные языкиИнформатикаИскусствоИсторияКомпьютерыКулинарияКультураЛексикологияЛитератураЛогикаМаркетингМатематикаМашиностроениеМедицинаМенеджментМеталлы и СваркаМеханикаМузыкаНаселениеОбразованиеОхрана безопасности жизниОхрана ТрудаПедагогикаПолитикаПравоПриборостроениеПрограммированиеПроизводствоПромышленностьПсихологияРадиоРегилияСвязьСоциологияСпортСтандартизацияСтроительствоТехнологииТорговляТуризмФизикаФизиологияФилософияФинансыХимияХозяйствоЦеннообразованиеЧерчениеЭкологияЭконометрикаЭкономикаЭлектроникаЮриспунденкция

Директивы компиляции по условию




Макродирективы

 

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

       1.Директивы повторения WHILE, REPT, IRP и IRPC предназначены для создания макросов, содержащих несколько идущих подряд одинаковых последовательностей строк. При этом возможна частичная модификация этих строк.

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

Директивы WHILE и REPT

       Директивы WHILE и REPT применяются для повторения определенное количество раз некоторой последовательности строк. Эти директивы имеют следующий синтаксис:

WHILE константное_выражение

последовательность строк

ENDM

REPT константное_выражение

последовательность строк

ENDM

       Обратите внимание на то, что последовательность повторяемых строк в обеих директивах ограничена директивой ENDM.

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

       Директива REPT, подобно директиве WHILE, повторяет последовательность строк столько раз, сколько это определено значением константное_выражение. Отличие этой директивы от WHILE состоит в том, что она автоматически уменьшает на единицу значение константное_выражение после каждой итерации.

       Пример. В нем демонстрируется применение директив WHILE и REPT для резервирования области памяти в сегменте данных. Имя идентификатора и длина области задаются в качестве параметров соответствующих макросов def_sto_l и def_sto_2. Заметьте, что счетчик повторений в директиве REPT уменьшается автоматически после каждой итерации цикла.

       Использование директив повторения

def_sto_l macro id_table, ln:=<5>

;макрос резервирования памяти длиной ten

; Используется WHILE

id_table label byte

len=ln

       while len

                   db 0

                   len=len-l

       endm

endm

def_sto_2 macro id_table, len

;макрос резервирования памяти длиной len

;Используется REPT

id_table label byte

       rept len

                   db 0

       endm

endm

data segment para public 'data'

def_sto_l tab_l, 10

def_sto_2 tab_2, 10

data ends

init_ds macro

;Макрос настройки ds на сегмент данных

mov ax, data

mov ds. ax

endm

exit macro

;макрос конца программы

mov ax, 4c00h

int 21h

endm

code segment para public ‘code’

assume cs:code, ds:data

main proc

init_ds

exit

main endp

code ends

end main

       Таким образом, директивы REPT и WHILE удобно применять для «размножения » в тексте программы последовательности одинаковых строк без внесения в эти строки каких-либо изменений на этапе трансляции.

Директива IRP

       Директива IRP имеет следующий синтаксис:

IRP формальный_аргумент, <строка_символов_1,…, строка_символов_n>

последовательность строк

ENDM

       Действие данной директивы заключается в том, что она повторяет последовательность строк п раз, то есть столько раз, сколько строк символов заключено в угловые скобки во втором операнде директивы IRP. Повторение последовательности строк сопровождается заменой в этих строках формального аргумента очерёдной строкой символов из второго операнда. Так, при первой генерации последовательности строк формальный аргумент в них заменяется первой строкой символов (то есть аргументом строка_символов_1). Если есть вторая строка символов (строка_символов_2), это приводит к генерации второй копии последовательности строк, в которой формальный аргумент заменяется второй строкой символов. Эти действия продолжаются до последней строки символов (строка_символов_n) включительно.

       К примеру, рассмотрим результат определения в программе такой конструкции:

irp ini, <1, 2, 3, 4, 5>

db ini

endm

Макрогенератором будет сгенерировано следующее макрорасширение:

db 1

db 2

db 3

db 4

db 5

Директива IRPC

       Директива IRPC имеет следующий синтаксис:

IRPC формальный_аргумент, строка_символов

последовательность строк

ENDM

 

       Действие данной директивы подобно действию директивы IRP, но отличается тем, что она на каждой очередной итерации заменяет формальный аргумент очередным символом из строки символов. Понятно, что количество повторений последовательности строк будет определяться количеством символов в строке символов. К примеру,

irpc rg. abcd

push rg&x

endm

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

push ax

push bx

push ex

push dx

       Если строка символов, задаваемая в директиве IRP, содержит спецсимволы вроде точек и запятых, то она должна быть заключена в угловые скобки:<ab, , cd>.

Директивы условной компиляции

       Последний тип макросредств — директивы условной компиляции. Существует два вида этих директив:

       1. директивы компиляции по условию позволяют проанализировать определенные условия в ходе генерации макрорасширения и при необходимости изменить этот процесс;

       2. директивы генерации ошибок по условию контролируют ход генерации макро-расширения с целью генерации или обнаружения определенных ситуаций, которые могут интерпретироваться как ошибочные.

       С этими директивами применяются упомянутые ранее директивы управления процессом генерации макрорасширений EXITM и GOTO.

       Директива EXITM не имеет операндов, она немедленно прекращает процесс генерации макрорасширения, как только встречается в макроопределении.

       Директива GOTO имя_метки переводит процесс генерации макроопределения в другое место, прекращая тем самым последовательное разворачивание строк макроопределения. Метка, на которую передается управление, имеет специальный формат:

       :имя_метки

 

Директивы компиляции по условию

       Директивы компиляции по условию предназначены для выборочной трансляции фрагментов программного кода. Это означает, что в макрорасширение включаются не все строки макроопределения, а только те, которые удовлетворяют определенным условиям. Какие конкретно условия должны быть проверены, определяется типом условной директивы. Всего имеются 10 типов условных директив компиляции. Их логично попарно объединить в четыре группы:

       1. IF и IFE — условная трансляция по результату вычисления логического выражения;

       2. IFDEF и IFNDEF — условная трансляция по факту определения символического имени;

       3. IFB и IFNB – условная трансляция по факту определения фактического аргумента при вызове макрокоманды;

       4. IFIDN, IFIDNI, IFDIF и IFDIFI — условная трансляция по результату сравнения строк символов.

       Условные директивы компиляции имеют общий синтаксис и применяются в составе следующей синтаксической конструкции:

IFxxx логическое_выражение_или_аргументы

фрагмент_программы_1

ELSE

фрагмент_программы_2

ENDIF

       Заключение некоторых фрагментов текста программы (фрагмент_программы_1 и фрагмент_программы_2) между директивами IFxxx, ELSE и ENDIF приводит к их выборочному включению в объектный модуль. Какой именно из этих фрагментов будет включен в объектный модуль, зависит от конкретного типа условной директивы, задаваемого значением ххх, и значения условия, определяемого операндом (операндами) условной директивы логическое_выражение_или_аргумент(ы).

Директивы IF и IFE

       Синтаксис директив IF и IFE следующий:

       IF(E)логическое_выражение

       фрагмент_программы_1

                   ELSE

       фрагмент_программы_2

       ENDIF

       Обработка этих директив макроассемблером заключается в вычислении логического выражения и включении в объектный модуль первого (фрагмент_программы_1) или второго (фрагмент_программы_2) фрагмента программы в зависимости от того, в какой директиве (IF или IFE)это выражение встретилось.

       1.Если в директиве IF логическое выражение истинно, то в объектный модуль помещается первый фрагмент программы. Если логическое выражение ложно, то при наличии директивы ELSE в объектный код помещается второй фрагмент программы. Если же директивы ELSE нет, то вся часть программы между директивами IF и ENDIF игнорируется, и в объектный модуль ничего не включается. Ложным оно будет считаться, если его значение равно нулю, а истинным — при любом значении, отличном от нуля.

       2. Директива IFE аналогично директиве IF анализирует значение логического выражения. Но теперь для включения первого фрагмента программы в объектный модуль требуется, чтобы логическое выражение было ложным.

       Директивы IF и IFE очень удобно использовать для изменения текста программы в зависимости от некоторых условий. К примеру, составим макрос для определения в программе области памяти длиной не более 50 и не менее 10 байт.

;Использование условных директив IF и IFE

masm

model small

stack 256

def_tab_50 macro len

if len GE 50

       GOTO exit

Endif

if len LT 10

:exit

EXITM

endif

rept len

       db 0

endm

endm

.data

def_tab_50 15

def_tab_50 5

.code

main:

       mov ax, @data

       mov ds, ax

exit:

       mov ax, 4c00h

       int 21h

end main

       Условные директивы действуют только на шаге трансляции, и поэтому результат их работы можно увидеть лишь после макрогенерации, то есть в листинге программы.

       Другой интересный и полезный вариант применения директив IF и IFE — отладочная печать. Суть здесь в том, что в процессе отладки программы почти всегда возникает необходимость динамически отслеживать состояние определенных программно-аппаратных объектов, в качестве которых могут выступать переменные, регистры процессора и т. п. После этапа отладки отпадает необходимость в таких диагностических сообщениях. Для их устранения приходится корректировать исходный текст программы, после чего подвергать ее повторной трансляции. Но есть более изящный выход. Можно определить в программе некоторую переменную, к примеру debug, и использовать ее совместно с условными директивами IF или IFE:

debug equ i

. code

if debug

;любые команды и директивы ассемблера

;(вывод на печать или монитор)

endif

       На время отладки и тестирования программы вы можете заключить отдельные участки кода в своеобразные операторные скобки в виде директив IF и ENDIF, реагирующие на значение логической переменной debug. При значении debug =0 транслятор полностью проигнорирует текст внутри этих условных операторных скобок;при debug =1, наоборот, будут выполнены все действия, описанные внутри них.

Директивы IFDEF и IFNDEF

       Синтаксис директив IFDEF и IFNDEF следующий:

IF(N)DEF символическое_имя

фрагмент_программы_1

ELSE

фрагмент_программы_2

ENDIF

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

       1. Директива IFDEF проверяет, описано или нет в программе символическое имя, и если это так, то в объектный модуль помещается первый фрагмент программы (фрагмент_программы_1). В противном случае при наличии директивы ELSE в объектный код помещается второй фрагмент программы (фрагмент_программы_2). Если же директивы ELSE нет (и символическое имя в программе не описано), то вся часть программы между директивами IF и ENDIF игнорируется и в объектный модуль не включается.

       2. Действие IFNDEF обратно действию IFDEF. Если символического имени в программе нет, то транслируется первый фрагмент программы. Если оно присутствует, то при наличии ELSE транслируется второй фрагмент программы. Если ELSE отсутствует, а символическое имя в программе определено, то часть программы, заключенная между IFNDEF и ENDIF, игнорируется.

       В качестве примера рассмотрим ситуацию, когда в объектный модуль программы должен быть включен один из трех фрагментов кода в зависимости от значения некоторого идентификатора switch:

       – если switch =0, то сгенерировать фрагмент для вычисления выражения у=х*2**n;

       – если switch =1, то сгенерировать фрагмент для вычисления выражения у =х/2**n;

       – если идентификатор switch не определен, то ничего не генерировать.

       Соответствующий фрагмент исходной программы может выглядеть так:

ifndef sw ;если sw не определено, то выйти из макроса

EXITM

else ;иначе на вычисление

mov cl, n

       ife sw

                   sal x, cl ;умножение на степень 2 сдвигом влево

       else

                   sar x, cl ;деление на степень 2 сдвигом вправо

       endif

endif

       Как видим, эти директивы логически связаны с директивами IF и IFE, то есть их

можно применять в тех же самых случаях, что и последние.

Директивы IFB и IFNB

       Синтаксис директив IFB и IFNB следующий:

       IF(N)B аргумент

                   фрагмент_программы_1

       ELSE

                   фрагмент_программы_2

       ENDIF

       Данные директивы используются для проверки фактических параметров, передаваемых в макрос. При вызове макрокоманды они анализируют значение аргумента и в зависимости от того, равно оно пробелу или нет, транслируется либо первый фрагмент программы (фрагмент_программы_1), либо второй (фрагмент_программы_2). Какой именно фрагмент будет выбран, зависит от кода директивы.

       1. Директива IFB проверяет равенство аргумента пробелу. В качестве аргумента могут выступать имя или число. Если его значение равно пробелу (то есть фактический аргумент при вызове макрокоманды не был задан), то транслируется и помещается в объектный модуль первый фрагмент программы. В противном случае при наличии директивы ELSE в объектный код помещается второй фрагмент программы. Если же директивы ELSE нет, то при равенстве аргумента пробелу вся часть программы между директивами IFB и ENDIF игнорируется и в объектный модуль не включается.

       2. Действие IFNB обратно действию IFB. Если значение аргумента в программе не равно пробелу, то транслируется первый фрагмент программы. В противном случае при наличии директивы ELSE в объектный код помещается второй фрагмент программы. Если же директивы ELSE нет, то вся часть программы (при неравенстве аргумента пробелу)между директивами IFNB и ENDIF игнорируется и в объектный модуль не включается.

       Типичным примером применения этих директив являются строки в макроопределении, проверяющие, указывается ли фактический аргумент при вызове соответствующей макрокоманды:

show macro reg

ifb <reg>

display "не задан регистр"

exitm

endif

endm

 

       Если теперь в сегменте кода вызвать макрос SHOW без аргументов, то будет выведено сообщение о том, что не задан регистр, и генерация макрорасширения прекратится директивой EXITM.










Последнее изменение этой страницы: 2018-04-12; просмотров: 273.

stydopedya.ru не претендует на авторское право материалов, которые вылажены, но предоставляет бесплатный доступ к ним. В случае нарушения авторского права или персональных данных напишите сюда...