微波EDA网,见证研发工程师的成长!
首页 > 硬件设计 > 嵌入式设计 > AVR内部EEPROM读写范例

AVR内部EEPROM读写范例

时间:12-12 来源:互联网 点击:

/***********************************************

**** AVR 内部EEPROM读写范例 ***

**** 编译器:WINAVR20050214 ***

***********************************************/

/*

本程序简单的示范了如何使用ATMEGA16的EERPOM

EEPROM的简介

EEPROM的写操作

EEPROM的读操作

出于简化程序考虑,各种数据没有对外输出,学习时建议使用JTAG ICE硬件仿真器。

在打开调试文件到JTAG后,打开Debug -> JTAG ICE Options菜单,然后在JTAG ICE Properties中点击Dbug页面,将preserve eeprom选项选中。 在每次仿真调试时候,就保护EEPROM内容了。否则,会按照默认设置擦除EEPROM的内容。

由于定义了EEPROM变量,JTAG调试时会询问是否初始化EEPROM,请选择[否],EEPROM的数据也可以在view->memory,选Eeprom窗口下察看

*/

#i nclude

#i nclude

////时钟定为内部1MHz,F_CPU=1000000 时钟频率对程序的运行没什么影响

/*

GCCAVR(avr-libc)里面自带了EEPROM的读写函数。

下面列举部分常用函数(原型)

#define eeprom_is_ready() bit_is_clear(EECR, EEWE)

检测EEPROM是否准备好。OK返回1(返回EEWE位)

#define eeprom_busy_wait() do {} while (!eeprom_is_ready())

等待EEPROM操作完成

extern uint8_t eeprom_read_byte (const uint8_t *addr);

读取指定地址的一个字节8bit的EEPROM数据

extern uint16_t eeprom_read_word (const uint16_t *addr);

读取指定地址的一个字16bit的EEPROM数据

extern void eeprom_read_block (void *buf, const void *addr, size_t n);

读取由指定地址开始的指定长度的EEPROM数据

extern void eeprom_write_byte (uint8_t *addr, uint8_t val);

向指定地址写入一个字节8bit的EEPROM数据

extern void eeprom_write_word (uint16_t *addr, uint16_t val);

向指定地址写入一个字16bit的EEPROM数据

extern void eeprom_write_block (const void *buf, void *addr, size_t n);

由指定地址开始写入指定长度的EEPROM数据,但不支持部分AVR,原文如下:

ote This library will e not work with the following devices since these

devices have the EEPROM IO ports at different locations:

- AT90CAN128

- ATmega48

- ATmega88

- ATmega165

- ATmega168

- ATmega169

- ATmega325

- ATmega3250

- ATmega645

- ATmega6450

在程序中对EEPROM 操作有两种方式:

方式一:直接指定EERPOM 地址

即读写函数的地址有自己指定,用于需要特定数据排列格式的应用中

方式二:先定义EEPROM 区变量法

在这种方式下变量在EEPROM 存储器内的具体地址由编译器自动分配。

相对方式一,数据在EEPROM 中的具体位置是不透明的。

为EEPROM 变量赋的初始值,编译时被分配到.eeprom 段中,可用avr-objcopy 工具从.elf文件中提取并产生ihex 或binary 等格式的文件, 从而可以使用编程器或下载线将其写入到器件的EEPROM 中。实际上WINAVR 中MFILE 生成的MAKEFILE 已经为我们做了这一切。它会自动生成以 “.eep” 为后缀的文件,通常它是iHex 格式(这次测试发现 分配地址是从0x0000开始的,故增加了一个EEPROM变量Evalvoid[16]),如果同时使用方式1和2,请注意防止地址重叠,自己指定的地址应该选在后面。

*/

//全局变量

unsigned char EDATA;

unsigned char ORGDATA[16]={0x00,0x02,0x04,0x06,0x08,0x0A,0x0C,0x0E,

0x01,0x03,0x05,0x07,0x09,0x0B,0x0D,0x0F}; //原始数据

unsigned char CMPDATA[16]; //比较数据

//仿真时在watch窗口,监控这些全局变量。

//EEPROM 变量定义

unsigned char Evalvoid[16] __attribute__((section(".eeprom"))); //这个没用到

unsigned char Eval[16] __attribute__((section(".eeprom")));

int main(void)

{

eeprom_write_byte (0x40,0xA5); //向EEPROM的0x40地址写入数据 0xA5

EDATA=eeprom_read_byte (0x40); //读出,然后看看数据对不对?

//上面两句编译是有如下警告,但不必理会

//EEPROM_main.c:103: warning: passing arg 1 of `eeprom_write_byte' makes pointer from integer without a cast

//EEPROM_main.c:104: warning: passing arg 1 of `eeprom_read_byte' makes pointer from integer without a cast

eeprom_write_block (ORGDATA[0], Eval[0], 16); //块写入

//看看EEPROM数据是否是能失电永久保存,可以注释上面这句程序(不写入,只是读出),然后编译,烧写,断电(一段时间),上电,调试。

eeprom_read_block (CMPDATA[0],Eval[0], 16); //块读出,然后看看数据对不对?

while

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top