历史上的今天
返回首页

历史上的今天

今天是:2024年12月11日(星期三)

正在发生

2019年12月11日 | 通过PICC编译环境下,对PIC单片机程序进行操作研究

2019-12-11 来源:elecfans

PICC基本上符合ANSI标准,但是不支持函数的递归调用,其主要原因是PIC单片机特殊的堆栈结构。PIC单片机中的堆栈是硬件实现的,其深度已随芯片固定,无法实现需要大量堆栈操作的递归算法;另外在PIC单片机中实现软件堆栈的效率也不是很高。为此,PICC编译器采用一种“静态覆盖”技术,以实现对C语言函数中的局部变量分配固定的地址空间。经这样处理后产生出的机器代码效率很高。当代码量超过4KB后,C语言编译出的代码长度与全部用汇编代码实现的差别已经不是很大(<10%),当然前提是在整个C代码编写过程中需时时注意所编写语句的效率。


PICC中的变量类型和标准C语言一样,这里不再重复。为了使编译器产生最高效的机器码,PICC把单片机中数据寄存器的bank交由编程员自己管理,因此在定义用户变量时必须自己决定这些变量具体放在哪一个bank中。如果没有特别指明,所定义的变量将被定位在bank0。定义在其他bank内的变量前面必须加上相应的bank序号,例如:

bank1 unsigned char temp;//变量定位在bank1中档系列PIC单片机数据寄存器的一个bank大小为128B,除前面若干字节的特殊功能寄存器区域,在C语言中某一bank内定义的变量字节总数不能超过可用RAM字节数。如果超过bank容量,在最后连接时会报错,大致信息如下:

Error[000]:Can’t find 0x12C words for psect rbss_1 insegmentBANK1

链接器提示,总共有0x12c(300)字节准备放到bank1中,但bank1容量不够。虽然变量所在的bank定位必须由编程员自己决定,但编写源程序时在进行变量存取操作前无需再特意编写设定bank的指令。C编译器会根据所操作的对象自动生成对应bank设定的汇编指令。为避免频繁的bank切换以提高代码效率,尽量把实现同一任务的变量定位在同一个bank内;对不同bank内的变量进行读写操作时也尽量把位于相同bank内的变量归并在一起进行连续操作。


bit型位变量只能是全局的或静态的。PICC将把定位在同一bank内的8个位变量合并成一个字节存放于一个固定地址。PICC对整个数据存储空间实行位编址,0x000单元第0位位地址是0x0000,以此类推,每个字节有8个位地址。如果一个位变量flag1被编址为0x123,那么实际的存储空间位于:

字节地址=0x123/8 = 0x24

位偏移=0x123%8 = 3

即flag1位变量位于地址为0x24字节的第3位。在程序调试时如果要观察flag1的变化,必须观察地址为0x24的字节而不是0x123。PICC在编译原代码时只要有可能,对普通变量的操作也将以最简单的位操作指令来实现。假设一个字节变量tmp最后被定位在地址0x20,那么tmp | =0x80=>bsf 0x20.7

另外,函数可以返回一个位变量,返回的位变量将存放于单片机的进位位中返回。

PICC在编译C源程序时,将指向RAM的指针操作最终用FSR来实现间接寻址。FSR能够直接连续寻址的范围是256B,所以一个指针可以同时覆盖2个bank的存储区域(bank0/1或bank2/3,一个bank区域是128 B)。要覆盖最大512B的内部数据存储空间,在定义指针时必须明确指定该指针适用的寻址区域。例如:

通过PICC编译环境下,对PIC单片机程序进行操作研究

既然定义的指针有明确的bank适用区域,在对指针变量赋值时就必须实现类型匹配,否则将产生错误,例如:

通过PICC编译环境下,对PIC单片机程序进行操作研究

程序语句:

pointer() =buff;//错误!试图将bank2内的变量地址赋给指向bank0/1的指针

若出现此类错误的指针操作,PICC在最后链接时会告知类似于下面的信息:

Fixup overflow in expression (…)

如果一组变量是已经被定义在ROM区的常数,那么指向其的指针可以这样定义:

const unsigned char company[]="software"

因为在PIC单片机这一特定的架构上实现函数指针调用的效率不高,因此,除非特殊算法的需要,建议大家尽量不要使用函数指针。


中档系列的PIC单片机程序空间有分页的概念,但用C语言编程时基本不用过多关心代码的分页问题。因为所有函数或子程序调用时的页面设定(如果代码超过一个页面)都由编译器自动生成的指令实现。


PICC决定了C源程序中的一个函数经编译后生成的机器码一定会放在同一个程序页面内。中档系列PIC单片机的一个程序页面的长度是2

KB,用C语言编写的任何一个函数最后生成的代码不能超过2

KB。如果为实现特定的功能确实要连续编写很长的程序,这时就必须把这些连续的代码拆分成若干函数,以保证每个函数最后编译出的代码不超过一个页面空间。


