小熊派驅動開發(fā)流程(以點亮LED燈為例)
原創(chuàng)??https://harmonyos.51cto.com??
一、流程總覽
創(chuàng)建驅動目錄并添加驅動源碼文件和編譯腳本–>添加驅動配置目錄和驅動配置文件
二、添加驅動文件
2.1 創(chuàng)建驅動文件夾
在./device/st/drivers路徑下新建一個led目錄,用于存放驅動源碼文件。
2.2 創(chuàng)建驅動源碼文件
新建led.c文件,編寫驅動源碼
#include "hdf_device_desc.h"
#include "hdf_log.h"
#include "device_resource_if.h"
#include "osal_io.h"
#include "osal.h"
#include "osal_mem.h"
#include "gpio_if.h"
#define HDF_LOG_TAG led_driver // 打印日志所包含的標簽,如果不定義則用默認定義的HDF_TAG標簽
#define LED_WRITE_READ 1 // 讀寫操作碼1
enum LedOps {
LED_OFF,
LED_ON,
LED_TOGGLE,
};
struct Stm32Mp1ILed {
uint32_t gpioNum;
};
static struct Stm32Mp1ILed g_Stm32Mp1ILed;
uint8_t status = 0;
// Dispatch是用來處理用戶態(tài)發(fā)下來的消息
int32_t LedDriverDispatch(struct HdfDeviceIoClient *client, int cmdCode, struct HdfSBuf *data, struct HdfSBuf *reply)
{
uint8_t contrl;
HDF_LOGE("Led driver dispatch");
if (client == NULL || client->device == NULL)
{
HDF_LOGE("Led driver device is NULL");
return HDF_ERR_INVALID_OBJECT;
}
switch (cmdCode)
{
/* 接收到用戶態(tài)發(fā)來的LED_WRITE_READ命令 */
case LED_WRITE_READ:
/* 讀取data里的數據,賦值給contrl */
HdfSbufReadUint8(data,&contrl);
switch (contrl)
{
/* 開燈 */
case LED_ON:
GpioWrite(g_Stm32Mp1ILed.gpioNum, GPIO_VAL_LOW);
status = 1;
break;
/* 關燈 */
case LED_OFF:
GpioWrite(g_Stm32Mp1ILed.gpioNum, GPIO_VAL_HIGH);
status = 0;
break;
/* 狀態(tài)翻轉 */
case LED_TOGGLE:
if(status == 0)
{
GpioWrite(g_Stm32Mp1ILed.gpioNum, GPIO_VAL_LOW);
status = 1;
}
else
{
GpioWrite(g_Stm32Mp1ILed.gpioNum, GPIO_VAL_HIGH);
status = 0;
}
break;
default:
break;
}
/* 把LED的狀態(tài)值寫入reply, 可被帶至用戶程序 */
if (!HdfSbufWriteInt32(reply, status))
{
HDF_LOGE("replay is fail");
return HDF_FAILURE;
}
break;
default:
break;
}
return HDF_SUCCESS;
}
// 讀取驅動私有配置
static int32_t Stm32LedReadDrs(struct Stm32Mp1ILed *led, const struct DeviceResourceNode *node)
{
int32_t ret;
struct DeviceResourceIface *drsOps = NULL;
drsOps = DeviceResourceGetIfaceInstance(HDF_CONFIG_SOURCE);
if (drsOps == NULL || drsOps->GetUint32 == NULL) {
HDF_LOGE("%s: invalid drs ops!", __func__);
return HDF_FAILURE;
}
/* 讀取led.hcs里面led_gpio_num的值 */
ret = drsOps->GetUint32(node, "led_gpio_num", &led->gpioNum, 0);
if (ret != HDF_SUCCESS) {
HDF_LOGE("%s: read led gpio num fail!", __func__);
return ret;
}
return HDF_SUCCESS;
}
//驅動對外提供的服務能力,將相關的服務接口綁定到HDF框架
int32_t HdfLedDriverBind(struct HdfDeviceObject *deviceObject)
{
if (deviceObject == NULL)
{
HDF_LOGE("Led driver bind failed!");
return HDF_ERR_INVALID_OBJECT;
}
static struct IDeviceIoService ledDriver = {
.Dispatch = LedDriverDispatch,
};
deviceObject->service = (struct IDeviceIoService *)(&ledDriver);
HDF_LOGD("Led driver bind success");
return HDF_SUCCESS;
}
// 驅動自身業(yè)務初始的接口
int32_t HdfLedDriverInit(struct HdfDeviceObject *device)
{
struct Stm32Mp1ILed *led = &g_Stm32Mp1ILed;
int32_t ret;
if (device == NULL || device->property == NULL) {
HDF_LOGE("%s: device or property NULL!", __func__);
return HDF_ERR_INVALID_OBJECT;
}
/* 讀取hcs私有屬性值 */
ret = Stm32LedReadDrs(led, device->property);
if (ret != HDF_SUCCESS) {
HDF_LOGE("%s: get led device resource fail:%d", __func__, ret);
return ret;
}
/* 將GPIO管腳配置為輸出 */
ret = GpioSetDir(led->gpioNum, GPIO_DIR_OUT);
if (ret != 0)
{
HDF_LOGE("GpioSerDir: failed, ret %d\n", ret);
return ret;
}
HDF_LOGD("Led driver Init success");
return HDF_SUCCESS;
}
// 驅動資源釋放的接口
void HdfLedDriverRelease(struct HdfDeviceObject *deviceObject)
{
if (deviceObject == NULL)
{
HDF_LOGE("Led driver release failed!");
return;
}
HDF_LOGD("Led driver release success");
return;
}
// 定義驅動入口的對象,必須為HdfDriverEntry(在hdf_device_desc.h中定義)類型的全局變量
struct HdfDriverEntry g_ledDriverEntry = {
.moduleVersion = 1,
.moduleName = "HDF_LED",
.Bind = HdfLedDriverBind,
.Init = HdfLedDriverInit,
.Release = HdfLedDriverRelease,
};
// 調用HDF_INIT將驅動入口注冊到HDF框架中
HDF_INIT(g_ledDriverEntry);
2.3 創(chuàng)建編譯腳本文件
新建BUILD.gn文件,添加以下內容
import("http://drivers/adapter/khdf/liteos/hdf.gni")
hdf_driver("hdf_led") {
sources = [
"led.c",
]
}
2.4 修改drivers的編譯腳本使之編譯進內核
在/device/st/drivers/BUILD.gn的deps中加入"led"
三、添加驅動配置文件
驅動配置包含兩部分,HDF框架定義的驅動的私有配置信息和驅動設備描述
3.1 創(chuàng)建驅動配置文件夾
在./device/st/bearpi_hm_micro/liteos_a/hdf_config路徑下新建一個led目錄,用于存放led驅動配置文件。
3.2 創(chuàng)建驅動私有配置文件
在led文件夾下創(chuàng)建led_config.hcs文件,該文件為驅動配置文件。
添加以下內容
root {
LedDriverConfig {
led_gpio_num = 13;
match_attr = "st_stm32mp157_led"; //該字段的值必須和device_info.hcs中的deviceMatchAttr值一致
}
}
3.3 添加驅動設備描述
在device\st\bearpi_hm_micro\liteos_a\hdf_config\device_info\device_info.hcs文件中添加以下代碼
device_led :: device { // led設備節(jié)點
device0 :: deviceNode { // led驅動的DeviceNode節(jié)點
policy = 2; // policy字段是驅動服務發(fā)布的策略,在驅動服務管理章節(jié)有詳細介紹
priority = 10; // 驅動啟動優(yōu)先級(0-200),值越大優(yōu)先級越低,建議默認配100,優(yōu)先級相同則不保證device的加載順序
preload = 1; // 驅動按需加載字段
permission = 0777; // 驅動創(chuàng)建設備節(jié)點權限
moduleName = "HDF_LED"; // 驅動名稱,該字段的值必須和驅動入口結構的moduleName值一致
serviceName = "hdf_led"; // 驅動對外發(fā)布服務的名稱,必須唯一
deviceMatchAttr = "st_stm32mp157_led"; //驅動私有數據匹配的關鍵字,必須和驅動私有數據配置表中的match_attr值相等
}
}
如圖所示
- moduleName要與led.c中的**.moduleName = “HDF_LED”**參數一致
- serviceName要與led_config.hcs中的**match_attr = “st_stm32mp157_led”**參數一致
3.4 將寫好的驅動添加到板級配置入口
在在device\st\bearpi_hm_micro\liteos_a\hdf_config\hdf.hcs文件中添加如下代碼
#include "led/led_config.hcs"
四、小結
到此為止,以LED驅動開發(fā)的流程到此結束。
??https://harmonyos.51cto.com??