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