《林夕依然AVR知识学习记录文本移植记录文本1.doc》由会员分享,可在线阅读,更多相关《林夕依然AVR知识学习记录文本移植记录文本1.doc(21页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、,林夕依然ATmega16学习笔记例程移植到Atmel Studio 6.2笔记相信你已经有了一定的C语言及数字电路基础,以及能定下心来,决心学点东西。如果做不到的话,麻烦您把我轻轻的放开,该干啥干啥去。前言最近学习AVR,网上找了N多资料后,发现就算林夕依然ATmega16学习笔记最合俺的心意。优点如下:1、 不讲原理,直接操作,适合俺这种不愿背书的差生。2、 每个例程都附有源程序及PROTEUS仿真文件,省了买零件的金钱及焊板的时间。缺点当然也有:1、 太省事了,初学者容易只跑跑例程,不求甚解(这也是市售所有学习板的通病)。2、 基于ICCAVR,而不是最新的AtmelStudio。3、
2、理论知识太少(也算是优点)。还有一条就是这玩意是用EXE文件打包的,初次下载时,我不敢打开,生怕会给我强装软件或者开个后门什么的。后来在网上下载了一个开发板的附带光盘文件,发现里面也带着这玩意。心想这里面不该会有问题呀,才不是太放心的打开,打开后发现相见恨晚呀!好吧,先把这些例程一个一个琢磨完吧。不过看程序时发现想打瞌睡,也难怪,看这玩意也太没挑战性了。既然我的电脑中装了AtmelStudio,即然ICCAVR迟早要过时,不如把这里面的例程移植到AtmelStudio中吧!顺便也可以学习一下编程。笔记不妨公开,俺也好刷一点成在感。本文中所有例程均在AtmelStudio6.2+Proteus7
3、.8中调试通过。本文是我个人学习时的笔记,希望后来者能少走一点弯路。学习MCU的惟一捷径是多读例程,多写程序,别无他法。林夕依然ATmega16学习笔记下载链结如下:链接: 密码:mzah实验1:8种LED点亮模式1、 移植到AtmelStudio中的步骤打开Atmel Studio 6.2,新建一个项目。不知道啥意思,我选第二个后,在下面起好项目名称及选好项目目录后OK。这个好办,按CPU型号选就好了。终于进入编辑窗口了。一言蔽之,提取包中所有.c(C源码)、.h(C头文件)、.dsn(PROTEUS仿真文件)文件。AtmelStudio自己生成的那个.c文件要删掉,否则你懂的。移植的时候,
4、要把头文件换一下,否则用不了所有的iom16v.h替换为avr/io.h所有#include 全部注释掉。有这一句的文件中,全部增加#define BIT(x)(1 (x)当然了,移植的时候,我们懒得逐一去改BIT,才这么将就一下。那么AtmelStudio与BIT()等价的命令是什么呢?打开sfr_defs.h文件,里面有一行:没错,AtmelStudio中与BIT等价的命令是_BV。再把随项目自动生成的那个C文件(含main函数的那个)删掉。main函数要改成int型。嗯,C99是这样规定的,新出的标准C编译器都得这么干,否则会有警告。某个不仅仅是五百强的公司的编程规范中可是明文规定编译时
5、不准有任何警告。按F7编译在项目目录中找到生成的hex文件,一般在DEBUG子目录里,用Proteus仿真。2、Proteus仿真步骤假设已经装好Proteus,我装的是7.8版本。双击打开包中的DSN文件。如上图所示,双击CPU,弹出下框。在Program File中找到刚才编译出来的hex文件。然后点击OK。点击程序左下角的三角形图标开始仿真我们现在可以愉悦的看着LED图标在欢快的上下滚动。3、知识点作为开山第一个实验,知识点会多一些,我会很认真的写,希望你能看懂。AtmelStudio不支持直接对单个位进行操作,若要对单个位进行操作,则要通过与或非的办法。小知识:ICCAVR位操作的宏定
6、义C语言位操作的功能是其比较出色的地方,现在许多微处理器都支持C编译器也可看出。汇编语言的位操作也很直观,但在某些C编译器对位操作却不能象汇编中对单个的位进行直接处理,而要转化为对一个字节的操作,在ICCAVR中这种限制充分体现出来。如要对PORTA的PORTA0置1,则应写成PORTA|=0X01;这样的位处在字节的两端还比较直观,但若是要置PORTA的第3位,第6位置1,则写出的程序可读性不强了。于是便有了一些宏操作定义#define BIT(x) (1 (x)(需要包含头文件macros.h)才可直接使用。下面给出两种置1和清0的宏定义:置1:1,#define set(x) (1(x)
7、2,#define setb(temp,x) temp|=(1(x)清0:1,#define clr(x) (1(x)2,#define clrb(temp,x) temp&=(1(x)比较上述置1的两种宏定义,第1种在写1时需写成 PORTA|=set(x); x表示要处理的位,第2种更象函数,使用起来也比较直观:setb(PORTA,x), 可读性比较强 但是第2种却并非是万能的,以MEGA8515为例,USART的UBRRH和UCSRC共用一个I/O location,在写UBRRH的时候最高位必须为0,写UCSRC时则正好相反。上述第2种的置1和清0操作只能对1 个BIT进行操作,所以
8、是不能处理UBRRH和UCSRC的写入工作的,而第1种置1和清0操作却可以通过或/与进行多个位操作的,比如写入UCSRC 0x01可以写作:UCSRC|=set(7)|set(0); 而如果用第2种宏定义则让它太为难了。两种方法有其利弊,使用的时候靠自己衡量了。#define checkbit(var,bit) (var&(0x01(bit) /*定义查询位函数*/#define setbit(var,bit) (var|=(0x01(bit) /*定义置位函数*/#define clrbit(var,bit) (var&=(0x01(bit) /*定义清零位函数*/#define TGLBI
9、T(reg,bit) reg=bit /*定义取反函数*/实验2:有源蜂鸣器驱动实验实验二很简单,移植也很简单,只要把所有#include 全部注释掉。所有的iom16v.h替换为avr/io.h,因为没有使用BIT(),所以不用加相应的宏定义。main函数改为int型。程序就是这么个样子。void bell(void) PORTD&=(1PD5); DelayMs(50); PORTD|=(1PD5); DelayMs(50);int main(void) DDRD=0XFF; PORTD=0XFF; while(1) bell(); 其实bell()可以改得更简单一些,因为编译器不支持端口
10、位操作,但是PORTD只接了个蜂鸣器,所以大可以把整个PORTD取反。PORTD=PORTD;虽然说编译器不支持端口位操作,不过我们可以牺牲点效率,用与或非的办法实现端口位操作。PORTD = _BV(PD5);可能会有人问PD5是什么意思,怎么定义的?在PD5上右击,点击Goto Implementation,会自动跳到PD5的定义。恩,PD5=5;下面是我改动过后的bell(),每一种方法都是可行的。void bell(void)/方法一、原例,不停的输出高低电平PORTD&=(1PD5);/PD5等价于数值5,15表示把1左移五位。即000000015 = 00100000DelayMs
11、(50);PORTD|=(1PD5);DelayMs(50);/方法二、对PORTD不停的取反PORTD=PORTD;/对整个端口取反DelayMs(50);/方法三、循环对单个端口置高低电平/编译器不支持对端口进行位操作,需要使用标准C语言中位操作的技巧,可以参看谭版C程序设计,位操作一章。PORTD |= _BV(PD5);/单独把PD5置高电平。DelayMs(50);PORTD &= _BV(PD5);/单独把PD5置低电平DelayMs(50);/方法四、对单个位取反PORTD = _BV(PD5);/对单个端口进行取反操作DelayMs(50);实验3:按键扫描(用KEY选择对应L
12、ED点亮)继续按实验一的方法处理:所有#include 全部注释掉,有这个头文件的地方增加宏定义#define BIT(x)(1 (x)。所有的iom16v.h替换为avr/io.h。main函数改为int型。这个程序很简单,先初始化端口,然后循环运行key_scan函数。key_scan函数扫描所有有开关的端口,如果其中有一个为低电平的话,则将对应LED点亮,之后检查PINA有没有电平变化,如果有的话,跳出while,然后重复上述流程。因为开关无论抖动几次,都不会影响结果,所以不需要防抖动。实验4:按键扫描+8种LED亮灭模式控制移植方法与实验3一样一样的。程序是实验3与实验1的结合版本,很
13、简单。实验5:按键扫描(用KEY选择LED点亮模式)实验6:无源蜂鸣器驱动实验(ICC)依电脑配置及设置不同,仿真时可能会有下列警告,表示仿真时间与实际时间不符。若把两个电阻属性改为数字的,警告会消失,不过声音会异常。实验7:外部中断INT0实验这个实验就麻烦多了。先按照之前的套路:所有#include 全部注释掉,有这个头文件的地方增加宏定义#define BIT(x)(1 (x)。所有的iom16v.h替换为avr/io.h。main函数改为int型。因为main()调用了SEI(),所以要增加宏定义:#define SEI()asm(sei)AtmelStudio的中断函数与ICCAVR
14、不一样,需要改,如下图:前面还要增加一个头文件:#include ISR表示是中断服务函数,所有中断服务函数必须叫这个名字,INT0_vect表示INT0入口,中断入口可以在iom16.h文件中找到(此为举例,实际CPU不同,文件名也会不同)。正常情况下,这样就大功告成了,实际仿真结果却与预期值大相径庭。预期值是程序运行后D0慢闪,按下INT0键后,这一组LED花样闪烁。实际仿真结果却是按下键后似乎毫无反应,从逻辑分析仪上看,好像又有一点变化。我花费了好几个小时找资料排查,毫无所获,分析代码也没有发现任何问题,修改程序定位的结果DelayMs()好像没有执行。但是在main()中调用该函数却毫
15、无问题。DelayMs()仅仅只是一个延时N毫秒的程序,其实我们完全可以用AtmelStudio自带的库函数替代。在main.c与functions.c文件头上增加如下定义及头文件#define F_CPU 3686400UL#define _DELAY_BACKWARD_COMPATIBLE_#include F_CPU表示CPU时钟频率,最后的UL代表无符长整型。后面一句可以参看下面的文章:现在我们可以愉快的在这两个函数中调用_delay_ms()函数了,但是改后,结果还没有改善。好吧,现在把程序复原,然后回过头把这个程序再读一遍,发现这中断服务函数也太长了吧,长就算了,哪有中断服务程序加
16、这么延时的?再者说,这函数内的变量也定义太多了吧?一个i不就得了。但是就算这样,也不该会出这种问题吧!接着上网查资料,死了无数脑细胞后,发现原来是AtmelStudio编译时,认为延时函数没有做任何操作,将其忽略掉了。解决起来也很简单,在延时函数的变量前加入关键字volatile就好了。一次通过好吧,今天到此为止,让我一个人静静。实验8:7种LED亮灭模式+INT0中断+蜂鸣器本节需要改动的地方与上一节一样,调试一次通过。本节的知识点在于PWM功能、内部控制寄存器的调用。实验9:蜂鸣器产生音节节拍这一章按照第7节的方法,很顺利,也没有增加新的知识点。唯一的小插曲是main()多定义的了一个变量j,注释掉就没警告了。实验10:按键扫描(KEY控制对应LED亮灭)这一节,简单的令人发指了。