RM新时代网站-首页

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

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

3天內不再提示

【產品應用】用 AWTK 和 AWPLC 快速開發(fā)嵌入式應用程序 (5)- 自定義功能塊(下)

ZLG致遠電子 ? 2022-11-04 10:12 ? 次閱讀

AWPLC 是 ZLG 自主研發(fā)的 PLC 系統(tǒng)(兼容 IEC61131-3),本文以定時器為例介紹一下如何擴展自定義功能塊,以及代碼生成器的用法。

fc17e518-5ab3-11ed-b116-dac502259ad0.jpg ?背景

AWTK全稱 Toolkit AnyWhere,是 ZLG 開發(fā)的開源 GUI 引擎,旨在為嵌入式系統(tǒng)、WEB、各種小程序、手機和 PC 打造的通用 GUI 引擎,為用戶提供一個功能強大、高效可靠、簡單易用、可輕松做出炫酷效果的 GUI 引擎。

AWPLC是 ZLG 自主研發(fā)的 PLC 系統(tǒng)(兼容 IEC61131-3),其中 AWPLC 的運行時庫(Runtime)基于 ZLG TKC 開發(fā),可以移植到到任何主流 RTOS嵌入式系統(tǒng)。AWPLC 的集成開發(fā)環(huán)境(IDE)基于 AWTK 開發(fā),可以運行在 Windows、MacOS 和 Linux 系統(tǒng)之上。AWPLC 的主要目標之一是把 PLC 中低代碼開發(fā)方法引入到嵌入式軟件,從而提高嵌入式軟件的開發(fā)效率和可靠性。

fc17e518-5ab3-11ed-b116-dac502259ad0.jpg ?簡介

在前一篇文章中,我們介紹了自定義 AWPLC 功能塊的基本方法,但是有些部分的內容并沒有提到,比如:

1. 功能塊的部分虛函數的實現(xiàn)。這些函數在不同功能塊中的實現(xiàn)是不同的,所以要做成虛函數,但是在各個功能塊中的實現(xiàn)又是相似的,不得不去寫一遍。比如 get_prop 這個函數,它在 ZTIMER 中的實現(xiàn)如下:

staticret_taw_plc_fb_ztimer_get_prop(aw_plc_fb_t*fb,constchar*name,value_t*v){
aw_plc_fb_ztimer_t*ztimer=AW_PLC_FB_ZTIMER(fb);

if(tk_str_eq(name,AW_PLC_FB_ZTIMER_PROP_IN)){
value_set_bool(v,ztimer->in);
returnRET_OK;
}

if(tk_str_eq(name,AW_PLC_FB_ZTIMER_PROP_PT)){
value_set_uint64(v,ztimer->pt);
returnRET_OK;
}

if(tk_str_eq(name,AW_PLC_FB_ZTIMER_PROP_Q)){
value_set_bool(v,ztimer->q);
returnRET_OK;
}

if(tk_str_eq(name,AW_PLC_FB_ZTIMER_PROP_ET)){
value_set_uint64(v,ztimer->et);
returnRET_OK;
}

if(tk_str_eq(name,AW_PLC_FB_ZTIMER_PROP_COUNT)){
value_set_uint32(v,ztimer->count);
returnRET_OK;
}

returnRET_NOT_FOUND;

}

*這樣的代碼看起來很簡單,但是恰恰容易出錯,更容易讓人厭倦,沒有什么樂趣。

2. API 和結構的注釋。我們來看看 ZTIMER 的結構注釋:

