单片机方案开发商深圳英锐恩分享基于PICC编译环境编写PIC单片机程序。
摘 要:Microchip 公司生产的PIC系列单片机具有实用、低价、简单易学、低功耗、高速度、体积小、功能强等特点,体现了单片机发展的一种新趋势,而PICC具有许多特殊的性质,并且进行了C语言的扩展,从而可以更轻松地完成编程任务。本文简单介绍了PIC系列单片机在国内的发展情况,以HiTech Software公司的HiT ech PICC编译器为例介绍了PICC和标准C的异同及HiTech PICC语言的特点,详细介绍了PICC中的变量、指针、函数以及C与汇编混合编程的一些相关知识,并列举了许多例子以便读者理解。此外还着重介绍了用PICC开发PIC 系列单片机时应注意的一些问题。
关键词:PIC;PICC编译器;C与汇编混合编程;HiTech;单片机
目前,Microchip 公司生产的PIC系列单片机以其低成本、低功耗、高性能、开发速 度快且一次性用户可编程等优点迅速占领了国内市场,成为国内销售量最大的单片机,但国 内介绍他的C语言开发工具的书籍和文章却比较少,而且用的人也不多,在用其开发的过程 中给广大程序员带来了许多困难和不便。
Microchip 公司自己没有针对中低档系列PIC单片机的C 语言编译器,但很多专业的 第三方 公司有众多支持PIC 单片机的C 语言编译器提供,常见的有Hitech,CCS,IAR,Bytecraft 等公司。Hitech 公司的PICC 编译器稳定可靠,编译生成的代码效率高,在用PIC 单片 机进行系统设计和开发的工程师群体中得到广泛认可。因此,本文主要以Hi-Tech PICC为基 础,介绍PIC的C语言的基本特点。
1 HiTech PICC和 ANSI C 的异同及HiTech PICC语言的特点
除了PICC不支持函数的递归调用外,PICC 基本上符合ANSI 标准,其主要原因是因为PIC 单片机特殊的堆栈结构。PIC 单片机中的堆栈是硬件实现的,其深度已随芯片而固定,无法 实现需要大量堆栈操作的递归算法;另外在PIC 单片机中实现软件堆栈的效率也不是很高, 为此,PICC 编译器采用一种“静态覆盖”技术以实现对C 语言函数中的局部变量分配固定 的地
址空间。经这样处理后产生出的机器代码效率很高,当代码量超过4 kB后,C 语言编译 出的代码长度和全部用汇编代码实现时的差别已经不是很大(<10%),当然前提是在整个C代码编写过程中需时时注意所编写语句的效率。
2 PICC中的变量
PICC中的变量类型和标准C一样,这里不再重复。为了使编译器产生最高效的机器码,PICC把单片机中数据寄存器的bank 问题交由编程员自己管理,因此在定义用户变量时必须自己 决定这些变量具体放在哪一个bank 中。如果没有特别指明,所定义的变量将被定位在bank0。定义在其他bank 内的变量前面必须加上相应的bank 序号,例如:
bank1 unsigned char temp;//变量定位在bank1 中
中档系列PIC单片机数据寄存器的一个bank 大小为128 B,刨去前面若干字节的特殊功能寄 存器区域,在C语言中某一bank内定义的变量字节总数不能超过可用RAM字节数。如果超过ba nk 容量,在最后连接时会报错,大致信息如下:
连接器提示总共有0x12C(300)个字节准备放到bank1
中但bank1 容量不够。虽然变量所 在的bank定位必须由编程员自己决定,但在编写源程序时进行变量存取操作前无需再特意编 写设定bank
的指令。C 编译器会根据所操作的对象自动生成对应bank 设定的汇编指令。为 避免频繁的bank
切换以提高代码效率,尽量把实现同一任务的变量定位在同一个bank 内; 对不同bank 内的变量进行读写操作时也尽量把位于相同bank
内的变量归并在一起进行连续 操作。
bit 型位变量只能是全局的或静态的。PICC 将把定位在同一bank 内的8 个位变量合并成一 个字节存放于一个固定地址。PICC 对整个数据存储空间实行位编址,0x000 单元的第0 位 是位地址0x0000,以此后推,每个字节有8 个位地址。如果一个位变量flag1 被编址为0x12 3,那么实际的存储空间位于:
字节地址=0x123/8 = 0x24
位偏移=0x123%8 = 3
即flag1
位变量位于地址为0x24 字节的第3 位。在程序调试时如果要观察flag1 的变化, 必须观察地址为0x24 的字节而不是0x123。PICC
在编译原代码时只要有可能,对普通变量 的操作也将以最简单的位操作指令来实现。假设一个字节变量tmp 最后被定位在地址0x20,那么
另外,函数可以返回一个位变量,实际上此返回的位变量将存放于单片机的进位位中带出返回。
3 PICC中的指针
3.1 指向RAM的指针
PICC在编译C原程序时将指向RAM的指针操作最终用FSR来实现间接寻址。FSR能够直接连续寻址的范围是256
B,所以一个指针可以同时覆盖两个bank 的存储区域(bank0/1或bank2/3,一个bank区域是128 B)。要覆盖最大512
B的内部数据存储空间,在定义指针时必须明确指 定该指针所适用的寻址区域。例如:
既然定义的指针有明确的bank 适用区域,在对指针变量赋值时就必须实现类型匹配,否则 将产生错误,例如:
若出现此类错误的指针操作,PICC 在最后连接时会告知类似于下面的信息:
Fixup overflow in expression (…)
3.2 指向ROM常数的指针
如果一组变量是已经被定义在ROM 区的常数,那么指向他的指针可以这样定义:
3.3 指向函数的指针
因为在PIC 单片机这一特定的架构上实现函数指针调用的效率不高,因此,除非特殊算法的需要,建议大家尽量不要使用函数指针。
4PICC中的子程序和函数
中档系列的PIC 单片机程序空间有分页的概念,但用C语言编程时基本不用太多关心代码的分页问题。因为所有函数或子程序调用时的页面设定(如果代码超过一个页面)都由编译器自动生成的指令实现。
4.1 函数的代码长度限制
PICC决定了C源程序中的一个函数经编译后生成的机器码一定会放在同一个程序页面内。 中档系列的PIC单片机其一个程序页面的长度是2 kB,用C语言编写的任何一个函数最后生成的代码不能超过2 kB。如果为实现特定的功能确实要连续编写很长的程序,这时就必须把这些连续的代码拆分成若干函数,以保证每个函数最后编译出的代码不超过一个页面空间。
4.2 调用层次的控制
PIC单片机采用硬件堆栈,所以编程时函数的调用层次会受到一定限制。一般PIC系列的中档单片机硬件堆栈深度为8级。编程员必须自己控制子程序调用时的嵌套深度以符合这一限制要求。PICC
在最后编译连接成功后可以生成一个连接定位映射文件(*map),在此文件中有详细的函数调用嵌套指示图“call
graph”,有些函数调用是编译代码时自动加入的库函数,这些函数调用从C源程序中无法直接看出,但在嵌套指示图上则一目了然。
5C语言和汇编语言混合编程
单片机的一些特殊指令操作在标准的C 语言语法中没有直接对应的描述,例如PIC 单片机的清看门狗指令“clrwdt”和休眠指令“sleep”;单片机系统强调的是控制的实时性,为了实现这一要求,有时必须用汇编指令实现部分代码以提高程序运行的效率。在C程序中嵌入汇 编指令有2种方法:
1) 如果只需要嵌入少量几条的汇编指令,PICC 提供了一个类似于函数的语句:
asm("clrwdt");
这是在C原程序中直接嵌入汇编指令的最直接最容易的方法。
2) 如果需要编写一段连续的汇编指令,PICC 支持另外一种语法描述:用“#asm”开始汇 编指令段,用“#endasm”结束。例如:
5.1 汇编指令寻址C语言定义的全局变量
所有C中定义的符号在编译后将自动在前面添加一下划线符“_”,因此,若要在汇编指令
中寻址C
语言定义的各类变量,一定要在变量前加上“_”符号,例如上例中的count是在C中定义的无符号全局变量,在汇编语言中只需在其前面加上“_”符号就可对他进行访问了。另外,对于C中定义的多字节全局变量,例如在C中有如下定义:
那么在汇编里访问他时就得分字节访问,例如:
5.2 汇编指令寻址C函数的局部变量
前面已经提到,PICC 对自动型局部变量(包括函数调用时的入口参数)采用一种“静态覆 盖”技术对每一个变量确定一个固定地址(位于bank0),因此嵌入的汇编指令对其寻址时只需采用数据寄存器的直接寻址方式即可, 所以关键是要知道这些局部变量的寻址符号。建议读者先编写一小段C代码,其中有最简单的局部变量操作指令,把此C源代码编译成对应的PICC 汇编指令;查看C编译器生成的汇编指令是如何寻址这些局部变量的,自己编写的行内汇编指令就采用同样的寻址方式。
相比于汇编语言,用C语言编程的优势是毋庸置疑的:开发效率大大提高、人性化的语句指令加上模块化的程序易于日常管理和维护、程序在不同平台间的移植方便。所以既然用了C语言编程,就尽量避免使用嵌入汇编指令或整个地编写汇编指令模块文件。例如:
变量的循环右移操作用C语言实现非常不方便,PIC
单片机已有对应的移位操作汇编指令, 因此用嵌入汇编的形式实现效率最高。同时对移位次数的控制,实质上说变量count1
的递减判零也可以直接用汇编指令实现,这样可节约代码,但用标准的C语言描述更直观,更易于维护。
6 注意事项
1)既然所有的局部变量将占用bank0 的存储空间,因此用户自己定位在bank0 内的变量 字节数将受到一定的限制,在实际使用时需注意。
2)当程序中把非位变量进行强制类型转换成位变量时,要注意编译器只对普通变量的最 低位做判别:若最低位是0,则转换成位变量0;最低位是1,则转换成位变量1。
3)由于PIC系列单片机的内部资源十分有限,所以在允许的条件下应尽量使用无符号字符型变量以节约空间。
4)PICC 对绝对定位的变量不保留地址空间,例如:
所以请读者慎用。
5)尽量使用全局变量进行参数传递,使用全局变量最大的好处是寻址直观,只需在C语 言定义的变量名前增加一个下划线符即可在汇编语句中寻址;使用全局变量进行参数传递的效率也比形参高。
6)对于多字节变量(如int型、float型变量等)PICC 遵循Littleendian 标准,即低字节放在存储空间的低地址,高字节放在高地址,编程时需注意。
7 结语
一般C语言产生的代码是比较繁琐的,所以要想写出高质量实用的C语言程序,就必须对单片机体系结构和硬件资源作详尽地了解,但用C 语言开发PIC系列单片机系统软件具有编写代码效率高、软件调试直观、维护升级方便、代码的重复利用率高、便于跨平台的代码移植等优点,因此C 语言编程在单片机系统设计中的应用必将越来越广泛。