在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

CAPL腳本使用介紹

汽車ECU開發 ? 來源:汽車ECU開發 ? 2024-04-01 11:23 ? 次閱讀

之前斷斷續續分享過一些Vector工具的使用總結,如下所示,包括CANape、CANoe、CANalyzer。

然而里面還有一項最重要的,也是平時使用過程中,提效很明顯的CAPL腳本的使用,覆蓋整車各節點的模擬、軟件刷寫、診斷測試等等,今天就來理一理CAPL腳本。 首先理一理基本語法和常用語句。在打開CAPl編輯界面時,下默認組成部分有:Include、Variable、System、Value Objects,其中Include為需要包含的已存在的頭文件,一般不配置;Variable為申明與定義全局變量,需要定義的變量包括需要發送的信號以及其數據類型。CAPL中的數據類型有: 無符號整型:byte(1字節),word(2字節),dword(4字節)

有符號整型:int(2字節),long(4字節)

浮點數:float(8字節),double(16字節)

CAN消息類型:Message;定時器類型:timer(單位為s),msTimer(單位為ms);

單個字符:char(1字節)。

除了界面基礎的信息外,在CAPL腳本中,我們大量使用官方定的的一些接口,這些接口通常需要查看help文檔或者是CAPL的手冊,下面是梳理的一些常用接口。

1、定時器

CAPL中的定時器的使用相當頻繁,比如測試時需要向定時發送某條CAN報文時就需要用定時器;定時器的聲明:

msTimer myTimer1;//聲明了一個ms定時器,定時單位是毫秒
timermyTimer2;//聲明了一個以秒為單位的定時器;

設置定時器:

setTimer(myTimer1,400);//設置定時器myTimer1為一個400ms定時器;
setTimerCyclic(myTimer2,2);//設置定時器myTimer2為一個2s為周期的循環定時器;

設置定時器定時事件,即當定時器計時時間到時將要執行的操作:

on timer myTimer1
{
.......
}

2、信息的操作和發送

//CAN消息發送:
message0x7ffMsg;//聲明一個message,ID=0x7ff
Msg.dlc=8;//設置其DLC=8;
Msg.id=0x100;//更改其ID=0x100;
Msg.byte(0)=55;//設置數據場的第一個字節為55
output(Msg);//發送Msg


//CANFD消息發送:
  msg1.FDF=1;
  msg1.BRS=1;
  msg1.dlc=8;
Msg.id=0x100;//更改其ID=0x100;
  msg1.byte(0)=0x44;
  msg1.byte(10)=0x10;
  msg1.byte(11)=0x11;
  output(Msg);//發送Msg

3、節點上下線操作

節點是在dbc中定義的,如VCU,BMS,MCU等,有時需要將它們離線,離線后不再向總線上發送報文,在線時可以向總線上發送報文。

節點上線:

voidtestSetEcuOnline(dbNodeaNode);
void testSetEcuOnline(char aNodeName[]);

節點下線:

voidtestSetEcuOffline(dbNodeaNode);
void testSetEcuOffline(char aNodeName[]);

4、檢查錯誤幀

進行CAN通訊的測試時,檢查錯誤幀是很常見的,要用CAPL腳本實現自動檢測錯誤幀也不困難,它的核心就是調用錯誤檢查函數ChkStart_ErrorFrameOccured(),該函數一旦被調用,CANoe就會從此函數被調用時開始持續檢測總線上有沒有出現錯誤幀。

下面是一個小的例子

dword chechId;
dword numCheckEvents;
checkId=ChkStart_ErrorFrameOccured();//開始檢測錯誤幀
TestAddCondition(checkId);//添加檢測條件,如果出現了錯誤幀,則輸出報告中會記錄下來
TestWaitForTimeout(5000);//持續檢測5s
checkControl_Stop(checkId);//停止檢測錯誤幀
numCheckEvents=ChkQuery_NumEvents(checkId);//對5s內的檢測情況進行獲取,若函數返回0則沒有出現錯誤幀
if(numCheckEvents>0)
     TestStepFail("Error Frames Occured");

5、添加事件信號

這種事件信號相當于信號量機制,一般使用在需要等待某個或者是多個條件滿足時進行下一步操作。

具體做法是:在一個位置添加需要等待的事件,程序中的其他地方,如果某個事件發生了(如周期超界等),提供該事件的供應,則等待的程序段獲得了該事件,繼續執行下面的操作。主要使用的函數有以下幾個:

//供應text事件
long TestSupplyTextEvent( char aText[] );
//添加text事件
long TestJoinTextEvent(char[]aText);
//等待text事件,有一個出現則程序執行下一步
long TestWaitForAnyJoinedEvent(dword aTimeout);
//等待text事件,所有等待事件都出現則程序執行下一步
long TestWaitForAllJoinedEvents(dword aTimeout);

以下是一個例子:

TestJoinTextEvent("Test finished");
TestJoinTextEvent("Error Frame Occured");
TestWaitForAnyJoinedEvents(20000);
或者:
TestWaitForAllJoinedEvents(20000);
在系統事件on errorFrame中:
on errorFrame
{
   TestSupplyTextEvent("Error Frame occured");
}
在系統的on message 中:
on message 0x400
{
   TestSupplyTextEvent("Test Finished")
}

6、回調函數

CAPL中也有類似于C語言中的回調函數的機制,如檢測報文周期和錯誤幀的函數中就可以使用,當周期超界或者總線出現錯誤幀就會自動調用回調函數執行一些操作;如:

ErrChkId=ChkStart_ErrorFramesOccured("Callback_ErrorFrameOccured");//檢查錯誤幀,如果發現錯誤幀就調用回調函數
回調函數設計如下:
void Callback_errorFrameOccured(dword chk_id)
{
  float t;
  t=timeNow()/100000.0;//記錄出現錯誤幀的時間
  testStep("ErrorFrameTimeStamp","%.6f s",t);//打印該事件戳
  TestSupplyTextEvent("ErrorFrameOccured");//供應Text事件
}

7、監視總線的情況,這一般會用在查看一段時間內,總線上有沒有出現通訊異常的情況。需要使用函數ChkStart_NodeBabbling( ). 如,檢測一段時間內總線有沒有出現停止通訊的情況:

CheckId=ChkStart_NodeBabbling(CAN::PT_MCU,0);//立即開始檢查總線狀態
testWaitForTimeout(2000);//延時2s
ChkControl_Stop(CheckId);//停止檢測
QueryNumberEvents=ChkQuery_NumEvents(CheckId);//如果在2s內總線停止通訊,則QueryNumberEvents!=0

8、關于獲取關鍵時間點

(1)CANoe中獲取定時器當前計時值的函數為:timerToElapse();該函數原型如下:

long timerToElapse(timer);
long timerToElpase(msTimer);                

(2)獲取等待某個事件的時間,需要使用函數TestGetLastWaitElapsedTimeNS(),其原型如下:

float TestGetLastWaitElapsedTimeNS();

(3)獲取當前的仿真時間點:

float timeNowFloat();

(4)等待指定報文:

long TestWaitForMessage(dbMessage aMessage,dword aTimeout);
long TestWaitForMessage(dword aMessageId,dword aTimeout);

若在aTimeout時間內等到了指定ID的報文,函數返回1,否則返回0;

(5)獲取報文的數據,等到了報文之后,如果想知道報文的具體內容可以使用函數:

message msg;
long result;
result=TestGetWaitEventMsgData(msg);
.....處理msg.....

9、多總線測試

設置總線背景,一般都總線測試都會有兩路及以上的CAN,這時若要通過CAPL腳本獲取某個CAN通道上的報文時,就需要先設置好總線背景,即將總線設置為值監聽某一路的CAN通道。下面是一個例子:

void BusContextConfiguration(char yBus[])
{
   yBusContext=GetBusNameContext(yBus);//這里的yBusContext為全局變量
   SetBusContext(yBusContext);
}
//使用:
BusContextConfiguration("CAN1");//將總線監聽設為CAN1

此時等待某一路的CAN報文可是這樣實現:

res=testWaitForMessage(CAN1::NM_IPU,600);//等待CAN1上的名稱為NM_IPU的報文,等待事件為600ms

10、診斷報文的發送和接收

request_A.SendRequest();//診斷請求
TestWaitForDiagResponse(request_A,5000);//診斷接收

11、

將診斷請求 / 響應寫入報告

TestReportWriteDiagObject (diagRequest req);


TestReportWriteDiagObject (diagResponse resp);


TestReportWriteDiagResponse (diagRequest req);

12、獲取診斷請求 / 響應的原始數據