PIC单片机采用硬件堆栈,所以编程时函数的调用层次会受到一定限制。一般PIC系列的中档单片机硬件堆栈深度为8级。程序员必须自己控制子程序调用时的嵌套深度以符合这一限制要求。PICC在最后编译链接成功后可以生成一个链接定位映射文件(*.map),在此文件中有详细的函数调用嵌套指示图“callgraph”,有些函数调用是编译时自动加入的库函数,这些函数调用从C源程序中无法直接看出,但在嵌套指示图上则一目了然。


单片机的一些特殊指令操作在标准的C语言语法中没有直接对应的描述,例如PIC单片机的清看门狗指令“clrwdt”和休眠指令“sleep”;单片机系统强调的是控制的实时性,为了实现这一要求,有时必须用汇编指令实现部分代码以提高程序运行的效率。在C程序中嵌入汇编指令有2种方法。


① 如果只需要嵌入少量几条汇编指令,PICC提供了一个类似于函数的语句:

asm("clrwdt");

这是在C源程序中直接嵌入汇编指令的最直接最容易的方法。

② 如果需要编写一段连续的汇编指令,PICC支持另外的一种语法描述:用“#asm”来开始汇编指令段,用“#endasm”结束。例如:

# asm

通过PICC编译环境下,对PIC单片机程序进行操作研究

5.1  汇编指令寻址C语言定义的全局变量

所有C语言中定义的符号在编译后将自动在前面添加下划线“_”。因此,若要在汇编指令中寻址C语言定义的各类变量,一定要在变量前加上“_”符号,例如上例中的count是在C语言中定义的无符号全局变量,在汇编语言中只需在其前面加上“_”符号就可进行访问了。另外,对于C语言中定义的多字节全局变量,例如C语言中的如下定义:

int advalue;

在汇编语言里访问时就得分字节访问,例如:

asm(“movf_advalue+0.0”);//把advalue低字节中的数送到w里

asm(“rrf_advalue+1”)//把advalue高字节中的数左移一位


5.2  汇编指令寻址C函数的局部变量

前面已经提到,PICC对自动型局部变量(包括函数调用时的入口参数)采用一种“静态覆盖”技术,对每一个变量确定一个固定地址(位于bank0),嵌入的汇编指令对其寻址时只需采用数据寄存器的直接寻址方式即可,因此关键是要知道这些局部变量的寻址符号。建议读者先编写一小段C代码,其中有最简单的局部变量操作指令,把此源代码编译成对应的PICC汇编指令;查看C编译器生成的汇编指令是如何寻址这些局部变量的,自己编写的行内汇编指令就采用同样的寻址方式。


相对于汇编语言,用C语言编程的优势是毋庸置疑的:开发效率大大提高、人性化的语句指令及模块化的程序易于日常管理和维护、程序在不同平台间移植方便。所以既然使用C语言编程,就应该尽量避免嵌入汇编指令或编写汇编指令模块文件。例如:

通过PICC编译环境下,对PIC单片机程序进行操作研究

变量的循环右移操作用C语言实现非常不方便,PIC单片机已有对应的移位操作汇编指令,因此用嵌入汇编的形式实现效率最高。对移位次数的控制,实际上变量count1的递减判零也可以直接用汇编指令实现,这样可节约代码,但用标准C语言描述更直观、更易于维护。


6  注意事项

① 既然所有的局部变量将占用bank0的存储空间,因此用户自己定位在bank0内的变量字节数将受到一定的限制,在实际使用时需注意。

②当程序中把非位变量进行强制类型转换成位变量时,要注意编译器只对普通变量的最低位做判别:若最低位是0,则转换成位变量0;若最低位是1,则转换成位变量1。

③ 由于PIC系列单片机的内部资源十分有限,所以在允许的条件下应尽量使用无符号字符型变量,以节约空间。

④ PICC对绝对定位的变量不保留地址空间,例如:unsigned char advalue @ 0x20 ;//advalue定位在地址0x20,相当于汇编语言中的伪指令advalue EQU 20H所以请读者慎用。

⑤尽量使用全局变量进行参数传递,使用全局变量最大的好处是寻址直观,只需在C语言定义的变量名前增加一个下划线符即可在汇编语句中寻址;使用全局变量进行参数传递的效率也比形参高。

⑥ 对于多字节变量(如int型、float型变量等)PICC遵循Littleendian标准,即低字节放在存储空间的低地址,高字节放在高地址,编程时需注意。


结语

一般C语言产生的代码是比较繁琐的,所以要写出高质量、实用的C语言程序,就必须对单片机体系结构和硬件资源作详尽的了解。用C语言开发PIC系列单片机系统软件具有编写代码效率高、软件调试直观、维护升级方便、代码的重复利用率高、便于跨平台的代码移植等优点,因此C语言编程在单片机系统设计中的应用必将越来越广泛。

推荐阅读

史海拾趣

Astro Industries Inc公司的发展小趣事

Astro Industries Inc的创始人在电子行业拥有丰富的经验,他们看到了市场对于高性能电子产品的迫切需求。于是,公司从创立之初就专注于技术创新,投入大量资源进行研发。经过不懈的努力,Astro Industries Inc成功推出了一款具有颠覆性的电子产品,其性能远超当时市场上的同类产品,迅速获得了市场的认可。

