单片机要如何寻址?
当前,单片机种类很多,且实际工作中仅应用一种单片机也是不现实的,必得对常用的几种单片机有所了解。
尽管现在单片机编程大多使用C语言,但必得对单片机的内核结构、存储结构及指令集有一定的了解,才有可能写出优秀的程序代码。对于单片机指令的学习,寻址方式的学习是其中的一个重点和难点,寻址方式的正确理解不仅对汇编编程至关重要,而且有助于对于单片机内核结构(如RISC和CISC的区别)、存储结构的更深刻理解。
但是,不同单片机都提供了一些不同的寻址方式,且即使同样的寻址方式在不同的单片机中也有不同的名称,使得寻址方式显得混乱,不一致,不易理解。
不过,经过仔细的对比、学习和分析,我发现,其实所有的寻址方式,都可以归为以下六类:
1)立即寻址
2)无址寻址
3)寄存器直接寻址
4)寄存器间接寻址
5)内存直接寻址
6)内存间接寻址
下面对以上六类指令一一分解:
1、立即寻址,即在指令中直接给出实际的操作数数值,如MOV R1,#0
2、无址寻址,就是说指令中根本没有给出操作数或操作数地址,其原因可能是以下两者之一:其一,本指令确实不需要操作数,如NOP指令;其二,本指令本身就是专门为某一个操作数制造的,它不可能用来操作其他的操作数,如CLRA指令(由其名称可知,它用来清零累加器A)。
3、寄存器直接寻址,这是一种十分重要,十分常见,也十分易于理解、易于掌握和应用的寻址方式,就是在指令中直接给出寄存器编号(在汇编中,给出的是寄存器的标号,也可以视作寄存器的名称)作为操作数的寻址方式,而待操作的数据就存在此寄存器中。如指令MOV R1,R2就是一条双操作数都采用寄存器直接寻址的指令。

另外,这种寻址方式在一些处理器构架中又会有一些细微变化,如在ARM中由于大量使用寄存器直接寻址方式,为了增强此寻址方式的威力,ARM处理器中的寄存器直接寻址加入了移位功能,即在实际指令执行中的数据不是简单的寄存器中直接存储的数据,而是此数据经过移位后得到的数据。
4、寄存器间接寻址,如同寄存器直接寻址,都是在指令编码中给出寄存器编号(在汇编中表现为寄存器名称);但与寄存器直接寻址不同的是,此处寄存中存储的并非操作数本身,而是操作数在内存中的位置,即操作数的地址(相当于C语言中的指针),在内存中此地址处存储的才是真正的操作数。所以,CPU要获取此操作数,在微观上需要进行两次数据获取操作,第一次从寄存器中获取操作数的地址,第二此从刚才获取的地址处在内存中获取真正的操作数。在多数处理器中,指令使用寄存器间接寻址比使用寄存器直接寻址运行要耗费更多的指令周期。如指令 MOV R1,@R2将R1中的数据复制到R2中地址所指向的内存中。
寄存器间接寻址是变种最多、扩展最多的寻址方式,正因如此,他也是处理器中最为灵活、最为强大、最为难于掌握的寻址方式。不仅不同处理器对它进行了不同的扩展,而且即使对于完全相同的扩展,不同汇编中也可能给出相差极大的书写格式,这就更加导致了寄存器间接寻址难于理解、难于掌握。
其扩展主要有以下形式:1)自增、自减型,@R1++、@R1--、@(--R1)、@(++R1),看到它们的书写形式大家应该也大体明白它们的意思了,当然具体的汇编书写格式那就根据不同的处理器和汇编器变得五花八门了,比如有的不用@号而用中括号,有的只用一个加减号而不是两个,有的只提供了四种变体中的一种或几种,情况不一而足,但其实质内容却大体一致。2)索引型(或叫偏移型),就是指寄存器中存储的地址还不是操作数的最终地址,须得对此地址加一个数才是最终的操作数地址。这种情况就更加复杂了,其一,书写格式更加纷乱,如我就见过X(R1)、[R1,X]、(R1+X)等多种形式;其二,偏移量X本身也涉及寻址方式这个问题,最常用的当然是立即寻址和寄存器直接寻址两种,但也不排除有更加复杂的寄存器间接寻址这种方式的使用,要是这个X本身要是也能使用寄存器间接寻址的变种——索引寻址,那可就真是热闹了(试想一下,这不就是迭代了吗)。
5、内存直接寻址,就是在指令中直接给出操作数的地址——存储于内存中的位置,表现在汇编中,为了区分立即数和内存地址,一般对地址加&或$前缀。如 MOV R1,$100
这个内存地址可以可以分两种情况给出:其一,直接给出完整的地址;其二,仅给出一个偏移量,将此偏移量加上PC才能得到操作数的地址。其中,前者一般叫做绝对寻址,后者一般叫做相对寻址(当然,有些汇编
- 基于自适应技术的CPU供电电路系统(10-27)
- 嵌入式CPU卡在医用便携式监护仪中的应用及设计(09-23)
- 四核Vs八核移动处理器 性能差异并不大(03-10)
- 可编程逻辑控制器(PLC)基本操作及功能简介(03-07)
- 评测:采用AMD APU平台的联想启天M5800——均衡+全能(02-17)
- 智能手机省电秘诀:看如何从设计源头来降低功耗(02-14)
