Структура ассемблерной программы




АССЕМБЛЕР AVR-МИКРОКОНТРОЛЛЕРОВ

Особенности ассемблера

Здесь представлена основная информация по ассемблеру всей серии AVR, т.к. все микроконтроллеры этой серии программно совместимы.

Ассемблер – это инструмент, с помощью которого создаётся программа для микроконтроллера. Ассемблер транслирует ассемблируемый исходный код программы в объектный код, который может быть непосредственно введен в программную память микроконтроллера, а также использоваться в симуляторах или эмуляторах AVR.

При работе с ассемблером нет никакой необходимости в непосредственном соединении с микроконтроллером.

Исходный файл, с которым работает ассемблер, должен содержать мнемоники, директивы и метки.

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

Строка программы может быть в одной из четырёх форм:

[Метка:] директива [операнды] [Комментарий]

[Метка:] команда [операнды] [Комментарий]

Комментарий

Пустая строка

Комментарий имеет следующую форму:

; [Текст]

Таким образом любой текст после символа «;» игнорируется ассемблером и имеет значение только для пользователя.

Операнды можно задавать в различных форматах:

– десятичный (по умолчанию): 10,255

– шестнадцатеричный (два способа): 0x0а, $0а

– двоичный: 0b00001010, 0b11111111

– восьмеричный (впереди ноль): 010, 077

Директивы ассемблера

Ассемблер поддерживает множество директив. Директивы не транслируются непосредственно в коды операции. Напротив, они используются, чтобы корректировать местоположение программы в памяти, определять макрокоманды, инициализировать память и так далее. То есть это указания самому ассемблеру, а не команды микроконтроллера.

Директивы соответствуют второй версии компилятора ассемблера avrasm2.exe компании «Atmel».

Директивы ассемблера приведены в таблице 8.

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

.[директива]

То есть перед директивой должна стоять точка. Иначе ассемблер воспринимает это как метку.

Таблица 8. Директивы ассемблера

Директива Описание
BYTE Зарезервировать байт под переменную
CSEG Сегмент кодов
DB Задать постоянным(и) байт(ы) в памяти
DEF Задать символическое имя регистру
DEVICE Задать для какого типа микроконтроллера компилировать
DSEG Сегмент данных
DW Задать постоянное(ые) слово(а) в памяти
EQU Установить символ равный выражению
ESEG Сегмент EEPROM
EXIT Выход из файла
INCLUDE Включить исходный код из другого файла
LIST Включить генерацию.lst - файла
NO.LIST Выключить генерацию.lst - файла
ORG Начальный адрес программы
SET Установить символ, равный выражению

 

Дадим несколько пояснений наиболее важным директивам ассемблера.

Директива CSEG указывает на начало сегмента кодов. Ассемблируемый файл может иметь несколько кодовых сегментов, которые будут объединены в один при ассемблировании. Синтаксис:

.CSEG

Пример:

.DSEG; Начало сегмента данных

vartab:.BYTE 4; Резервируется 4 байта в ОЗУ

.CSEG; Начало сегмента кодов

const:.DW 2; Записать 0x0002 в программной памяти

mov r1,r0; Что-то делать

 

Директива DSEG указывает на начало сегмента данных. Ассемблируемый файл может содержать несколько сегментов данных, которые потом будут собраны в один при ассемблировании. Обычно сегмент данных состоит лишь из директив BYTE и меток. Синтаксис:

.DSEG

Пример:

.DSEG; Начало сегмента данных

varl:.BYTE 1; Резервировать 1 байт под переменную varl

table:.BYTE tab_size; Резервировать tab_size байтов.

.CSEG

ldi r30,low(var1)

ldi r31,high(var1)

ld r1,Z

 

Директива ESEG указывает на начало сегмента EEPROM памяти.

Ассемблируемый файл может содержать несколько EEPROM сегментов, которые будут собраны в один сегмент при ассемблировании. Обычно сегмент EEPROM состоит из DB и DW директив (и меток). Сегмент EEPROM памяти имеет свой собственный счетчик. Директива ORG может использоваться для размещения переменных в нужной области EEPROM. В данной версии комплекса программирование EEPROM может осуществляться только непосредственно из программы учащегося. Прямое программирование в процессе прошивки не предусмотрено.

Синтаксис:

.ESEG

Пример:

.DSEG; Начало сегмента данных

varl:.BYTE 1; Резервировать 1 байт под переменную varl

table:.BYTE tab_size; Зарезервировать tab_size байт

.ESEG

eevarl:.DW 0xffff; Записать 1 слово в EEPROM

 

Директива ORG присваивает значения локальным счетчикам. Используется только совместно с директивами.CSEG,.DSEG,.ESEG.

Синтаксис:

.ORG адрес

Пример:

.DSEG; Начало сегмента данных

.ORG 0x37; Установить адрес ОЗУ на 37h

variable:.BYTE 1; Зарезервировать байт СОЗУ по адресу 37h

.CSEG

.ORG 0x10; Установить счетчик команд на адрес 10h

mov r0,rl; Чего-нибудь делать

 

Директива DB резервирует ресурсы памяти (байты) в программной памяти или в EEPROM. Директиве должна предшествовать метка. DB задает список выражений и должна содержать по крайней мере одно выражение. Размещать директиву следует в сегменте кодов или в EEPROM сегменте.

Список выражений представляет собой последовательность выражений, разделенных запятыми. Каждое выражение должно быть величиной между
–128 и 255.

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

Синтаксис:

LABEL:.DB список выражений

Пример:

.CSEG

сonsts:.DB 0, 255, 0b01010101, -128, 0хаа

.ESEG

const2:.DB 1,2,3

 

Директива DW резервирует ресурсы памяти (слова) в программной памяти или в EEPROM. Директиве должна предшествовать метка. DW задает список выражений и должна содержать по крайней мере одно выражение. Размещать директиву следует в сегменте кодов или в EEPROM сегменте.

Список выражений представляет собой последовательность выражений, разделенных запятыми. Каждое выражение должно быть величиной между
–32768 и 65535.

Синтаксис:

LABEL:.DW список выражений

Пример:

.CSEG

varlist:.DW 0, 0xffff, 0b1001110001010101, -32768, 65535

.ESEG

eevarlst:.DW 0, 0xffff, 10

 

Директива DEF позволяет присвоить символическое имя регистру. Регистр может иметь несколько символических имен.

Синтаксис:

.DEF Имя = Регистр

Пример:

. DEF temp=R16.DEF ior=R0

.CSEG

ldi temp,0xf0; Загрузить 0xf0 в регистр temp

in ior,0x3f; Прочитать SREG в регистр ior

eor temp, ior

 

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

Синтаксис:

.EQU метка = выражение

Пример:

.EQU io_offset = 0x23

.EQU porta = io_offset + 2

.CSEG; Начало сегмента кодов

clr r2; Очистить регистр r2

out porta,r2; Записать в порт А

 

Директива INCLUDE предлагает Ассемблеру начать читать из другого файла. Ассемблер будет ассемблировать этот файл до конца файла или до директивы EXIT. Включаемый файл может сам включать директивы INCLUDE.

Синтаксис:

.INCLUDE "имя файла"

Пример:

; iodefs.asm:

.EQU sreg = 0x3f; Регистр статуса

.EQU sphigh = 0хЗе; Старший байт указателя стека.

.EQU splow = 0x3d;; Младший байт указателя стека.

; incdemo.asm

.INCLUDE iodefs.asm; Включить файл «iodefs.asm»

in r0,sreg; Прочитать регистр статуса

 

EXIT – выйти из файла.

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

Синтаксис:

.EXIT

Пример:

.EXIT; выйти из этого файла

 

DEVICE – указать, для какого микроконтроллера ассемблировать.

Директива позволяет пользователю сообщить ассемблеру, для какого типа устройства пишется программа. Если ассемблер встретит команду, которая не поддерживается указанным типом микроконтроллера, то будет выдано сообщение. Также сообщение появится в случае, если размер программы превысит объем имеющейся в этом устройстве памяти.

Синтаксис:

.DEVICE AT90S1200 |AT90S2313 | AT90S2323 | AT90S2333 | AT90S2343 | AT90S4414 | AT90S4433 | AT90S4434 | AT90S8515 | AT90S8534 | AT90S8535 | ATtinyl1 | ATtinyl2 | ATtiny22 |

ATmega64 | ATmega128| Atmega8535

Пример:

.DEVICE АТmega8535; использовать АТmega8535

.CSEG

.ORG 0000

jmp label1; При ассемблировании появится сообщение, что

; ATmega8535 не поддерживает команду jmp в

; таблице векторов прерываний

 

Структура ассемблерной программы

Программа, написанная на ассемблере, должна иметь определенную структуру.

Предлагается следующий шаблон (для ATmega8535)

;****************************************************

; название программы,

; краткое описание, необходимые пояснения

;****************************************************

; *********** подключаемые дополнительные файлы

.include "m8535def.inc"; файл описания ATmega8535

.include «имя_файла1.расширение»; включение дополнительных

.include «имя_файла2.расширение»; файлов

;****** глобальные константы

equ имя1 = хххх;

equ имя2 = nnnn

;****** глобальные регистровые переменные

def имя1 = регистр

def имя2 = регистр

;******* сегмент данных

.dseg