/**
*@classaw_plc_fb_ztimer_t
*@parentaw_plc_fb_t
*@annotation["fb"]
*循環(huán)定時器。
*
*>當輸入 IN 為 TRUE 時,開始計時,輸出 Q 為 FALSE,ET 開始記錄過去的時間。
*>定時時間到時,COUNT 增加 1,輸出 Q 在本次循環(huán)為 TRUE,ET 重置為0。
*>輸入 IN 為 FALSE 時重置定時器。
*/
typedefstruct_aw_plc_fb_ztimer_t{
aw_plc_fb_tfb;

/**
*@property{bool_t}in
*@annotation["in"]
*為 TRUE 開始計時,為 FALSE 時重置定時器。
*/
bool_tin:1;

/**
*@property{iec_time_t}pt
*@annotation["in"]
*預設時間(ms)。
*/
iec_time_tpt;

/**
*@property{bool_t}q
*@annotation["default","out"]
*定時時間是否到(僅在時間到的當次循環(huán)為 TRUE)。
*/
bool_tq:1;

/**
*@property{iec_time_t}et
*@annotation["out"]
*過去時間(ms)。
*/
iec_time_tet;

/**
*@property{uint32_t}count
*@annotation["out"]
*定時器時間到的次數。
*/
uint32_tcount;

/**
*@property{bool_t}prev_in
*@annotation["private"]
*前一次的輸入。
*/
bool_tprev_in:1;

/**
*@property{uint8_t}state
*@annotation["private"]
*狀態(tài)。
*/
uint8_tstate;

/**
*@property{iec_time_t}current_time
*@annotation["private"]
*當前時間(ms)。
*/
iec_time_tcurrent_time;

/**
*@property{iec_time_t}start_time
*@annotation["private"]
*開始時間(ms)。
*/
iec_time_tstart_time;

}aw_plc_fb_ztimer_t;

*上面的代碼看起來很美觀,讀起來很舒服,但是寫起來卻是有些費勁。3. IDE 需要功能塊的描述信息,以方便把它呈現(xiàn)到界面上。比如 ZTIMER 的描述信息如下:

{
"type":"fb_zlg_misc.ztimer",
"real_type":"ZTIMER",
"helpUrl":"https://developer.zlg.cn",
"style":"fb",
"desc":"循環(huán)定時器。\n\n>當輸入 IN 為 TRUE 時,開始計時,輸出 Q 為 FALSE,ET 開始記錄過去的時間。
\n>定時時間到時,COUNT 增加 1,輸出 Q 在本次循環(huán)為 TRUE,ET 重置為0。\n>輸入 IN 為 FALSE 時重置定時器。",
"ins":[
{
"name":"IN",
"desc":"為 TRUE 開始計時,為 FALSE 時重置定時器。",
"min_connections":1,
"max_connections":1,
"data_type":"BOOL"
},
{
"name":"PT",
"desc":"預設時間(ms)。",
"min_connections":1,
"max_connections":1,
"data_type":"TIME"
}
],
"outs":[
{
"name":"Q",
"desc":"定時時間是否到(僅在時間到的當次循環(huán)為 TRUE)。",
"data_type":"BOOL"
},
{
"name":"ET",
"desc":"過去時間(ms)。",
"data_type":"TIME"
},
{
"name":"COUNT",
"desc":"定時器時間到的次數。",
"data_type":"DWORD"
}
]

}

*這個 JSON 文件中的內容,和前面結構的注釋很相似,除了呈現(xiàn)的格式不同,同時還加了一些新內容。4. IDE 需要的文檔。功能塊需要提供一個 markdown 文檔,這個文檔會被轉換成 html,在用戶查看幫助時顯示給用戶。ZTIMER 的文檔內容如下:

#ZTIMER

##功能

循環(huán)定時器。

>當輸入 IN 為 TRUE 時,開始計時,輸出 Q 為 FALSE,ET 開始記錄過去的時間。
>定時時間到時,COUNT 增加 1,輸出 Q 在本次循環(huán)為 TRUE,ET 重置為0。
>輸入 IN 為 FALSE 時重置定時器。

##輸入

* IN **BOOL**為 TRUE 開始計時,為 FALSE 時重置定時器。
* PT **TIME**預設時間(ms)。

##輸出

* Q **BOOL**定時時間是否到(僅在時間到的當次循環(huán)為 TRUE)。
* ET **TIME**過去時間(ms)。

* COUNT **DWORD**定時器時間到的次數。*這個文檔的內容和前面結構的注釋,除了形式不同,內容是差不多的。很抱歉貼了這么代碼,希望您并沒仔細去讀它們。不要被這些代碼嚇到,它們都是自動生成的。如果手工去寫這些代碼,一天能寫一個功能塊就不錯了,不但辛苦而且容易出錯。這些工作必須自動完成!所以 AWPLC 中提供了一個代碼生成器,實測這個代碼生成器讓工作效率提高 10倍,幸福指數提高 10倍。

在進入正題前,我們先聊一下代碼生成器的基本知識。


