1、案例簡介
該程序是基于OpenHarmony標準系統(tǒng)編寫的基礎外設類:ADC驅(qū)動。
2、基礎知識
2.1、ADC簡介
ADC(Analog to Digital Converter),即模擬-數(shù)字轉(zhuǎn)換器,可將模擬信號轉(zhuǎn)換成對應的數(shù)字信號,便于存儲與計算等操作。除電源線和地線之外,ADC只需要1根線與被測量的設備進行連接。
2.2、ADC平臺驅(qū)動
在HDF框架中,同類型設備對象較多時(可能同時存在十幾個同類型配置器),若采用獨立服務模式,則需要配置更多的設備節(jié)點,且相關服務會占據(jù)更多的內(nèi)存資源。相反,采用統(tǒng)一服務模式可以使用一個設備服務作為管理器,統(tǒng)一處理所有同類型對象的外部訪問(這會在配置文件中有所體現(xiàn)),實現(xiàn)便捷管理和節(jié)約資源的目的。ADC模塊即采用統(tǒng)一服務模式。如下圖所示:
ADC模塊各分層的作用為:
接口層:提供打開設備,寫入數(shù)據(jù),關閉設備的能力。
核心層:主要負責服務綁定、初始化以及釋放管理器,并提供添加、刪除以及獲取控制器的能力。
適配層:由驅(qū)動適配者實現(xiàn)與硬件相關的具體功能,如控制器的初始化等。
在統(tǒng)一模式下,所有的控制器都被核心層統(tǒng)一管理,并由核心層統(tǒng)一發(fā)布一個服務供接口層,因此這種模式下驅(qū)動無需再為每個控制器發(fā)布服務。
詳細資料請參考官網(wǎng)地址:ADC平臺驅(qū)動
2.3、ADC應用程序
ADC模塊提供的主要接口如表1所示,具體API詳見//drivers/hdf_core/framework/include/platform/adc_if.h。
ADC驅(qū)動API接口功能介紹如下所示:
接口名 | 接口描述 |
---|---|
DevHandle AdcOpen(uint32_t number) | 打開ADC設備 |
void AdcClose(DevHandle handle) | 關閉ADC設備 |
int32_t AdcRead(DevHandle handle, uint32_t channel, uint32_t *val) | 讀取AD轉(zhuǎn)換結(jié)果值 |
使用ADC設備的一般流程如下所示:
詳細資料請參考官網(wǎng)地址:ADC應用程序
3、程序解析
3.1、準備工作
查看《凌蒙派-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ū)動設備描述,具體內(nèi)容如下:
#include "adc_config.hcs"
root { device_info { platform :: host { device_adc :: device { device0 :: deviceNode { // ADC控制器信息描述 policy = 2; // 對外發(fā)布服務,必須為2,用于定義ADC管理器的服務 priority = 50; permission = 0644; moduleName = "HDF_PLATFORM_ADC_MANAGER"; // 這與drivers/hdf_core/framework/support/platform/src/adc/adc_core.c的g_adcManagerEntry.moduleName對應,它主要負責ADC的管理 serviceName = "HDF_PLATFORM_ADC_MANAGER"; } device1 :: deviceNode { policy = 0; // 等于0,不需要發(fā)布服務 priority = 55; // 驅(qū)動驅(qū)動優(yōu)先級 permission = 0644; // 驅(qū)動創(chuàng)建設備節(jié)點權(quán)限 moduleName = "linux_adc_adapter"; // 用于指定驅(qū)動名稱,必須是linux_adc_adapter deviceMatchAttr = "linux_adc_adapter"; // 用于配置控制器私有數(shù)據(jù),必須與adc_config.hcs中對應控制器保持一致 } } } }}
ADC實際驅(qū)動是//drivers/hdf_core/adapter/khdf/linux/platform/adc/adc_iio_adapter.c,template adc_device定義的各項關鍵變量是由adc_iio_adapter.c決定,不可修改。
adc_iio_adapter.c實際是對Linux IIO子系統(tǒng)進行操作來控制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、應用程序
3.5.1、adc_test.c
添加平臺驅(qū)動ADC的頭文件,具體內(nèi)容如下:
#include "adc_if.h" // ADC標準接口頭文件
程序可通過,具體內(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設備 handle = AdcOpen(m_adc_device); if (handle == NULL) { PRINT_ERROR("AdcOpen failed\n"); return -1; }
// 進行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);
// 關閉ADC設備 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、參與應用程序編譯
編輯//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
+關注
關注
99文章
6531瀏覽量
545390 -
驅(qū)動
+關注
關注
12文章
1848瀏覽量
85464 -
應用程序
+關注
關注
38文章
3289瀏覽量
57815 -
OpenHarmony
+關注
關注
25文章
3744瀏覽量
16470
發(fā)布評論請先 登錄
相關推薦
評論