ANSI C与C51.docx

上传人:b****5 文档编号:3257584 上传时间:2022-11-21 格式:DOCX 页数:18 大小:32.18KB
下载 相关 举报
ANSI C与C51.docx_第1页
第1页 / 共18页
ANSI C与C51.docx_第2页
第2页 / 共18页
ANSI C与C51.docx_第3页
第3页 / 共18页
ANSI C与C51.docx_第4页
第4页 / 共18页
ANSI C与C51.docx_第5页
第5页 / 共18页
点击查看更多>>
下载资源
资源描述

ANSI C与C51.docx

《ANSI C与C51.docx》由会员分享,可在线阅读,更多相关《ANSI C与C51.docx(18页珍藏版)》请在冰豆网上搜索。

ANSI C与C51.docx

ANSIC与C51

Keil C51 vs 标准C 1

深入理解并应用C51对标准ANSIC的扩展是学习C51的关键之一。

因为大多数扩展功能都是直接针对8051系列CPU硬件的。

大致有以下8类:

l8051存储类型及存储区域

l存储模式

l存储器类型声明

l变量类型声明

l位变量与位寻址

l特殊功能寄存器(SFR)

lC51指针

l函数属性

具体说明如下(8031为缺省CPU)。

1.第一节KeilC51扩展关键字

C51V4.0版本有以下扩展关键字(共19个):

_at_ idatasfr16alieninterruptsmall

bdatalarge_task_Codebitpdata

usingreentrantxdatacompactsbitdatasfr

2.第二节内存区域(MemoryAreas):

1.1.PragramArea:

由Code说明可有多达64kBytes的程序存储器

2.2.InternalDataMemory:

内部数据存储器可用以下关键字说明:

data:

直接寻址区,为内部RAM的低128字节00H~7FH

idata:

间接寻址区,包括整个内部RAM区00H~FFH

bdata:

可位寻址区, 20H~2FH

3.3.ExternalDataMemory

外部RAM视使用情况可由以下关键字标识:

xdata:

可指定多达64KB的外部直接寻址区,地址范围0000H~0FFFFH

pdata:

能访问1页(25bBytes)的外部RAM,主要用于紧凑模式(CompactModel)。

4.4.SpeciacFunctionRegisterMemory

8051提供128Bytes的SFR寻址区,这区域可位寻址、字节寻址或字寻址,用以控制定时器、计数器、串口、I/O及其它部件,可由以下几种关键字说明:

sfr:

字节寻址比如sfrP0=0x80;为PO口地址为80H,“=”后H~FFH之间的常数。

sfr16:

字寻址,如sfr16T2=0xcc;指定Timer2口地址T2L=0xccT2H=0xCD

sbit:

位寻址,如sbitEA=0xAF;指定第0xAF位为EA,即中断允许

还可以有如下定义方法:

sbit0V=PSW^2;(定义0V为PSW的第2位)

sbit0V=0XDO^2;(同上)

或bit0V-=0xD2(同上)。

.第三节存储模式

存储模式决定了没有明确指定存储类型的变量,函数参数等的缺省存储区域,共三种:

1.1.Small模式

所有缺省变量参数均装入内部RAM,优点是访问速度快,缺点是空间有限,只适用于小程序。

2.2.Compact模式

所有缺省变量均位于外部RAM区的一页(256Bytes),具体哪一页可由P2口指定,在STARTUP.A51文件中说明,也可用pdata指定,优点是空间较Small为宽裕速度较Small慢,较large要快,是一种中间状态。

3.3.large模式

所有缺省变量可放在多达64KB的外部RAM区,优点是空间大,可存变量多,缺点是速度较慢。

提示:

存储模式在C51编译器选项中选择。

4.第四节存储类型声明

变量或参数的存储类型可由存储模式指定缺省类型,也可由关键字直接声明指定。

各类型分别用:

code,data,idata,xdata,pdata说明,例:

datauar1

charcodearray[]=“hello!

”;

unsignedcharxdataarr[10][4][4];

5.第五节变量或数据类型

C51提供以下几种扩展数据类型:

bit位变量值为0或1

sbit从字节中定义的位变量0或1

sfrsfr字节地址0~255

sfr16sfr字地址0~65535

其余数据类型如:

char,enum,short,int,long,float等与ANSIC相同。

6.第六节位变量与声明

1.1.bit型变量

bit型变量可用变量类型,函数声明、函数返回值等,存贮于内部RAM20H~2FH。

注意:

(1)用#pragmadisable说明函数和用“usign”指定的函数,不能返回bit值。

(2)一个bit变量不能声明为指针,如bit*ptr;是错误的

(3)不能有bit数组如:

bitarr[5];错误。

2.2.可位寻址区说明20H-2FH

可作如下定义:

intbdatai;

charbdataarr[3],

然后:

sbitbito=in0;sbitbit15=I^15;

sbitarr07=arr[0]^7;sbitarr15=arr[i]^7;

7.第七节KeilC51指针

C51支持一般指针(GenericPointer)和存储器指针(Memory_SpecificPointer).

1.1.一般指针

一般指针的声明和使用均与标准C相同,不过同时还可以说明指针的存储类型,例如:

long*state;为一个指向long型整数的指针,而state本身则依存储模式存放。

char*xdataptr;ptr为一个指向char数据的指针,而ptr本身放于外部RAM区,以上的long,char等指针指向的数据可存放于任何存储器中。

一般指针本身用3个字节存放,分别为存储器类型,高位偏移,低位偏移量。

2.2.存储器指针

基于存储器的指针说明时即指定了存贮类型,例如:

chardata*str;str指向data区中char型数据

intxdata*pow;pow指向外部RAM的int型整数。

这种指针存放时,只需一个字节或2个字节就够了,因为只需存放偏移量。

3.3.指针转换

即指针在上两种类型之间转化:

l当基于存储器的指针作为一个实参传递给需要一般指针的函数时,指针自动转化。

l如果不说明外部函数原形,基于存储器的指针自动转化为一般指针,导致错误,因而请用“#include”说明所有函数原形。

l可以强行改变指针类型。

8.第八节KeilC51函数

C51函数声明对ANSIC作了扩展,具体包括:

1.1.中断函数声明:

中断声明方法如下:

voidserial_ISR()interrupt4[using1]

{

/*ISR*/

}

为提高代码的容错能力,在没用到的中断入口处生成iret语句,定义没用到的中断。

/*definenotusedinterrupt,sogenerate"IRET"intheirentrance*/

voidextern0_ISR()interrupt0{}/*notused*/

voidtimer0_ISR()interrupt1{}/*notused*/

voidextern1_ISR()interrupt2{}/*notused*/

voidtimer1_ISR()interrupt3{}/*notused*/

voidserial_ISR()interrupt4{}/*notused*/

2.2.通用存储工作区

3.3.选通用存储工作区由usingx声明,见上例。

4.4.指定存储模式

由smallcompact及large说明,例如:

voidfun1(void)small{}

提示:

small说明的函数内部变量全部使用内部RAM。

关键的经常性的耗时的地方可以这样声明,以提高运行速度。

5.5.#pragmadisable

在函数前声明,只对一个函数有效。

该函数调用过程中将不可被中断。

6.6.递归或可重入函数指定

在主程序和中断中都可调用的函数,容易产生问题。

因为51和PC不同,PC使用堆栈传递参数,且静态变量以外的内部变量都在堆栈中;而51一般使用寄存器传递参数,内部变量一般在RAM中,函数重入时会破坏上次调用的数据。

可以用以下两种方法解决函数重入:

a、在相应的函数前使用前述“#pragmadisable”声明,即只允许主程序或中断之一调用该函数;

b、将该函数说明为可重入的。

如下:

voidfunc(param...)reentrant;

KeilC51编译后将生成一个可重入变量堆栈,然后就可以模拟通过堆栈传递变量的方法。

由于一般可重入函数由主程序和中断调用,所以通常中断使用与主程序不同的R寄存器组。

另外,对可重入函数,在相应的函数前面加上开关“#pragmanoaregs”,以禁止编译器使用绝对寄存器寻址,可生成不依赖于寄存器组的代码。

7.7.指定PL/M-51函数

由alien指定。

4.第四章KeilC51高级编程

本章讨论以下内容:

l绝对地址访问

lC与汇编的接口

lC51软件包中的通用文件

l段名转换与程序优化

1.第一节绝对地址访问

C51提供了三种访问绝对地址的方法:

1.1.绝对宏:

在程序中,用“#include”即可使用其中定义的宏来访问绝对地址,包括:

CBYTE、XBYTE、PWORD、DBYTE、CWORD、XWORD、PBYTE、DWORD

具体使用可看一看absacc.h便知

例如:

rval=CBYTE[0x0002];指向程序存贮器的0002h地址

rval=XWORD[0x0002];指向外RAM的0004h地址

2.2._at_关键字

直接在数据定义后加上_at_const即可,但是注意:

(1)绝对变量不能被初使化;

(2)bit型函数及变量不能用_at_指定。

例如:

idatastructlinklist_at_0x40;指定list结构从40h开始。

xdatachartext[25b]_at_0xE000;指定text数组从0E000H开始

提示:

如果外部绝对变量是I/O端口等可自行变化数据,需要使用volatile关键字进行描述,请参考absacc.h。

3.3.连接定位控制

此法是利用连接控制指令codexdatapdata\databdata对“段”地址进行,如要指定某具体变量地址,则很有局限性,不作详细讨论。

2.第二节KeilC51与汇编的接口

1.1.模块内接口

方法是用#pragma语句具体结构是:

#pragmaasm

汇编行

#pragmaendasm

这种方法实质是通过asm与ndasm告诉C51编译器中间行不用编译为汇编行,因而在编译控制指令中有SRC以控制将这些不用编译的行存入其中。

2.2.模块间接口

C模块与汇编模块的接口较简单,分别用C51与A51对源文件进行编译,然后用L51将obj文件连接即可,关键问题在于C函数与汇编函数之间的参数传递问题,C51中有两种参数传递方法。

(1)通过寄存器传递函数参数

最多只能有3个参数通过寄存器传递,规律如下表:

参数数目charintlong,float一般指针

123R7R5R3R6&R7R4&R5R2&R3R4~R7R4~R7R1~R3R1~R3R1~R3

(2)通过固定存储区传递(fixedmemory)

这种方法将bit型参数传给一个存储段中:

 ?

function_name?

BIT

将其它类型参数均传给下面的段:

function_name?

BYTE,且按照预选顺序存放。

至于这个固定存储区本身在何处,则由存储模式默认。

(3)函数的返回值

函数返回值一律放于寄存器中,有如下规律:

returntypeRegistev说明

bit标志位由具体标志位返回

char/unsignedchar1_byte指针R7单字节由R7返回

int/unsignedint2_byte指针R6&R7双字节由R6和R7返回,MSB在R6

long&unsignedlong R4~R7MSB在R4,LSB在R7

floatR4~R732BitIEEE格式

一般指针R1~R3存储类型在R3高位R2低R1

(4)SRC控制

该控制指令将C文件编译生成汇编文件(.SRC),该汇编文件可改名后,生成汇编.ASM文件,再用A51进行编译。

3.第三节KeilC51软件包中的通用文件

在C51\LiB目录下有几个C源文件,这几个C源文件有非常重要的作用,对它们稍事修改,就可以用在自己的专用系统中。

1.1.动态内存分配

init_mem.C:

此文件是初始化动态内存区的程序源代码。

它可以指定动态内存的位置及大小,只有使用了init_mem()才可以调回其它函数,诸如malloccalloc,realloc等。

calloc.c:

此文件是给数组分配内存的源代码,它可以指定单位数据类型及该单元数目。

malloc.c:

此文件是malloc的源代码,分配一段固定大小的内存。

realloc.c:

此文件是realloc.c源代码,其功能是调整当前分配动态内存的大小。

2.2.C51启动文件STARTUP.A51

启动文件STARTUP.A51中包含目标板启动代码,可在每个project中加入这个文件,只要复位,则该文件立即执行,其功能包括:

l定义内部RAM大小、外部RAM大小、可重入堆栈位置

l清除内部、外部或者以此页为单元的外部存储器

l按存储模式初使化重入堆栈及堆栈指针

l初始化8051硬件堆栈指针

l向main()函数交权

开发人员可修改以下数据从而对系统初始化

常数名 意义

IDATALEN待清内部RAM长度

XDATASTART指定待清外部RAM起始地址

XDATALEN待清外部RAM长度

IBPSTACK是否小模式重入堆栈指针需初始化标志,1为需要。

缺省为0

IBPSTACKTOP指定小模式重入堆栈顶部地址

XBPSTACK是否大模式重入堆栈指针需初始化标志,缺省为0

XBPSTACKTOP指定大模式重入堆栈顶部地址

PBPSTACK是否Compact重入堆栈指针,需初始化标志,缺省为0

PBPSTACKTOP指定Compact模式重入堆栈顶部地址

PPAGEENABLEP2初始化允许开关

PPAGE 指定P2值

PDATASTART待清外部RAM页首址

PDATALEN待清外部RAM页长度

提示:

如果要初始化P2作为紧凑模式高端地址,必须:

PPAGEENAGLE=1,PPAGE为P2值,例如指定某页1000H-10FFH,则PPAGE=10H,而且连接时必须如下:

L51PDATA(1080H),其中1080H是1000H-10FFH中的任一个值。

以下是STARTUP.A51代码片断,红色是经常可能需要修改的地方:

;------------------------------------------------------------------------------

;ThisfileispartoftheC51Compilerpackage

;CopyrightKEILELEKTRONIKGmbH1990

;------------------------------------------------------------------------------

;STARTUP.A51:

Thiscodeisexecutedafterprocessorreset.

;

;TotranslatethisfileuseA51withthefollowinginvocation:

;

;A51STARTUP.A51

;

;TolinkthemodifiedSTARTUP.OBJfiletoyourapplicationusethefollowing

;L51invocation:

;

;L51,STARTUP.OBJ

;

;------------------------------------------------------------------------------

;

;User-definedPower-OnInitializationofMemory

;

;WiththefollowingEQUstatementstheinitializationofmemory

;atprocessorresetcanbedefined:

;

; ;theabsolutestart-addressofIDATAmemoryisalways0

IDATALENEQU80H;thelengthofIDATAmemoryinbytes.

;

XDATASTARTEQU0H;theabsolutestart-addressofXDATAmemory

XDATALENEQU0H;thelengthofXDATAmemoryinbytes.

;

PDATASTARTEQU0H;theabsolutestart-addressofPDATAmemory

PDATALENEQU0H;thelengthofPDATAmemoryinbytes.

;

;Notes:

TheIDATAspaceoverlapsphysicallytheDATAandBITareasofthe

;8051CPU.AtminimumthememoryspaceoccupiedfromtheC51

;run-timeroutinesmustbesettozero.

;------------------------------------------------------------------------------

;

;ReentrantStackInitilization

;

;ThefollowingEQUstatementsdefinethestackpointerforreentrant

;functionsandinitializedit:

;

;StackSpaceforreentrantfunctionsintheSMALLmodel.

IBPSTACKEQU0;setto1ifsmallreentrantisused.

IBPSTACKTOPEQU0FFH+1;settopofstacktohighestlocation+1.

;

;StackSpaceforreentrantfunctionsintheLARGEmodel.

XBPSTACKEQU0;setto1iflargereentrantisused.

XBPSTACKTOPEQU0FFFFH+1;settopofstacktohighestlocation+1.

;

;StackSpaceforreentrantfunctionsintheCOMPACTmodel.

PBPSTACKEQU0;setto1ifcompactreentrantisused.

PBPSTACKTOPEQU0FFFFH+1;settopofstacktohighestlocation+1.

;

;------------------------------------------------------------------------------

;

;PageDefinitionforUsingtheCompactModelwith64KBytexdataRAM

;

;ThefollowingEQUstatementsdefinethexdatapageusedforpdata

;variables.TheEQUPPAGEmustconformwiththePPAGEcontrolused

;inthelinkerinvocation.

;

PPAGEENABLEEQU0;setto1ifpdataobjectareused.

PPAGE EQU0;definePPAGEnumber.

;

;------------------------------------------------------------------------------

3.3.标准输入输出文件

putchar.c

putchar.c是一个低级字符输出子程,开发人员可修改后应用到自己的硬件系统上,例如向CLD或LEN输出字符。

缺省:

putchar.c是向串口输出一个字符XON|XOFF是流控标志,换行符“\*n”自动转化为回车/换行“\r\n”。

getkey.c

getkey函数是一个低级字符输入子程,该程序可用到自己硬件系统,如矩阵键盘输入中,缺省时通过串口输入字符。

4.4.其它文件

还包括对Watch-Dog有独特功能的INIT.A51函数以及对8×C751适用的函数,可参考源代码。

4.第四节段名协定与程序优化

1.1.段名协定(SegmentNamingConventions)

C51编译器生成的目标文件存放于许多段中,这些段是代码空间或数据空间的一些单元,一个段可以是可重定位的,也可以是绝对段,每一个可重定位的段都有一个类型和名字,C51段名有以下规定:

每个段名包括前缀与模块名两部分,前缀表示存储类型,模块名则是被编译的模块的名字,例如:

CO?

main1:

表示main1模块中的代码段中的常数部分

PR?

function1?

module表module模块中函数function1的可执行段,具体规定参阅手册。

2.2.程序优化

C51编译器是一个具有优化功能的编译器,它共提供六级优化功能。

确保生成目标代码的最高效率(代码最少,运行速度最快)。

具体六级优化的内容可参考帮助。

在C51中提供以下编译控制指令控制代码优化:

OPTIMIZE(SJXE):

尽量采用子程序,使程序代码减少。

NOAREGS:

不使用绝对寄存器访问,程序代码与寄存器段独立。

NOREGPARMS:

参数传递总是在局部数据段实现,程序代码与低版本C51兼容。

OPTIMIZE(SIZE)AKOPTIMIZE(speed)提供6级优化功能,缺省为:

OPTIMIZE(6,SPEED)。

5.第五章KeilC51库函数参考

C51强大功能及其高效率的重要体现之一在于其丰富的可直接

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 经管营销 > 金融投资

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1