.org ххх; адрес первого зарезервированного байта

label1:.BYTE 1; резервировать 1 байт под переменную label1

label2:.BYTE m; резервировать m байт под переменную label2

.****** сегмент ЕЕPROM (ЭСППЗУ)

.eseg

.org ххх; адрес первого зарезервированного байта

.db выражение1,выражение2,..;записать список байтов в EEPROM

.dw выражение1,выражение2,...;записать список слов в EEPROM

;****** сегмент кодов

.cseg

.org $000; адрес начала программы в программной памяти

.****** вектора прерываний (если они используются)

rjmp reset; прерывание по сбросу

.org $001

rjmp INT0; обработчик внешнего прерывания 0

.org $002

rjmp INT1; обработчик внешнего прерывания 1

.org adrINTx; адрес следующего обработчика прерываний

rjmp INTx; обработчик прерывания х

......; далее по порядку располагать обработчики остальных

; прерываний

 

;******* начало основной программы

main: <команда> хххх

.........

;******* подпрограммы

;******** подпрограмма 1

subr1: <команда> хххх

.........

ret

;******* подпрограмма 2

subr2: <команда> хххх

.........

ret

 

;******* программы обработчиков прерываний

INT0: <команда> хххх

.........

reti

INT1: <команда> хххх

.........

reti

INTx: <команда> хххх

.........

reti

; конец программы никак не обозначается.

 

В комплексе предусмотрена работа с однофайловыми программами для учебных целей, поэтому подключение внешних файлов, кроме.include "m8535def.inc", смысл имеет только для внешних сред программирования типа AVRStudio.

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

Примеры программ

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

Рис. 8. Алгоритм программы № 1

Задача следующая: вычесть из числа 5 число 3. Если включен тумблер на входе РА2, то на индикацию выдать результат вычитания. Если тумблер отключен – на индикацию вывести цифру ноль.

При работе с портами ввода/вывода следует учитывать, что направление передачи данных через отдельные выводы задается с помощью регистров DDR (DDRA, DDRB, DDRC, DDRD). Если вывод порта сконфигурирован как выход, то его переключение производится через регистр PORT (PORTA, PORTB, PORTC, PORTD), если вывод сконфигурирован как вход, то его опрос следует производить через регистр входных данных PIN (PINA, PINB, PINC, PIND).

Алгоритм программы (рис. 8) соответствует программе № 1, использующей директиву equ ассемблера.

 

;Программа №1.Использование директивы equ

.include "m8535def.inc";включить файл-описание ATmega8535

.dseg;сегмент данных

.equ cod0=$64;присвоение имен ячейкам ОЗУ

.equ cod1=$65

.equ cod2=$66

.equ cod3=$67

.equ cod4=$68

.equ cod5=$69

.equ cod6=$6a

.equ cod7=$6b

.equ cod8=$6c

.equ cod9=$6d

 

.cseg;сегмент кодов

.org 0;адрес начала программной памяти

rjmp reset;вектор сброса

.org $30;начало программы

reset: ldi r16,$00;определение стека с вершиной по адресу $00ff

out sph,r16

ldi r16,$ff

out spl,r16

ldi zl,$64;задание адреса начала зарезервированных ячеек

ldi zh,$00

 

ldi r16,$ff;настроить порт С на выход

out ddrc,r16

ldi r16,00;настроить порт А на вход

out ddra,r16

ldi r16,$ff;настроить порт В на выход

out ddrb,r16

ldi r16,$f0;настроить порт D: биты 0…3 на вход,

out ddrd,r16;остальные на выход

sbi portd,7;выдать 1 на разряд 7 порта D

 

ldi r17,$3f;задание семисегментных кодов

sts cod0,r17

ldi r17,$06

sts cod1,r17

ldi r17,$5b

sts cod2,r17

ldi r17,$4f

sts cod3,r17

ldi r17,$66

sts cod4,r17

ldi r17,$6d

sts cod5,r17

ldi r17,$7d

sts cod6,r17

ldi r17,$07

sts cod7,r17

ldi r17,$7f

sts cod8,r17

ldi r17,$6f

sts cod9,r17

 

ldi r17,5;задание уменьшаемого

ldi r18,3;задание вычитаемого

 

m1: sbis pina,2;если включен тумблер, то пропустить

rjmp m2;следующую команду

mov r20,r17;в r20 поместить уменьшаемое

sub r20,r18;вычесть вычитаемое

rjmp vv

m2: ldi r20,0

 

vv: push zl;сохранить zl в стеке

add zl,r20;сложить zl с результатом

ld r0,z;семисегментный код результата переслать в r0

pop zl;извлечь z1 из стека

out portc,r0;выдать результат на индикацию

rjmp m1

 

