API
嵌入式编程的代码可以简单地分为两部分,一是与硬件无关的算法部分,对其编程与普通C编程没有区别;二是与硬件相关的寄存器/端口操作部
分。不同的MCU实现方法各有不同。在AVR-GCC里则通过一系列的API来解决。当然,用户也可以定义自己的API。在此简单地介绍目前AVR-
GCC里定义的API,以及AVR-GCC的工作过程。
一.存储器API
AVR具有三种存储器:FLASH,SRAM和EEPROM。
AVR-GCC将程序代码放在FLASH,数据放在SRAM。
I.程序存储器
如果要将数据(如常量,字符串,等等)放在FLASH里,用
户需要指明数据类型__attribute__((progmem))。为了方便使用,AVR-GCC定义了一些更直观的符号,如下表所示。
类
型 定义
prog_void void __attribute__((progmem))
prog_char char
__attribute__((progmem))
prog_int int
__attribute__((progmem))
prog_long long
__attribute__((progmem))
prog_long_long long long
__attribute__((progmem))
PGM_P prog_char const*
PGM_VOID_P prog_void
const*
提供的库函数有:
1.__elpm_inline
用法:uint8_t
__elpm_inline(uint32_t addr);
说明:执行ELPM指令从FLASH里取数。参数为32位地址,返回一个8位数据。
2.__lpm_inline
用
法:uint8_t __elpm_inline(uint16_t addr);
说明:执行LPM指令从FLASH里取数。参数为16位地址,
返回一个8位数据。
3.memcpy_P
用法:void* memcpy_P(void* dst, PGM_VOID_P src,
size_t n);
说明:memcpy的特殊版本。完成从FLASH取n个字节的任务。
4.PRG_RDB
用
法:uint8_t PGR_RDB(uint16_t addr);
说明:此函数简单地调用__lpm_inline
5.PSTR
用
法:PSTR(s);
说明:参数为字符串。功能是将其放在FLASH里并返回地址。
6.strcmp_P
用法:int
strcmp(char const*, PGM_P);
说明:功能与strcmp()类似。第二个参数指向程序存储器内的字符串。
7.strcpy_P
用
法:char* strcpy_P(char*, PGM_P);
说明:功能与strcpy()类似。第二个参数指向程序存储器内的字符串。
8.strlen_P
用
法:size_t strlen_P(PGM_P);
说明:功能与strlen()类似。第二个参数指向程序存储器内的字符串。
9.strncmp_P
用
法:size_t strncmp_P(char const*, PGM_P, size_t);
说明:功能与strncmp()类似。第二个
参数指向程序存储器内的字符串。
10.strncpy_P
用法:size_t strncpy_P(char*, PGM_P,
size_t);
说明:功能与strncpy()类似。第二个参数指向程序存储器内的字符串。
II.EEPROM
AVR内部有
EEPROM,但地址空间与SRAM的不相同。在访问时必须通过I/O寄存器来进行。EEPROM
API封装了这些功能,为用户提供了高级接口。使用时要包含eeprom.h。在程序里定义EEPROM数据的例子如下:
static
uint8_t variable_x __attribute__((section(".eeprom"))) = 0;
不同的AVR器件具
有不同数目的EEPROM。链接器将针对不同的器件分配存储器空间。
1. eeprom_is_ready
用法:int
eeprom_is_ready(void);
说明:此函数用于指示是否可以访问EEPROM。如果EEPROM正在执行写操作,则在4ms内无
法访问。此函数查询相应的状态位来指示现在是否可以访问EEPROM。
2. eeprom_rb
用法:uint8_t
eeprom_rb(uint16_t addr);
说明:从EEPROM里读出一个字节的内容。参数addr用于指示要读出的地址。
_EEGET(addr)调用此函数。
3. eeprom_read_block
用法:void
eeprom_read_block(void* buf, uint16_t addr, size_t n);
说明:读出一块EEROM的内
容。参数addr为起始地址,n表明要读取的字节数。数据被读到SRAM的buf里。
4. eeprom_rw
用
法:unint16_t eeprom_rw(uint16_t addr);
说明:从EEPROM里读出一个16位的数据。低字节为低8位,高
字节为高8位。参数addr为地址。
5. eeprom_wb
用法:void eeprom_wb(uint16_t
addr, uint8_t val);
说明:将8位数据val写入地址为addr的EEPROM存储器里。_EEPUT(addr,val)调
用此函数。
二.中断API
由于C语言设计目标为硬件无关,因此各种编译器在处理中断时使用的方法都是编译器设计者自己的方法。
在
AVR-GCC环境里,向量表已经预先定义,并指向具有预定义名称的中断例程。通过使用合适的名称,用户例程就可以由相应的中断所调用。如果用户没有定义
自己的中断例程,则器件库的缺省例程被加入。
除了中断向量表的问题,编译器还必须处理相关寄存器保护的问题。中断API解决了细节问题。用户只要
将中断例程定义为INTERRUPT()或SIGAL()即可。而对于用户没有定义的中断,缺省例程的处理是reti指令。
函数定义可参见
interrupt.h,中断信号符号表参见sig-avr.h。
1. cli
用法:void cli(void);
说
明:通过置位全局中断屏蔽位来禁止中断。其编译结果仅为一条汇编指令。
2. enable_external_int
用
法:void enable_external_int(uint8_t ints);
说明:此函数访问GIMSK寄存器(对于MEGA器件则是
EIMSK寄存器)。功能与宏outp()一样。
3. INTERRUPT
用法:INTERRUPT(signame)
说
明:定义中断源signame对应的中断例程。在执行时,全局屏蔽位将清零,其他中断被使能。ADC结束中断例程的例子如下所示:
INTERRUPT(SIG_ADC)
{
}
4. sei
用
法:void sei(void);
说明:通过清零全局中断屏蔽位来使能中断。其编译结果仅为一条汇编指令。
5. SIGNAL
用
法:SIGNAL(signame)
说明:定义中断源signame对应的中断例程。在执行时,全局屏蔽位保持置位,其他中断被禁止。ADC结束
中断例程的例子如下所示:
SIGNAL(SIG_ADC)
{
}
6. timer_enable_int
用
法:void timer_enable_int(uint8_t ints);
说明:此函数操作TIMSK寄存器。也可以通过outp()来设
置。
四.I/O API
I.I/O端口API
1. BV
用法:BV(pos);
说明:将位定义转换成屏
蔽码(MASK)。与头文件io.h里的位定义一起使用。例如,置位WDTOE和WDE可表示为"BV(WDTOE) | BV(WDE)"
2. bit_is_clear
用
法:uint8_t bit_is_clear(uint8_t port, uint8_t bit);
描述:如果port的bit位清零则返
回1。此函数调用sbic指令,故port应为有效地址。
3.bit_is_set
用法:uint8_t
bit_is_set(uint8_t port, uint8_t bit);
描述:如果port的bit位置位则返回1。此函数调用sbis
指令,故port应为有效地址。
4.cbi
用法:void cbi(uint8_t port, uint8_t bit);
说
明:清零port的bit位。bit的值为0~7。如果port为实际I/O寄存器,则此函数生成一条cbi指令;否则,函数生成相应的优化代码。
5.inp
用
法:uint8_t inp(uint8_t port);
说明:从端口port读入8比特的数值。如果port为常数,则函数生成一条in指
令;若为变量,则函数用直接寻址指令。
6.__inw
用法:uint16_t __inw(uint8_t port);
说明:
从I/O寄存器读入16位的数值。此函数用于读取16位寄存器(ADC,ICR1,OCR1,TCNT1)的值,因为读取这些寄存器需要合适的步骤。由于
此函数只产生两条汇编指令,因此要在中断禁止时使用,否则有可能由于中断插入到指令之间造成读取错误。
7.__inw_atomic
用
法:uint16_t __inw_atomic(uint8_t port);
说明:以原子语句方式读取16位I/O寄存器的数值。此函数首先
禁止中断,读取数据之后再恢复中断状态,因此可以安全地应用在各种系统状态。
8.loop_until_bit_is_clear
用
法:oidoid loop_until_bit_is_clear (uint8_t port, uint8_t bit);
说明:此函数简
单地调用sbic指令来测试端口port的bit位是否清零。port必须为有效端口。
9.loop_until_bit_is_set
用
法:oidoid loop_until_bit_is_set (uint8_t port, uint8_t bit);
说明:此函数简单地
调用sbis指令来测试端口port的bit位是否置位。port必须为有效端口。
10.outp
用法:void
outp(uint8_t val, uint8_t port);
说明:将val写入端口port。如果port为常数,则函数生成一条out
指令;若为变量,则函数用直接寻址指令。
11.__outw
用法:void __outw(uint16_t val, uint8_t
port);
说明:将16位的val写入端口port。此函数适合于操作16位寄存器,如ADC,ICR1,OCR1,TCNT1。由于此函数
只产生两条汇编指令,因此要在中断禁止时使用,否则有可能由于中断插入到指令之间造成读取错误。
12.__outw_atomic
用
法:void __outw_atomic(uint16_t val, uint8_t port);
说明:将16位的val写入端口
port。此函数适合于操作16位寄存器,如ADC,ICR1,OCR1,TCNT1。此函数首先禁止中断,读取数据之后再恢复中断状态,因此可以安全地
应用在各种系统状态。
13.sbi
用法:void sbi(uint8_t port, uint8_t bit);
说明:置位
port的bit位。bit的值为0~7。如果port为实际I/O寄存器,则此函数生成一条 sbi指令;否则,函数生成相应的优化代码。
五.
看门狗API
以下函数操作看门狗。宏定义参见wdt.h。
用户可以通过起动代码初始化看门狗。WDTCR的缺省值为0。如果你希望将其设
置为其他值,则需要在链接命令里加入相应的命令。使用的符号为__init_wdtcr__。如下为将WDTCR设置为0x1f的例子:
avr-ld
-defsym __init_wdtcr__=0x1f
1. wdt_disable
用法:void
wdt_disable(void);
说明:关闭看门狗。
2. wdt_enable
用法:void
wdt_disable(unit8_t timeout);
说明:使能看门狗。看门狗溢出时间为timeout。
timeout 周期
0 16K
CLK
1 32K CLK
2 64K CLK
3 128K CLK
4 256K
CLK
5 512K CLK
6 1024K CLK
7 2048K CLK
3. wdt_reset
用
法:void wdt_reset(void);
说明:产生喂狗指令wdr。
三.应用程序启动过程(Start Up)
标
准库文件包含一个启动模块(Start Up Module),用于为真正执行用户程序做环境设置。
启动模块完成的任务如下:
1. 提
供缺省向量表
2. 提供缺省中断程序入口
3. 初始化全局变量
4. 初始化看门狗
5. 初
始化寄存器MCUCR
6. 初始化数据段
7. 将数据段.bss的内容清零
8. 跳转到
main()。(不用调用方式,因为main()不用返回)
启动模块包含缺省中断向量表,其内容为预先定义好的函数名称。这些函数名称可以由程序
员重载。中断向量表的第一个内容为复位向量,执行结果是将程序跳转到_init_。在启动模块里,_init_表示的地址与_real_init_指向的
地址相同。如果要加入客户代码,则需要在程序里定义一个_init_函数。在此函数的末尾跳转到_real_init_。具体实现如下:
void
_real_init_(void);
void _init_(void) __attribute__((naked));
void
_init_(void)
{
// 用户代码
//
最后的代码必须为:
asm ("rjmp _real_init_");
}
在_real_init_部
分,系统将设置看门狗和MCUCR寄存器。启动模块并没有真正取用相应寄存器的设置数值(以符号
_init_wdctr_,_init_mcucr_,_init_emcucr_表示),而是通过地址来取得其值。因而用户可以通过链接器的
--defsym选项来设置这些符号的地址。如果用户没有定义,则启动模块将使用缺省值。
接下来系统将从程序存储器里把具有初值的全局变量加载到
数据存储器SRAM。然后是将数据段.bbs清零。此数据段包含所有没有的初值的非AUTO变量。
最后,系统跳转到main()函数,用户代码开
始执行。系统对此特殊函数加入一些特殊的处理。进入此函数后,堆栈指向SRAM的末尾。
阅读(1697) | 评论(0) | 转发(0) |