fc17e518-5ab3-11ed-b116-dac502259ad0.jpg ?代碼生成器基本知識

* 編寫能編寫代碼的代碼。-- 《程序員修煉之道》

代碼生成器是一個普通程序,它能夠生成另外的目標代碼??梢圆灰a生成器,直接編寫目標代碼嗎?通常情況下是可以的,但是這違背了優(yōu)秀程序員的第一美德-懶惰。因為懶惰,所以能讓計算機做的事,優(yōu)秀程序員是不會自己去做的。

這里所說的目標代碼,也并非一定是嚴格意義上的代碼,也可能是另外一些數據。當然,有時候要嚴格區(qū)分數據和代碼,本身就是一件困難的事情。不過,這不是我們要說的重點,重點是通過代碼生成器提升我們的工作效率。*一個人的數據就是另外一個人的代碼。--《編程珠璣 II》1.代碼生成器的分類要說分類,就要先說分類的標準,在不同的分類標準和分類依據下,分出的類別迥異?!冻绦騿T修煉之道》里提到的一個分類標準具有極強的實用意義,這里我們重點介紹一下。它根據生成的目標代碼是否需要二次修改來分類,將代碼生成器分為兩類:

被動代碼生成器目標代碼生成之后,需要進行修改和完善,然后獨立發(fā)展和維護,與代碼生成器再與關系。比如 IDE 的 Wizard 就是此例。前面提到的自定義控件生成器,代碼生成之后,你需要在上面添加需要的功能。如果過了一段時間,你想為控件添加一個新的屬性,可能會遇到一點麻煩,要么手工添加;要么重新生成代碼,然后把之前修改的代碼重新加上,無論哪種方式都不是愉快的方式。被動代碼生成器雖然有它的缺陷,但是仍然可以給我們帶來很大幫助。

主動代碼生成器目標代碼生成之后,不需要進行修改和完善,每次都重新生成,如果需要修改,修改元數據和代碼生成器。比如編譯器就是此例。前面提到的 MVVM 的 ViewModel 和 AWFlow 應用代碼生成也屬于此類。如果可以,優(yōu)先使用主動代碼生成器。

2.基本形式fc2ae01e-5ab3-11ed-b116-dac502259ad0.png

這是代碼生成器的基本形式:代碼生成器讀取元數據,生成目標代碼。元數據是描述數據的數據,這里是描述目標代碼的數據,也就是控制目標代碼的參數。一般情況下,目標代碼整體結構由代碼生成器決定,而變化的部分由元數據決定。

代碼生成器本身一個很有意思的話題,有機會可以專門來聊聊,本文就不扯遠了。


fc17e518-5ab3-11ed-b116-dac502259ad0.jpg AWPLC中的代碼生成器

按前面代碼生成器的分類方式,AWPLC 里實現(xiàn)了一個主動代碼生成器,實現(xiàn)成主動代碼生成器是很重要的,AWPLC 還在快速迭代中,有些接口可能會變化,主動代碼生成器保證,即使接口有變化,也只需要運行一些腳本,重新生成目標文件即可。

1.基本架構

AWPLC 功能塊代碼生成器架構如下圖所示。其中功能塊描述文件就是前面所說的元數據,代碼生成器用它生成前面介紹的各種代碼和數據。fc37ec64-5ab3-11ed-b116-dac502259ad0.png

2.功能塊描述文件格式

描述文件用標準的 JSON 格式,其內容包括兩個部分:

2.1基本信息

基本信息包括:

  • name 功能塊的名稱。英文小寫,必須是合格的 C 語言變量名;
  • category 功能塊所屬的分類。各層級之間用/分隔,它決定了生成文件的位置;
  • is_function_block true 表示功能塊,false 表示函數;
  • impl 具體實現(xiàn)的源文件;
  • author 作者聯(lián)系方式;
  • version 版本號;;
  • date 更新時間;
  • desc 功能描述;
  • properties 屬性列表。具體定義如下。

示例:

"name":"ztimer",
"category":"zlg/misc",
"is_function_block":true,
"impl":"input/zlg/misc/ztimer.c",
"author":"LiXianJing",
"desc":"循環(huán)定時器。\n\n>當輸入 IN 為 TRUE 時,開始計時,輸出 Q 為 FALSE,ET 開始記錄過去的時間>。\n>定時時間到時,COUNT 增加 1,輸出 Q 在本次循環(huán)為 TRUE,ET 重置為0。\n>輸入 IN 為 FALSE 時重
置定時器。",