Программа № 2 отличается от программы № 1 резервированием по одному байту оперативной памяти под семисегментные коды цифр от 0 до 9.

 

;Программа №2. Использование оперативной памяти под переменные

.include "m8535def.inc";подключение описания ATmega8535

.dseg;сегмент данных

.org $64;адрес первого зарезервированного байта

cod0:.byte 1;резервирование по одному байту

cod1:.byte 1;под переменные

cod2:.byte 1

cod3:.byte 1

cod4:.byte 1

cod5:.byte 1

cod6:.byte 1

cod7:.byte 1

cod8:.byte 1

cod9:.byte 1

 

.cseg;сегмент кодов

.org $0;адрес начала программной памяти

rjmp reset;вектор сброса

reset: ldi r16,$00;определение стека с вершиной по адресу $00ff

out sph,r16

ldi r16,$ff

out spl,r16

ldi zl,$64;задание адреса начала зарезервированных ячеек

ldi zh,$00

 

ldi r16,$ff;настроить порт С на выход

out ddrc,r16

ldi r16,00;настроить порт А на вход

out ddra,r16

ldi r16,$ff;настроить порт В на выход

out ddrb,r16

ldi r16,$f0;настроить порт D: биты 0…3 на вход,

out ddrd,r16;остальные на выход

sbi portd,7;выдать 1 на разряд 7 порта D

 

ldi r17,$3f;задание семисегментных кодов

sts cod0,r17

ldi r17,$06

sts codl,r17

ldi r17,$5b

sts cod2,rl7

ldi r07,$4f

sts cod3,rl7

ldi r17,$66

sts cod4,rl7

ldi r17,S6d

sts cod5,r17

ldi r17,$7d

sts cod6,r17

ldi r17,$07

sts cod7,r17

ldi r17,$7f

sts cod8,r17

ldi rl7,$6f

sts cod9,r17

 

ldi r17,5;задание уменьшаемого

ldi r18,3;задание вычитаемого

 

m1: sbis pina,2;если включен тумблер, то пропустить

rjmp m2;следующую команду

mov r20,r17;в г20 поместить уменьшаемое

sub r20,r18;вычесть вычитаемое

rjmp vv

m2: ldi r20,0

 

vv: push zl;сохранить zl в стеке

add z1,r20;сложить zl с результатом

ld r0,z;семисегментный код результата переслать в г0

pop z1;извлечь zl из стека

out portc,r0;выдать результат на индикацию

rjmp m1

 

Программа № 3 самая короткая. Она использует директиву.dw для определения слов в программной памяти. В программе используется команда LPM ассемблера. По этой команде загружается байт, адресуемый регистром Z в регистр R0. Команда обеспечивает доступ к любому байту памяти программы, организованной как 16-битное слово. Младший бит регистра Z определяет, осуществляется ли доступ к младшему байту слова (0) или к старшему (1).

 

;Программа №3. Использование директивы.dw

.include "m8535def.inc";подключение описания ATmega8535

.cseg

.org 0

rjmp reset

.dw $063f,$4f5b,$6d66,$077d,$617f,$7c77,$5e39,$7179

;семисегментные коды цифр от 0 до F

reset: ldi r16,$00;определение стека с вершиной по адресу $00ff

out sph,r16

ldi r16,$ff

out spl,r16

 

ldi rl6,$ff;настроить порт С на выход

out ddrc,r16

ldi r16,00;настроить порт А на вход

out ddra,r16

ldi r16,$ff;настроить порт В на выход

out ddrb,r16

ldi r16,$f0;настроить порт D: биты 0...3 на вход,

out ddrd,r16;остальные на выход

sbi portd,7;выдать 1 на разряд 7 порта D

 

ldi zl,02;установить адрес семисегментного кода нуля

ldi zh,00;в регистр Z

ldi r17,5;задание уменьшаемого

ldi r18,3;задание вычитаемого

 

m1: sbis pina,2;если включен тумблер, то пропустить

rjmp m2;следующую команду

mov r20,r17;в r20 поместить уменьшаемое

sub r20,r18;вычесть вычитаемое

rjmp vv

m2: ldi r20,0;присвоить результату значение 0

 

vv: push z1;сохранить z1 в стеке

add z1,r20;сложить z1 с результатом

lpm;загрузить байт, адресуемый регистром Z, в регистр 0

pop z1;извлечь z1 из стека

out portc,r0;выдать результат на индикацию

rjmp m1



Поделиться:




Поиск по сайту

©2015-2024 poisk-ru.ru
Все права принадлежать их авторам. Данный сайт не претендует на авторства, а предоставляет бесплатное использование.
Дата создания страницы: 2016-02-16 Нарушение авторских прав и Нарушение персональных данных


Поиск по сайту: