CANOE CAPL编程 HEX文件读取

 最近在学习CAPL编程,简单的操作基本掌握了。其实CAPL就是C语言的简化版本,少了挺多特性的。刚开始上手有点不太懂,后面看了几个别人的实例后就手痒痒想动手做点什么。刚好在工作上遇到一些问题,想要用CAPL编写一个UDS刷写器。但是网上只能找到S19文件读写的例子。而我们的固件都是HEX文件的。那么我就动手做了一个HEX文件解析器。

首先用Excel打开一个hex文件,通过分列很容易就能得到我的HEX文件里面只有00,01,04三种码。那么我们在解析的时候只需要对这三种不同状态的进行不同的解析即可。

接下来就是对进行编程了,首先创建一个Block的结构体,根据经验我们知道,一个数据块有:开始的地址,数据的长度,以及数据Buffer。三个结果。而一个hex文件里面可能有多个数据块,因此我们接下来声明了5个数据块。这个需要根据变量进行调整。然后根据hex文件初始化数据块的个数。再声明变量 HexBlockTotalNumber 为数据块的总数。

variables
{
  struct Block {
  dword   BlockStartAddr;       //数据开始的地址
  dword BlockDataLength;        //数据的长度
  byte dataBuffer[0x020FFFF];   //数据区域(单块数据的Buffer,如果hex文件很大,则需要把次参数调大)
  };  
  struct Block hexfile[5];      //创建5个数据块(主要根据HEX文件进行调整,hex有几个数据块就需要设置几个)
  int HexBlockTotalNumber = 0;  //数据块总数
  dword t1;                     //计算解析所用的时间
}


/*********************************************************************************
  *Function:  //char2byte
  * Description:  //把单个字符转换为Byte的函数
  *Input:  //ch:ASCII编码字符,取值为0到F
  *Return:  //val,为byte类型
**********************************************************************************/
byte char2byte(char ch)
{
   byte  val;
   val = 0;
   if ( ch >= '0' && ch <= '9')
   {
      val = ch - '0';      
   }
   if ( ch >= 'a' && ch <= 'f')
   {
      val = (ch - 'a') + 10;      
   }
   if ( ch >= 'A' && ch <= 'F')
   {
      val = (ch - 'A') + 10;       
   }
   return val;
}

/*********************************************************************************
  *Function:  //Read_hexFile
  * Description:  //解码HEX文件,只支持0x00,0x04,0x01类型
  *Input:  //Filename:需要解码的文件名
  *Output:  //hexfile
  *Return:  //void
**********************************************************************************/
//读取HEXFILE
void Read_hexFile(char Filename[])
{
  
  long file_handle;
  char RowData[128];                        //逐行读取,每行数据缓存,当每行数据大于128时,需要将其调整
	dword   i;
	dword   RowDataByte;											//单块数据块字节数
	qword   OffsetAddress;										//扩展线性地址 	
	qword   ReAddr;												//上一数据行起始地址 
	dword   Len;													//HEX每行有效数据字节数
	dword   ReLen;												//HEX前一次数据长度
	dword   Addr;													  //HEX每行起始地址
	dword    Type;		                    //HEX每行类型,有00,01,04四种类型
  RowDataByte = 0;i = 0;Len = 0;ReLen = 0;Addr=0;Type = 0;ReAddr = 0;
  file_handle = OpenFileRead(Filename,0);
  HexBlockTotalNumber = 0;
  if(file_handle!=0)
  { // Read all lines
    while ( fileGetStringSZ(RowData,elcount(RowData),file_handle)!=0 ){
      //判断首字符是否为:号
      if(RowData[0] == ':'){
        Len = (char2byte(RowData[1])*0x10+char2byte(RowData[2]));
        Addr = char2byte(RowData[3])*0x1000+char2byte(RowData[4])*0x100+char2byte(RowData[5])*0x10+char2byte(RowData[6]);
        Addr |= (OffsetAddress << 16);
        Type = char2byte(RowData[7])*0x10+char2byte(RowData[8]);
        //以下为打印解析的过程,打印解析时候的变量
        //write("RowData:%s,HexBlockTotalNumber:%d,ReLen:%X,ReAddr:%X,Addr:%X,RowDataByte:%X",RowData,HexBlockTotalNumber,ReLen,ReAddr,Addr,RowDataByte);
        switch(Type){
          case 0x00:											  //数据
            if (Addr  > (ReLen + ReAddr)){  //判断为新数据块
                if(RowDataByte == 0)			  //是否为首行数据字节数
  							{ 
  								hexfile[HexBlockTotalNumber].BlockStartAddr = Addr;         //记录新数据块的起始地址
  							}
  							else										    //不是首行
  							{
  								hexfile[HexBlockTotalNumber].BlockDataLength = RowDataByte; //数据长度  
  								RowDataByte = 0;					//重新开始计数
  								hexfile[HexBlockTotalNumber].BlockStartAddr = Addr;		      //记录新数据块的起始地址
                  
                  HexBlockTotalNumber++; 
  							}
            }
             for(i = 0; i< Len ; i++)
                {
                  //储存buffer,注意没有对crc进行校验。
                  hexfile[HexBlockTotalNumber].dataBuffer[RowDataByte++]=(char2byte(RowData[2*i+9])*0x10+char2byte(RowData[2*i+10]));
                }  
            ReAddr = Addr;									//保存当前地址,下一次使用  
						ReLen  = Len;									  //保存当前长度,下一次使用   
           break;
          case 0x04:										   	//扩展线性地址记录
						OffsetAddress =  char2byte(RowData[9])*0x1000+char2byte(RowData[10])*0x100+char2byte(RowData[11])*0x10+char2byte(RowData[12]);		//偏移地址	 
					break;
          case 0x01:										   	//地址,结束 
           hexfile[HexBlockTotalNumber].BlockDataLength = RowDataByte;        //数据长度
           HexBlockTotalNumber++;
          break;          
        }        
      }

    }
  write("Hex文件读取成功, 数据分块:%d",HexBlockTotalNumber);
  for(i = 0; i < HexBlockTotalNumber; i++)
		{
			write("数据块:%d,  起始地址:0x%X, 结束地址:0x%X, 数据长度:%6d字节\r\n", i+1, hexfile[i].BlockStartAddr, hexfile[i].BlockStartAddr + hexfile[i].BlockDataLength - 1, hexfile[i].BlockDataLength);
		}
  fileClose(file_handle);
  }
  else{
     write("OpenFileRead,error occurs");
  }

}
on key 'f'
{
  t1 = timeNow();
  Read_hexFile("test.hex");
  write("%d",t1-timeNow());
  write(hexfile[0].dataBuffer);
}

