i C51指针与A51汇编接口之间关系研究-嵌入式MCU技术-电子人社区 -

电子人社区

 找回密码
 立即注册
今日 : 0|主题 : 4177|排名 : 314 
打印 上一主题 下一主题

C51指针与A51汇编接口之间关系研究

发表于 2019-1-13 14:18:49 | 93910 只看该作者 回帖奖励 |倒序浏览 |阅读模式

[复制链接]
发表于 2019-1-13 14:18:49 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
电子人社区网讯:   最近在研究单片机C51对汇编的接口问题。char和int等都比较简单,使用寄存器或固定地地址传值都是可以的,具体可以参考keil的C51 user‘s guide。本篇短文主要重点讨论一下A51下如何遵循C51的接口标准来实现C51的指针。主要原因是,现在用C51的人越来越多,大家都图省事和方便。网上面有关A51的资料少得可怜,知道用汇编来实现代码优化的少之又少。本人是一直坚持用汇编写东西的。在嵌入式领域,很多东西都与硬件有关,多知道点底层东西还是有好处。
  本文引用地址: http://www.21ic.com/app/mcu/201812/781548.htm
  使用工具主要为keil,在windows 7环境下。C51测试程序如下:
  #include
  unsignedpos;
  bitabc(unsigned*pos){
  *pos=100;
  return0;
  }
  voidmain(){
  while(1){
  abc(&pos);}
  这是测试用源程序,很明显程序中使用了int型指针 pos。为了能够生成汇编代码,加入编译器控制:
  #pragmaSRC
  #pragmaSMALL
  #include
  unsignedpos;
  bitabc(unsigned*pos){
  *pos=100;
  return0;
  }
  voidmain(){
  while(1){
  abc(&pos);
  }
  }
  注意,#pragma必需在文件的开始位置,否则会报错。另外编译器可能会跳“ EXCEPTION 0021H: PATH OR FILE NOT FOUND:main.obj”,没有关系,我们的目标是生成汇编的src文件。编译后生成src如下:
  ;main.SRCgeneratedfrom:main.c
  ;COMPILERINVOKEDBY:
  ;D:ProgramFilesKeilC51BINC51.EXEmain.cBROWSEDEBUGOBJECTEXTENDTABS(2)
  $NOMOD51
  NAMEMAIN
  ;寄存器及内存声明
  P0DATA080H
  P1DATA090H
  P2DATA0A0H
  P3DATA0B0H
  T0BIT0B0H.4
  ACBIT0D0H.6
  T1BIT0B0H.5
  T2BIT090H.0
  EABIT0A8H.7
  IEDATA0A8H
  EXF2BIT0C8H.6
  RDBIT0B0H.7
  ESBIT0A8H.4
  IPDATA0B8H
  RIBIT098H.0
  INT0BIT0B0H.2
  CYBIT0D0H.7
  TIBIT098H.1
  INT1BIT0B0H.3
  RCAP2HDATA0CBH
  PSBIT0B8H.4
  SPDATA081H
  T2EXBIT090H.1
  OVBIT0D0H.2
  RCAP2LDATA0CAH
  C_T2BIT0C8H.1
  WRBIT0B0H.6
  RCLKBIT0C8H.5
  TCLKBIT0C8H.4
  SBUFDATA099H
  PCONDATA087H
  SCONDATA098H
  TMODDATA089H
  TCONDATA088H
  IE0BIT088H.1
  IE1BIT088H.3
  BDATA0F0H
  CP_RL2BIT0C8H.0
  ACCDATA0E0H
  ET0BIT0A8H.1
  ET1BIT0A8H.3
  TF0BIT088H.5
  ET2BIT0A8H.5
  TF1BIT088H.7
  TF2BIT0C8H.7
  RB8BIT098H.2
  TH0DATA08CH
  EX0BIT0A8H.0
  IT0BIT088H.0
  TH1DATA08DH
  TB8BIT098H.3
  EX1BIT0A8H.2
  IT1BIT088H.2
  TH2DATA0CDH
  PBIT0D0H.0
  SM0BIT098H.7
  TL0DATA08AH
  SM1BIT098H.6
  TL1DATA08BH
  SM2BIT098H.5
  TL2DATA0CCH
  PT0BIT0B8H.1
  PT1BIT0B8H.3
  RS0BIT0D0H.3
  PT2BIT0B8H.5
  TR0BIT088H.4
  RS1BIT0D0H.4
  TR1BIT088H.6
  TR2BIT0C8H.2
  PX0BIT0B8H.0
  PX1BIT0B8H.2
  DPHDATA083H
  DPLDATA082H
  EXEN2BIT0C8H.3
  RENBIT098H.4
  T2CONDATA0C8H
  RXDBIT0B0H.0
  TXDBIT0B0H.1
  F0BIT0D0H.5
  PSWDATA0D0H
  ?PR?_abc?MAINSEGMENTCODE
  ?PR?main?MAINSEGMENTCODE
  ?DT?MAINSEGMENTDATA
  EXTRNCODE(?C?ISTPTR)
  EXTRNCODE(?C_STARTUP)
  PUBLICpos
  PUBLICmain
  PUBLIC_abc
  RSEG?DT?MAIN
  pos:DS2
  ;#pragmaSRC
  ;#pragmaSMALL
  ;#include
  ;
  ;unsignedpos;
  ;
  ;bitabc(unsigned*pos){
  RSEG?PR?_abc?MAIN
  _abc:
  ;SOURCELINE#7
  ;----Variable'pos?040'assignedtoRegister'R1/R2/R3'----
  ;*pos=100;
  ;SOURCELINE#8
  ;---------关键代码-------------------
  CLRA
  MOVB,#064H
  LCALL?C?ISTPTR;return0;
  ;SOURCELINE#9
  CLRC;}
  ;SOURCELINE#10
  ?C0001:RET
  ;ENDOF_abc;
  ;voidmain(){
  RSEG?PR?main?MAIN
  main:
  USING0;SOURCELINE#12
  ?C0002:;while(1){
  ;SOURCELINE#13
  ;abc(&pos);
  ;SOURCELINE#14
  ;---------关键代码-------------------
  MOVR3,#00H
  MOVR2,#HIGH(pos)
  MOVR1,#LOW(pos)
  LCALL_abc;}
  ;SOURCELINE#15
  SJMP?C0002
  ;ENDOFmainEND
  对于A51的程序格式,这里不多解释,有兴趣可以自己去Keil官网学习。这里主要说一下指针的C51汇编接口。代码中内存和寄存器声明,这个不重要,可以跳过。主要看标记为关键代码的部分。我们可以看到,在main中,在调用abc函数前,主程序初始了三个寄存器,分别是R1R2R3,代码如下:
  MOVR3,#00H
  MOVR2,#HIGH(pos)
  MOVR1,#LOW(pos)
  这里得着重讲一下。很明显,这是寄存器传值。依据C51规定,指针传递使用R1R2R3寄存器。其中,Mem type in R3, MSB in R2, LSB in R1。也就是说,R3表指针类型,R1为指针内容低位,R2为内容高位。从上面三个语句可以看出,程序将类型设为0,将pos地址高位传给R2,低位给R1。接下来再看abc函数中的操作
  CLR      A

来源:网络转载

 

                                                   转载请注明:电子人社区

0条回复
跳转到指定楼层

发表回复

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|电子人物联网  

Copyright 2013 最新最精彩-社区论坛 版权所有 All Rights Reserved.

QQ|电子人物联网  

GMT+8, 2024-5-2 02:43 , Processed in 0.104687 second(s), 31 queries.

快速回复 返回顶部 返回列表