Передача параметров через стек
Передача параметров через регистры
Передача через COMMON область
Введение
Передача параметров из программы в подпрограмму
Часто необходимо не только "вызывать" подпрограмму, но и передавать в подпрограмму параметры. Формальным параметрам подпрограммы должны быть присвоены значения фактических параметров программы.
Решается проблема тремя способами:
а) передача через COMMON область
б) передача через регистры
в) передача через стек
COMMON область – участок ОЗУ, который ( по договорённости программистов пишущих программу и подпрограмму) зарезервирован для передачи данных между программами и подпрограммами. Программа пишет фактические значения в эту область, подпрограмма – считывает из этой области.
Недостатки:
- дополнительная, нежелательная связь между программой и подпрограммой
- адреса не могут быть изменены без одновременной перекомпиляции - как программы, так и подпрограммы.
Достоинства:
- относительно быстрый способ передать большие данные большого объёма.
Применение:
- передача параметров в тех случаях, когда надо передать большие объемы данных, скорость критична, и ради скорости можно пожертвовать удобствами раздельной компиляции программ и подпрограмм.
В качестве COMMON области используется не участок ОЗУ, а регистр.
Достоинство:
- максимальная скорость
Недостаток:
- ограниченное число передаваемых данных
Чем больше объем регистров процессора, тем чаще удастся использовать этот способ и тем большую реактивность ЭВМ можно получить. Поэтому в процессорах, специализированных под решение задач с повышенной производительностью используются регистровые массивы повышенного объема.
Не самый быстрый, но самый удобный способ. Удобство состоит в том, что не нужно согласовывать адреса между программой и подпрограммой. Программа записывает значения фактических параметров в системный стек, а подпрограмма - считывает эти значения из стека и присваивает их переменным формальных параметров.
Возьмем пример программы из [3.12.2] и добавим к программе передачу параметров через стек.
Начало программы |
CLI |
PUSH R1 |
… |
PUSH Rn |
MOVE reg1 FactPar1 (1) |
PUSH reg1 (2) |
… (2) |
MOVE reg1 FactParN)(2) |
PUSH reg1 (2) |
CALL #Adr |
STI |
Продолжение |
РOP reg2 (3) |
POP reg3 (3) |
POP reg1 (4) |
MOVE FormParN reg1(5) |
… (5) |
POP reg1 (5) |
MOVE FormPar1 reg1(5) |
PUSH reg3 (6) |
PUSH reg2 (6) |
STI |
Тело подпрограммы |
CLI |
RET |
Как мы видим – много общего с тем что мы рассматривали [3.12.2], отметим только новые моменты:
(1) - Сохранить в стеке можно только регистр. Регистр reg1 служит в качестве буфера. ( Может быть использован любой свободный регистр). В регистр записываем значение первого фактического параметра.
(2) – Первый фактический параметр сохраняется в стеке. Далее – все остальные фактические параметры.
(3) – Последние из двух сохранённых в стеке регистров, это регистры, сохранённые командой CALL. Считываем их на временное хранение в reg2 и reg3. ( Могут быть использованы любые два свободных регистра. ) Теперь, следующее считанное из стека значение – окажется последним из сохранённых в стеке фактических параметров.
(4) - Чтение из стека тоже осуществляется через буферный регистр. ( Может быть использован любой свободный регистр)
(5) – Присваиваем значение последнему из формальных параметров, далее всем остальным.
(6) – Теперь можно ( и нужно) освободить reg2 и reg3, возвращаем в стек значения регистров, в своё время сохраняемых в стеке командой CALL.