2.2屬性描述對于每個屬性,又包括下列信息:

  • name 屬性名;
  • desc 屬性描述;
  • type 實際的數據類型;
  • data_type(可選)用于在 IDE 中時類型檢查,缺省為 type 對應的 IEC 的數據類型,但是有時可用 ANY_INT 和 ANY_NUM 等來放寬類型檢查;
  • annotation 用于額外的標識。目前主要用于指定輸入輸出等特性。

示例:

{
"name":"count",
"desc":"定時器時間到的次數。",
"type":"uint32_t",
"annotation":{
"out":true
}

},

2.3使用方法

代碼生成器用 nodejs 編寫,需要安裝 nodejs。具體用法如下:

node gen.js 描述文件名。

如:

nodegen.jsinput/zlg/misc/ztimer.json

上面介紹了用 C 語言開發(fā)原生功能塊的方法。當然,也可以用 IEC 61131-3 中一些語言開發(fā)功能塊,除此之外,AWPLC 還會支持用 AWBlock 開發(fā)功能塊,在后續(xù)文章中,我們將一一介紹,敬請關注。AWPLC 目前還處于開發(fā)階段的早期,寫這個系列文章的目的,除了用來驗證目前所做的工作外,還希望得到大家的指點和反饋。如果您有任何疑問和建議,請在評論區(qū)留言。

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

    關注

    5082

    文章

    19104

    瀏覽量

    304803
