51单片机外扩RAM,ROM操作。

      这是51单片机内置的功能,可以让我们少写很多时序,让代码变的整洁好看。

      这里先拓展点小知识:RAM,ROM的区别,地址数据总线,控制总线。

                1.RAM是一种掉电会丢失数据的可读可写的存储器。//其实题目上写的是外扩ram,但很多时候我们很少来外扩ram,          都是外扩各种外围器件,比如ADC,DAC,等等等

                2.ROM是掉电不丢失,只读的存储器。

                3.51单片机P0口是低8地址和数据总线复用。P2口是高8位地址总线。

              4.数据总线是用来传输数据的,地址总线是用来选择地址的,51单片机地址总线是16根,也就是外扩最大内存为    2^16=64K;

                5.控制总线就是单片机发出的各种控制信号线,比如读写时序产生WR/RD,低8位地址锁存信号线ALE等...

        然后我们进入正题。如果是学过单片机原理的小伙伴们应该知道这两个操作指令 MOVX,MOVC,如果用我们用汇编写外扩RAM、ROM,就是用这两个操作指令,当出现这两个操作指令的时候,单片机的相关引脚就会产生对应的时序,这样我们把相关外扩器件与之相连的时候,就不用去管时序的问题了。

                MOVX  A,@DPTR    //读外扩ram,产生读时序,由WR/p3.6产生

                MOVX  @DPTR,A    //写外扩ram,产生写时序,由RD/p3.7产生

                MOVC  A,@DPTR    //读外扩rom,产生读时序,由PSEN产生

                我这里这介绍RAM的读写。当我们发出一条MOVX指令的时候,先发出16位地址,产生控制时序,然后在写(读)数据,这个时候,我们低8位地址总线就与数据总线冲突了(因为都是用的P0口),使用如果我们要时候低8位地址去限定的时候,就可以就一个锁存器,来锁存低8位地址(这个时候要用到锁存信号引脚ALE),再比如我们外围器件不多的时候,或者外扩ram不超过2^8=256B的时候,完全可以不管低8位。

                然后就是我们读写控制线都有效时序都是由高到低,当数据读写完后,在变为高电平,你们要注意自己控制的外围器件的时序,看要不要加非门处理一下读写时序,比如控制LCD1602的时候。

                我们大致了解了这个控制过程(想详细了解的建议去看书,单片机原理),我们在开发的过程中很少用汇编写项目,下面我们就来看看C语言的写法。

                首先我们来看一看下面的关键字:                 

                        code   以MOVC @A+DPTR 读取的程序内存

                        data   可以直接存取的内部数据存储器

                        idata   以 MOV@Rn 存取的内部数据存储器

                        bdata   可以位寻址(BitAddressable)的内部存储器

                        xdata   以MOVX @DPTR 存取的外部数据存储器

                        pdata   以MOVX @Rn 存取的外部数据存储器 

                我们一般都只用到 xdata,我们在头文件<absacc.h>中可以看到这样一个定义

                 #define XBYTE ((unsigned char voalite xdata*)0)

                这样一个宏定义,这个宏定义我按我自己的理解去演示一遍,

                当我们用XBYTE[0X1000]=1;

                想当于  MOV @DPTR,1000H

                            MOV  A,#1

                            MOVX @DPTR,A

                把1写到外部ram地址位0x1000中,如果是 i=XBYTE[0X1000];这就是读了,意思是差不多的。

                 这个地址0x1000,就是通过地址总线发出去的,P0发送0x00也就是低8位,P2发送0x10也就是高8位,发送地址后,控制线发出对应的控制时序,ALE发送锁存,WR发出一个下降沿,这个时候在通过数据总线P0发送1出去。WR在复位到高电平。这就是大概流程。

          当我们不是外扩ram的时候 XBYTE[]中的地址应该怎么去填呢,我们来举的DAC0832的例子大笑,借用的别人的图


  

        我们0832是在片选CS为低的时候,给WR低电平就可以工作了。

         上图我们可以明显体会到用外扩ram这种编程方法的便利,以及节省很多资源了(用i/o直接操作的话,不加额外的器件,连接两个DAC起码要16个引脚)。

          当我们想让dac0832(1)工作,但片2不工作,就要让P2.7=0,P2.6=0,P2.5=1。是不是这样的。

          所以我们片1的地址高3位就出来了,其余的地址都影响不到它,所以我们称为地址无关位,我们任意填什么都可以,为了方便,我就填0。那么片1的地址  是不是就是XBYTE[0X2000] (001 0 0000 0000 0000),是不是就是这样的,

           同理片2的就为 XBYTE[0X8000] (100 0 0000 0000 0000) 。

            当我们写 XBYTE[0x2000]=0xff;这句话,是不是就启动了dac0832,并写入了0xff。

            其实大家可以去看书上单片机外扩程序存储器的工作时序图,多想想这个过程,就可以很轻松的去运用他了,可以让自己轻松很多,并且玩法有很多很多。

猜你喜欢

转载自blog.csdn.net/A_wangc/article/details/79559507
今日推荐