long diagGetPrimitiveByte( diagRequest request, DWORD bytePos);


long diagGetPrimitiveByte( diagResponse response, DWORD bytePos);

13、獲取診斷請求 / 響應的參數

long diagGetParameter (diagResponse obj, char parameterName[], double output[1])


long diagGetParameter (diagRequest obj, char parameterName[], double output [1])


double diagGetParameter (diagResponse obj, char parameterName[])


double DiagGetParameter (diagRequest obj, char parameterName[])


long diagGetParameter (diagResponse obj, long mode, char parameterName[], double output[1])


long DiagGetParameter (diagRequest obj, long mode, char parameterName[], double output [1])


double diagGetParameter (diagResponse obj, long mode, char parameterName[])


double diagGetParameter (diagRequest obj, long mode, char parameterName[])

最后分享最近剛使用CAPL腳本的一些注意點以及一個示例

第一CAPL 的局部變量是靜態局部變量。經過使用發現,在 variables{ }之外,事件或者函數內部定義的局部變量是靜態局部變量,其值不會因為退出本事件或者函數,而變為初始值。所以如果真的需要一個局部變量,在每次退出之前,重新使用賦值語句賦為初始值。

第二建議使用系統變量或者環境變量,這樣可以跨不同的capl腳本操作,比如檢測某個環境變量或者系統變量的變化,來執行一些動作。

on sysvar sysvar::EngineStateSwitch
{
  $EngineState::OnOff = @this;
if(@this)
  $EngineState::EngineSpeed = @sysvar::Engine::EngineSpeedEntry;
else
  $EngineState::EngineSpeed = 0;
}

第三個就是以太網轉CAN的capl腳本示例:

/*@!Encoding:1252*/


variables
{
  //
  // Constants
  //
  
  const WORD kPort         = 23; // UDP port number for instance
  const WORD kRxBufferSize = 1500;
  const WORD kTxBufferSize = 1500;
  
  //
  // Structure of UDP payload
  //
  
  _align(1) struct CANData
  {
    BYTE  dlc;
    BYTE  flags; // Bit 7 - Frame type (0 = standard, 1 = extended)
                 // Bit 6 - RTR bit ('1' = RTR bit is set)
    DWORD canId;
    BYTE  canData[8];
  };
  
  //
  // Global variables
  //
  
  UdpSocket gSocket;
  CHAR      gRxBuffer[kRxBufferSize];
  CHAR      gTxBuffer[kTxBufferSize];
  DWORD     gOwnAddress;
  DWORD     gModuleAddress= 0xFFFFFFFF; // default is the broadcast address 255.255.255.255  and the TCP/IP stack will build the Network broadcast address
}


//
// Measurement start handler
//


on start
{
  DWORD addresses[1];
  
  // get own IP address of the Windows TCP/IP stack
  IpGetAdapterAddress( 1, addresses, elcount(addresses) );
  gOwnAddress = addresses[0];
  
  // open UDP socket
  gSocket = UdpSocket::Open( 0, kPort ); 
  
  if (gSocket.GetLastSocketError() != 0)
  {
    write( "<%BASE_FILE_NAME%> Open UDP socket failed, result %d. Measurement stopped!", gSocket.GetLastSocketError() );
    stop();
    return;
  }


  if (gSocket.ReceiveFrom( gRxBuffer, elcount(gRxBuffer) ) != 0)
  {
    if (gSocket.GetLastSocketError() != 997) // ignore pending IO operation
    {
      write( "<%BASE_FILE_NAME%> UDPReceive failed, result %d. Measurement stopped!", gSocket.GetLastSocketError() );
      stop();
      return;
    }
  }


}
//
// On receive UDP data handler using CAPL Callback 
//
void OnUdpReceiveFrom( dword socket, long result, dword address, dword port, char buffer[], dword size)
{
  DWORD          dataOffset;
  struct CANData canData;
  message *      canMsg;
  
if(address==gOwnAddress)return;//ignoreownbroadcasts
  //
  // Store IP address of module to reach
  //
  
  if (gModuleAddress == 0)
  {
    gModuleAddress = address;
  }
  
  //
  // Handle received data
  //
  
  dataOffset = 0;
  while (dataOffset + __size_of(struct CANData) <= size)
  {
    memcpy( canData, buffer, dataOffset );
    
    canMsg.id      = (canData.canId & 0x1FFFFFFF) | ((canData.flags & 0x80) ? 0x80000000 : 0);
    canMsg.dlc     = canData.dlc & 0x0f;
    canMsg.rtr     = ((canData.flags & 0x40) ? 1 : 0);
    canMsg.byte(0) = canData.canData[0];
    canMsg.byte(1) = canData.canData[1];
    canMsg.byte(2) = canData.canData[2];
    canMsg.byte(3) = canData.canData[3];
    canMsg.byte(4) = canData.canData[4];
    canMsg.byte(5) = canData.canData[5];
    canMsg.byte(6) = canData.canData[6];
    canMsg.byte(7) = canData.canData[7];
    
    output( canMsg );
   
    dataOffset += __size_of(struct CANData);
  }
  


  //
  // Receive more data
  //
  if (gSocket.ReceiveFrom( gRxBuffer, elcount(gRxBuffer) ) != 0)
  {
    if (gSocket.GetLastSocketError() != 997) // ignore pending IO operation
    {
      write( "<%BASE_FILE_NAME%> UDPReceive failed, result %d. Measurement stopped!", gSocket.GetLastSocketError() );
      stop();
      return;
    }
  }
}