收藏 人收藏

    評論

    相關推薦

    AWTK-WEB 快速入門(2) - JS 應用程序

    導讀AWTK可以使用相同的技術棧開發(fā)各種平臺的應用程序。有時我們需要使用Web界面與設備進行交互,本文介紹一如何使用JS語言開發(fā)
    的頭像 發(fā)表于 12-05 01:04 ?89次閱讀
    <b class='flag-5'>AWTK</b>-WEB <b class='flag-5'>快速</b>入門(2) - JS <b class='flag-5'>應用程序</b>

    AWTK-WEB 快速入門(1) - C 語言應用程序

    導讀AWTK可以使用相同的技術棧開發(fā)各種平臺的應用程序。有時我們需要使用Web界面與設備進行交互,本文介紹一如何使用C語言開發(fā)
    的頭像 發(fā)表于 11-27 11:46 ?185次閱讀
    <b class='flag-5'>AWTK</b>-WEB <b class='flag-5'>快速</b>入門(1) - C 語言<b class='flag-5'>應用程序</b>

    TPS659xx應用程序自定義工具

    電子發(fā)燒友網站提供《TPS659xx應用程序自定義工具.pdf》資料免費下載
    發(fā)表于 11-06 10:02 ?0次下載
    TPS659xx<b class='flag-5'>應用程序</b><b class='flag-5'>自定義</b>工具

    創(chuàng)建自定義的基于閃存的引導加載程序(BSL)

    電子發(fā)燒友網站提供《創(chuàng)建自定義的基于閃存的引導加載程序(BSL).pdf》資料免費下載
    發(fā)表于 09-19 10:50 ?0次下載
    創(chuàng)建<b class='flag-5'>自定義</b>的基于閃存的引導加載<b class='flag-5'>程序</b>(BSL)

    七大嵌入式GUI盤點

    特點是支持跨平臺同步開發(fā),一次編程,到處編譯,跨平臺使用。 GUIX 是微軟的高級工業(yè)級GUI解決方案,專門針對深度嵌入式,實時和IoT應用程序而設計。微軟還提供了名為GUIX Studio
    發(fā)表于 09-02 10:58

    AWTK使用經驗】如何添加中文輸入法

    AWTK是基于C語言開發(fā)的跨平臺GUI框架?!?b class='flag-5'>AWTK使用經驗》系列文章將介紹開發(fā)AWTK過程中一些常見問題與解決方案,例如:如何加載外部資
    的頭像 發(fā)表于 06-20 08:25 ?1035次閱讀
    【<b class='flag-5'>AWTK</b>使用經驗】如何添加中文輸入法

    AWTK使用經驗】如何響應物理按鍵

    AWTK是基于C語言開發(fā)的跨平臺GUI框架?!?b class='flag-5'>AWTK使用經驗》系列文章將介紹開發(fā)AWTK過程中一些常見問題與解決方案,例如:如何加載外部資
    的頭像 發(fā)表于 06-06 08:25 ?790次閱讀
    【<b class='flag-5'>AWTK</b>使用經驗】如何響應物理按鍵

    AWTK使用經驗】如何自定義combo_box下拉框樣式

    AWTK是基于C語言開發(fā)的跨平臺GUI框架?!?b class='flag-5'>AWTK使用經驗》系列文章將介紹開發(fā)AWTK過程中一些常見問題與解決方案,例如:如何加載外部資
    的頭像 發(fā)表于 05-23 08:25 ?453次閱讀
    【<b class='flag-5'>AWTK</b>使用經驗】如何<b class='flag-5'>自定義</b>combo_box下拉框樣式

    HarmonyOS開發(fā)案例:【 自定義彈窗】

    基于ArkTS的聲明開發(fā)范式實現(xiàn)了三種不同的彈窗,第一種直接使用公共組件,后兩種使用CustomDialogController實現(xiàn)自定義彈窗
    的頭像 發(fā)表于 05-16 18:18 ?1353次閱讀
    HarmonyOS<b class='flag-5'>開發(fā)</b>案例:【 <b class='flag-5'>自定義</b>彈窗】

    AWTK 開源串口屏開發(fā)(18) - C 語言自定義命令

    如果AWTK-HMI內置模型無法滿足需求,可以使用C語言來擴展默認模型。本文通過一個簡單的例子,介紹一C語言擴展默認模型的方法。AWTK-HMI內置了不少模型,利用這些模型
    的頭像 發(fā)表于 05-11 08:24 ?436次閱讀
    <b class='flag-5'>AWTK</b> 開源串口屏<b class='flag-5'>開發(fā)</b>(18) - <b class='flag-5'>用</b> C 語言<b class='flag-5'>自定義</b>命令

    HarmonyOS實戰(zhàn)開發(fā)-深度探索與打造個性化自定義組件

    今天分享一 什么是自定義組件?及其自定義組件的實戰(zhàn)。 做過前端或者android開發(fā)的都知道自定義組件,鴻蒙中顯示在界面上的UI都稱為組件
    發(fā)表于 05-08 16:30

    微軟Dev Home應用提供自定義文件管理支持

    據悉,Microsoft 近期發(fā)布了 0.13 版 Dev Home 應用程序,除修復多項 BUG 外,還新增了自定義文件資源管理器功能。該應用支持用戶在應用內創(chuàng)建虛擬機,利用微軟旗下的 Hyper V 技術生成本地虛擬機。
    的頭像 發(fā)表于 04-26 11:15 ?454次閱讀

    HarmonyOS開發(fā)實例:【自定義Emitter】

    使用[Emitter]實現(xiàn)事件的訂閱和發(fā)布,使用[自定義彈窗]設置廣告信息。
    的頭像 發(fā)表于 04-14 11:37 ?995次閱讀
    HarmonyOS<b class='flag-5'>開發(fā)</b>實例:【<b class='flag-5'>自定義</b>Emitter】

    鴻蒙ArkUI實例:【自定義組件】

    組件是 OpenHarmony 頁面最小顯示單元,一個頁面可由多個組件組合而成,也可只由一個組件組合而成,這些組件可以是ArkUI開發(fā)框架自帶系統(tǒng)組件,比如?`Text`?、?`Button`?等,也可以是自定義組件,本節(jié)筆者簡單介紹一
    的頭像 發(fā)表于 04-08 10:17 ?631次閱讀

    博途用戶自定義庫的使用

    中經常使用的函數/函數/數據類型等存放到自定義庫中,方便自己使用及與別人共享。博途具有很強的庫管理功能,包括:庫版本管理,庫的更新及清掃等等。本系列文章我將給大家介紹項目庫、全局庫、庫的更新/清掃等
    的頭像 發(fā)表于 12-25 10:08 ?914次閱讀
    博途用戶<b class='flag-5'>自定義</b>庫的使用
    RM新时代网站-首页