Макросы и директивы компилятора FASM - страница 2
>macro stoschar [char]
> {
> mov al,char
> stosb
> }
Этот макрос принимает неограниченное количество аргументов, и каждый будет отдельно обработан с этими двумя инструкциями. Для примера >stoschar 1,2,3
будет скомпилирован как следующие инструкции:
>mov al,1
>stosb
>mov al,2
>stosb
>mov al,3
>stosb
Есть некоторые специальные директивы, располагаемые только в определениях макросов. Директива >local
определяет локальные имена, которые будут заменены новыми значениями каждый раз, когда используется макрос. Она должна сопровождаться именами, отделенными запятыми. Эта директива обычно нужна для констант или меток которые макрос определяет и использует внутри себя. Например:
>macro movstr
> {
> local move
> move:
> lodsb
> stosb
> test al,al
> jnz move
> }
При использовании этого макроса метка >move
всегда имеет свое уникальное значение и повторение этого макроса не вызовет ошибок.
Директивы >forward
, >reverse
и >common
делят макрос на блоки, каждый обрабатывается после того, как обработка предыдущих закончена. Они отличаются по поведению, только если макрос позволяет множественные группы аргументов. Блок инструкций, что следует за директивой >forward
, будет обработан для каждой группы аргументов с первого до последнего — наподобие обычному блоку (не заданному в соответствии с любой из этих директив). Блок, который следует за директивой >reverse
, будет обработан для каждой группы аргумента в обратном порядке — от последнего до первого. Блок, который следует за директивой >common
, будет обработан только однажды, сразу для всех групп аргументов. Значение >local
, определенное в одном из блоков доступно во всех следующих блоках при обработке той же самой группы аргументов где оно было определено, когда оно определено в блоке >common
, оно доступно во всех следующих блоках независимо какая группа аргументов обрабатывается.
Вот пример макроса, который создаст таблицу адресов к строкам, заданных этими строками:
>macro strtbl name,[string]
> {
> common
> label name dword
> forward
> local label
> dd label
> forward
> label db string,0
> }
Первый аргумент, переданный этому макросу, станет меткой для таблицы адресов, следующие аргументы должны быть строками. Первый блок обрабатывается только один раз и определяет метку, второй блок для каждой строки объявляет ее локальное имя и определяет запись таблицы, содержащую адрес той строки. Третий блок определяет данные каждой строки с соответствующей меткой. Директива, начинающая блок в макросе может сопровождаться первой инструкцией этого блока на той же самой строке, как показано в следующем примере:
>macro stdcall proc,[arg]
> {
> reverse
>push arg
> common
>call proc
> }
Этот макрос может использоваться для вызова процедур, использующих соглашение STDCALL, где аргументы сохраняются на стек в обратном порядке. Для примера >stdcall foo, 1,2,3
будет собран как:
>push 3
>push 2
>push 1
>call foo
Если некоторое имя в макросе имеет множественные параметры (одним из аргументов, огороженных в квадратные скобки или локальным именем, определенным в блоке, следующем за директивой >forward
или >reverse
) и используется в блоке, следующем за директивой common, это имя будет заменено всеми ее величинами, отделенными запятыми. Например следующий макрос передаст все дополнительные аргументы предварительно заданному макросу >stdcall
:
>macro invoke proc,[arg]
> {
>common
>stdcall [proc],arg
>}
Это может использоваться, чтобы вызывать косвенно (по указателю, находящемуся в памяти) процедуру, используя соглашение STDCALL.
Внутри макроса существует также специальный оператор >#
. Этот оператор связывает два имени в одно. Это может быть полезно, потому что это делается после того, как аргументы и локальные имена меняются на их настоящие значения. Следующий макрос произведет условный переход согласно аргументу >cond
:
>macro jif op1,cond,op2,label
> {
> cmp op1,op2
> j#cond label
> }
К примеру >jif ax,ae,10h,exit
будет скомпилировано как >cmp ax,10h
и >jae exit
.
Чтобы сделать макрос, ведущий себя в зависимости от типа аргумента, когда аргументы — строки или нет, Вы может использовать факт, что ассемблер отличает напрямую указанные строки от указанных строк в численных выражениях, но не отличает численное выражение, которому предшествуют знак