//
// Handler for CAN messages
//


on message *
{
  int i;
  struct CANData canData;
  
  if ((this.dir == RX) && (gModuleAddress != 0))
  {
    canData.canId = this.id & 0x1FFFFFFF;
    canData.flags = ((this.id & 0x80000000) ? 0x80 : 0x00) | ((this.rtr == 1) ? 0x40 : 0x00);
    canData.dlc   = this.dlc;
    
    for( i = 0; i < 8; i++ )
    {
      canData.canData[i] = (i < this.dlc) ? this.byte(i) : 0;
    }
    
    memcpy( gTxBuffer, canData );
    
    gSocket.SendTo( gModuleAddress, kPort, gTxBuffer, __size_of(struct CANData) );
  }
  else if (gModuleAddress == 0)
  {
    write( "<%BASE_FILE_NAME%> Tx not possible. Module to reach must send packets first." ); //Server simulation
  }
}
審核編輯:黃飛

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • C語言
    +關注

    關注

    180

    文章

    7608

    瀏覽量

    137140
  • 函數
    +關注

    關注

    3

    文章

    4338

    瀏覽量

    62752
  • 腳本
    +關注

    關注

    1

    文章

    391

    瀏覽量

    14892

原文標題:CAPL腳本使用介紹

文章出處:【微信號:eng2mot,微信公眾號:汽車ECU開發】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    ProF腳本指令介紹及解決方案

    INCA是車輛控制器標定的首選軟件之一,主要包含功能模型標定、測量數據分析(MDA)、標定數據管理(CDM)、控制器刷新(ProF)功能等。而本文將介紹常用卻又陌生的ProF腳本的擴展用法,通過編寫
    的頭像 發表于 10-08 11:13 ?4842次閱讀

    CANoe編寫CAPL測試腳本的幾點思考

    測試腳本的開發人員,需要考慮到測試執行者測試不同控制器時的參數配置。比如不同的網絡喚醒條件、不同的網絡管理消息、不同的時間參數等等。
    的頭像 發表于 01-02 10:42 ?2153次閱讀
    CANoe編寫<b class='flag-5'>CAPL</b>測試<b class='flag-5'>腳本</b>的幾點思考

    CAPL介紹-腳本編輯和常用基本事件#CANoe#CAPL#腳本

    編程語言
    北匯信息POLELINK
    發布于 :2023年01月06日 09:06:50

    CAPL編程語言 官方英文原版

    之前看過的一份CAPL編程語言資料,英文原版,分享給需要的小伙伴!!!
    發表于 01-14 18:19

    Lua腳本簡單介紹

    Lua簡單介紹Lua[1]是一個小巧的腳本語言。作者是巴西人。該語言的設計目的是為了嵌入應用程序中,從而為應用程序提供靈活的擴展和定制功能。Lua腳本能夠非常easy的被C/C++ 代碼調用,也能夠
    發表于 08-20 06:37

    Lua腳本簡單介紹

    Lua簡單介紹Lua[1]是一個小巧的腳本語言。作者是巴西人。該語言的設計目的是為了嵌入應用程序中,從而為應用程序提供靈活的擴展和定制功能。Lua腳本能夠非常easy的被C/C++ 代碼調用,也能夠
    發表于 08-20 08:06

    如何用CAPL通過RS232遠程控制ALR3220

    CAPL通過RS232遠程控制ALR3220程控電源
    發表于 12-28 06:37

    什么是腳本

    什么是腳本腳本是什么意思,腳本錯誤是什么意思電子發燒友深入為大家講解了腳本相關知識
    發表于 12-07 10:36 ?2825次閱讀

    什么是腳本腳本程序學習

    腳本中編寫VB腳本代碼。可以象使用系統函數一樣使用項目中完成的腳本。創建腳本時,確定其型號并定義傳送參數。“Function”類型的腳本
    的頭像 發表于 05-11 10:39 ?6799次閱讀
    什么是<b class='flag-5'>腳本</b>?<b class='flag-5'>腳本</b>程序學習

    CAPL通過RS232遠程控制ALR3220程控電源

    CAPL通過RS232遠程控制ALR3220程控電源
    發表于 01-05 15:10 ?20次下載
    用<b class='flag-5'>CAPL</b>通過RS232遠程控制ALR3220程控電源

    什么是CAPL編程?

    與Vspy的"C Code Interface"一樣;在CANoe的使用中,一樣提供了我們進行二次編程開發的工具——”CAPL Browser”。
    的頭像 發表于 06-18 10:13 ?2781次閱讀
    什么是<b class='flag-5'>CAPL</b>編程?

    CAPL編程語言快速入門

    CAPL是由Vector公司開發的類似于C語言的面向過程編程語言,是CANoe和CANalyzer中可用的編程語言。CAPL中程序塊的執行由事件控制,在專用的編譯器中開發和編譯,這樣可以訪問數據庫中
    的頭像 發表于 09-17 16:11 ?2760次閱讀
    <b class='flag-5'>CAPL</b>編程語言快速入門

    Linux主機排查腳本介紹

    介紹 HScan,本腳本旨在為安全應急響應人員對Linux主機排查,日志分析等提供便利,定制化在主機中執行命令 獲取腳本 git clone?https://github.com/HZzz2
    的頭像 發表于 06-28 09:44 ?570次閱讀
    Linux主機排查<b class='flag-5'>腳本</b><b class='flag-5'>介紹</b>

    ?CAPL在診斷中的應用,你值得了解!

    的過程中相信每位工程師都或多或少的要和“CAPL”打交道。學好CAPL的用法可以讓我們更加高效、便捷地使用CANoe。本文就CANoe中關于診斷的CAPL函數進行介紹
    的頭像 發表于 09-07 08:27 ?1059次閱讀
    ?<b class='flag-5'>CAPL</b>在診斷中的應用,你值得了解!

    Shell腳本檢查工具ShellCheck介紹

    ShellCheck是一個用于bash/sh shell腳本的靜態分析工具,可以輔助檢查腳本語法錯誤,給出建議增強腳本健壯性。
    的頭像 發表于 12-27 13:43 ?2210次閱讀
    Shell<b class='flag-5'>腳本</b>檢查工具ShellCheck<b class='flag-5'>介紹</b>
    主站蜘蛛池模板: 97干干| 免费国产不卡午夜福在线观看| 精品视频日本| 国产亚洲第一伦理第一区| 国产免费成人在线视频| 91黄色视屏| 日本高清免费aaaaa大片视频| 人人插人人| 欧美奇米| 亚洲乱码尤物193yw在线播放| 一区二区免费播放| 被男同桌摸内裤好爽视频| 国产黄色小视频在线观看| 一区二区三区网站| 99热都是精品| 色视频在线看| 极品美女写真菠萝蜜视频| 免费能直接在线观看黄的视频| 国产在线五月综合婷婷| 狠狠色影院| 一级高清| 国产午夜影院| 亚洲先锋资源| 性高清| 性欧美精品久久久久久久| 国产热视频| 久久精品国产99国产精品免费看| 亚洲天堂电影在线观看| 亚洲国产综合久久精品| 亚洲丁香| 欧美日韩看片| bt 另类 专区 欧美 制服| 狼人射综合| 亚欧精品一区二区三区| 免费播放一区二区三区| 婷婷久久久五月综合色| 色天使久久综合网天天| 国产激情电影综合在线看| 欧美午夜视频一区二区| 亚洲男人的天堂在线观看| 99久久精品国产免费|