1、案例簡介
該程序是基于OpenHarmony標(biāo)準(zhǔn)系統(tǒng)編寫的基礎(chǔ)外設(shè)類:ADC驅(qū)動。
2、基礎(chǔ)知識
2.1、ADC簡介
ADC(Analog to Digital Converter),即模擬-數(shù)字轉(zhuǎn)換器,可將模擬信號轉(zhuǎn)換成對應(yīng)的數(shù)字信號,便于存儲與計算等操作。除電源線和地線之外,ADC只需要1根線與被測量的設(shè)備進(jìn)行連接。
2.2、ADC平臺驅(qū)動
在HDF框架中,同類型設(shè)備對象較多時(可能同時存在十幾個同類型配置器),若采用獨立服務(wù)模式,則需要配置更多的設(shè)備節(jié)點,且相關(guān)服務(wù)會占據(jù)更多的內(nèi)存資源。相反,采用統(tǒng)一服務(wù)模式可以使用一個設(shè)備服務(wù)作為管理器,統(tǒng)一處理所有同類型對象的外部訪問(這會在配置文件中有所體現(xiàn)),實現(xiàn)便捷管理和節(jié)約資源的目的。ADC模塊即采用統(tǒng)一服務(wù)模式。如下圖所示:
ADC模塊各分層的作用為:
接口層:提供打開設(shè)備,寫入數(shù)據(jù),關(guān)閉設(shè)備的能力。
核心層:主要負(fù)責(zé)服務(wù)綁定、初始化以及釋放管理器,并提供添加、刪除以及獲取控制器的能力。
適配層:由驅(qū)動適配者實現(xiàn)與硬件相關(guān)的具體功能,如控制器的初始化等。
在統(tǒng)一模式下,所有的控制器都被核心層統(tǒng)一管理,并由核心層統(tǒng)一發(fā)布一個服務(wù)供接口層,因此這種模式下驅(qū)動無需再為每個控制器發(fā)布服務(wù)。
詳細(xì)資料請參考官網(wǎng)地址:ADC平臺驅(qū)動
2.3、ADC應(yīng)用程序
ADC模塊提供的主要接口如表1所示,具體API詳見//drivers/hdf_core/framework/include/platform/adc_if.h。
ADC驅(qū)動API接口功能介紹如下所示:
接口名 | 接口描述 |
---|---|
DevHandle AdcOpen(uint32_t number) | 打開ADC設(shè)備 |
void AdcClose(DevHandle handle) | 關(guān)閉ADC設(shè)備 |
int32_t AdcRead(DevHandle handle, uint32_t channel, uint32_t *val) | 讀取AD轉(zhuǎn)換結(jié)果值 |
使用ADC設(shè)備的一般流程如下所示:
詳細(xì)資料請參考官網(wǎng)地址:ADC應(yīng)用程序
3、程序解析
3.1、準(zhǔn)備工作
查看《凌蒙派-RK3568開發(fā)板排針說明表》(即Git倉庫的//docs/board/凌蒙派-RK3568開發(fā)板排針說明表v1.0.xlsx),選中ADC5(即ADC5)。
3.2、配置文件
3.2.1、device_info.hcs
創(chuàng)建config/device_info.hcs,用于驅(qū)動設(shè)備描述,具體內(nèi)容如下:
#include "adc_config.hcs"
root { device_info { platform :: host { device_adc :: device { device0 :: deviceNode { // ADC控制器信息描述 policy = 2; // 對外發(fā)布服務(wù),必須為2,用于定義ADC管理器的服務(wù) priority = 50; permission = 0644; moduleName = "HDF_PLATFORM_ADC_MANAGER"; // 這與drivers/hdf_core/framework/support/platform/src/adc/adc_core.c的g_adcManagerEntry.moduleName對應(yīng),它主要負(fù)責(zé)ADC的管理 serviceName = "HDF_PLATFORM_ADC_MANAGER"; } device1 :: deviceNode { policy = 0; // 等于0,不需要發(fā)布服務(wù) priority = 55; // 驅(qū)動驅(qū)動優(yōu)先級 permission = 0644; // 驅(qū)動創(chuàng)建設(shè)備節(jié)點權(quán)限 moduleName = "linux_adc_adapter"; // 用于指定驅(qū)動名稱,必須是linux_adc_adapter deviceMatchAttr = "linux_adc_adapter"; // 用于配置控制器私有數(shù)據(jù),必須與adc_config.hcs中對應(yīng)控制器保持一致 } } } }}
ADC實際驅(qū)動是//drivers/hdf_core/adapter/khdf/linux/platform/adc/adc_iio_adapter.c,template adc_device定義的各項關(guān)鍵變量是由adc_iio_adapter.c決定,不可修改。
adc_iio_adapter.c實際是對Linux IIO子系統(tǒng)進(jìn)行操作來控制ADC。
注意:
channelNum:表示通道數(shù)量
driver_channelX_name:必須是從0開始
3.2.3、參與配置樹編譯
編輯//vendor/lockzhiner/rk3568/hdf_config/khdf/hdf.hcs,將device_info.hcs添加配置樹中。具體內(nèi)容如下所示:
#include "../../samples/b04_platform_device_adc/config/device_info.hcs"
3.3、HDF驅(qū)動
ADC平臺驅(qū)動是//drivers/hdf_core/adapter/khdf/linux/platform/adc/adc_iio_adapter.c,用戶不必編寫HDF驅(qū)動。
3.4、參與Linux內(nèi)核編譯
編輯//kernel/linux/config/linux-5.10/arch/arm64/configs/rk3568_standard_defconfig,啟用CONFIG_DRIVERS_HDF_PLATFORM_ADC,具體內(nèi)容如下:
CONFIG_DRIVERS_HDF_PLATFORM_ADC=y
3.5、應(yīng)用程序
3.5.1、adc_test.c
添加平臺驅(qū)動ADC的頭文件,具體內(nèi)容如下:
#include "adc_if.h" // ADC標(biāo)準(zhǔn)接口頭文件
程序可通過,具體內(nèi)容如下:
int main(int argc, char* argv[]){ DevHandle handle = NULL; int32_t ret; uint32_t value;
// 解析參數(shù) parse_opt(argc, argv); printf("adc_device: %d\n", m_adc_device); printf("adc_channel: %d\n", m_adc_channel);
// 打開ADC設(shè)備 handle = AdcOpen(m_adc_device); if (handle == NULL) { PRINT_ERROR("AdcOpen failed\n"); return -1; }
// 進(jìn)行AD轉(zhuǎn)換并讀取轉(zhuǎn)換結(jié)果 ret = AdcRead(handle, m_adc_channel, &value); if (ret != 0) { PRINT_ERROR("AdcRead failed and ret = %d\n", ret); AdcClose(handle); return -1; }
printf("Adc Device(%d), Channel(%d) read successful and value = %d\n", m_adc_device, m_adc_channel, value);
// 關(guān)閉ADC設(shè)備 AdcClose(handle);
return 0;}
3.5.2、BUILD.gn
import("http://build/ohos.gni")import("http://drivers/hdf_core/adapter/uhdf2/uhdf.gni")
print("samples: compile rk3568_adc_test")ohos_executable("rk3568_adc_test") { sources = [ "adc_test.c" ] include_dirs = [ "$hdf_framework_path/include", "$hdf_framework_path/include/core", "$hdf_framework_path/include/osal", "$hdf_framework_path/include/platform", "$hdf_framework_path/include/utils", "$hdf_uhdf_path/osal/include", "$hdf_uhdf_path/ipc/include", "http://base/hiviewdfx/hilog/interfaces/native/kits/include", "http://third_party/bounds_checking_function/include", ]
deps = [ "$hdf_uhdf_path/platform:libhdf_platform", "$hdf_uhdf_path/utils:libhdf_utils", "http://base/hiviewdfx/hilog/interfaces/native/innerkits:libhilog", ]
cflags = [ "-Wall", "-Wextra", "-Werror", "-Wno-format", "-Wno-format-extra-args", ]
subsystem_name = "applications" part_name = "product_rk3568" install_enable = true}
3.5.3、參與應(yīng)用程序編譯
編輯//vendor/lockzhiner/rk3568/samples/BUILD.gn,開啟編譯選項。具體如下:
"b04_platform_device_adc/app:rk3568_adc_test",
4、程序編譯
建議使用docker編譯方法,運行如下:
hb set -root .hb set# 選擇lockzhiner下的rk3568編譯分支。hb build -f
5、運行結(jié)果
該程序運行結(jié)果如下所示:
# rk3568_adc_test -d 0 -c 5../../vendor/lockzhiner/rk3568/samples/b21_platform_device_adc/app/adc_test.c, main, 103, info: adc_device: 0../../vendor/lockzhiner/rk3568/samples/b21_platform_device_adc/app/adc_test.c, main, 104, info: adc_channel: 5Adc Device(0), Channel(5) read successful and value = 955#
可以將ADC引腳通過引線接入排針線中的GNU或3V3中,可以查看ADC的變化。
-
adc
+關(guān)注
關(guān)注
99文章
6705瀏覽量
549183 -
驅(qū)動
+關(guān)注
關(guān)注
12文章
1916瀏覽量
86914 -
應(yīng)用程序
+關(guān)注
關(guān)注
38文章
3335瀏覽量
59024 -
OpenHarmony
+關(guān)注
關(guān)注
29文章
3853瀏覽量
18596
發(fā)布評論請先 登錄
請問是否可以在通用Windows平臺中構(gòu)建OpenVINO? GenAI C++ 應(yīng)用程序?
設(shè)計雙極性輸入、全差分輸出ADC驅(qū)動器時需要考慮什么

基于小凌派RK2206開發(fā)板:OpenHarmony如何使用IoT接口控制ADC外設(shè)

MCP:連接AI與應(yīng)用程序的開放標(biāo)準(zhǔn)!

如何部署OpenVINO?工具套件應(yīng)用程序?
迅為RK3568開發(fā)板篇OpenHarmony實操HDF驅(qū)動控制LED-編寫應(yīng)用APP
OpenHarmony程序分析框架論文入選ICSE 2025

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

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

迅為iTOP-RK3568開發(fā)板驅(qū)動開發(fā)指南-第十八篇 PWM
ADC驅(qū)動器的理想之選:低噪聲全差分運放SC7516

【驅(qū)動教程】iTOP-RK3568開發(fā)板進(jìn)行講解第十三期,主要講解輸入子系統(tǒng),共計24 講
基于ArkTS語言的OpenHarmony APP應(yīng)用開發(fā):HelloOpenharmony

評論