write (byte v[]) {
   int i;
   for (i=0; i<48; i++) write("dataBuffer:%x",hexfile[0].dataBuffer[i]);
}

 

  • 23
    点赞
  • 158
    收藏
    觉得还不错? 一键收藏
  • 21
    评论
### 回答1: Canoe CAPL编程实例包括很多方面,例如网络通讯、数据传输、故障诊断和测试等。其中有一些比较典型的例子如下: 1、网络通讯实例: 在Canoe CAPL编程中,可以使用CAPL库函数对网络进行通讯。通过发送CAN消息可以在网络中进行数据的传输。例如,在测试中,我们可以模拟发送一条CAN消息,并且在接收到回传消息后进行判断,可以通过判断结果来进行测试。 2、数据传输实例: Canoe CAPL编程可以方便地进行数据的传输。例如,可以对多个CAN信号进行检测和处理,可以通过处理之后得到特定的数据内容。此外,还可以进行多种格式的数据处理,例如对BCD码、二进制码、十六进制码等进行解析。 3、故障诊断实例: Canoe CAPL编程可以实现对网络故障的诊断,通过在网络中发送和接收命令,可以进行网络的测试和检测,确保网络的健康。例如,在接收数据时,如果出现错误信息,则可以判断出网络出现了故障。 4、测试实例: Canoe CAPL编程也可以进行测试。例如,在汽车制造中,可以验证CAN网络的正确性,对CAN总线上的数据进行检查,以确保汽车的安全性。另外,还可以模拟传感器、控制单元等,以检测其他部件的正确性。 总之,Canoe CAPL编程实例是非常广泛的,包括了很多方面的内容,通过对这些实例进行学习和应用,可以达到更好地进行数据处理、网络通讯、故障诊断和测试的目的。 ### 回答2: Canoe CAPL编程是面向嵌入式系统的一种高级编程语言,能够帮助开发者进行控制单元的软件开发、测试和调试等工作。下面就介绍一下Canoe CAPL编程的实例。 我们可以通过Canoe CAPL编程来控制汽车中的某些功能模块,例如车门、车窗、车灯、音响等。可以编写一系列的函数,通过向不同的功能模块发送信号,来控制它们的状态。比如,我们可以定义一个函数,来控制车门的状态。通过编写如下代码: on key 'F4' { setDoorState(1); //打开车门 } on key 'F5' { setDoorState(0); //关闭车门 } void setDoorState(int state) { if(state == 1) { setSignal(DOOR_SIGNAL, 1); //发送打开车门的信号 } else { setSignal(DOOR_SIGNAL, 0); //发送关闭车门的信号 } } 在这个代码中,我们首先定义了两个按键,用来控制车门的状态。当用户按下F4键时,调用setDoorState函数,将车门打开;当用户按下F5键时,调用setDoorState函数,将车门关闭。setDoorState函数根据传入的状态参数,向DOOR_SIGNAL信号发送对应的数值(1为打开车门,0为关闭车门),从而实现控制车门状态的功能。 另外,Canoe CAPL编程还可以用于信号的模拟和调试。在开发过程中,我们可以通过编写一些模拟代码,来模拟接收到不同数据信号的情况,并进行相关的处理。这样就可以将信号的处理过程模拟出来,提前发现并解决潜在的问题。 总之,Canoe CAPL编程可以帮助汽车系统开发者快速地开发出控制单元的软件,并进行测试和调试,是一种非常实用的编程语言。 ### 回答3: Canoe CAPL编程实例是指用Vector公司的CANoe工具编写以CAPL语言为基础的程序来控制CANape的功能。 CANoe是一款用于开发、测试和分析基于CAN总线的系统的软件工具,支持多种总线,如CAN、LIN和FlexRay等。CAPLCANoe中一种专门用于CAN总线开发的高级编程语言,广泛应用于CAN总线控制程序的编写和测试。 一个典型的Canoe CAPL编程实例是用CAPL语言编写一个简单的CAN总线消息接收程序。该程序可以接收从CAN总线上发送过来的CAN消息,并将其显示在CANoe的消息窗口中。 这个程序的实现步骤如下: 1.定义变量:定义一个变量来保存接收到的CAN总线消息。 2.定义事件:通过事件触发来处理CAN总线消息。 3.创建回调函数:定义一个回调函数,用来处理CAN总线上接收到的消息,并将其保存到变量中。 4.初始化事件:初始化事件,以便接收CAN总线上的信息,然后在事件处理程序中调用回调函数。 5.启动测试:启动CANoe测试后,该程序就可以在CAN总线上接收消息并将其显示在CANoe的消息窗口中了。 总之,Canoe CAPL编程实例是一个重要的工具,可以帮助工程师们更好地开发、测试和分析CAN总线系统,从而提高生产效率和质量。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 21
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值