Frequency Electronics Inc公司的发展小趣事

为了进一步扩大市场份额,高频电子积极实施全球化战略。公司在全球范围内建立了销售网络和服务体系,与众多国际知名企业建立了长期合作关系。同时,高频电子还针对不同地区的市场需求,定制化开发符合当地标准的产品和服务。这些努力使得高频电子的产品和服务能够覆盖全球多个国家和地区,为公司带来了稳定的收入来源和持续增长的动力。

Eutech公司的发展小趣事

在全球化的浪潮下,EUtech公司积极寻求国际合作。他们与德国的氢探新能源公司建立了长期稳定的合作关系,共同研发燃料电池系统控制技术。通过共享资源、互补优势,双方不仅提高了产品的竞争力,还推动了整个行业的技术进步。

ADTech公司的发展小趣事

随着QCL技术的成熟和市场需求的增长,AdTech公司开始积极拓展国际市场。通过与全球多家知名企业建立合作伙伴关系,公司成功将产品打入欧洲、亚洲等多个地区。同时,AdTech还积极参与国际技术交流与合作,不断提升自身的研发能力和产品质量。这些举措不仅增强了公司的品牌影响力,也为其带来了更多的商业机会。

BEKA Associates Ltd公司的发展小趣事

随着全球电子市场的不断发展,BCD Semi(Diodes)积极寻求国际化拓展的机会。公司通过与国外知名企业的合作,成功进入了多个海外市场。在国际市场上,BCD Semi凭借其高品质的产品和专业的服务,赢得了客户的信赖和支持。同时,公司还积极参加国际电子展会和交流活动,与全球同行进行深入的交流与合作,不断提升自身的国际影响力。

ECLIPSE公司的发展小趣事

随着云计算、大数据和人工智能等技术的不断发展,Eclipse也面临着新的挑战和机遇。Eclipse团队正在积极研究新技术,探索新的应用场景。未来,Eclipse将继续保持开放和创新的精神,为开发者们提供更加优秀、高效的开发工具和服务。同时,Eclipse也将继续加强与其他开源项目和公司的合作,共同推动开源生态的发展。

问答坊 | AI 解惑

马上大四了,向大家询问就业职能取向问题。。。

学校只组织了C#和Java两种语言实训,但是我现在只对系统底层比较感兴趣,如果要学习驱动编写和系统安全相关(如破解,逆向等)这些,有相关的实训项目么?是否只有嵌入式可以选择了?如果学习嵌入式会对系统安全这方面有帮助么?烦请各位给个建议。 ...…

查看全部问答>

无源晶振为什么只有一个脚起振?? 高手进!

芯片能正常工作,但是27M无源晶振只有一个输出脚起振,输入脚为0电平。 我示波器是200M的,应该没问题。…

查看全部问答>

有没有检测PDA硬件性能的免费软件

需要检测PDA上的CPU频率,内存容量等硬件信息 我下载了一个Pocket PC Mark_ARM软件需要注册码,晕啊…

查看全部问答>

uclinux内核与版本问题

  我想知道用2.6的系统能不能编译2.4的uclinux吗!!…

查看全部问答>

请教PPC2003的开发环境

各位大侠:    小弟刚刚涉及PPC2003的开发,对开发环境和调试方法都一愁莫展,请各位大侠给予指点,谢谢了…

查看全部问答>

无线测试工程师,请发表你对新产品的见解

大家好,我是电子工程世界的编辑,赵思潇。因为我本身不是测试工程师出身,希望今后跟大家多探讨一些测试方面的问题。你们的见解对我来说非常重要。   上周,我参加了一个泰克的产品发布会。在此,想就这款产品,跟大家进行下讨论。   ...…

查看全部问答>

新手求助,关于DHT11,有经验的看一下

最近在做MSP430F149驱动DHT11采集温湿度然后在LCD1602上显示的实验,但硬件连接好后总是显示00.0,好像DHT11没工作,看了好久找不出程序问题,希望有经验的你帮下忙,谢谢。。。原程序如下: #include \"msp430x14x.h\" // 包含头文件#define ucha ...…

查看全部问答>

lauchpad g2553频率测量

做一个lauchpad  g2553频率测量,但是编译总是出错谁能给个程序参考下,频率捕获的…

查看全部问答>

如何在linux中改u-boot的环境变量--fw_setenv

由于项目的需要,我需要在linux中的应用程序里面更改u-boot 的环境变量,开始纠结了很久,不知道怎么去读取具体的nand flash物理地址,后来偶然中找到了 dd 指令,发现灰常的好用,直接 dd if=/dev/mtd1 of=/mtd1.env 读出。      & ...…

查看全部问答>

DSP仿真问题

我在21ic电子工程师论坛上下载了一个版主分享的DSP代码,我是把这个文件夹放在和之前的例程代码的文件夹同一个目录下,但是进行仿真时出错,不能下载到板子上,请问该怎样设置该文件夹??…

查看全部问答>