From bc3eb53a4abf91d01e1817e396b66b1b391460fe Mon Sep 17 00:00:00 2001 From: tianyangpeng Date: Mon, 23 Aug 2021 18:41:06 +0800 Subject: [PATCH] by typ Signed-off-by: tianyangpeng --- zh-cn/device-dev/driver/Readme-CN.md | 16 +- zh-cn/device-dev/driver/driver-peripherals.md | 2 +- .../driver/driver-platform-adc-develop.md | 317 ++++++++++++++ .../driver/driver-platform-develop.md | 12 + .../driver/driver-platform-gpio-develop.md | 285 +++++++++++++ .../driver/driver-platform-i2c-develop.md | 309 ++++++++++++++ .../driver/driver-platform-mipidsi-develop.md | 235 +++++++++++ .../driver/driver-platform-mmc-develop.md | 379 +++++++++++++++++ .../driver/driver-platform-pwm-develop.md | 275 ++++++++++++ .../driver/driver-platform-rtc-develop.md | 294 +++++++++++++ .../driver/driver-platform-sdio-develop.md | 309 ++++++++++++++ .../driver/driver-platform-spi-develop.md | 355 ++++++++++++++++ .../driver/driver-platform-uart-develop.md | 391 ++++++++++++++++++ .../driver-platform-watchdog-develop.md | 254 ++++++++++++ zh-cn/device-dev/driver/driver-platform.md | 2 +- ...5\345\212\241\346\250\241\345\274\217.png" | Bin 0 -> 40274 bytes ...5\345\212\241\346\250\241\345\274\217.png" | Bin 0 -> 52253 bytes ...5\345\212\241\346\250\241\345\274\217.png" | Bin 0 -> 58566 bytes 18 files changed, 3431 insertions(+), 4 deletions(-) create mode 100755 zh-cn/device-dev/driver/driver-platform-adc-develop.md create mode 100755 zh-cn/device-dev/driver/driver-platform-develop.md create mode 100755 zh-cn/device-dev/driver/driver-platform-gpio-develop.md create mode 100755 zh-cn/device-dev/driver/driver-platform-i2c-develop.md create mode 100755 zh-cn/device-dev/driver/driver-platform-mipidsi-develop.md create mode 100755 zh-cn/device-dev/driver/driver-platform-mmc-develop.md create mode 100755 zh-cn/device-dev/driver/driver-platform-pwm-develop.md create mode 100755 zh-cn/device-dev/driver/driver-platform-rtc-develop.md create mode 100755 zh-cn/device-dev/driver/driver-platform-sdio-develop.md create mode 100755 zh-cn/device-dev/driver/driver-platform-spi-develop.md create mode 100755 zh-cn/device-dev/driver/driver-platform-uart-develop.md create mode 100755 zh-cn/device-dev/driver/driver-platform-watchdog-develop.md create mode 100755 "zh-cn/device-dev/driver/figure/\346\227\240\346\234\215\345\212\241\346\250\241\345\274\217.png" create mode 100755 "zh-cn/device-dev/driver/figure/\347\213\254\347\253\213\346\234\215\345\212\241\346\250\241\345\274\217.png" create mode 100755 "zh-cn/device-dev/driver/figure/\347\273\237\344\270\200\346\234\215\345\212\241\346\250\241\345\274\217.png" diff --git a/zh-cn/device-dev/driver/Readme-CN.md b/zh-cn/device-dev/driver/Readme-CN.md index 6651085f35..32985b942c 100755 --- a/zh-cn/device-dev/driver/Readme-CN.md +++ b/zh-cn/device-dev/driver/Readme-CN.md @@ -7,7 +7,19 @@ - [驱动消息机制管理](driver-hdf-news.md) - [配置管理](driver-hdf-manage.md) - [HDF开发实例](driver-hdf-sample.md) -- [平台驱动](driver-platform.md) +- [平台驱动开发](driver-platform-develop.md) + - [GPIO](driver-platform-gpio-develop.md) + - [I2C](driver-platform-i2c-develop.md) + - [RTC](driver-platform-rtc-develop.md) + - [SDIO](driver-platform-sdio-develop.md) + - [SPI](driver-platform-spi-develop.md) + - [UART](driver-platform-uart-develop.md) + - [WATCHDOG](driver-platform-watchdog-develop.md) + - [MIPI_DSI](driver-platform-mipidsi-develop.md) + - [MMC](driver-platform-mmc-develop.md) + - [PWM](driver-platform-pwm-develop.md) + - [ADC](driver-platform-adc-develop.md) +- [平台驱动使用](driver-platform.md) - [GPIO](driver-platform-gpio-des.md) - [I2C](driver-platform-i2c-des.md) - [RTC](driver-platform-rtc-des.md) @@ -16,7 +28,7 @@ - [UART](driver-platform-uart-des.md) - [WATCHDOG](driver-platform-watchdog-des.md) - [MIPI DSI](driver-platform-mipidsi-des.md) -- [外设](driver-peripherals.md) +- [外设驱动开发](driver-peripherals.md) - [LCD](driver-peripherals-lcd-des.md) - [TOUCHSCREEN](driver-peripherals-touch-des.md) - [SENSOR](driver-peripherals-sensor-des.md) diff --git a/zh-cn/device-dev/driver/driver-peripherals.md b/zh-cn/device-dev/driver/driver-peripherals.md index 12db978ab6..8ef4078014 100644 --- a/zh-cn/device-dev/driver/driver-peripherals.md +++ b/zh-cn/device-dev/driver/driver-peripherals.md @@ -1,4 +1,4 @@ -# 外设 +# 外设驱动开发 - **[LCD](driver-peripherals-lcd-des.md)** diff --git a/zh-cn/device-dev/driver/driver-platform-adc-develop.md b/zh-cn/device-dev/driver/driver-platform-adc-develop.md new file mode 100755 index 0000000000..d4ca3f02c8 --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-adc-develop.md @@ -0,0 +1,317 @@ +# ADC + +- [概述](#1) +- [开发步骤](#2) +- [开发实例](#3) + +## 概述 + +ADC(Analog to Digital Converter),即模拟-数字转换器,是一种将模拟信号转换成对应数字信号的设备,在HDF框架中,ADC模块接口适配模式采用统一服务模式,这需要一个设备服务来作为ADC模块的管理器,统一处理外部访问,这会在配置文件中有所体现。统一服务模式适合于同类型设备对象较多的情况,如ADC可能同时具备十几个控制器,采用独立服务模式需要配置更多的设备节点,且服务会占据内存资源。 + +![image1](figure/统一服务模式.png) + +## 开发步骤 + +ADC模块适配的三个环节是配置属性文件,实例化驱动入口,以及实例化核心层接口函数。 + +1. **实例化驱动入口:** + + - 实例化HdfDriverEntry结构体成员。 + - 调用HDF_INIT将HdfDriverEntry实例化对象注册到HDF框架中。 + +2. **配置属性文件:** + + - 在device_info.hcs文件中添加deviceNode描述。 + - 【可选】添加adc_config.hcs器件属性文件。 + +3. **实例化ADC控制器对象:** + + - 初始化AdcDevice成员。 + - 实例化AdcDevice成员AdcMethod,其定义和成员说明见下 + +4. **驱动调试:** + - 【可选】针对新增驱动程序,建议验证驱动基本功能,例如挂载后的信息反馈,信号采集的成功与否等。 + +> ![](../public_sys-resources/icon-note.gif) **说明:** +> AdcMethod定义 +> +> ```c +> struct AdcMethod { +> int32_t (*read)(struct AdcDevice *device, uint32_t channel, uint32_t *val); +> int32_t (*start)(struct AdcDevice *device); +> int32_t (*stop)(struct AdcDevice *device); +> }; +> ``` +> +> 表1 AdcMethod结构体成员的回调函数功能说明 +> +> |函数成员|入参|出参|返回值|功能| +> |-|-|-|-|-| +> |read|**device**: 结构体指针,核心层ADC控制器;
**channel**:uint32_t,传入的通道号;|**val**:uint32_t指针,要传出的信号数据;|HDF_STATUS相关状态|读取ADC采样的信号数据| +> |stop |**device**: 结构体指针,核心层ADC控制器;|无|HDF_STATUS相关状态|关闭ADC设备| +> |start |**device**: 结构体指针,核心层ADC控制器;|无|HDF_STATUS相关状态|开启ADC设备| + + +## 开发实例 + +接下来以 adc_hi35xx.c 为示例, 展示需要厂商提供哪些内容来完整实现设备功能 + +1. 驱动开发首先需要实例化驱动入口,驱动入口必须为HdfDriverEntry(在 hdf_device_desc.h 中定义)类型的全局变量,且moduleName要和device_info.hcs中保持一致。HDF框架会将所有加载的驱动的HdfDriverEntry对象首地址汇总,形成一个类似数组的段地址空间,方便上层调用。 + + 一般在加载驱动时HDF会先调用Bind函数,再调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。 + +- ADC驱动入口参考 + + > ADC模块这种类型的控制器会出现很多个设备挂接的情况,因而在HDF框架中首先会为这类型的设备创建一个管理器对象,并同时对外发布一个管理器服务来统一处理外部访问。这样,用户需要打开某个设备时,会先获取到管理器服务,然后管理器服务根据用户指定参数查找到指定设备。 + > + > ADC管理器服务的驱动由核心层实现,**厂商不需要关注这部分内容的实现,这个但在实现Init函数的时候需要调用核心层的AdcDeviceAdd函数,它会实现相应功能。** + + ```c + static struct HdfDriverEntry g_hi35xxAdcDriverEntry = { + .moduleVersion = 1, + .Init = Hi35xxAdcInit, + .Release = Hi35xxAdcRelease, + .moduleName = "hi35xx_adc_driver",//【必要且与 HCS 里面的名字匹配】 + }; + HDF_INIT(g_hi35xxAdcDriverEntry); //调用HDF_INIT将驱动入口注册到HDF框架中 + + //核心层adc_core.c管理器服务的驱动入口 + struct HdfDriverEntry g_adcManagerEntry = { + .moduleVersion = 1, + .Init = AdcManagerInit, + .Release = AdcManagerRelease, + .moduleName = "HDF_PLATFORM_ADC_MANAGER",//这与device_info文件中device0对应 + }; + HDF_INIT(g_adcManagerEntry); + ``` + +2. 完成驱动入口注册之后,下一步请在device_info.hcs文件中添加deviceNode信息,并在adc_config.hcs中配置器件属性。deviceNode信息与驱动入口注册相关,器件属性值对于厂商驱动的实现以及核心层AdcDevice相关成员的默认值或限制范围有密切关系。 + + **统一服务模式**的特点是device_info文件中第一个设备节点必须为ADC管理器,其各项参数必须如下设置: + + |成员名|值| + |-|-| + |moduleName | 固定为 HDF_PLATFORM_ADC_MANAGER| + |serviceName| 无| + |policy| 具体配置为0,不发布服务| + |deviceMatchAttr| 没有使用,可忽略| + + **从第二个节点开始配置具体ADC控制器信息**,此节点并不表示某一路ADC控制器,而是代表一个资源性质设备,用于描述一类ADC控制器的信息。**本例只有一个ADC设备,如有多个设备,则需要在device_info文件增加deviceNode信息,以及在adc_config文件中增加对应的器件属性**。 + +- device_info.hcs 配置参考 + + ```c + root { + device_info { + platform :: host { + device_adc :: device { + device0 :: deviceNode { + policy = 0; + priority = 50; + permission = 0644; + moduleName = "HDF_PLATFORM_ADC_MANAGER"; + serviceName = "HDF_PLATFORM_ADC_MANAGER"; + } + device1 :: deviceNode { + policy = 0; // 等于0,不需要发布服务 + priority = 55; // 驱动启动优先级 + permission = 0644; // 驱动创建设备节点权限 + moduleName = "hi35xx_adc_driver"; //【必要】用于指定驱动名称,需要与期望的驱动Entry中的moduleName一致; + serviceName = "HI35XX_ADC_DRIVER"; //【必要】驱动对外发布服务的名称,必须唯一 + deviceMatchAttr = "hisilicon_hi35xx_adc";//【必要】用于配置控制器私有数据,要与adc_config.hcs中对应控制器保持一致 + } // 具体的控制器信息在 adc_config.hcs 中 + } + } + } + } + ``` + +- adc_config.hcs 配置参考 + + ```c + root { + platform { + adc_config_hi35xx { + match_attr = "hisilicon_hi35xx_adc"; + template adc_device { + regBasePhy = 0x120e0000;//寄存器物理基地址 + regSize = 0x34; //寄存器位宽 + deviceNum = 0; //设备号 + validChannel = 0x1; //有效通道 + dataWidth = 10; //信号接收的数据位宽 + scanMode = 1; //扫描模式 + delta = 0; //delta参数 + deglitch = 0; + glitchSample = 5000; + rate = 20000; + } + device_0 :: adc_device { + deviceNum = 0; + validChannel = 0x2; + } + } + } + } + ``` + +3. 完成驱动入口注册之后,最后一步就是以核心层AdcDevice对象的初始化为核心,包括厂商自定义结构体(传递参数和数据),实例化AdcDevice成员AdcMethod(让用户可以通过接口来调用驱动底层函数),实现HdfDriverEntry成员函数(Bind,Init,Release) + + +- 自定义结构体参考 + + > 从驱动的角度看,自定义结构体是参数和数据的载体,而且adc_config.hcs文件中的数值会被HDF读入通过DeviceResourceIface来初始化结构体成员,其中一些重要数值也会传递给核心层AdcDevice对象,例如设备号、总线号等。 + + ```c + struct Hi35xxAdcDevice { + struct AdcDevice device;//【必要】是核心层控制对象,具体描述见下面 + volatile unsigned char *regBase;//【必要】寄存器基地址 + volatile unsigned char *pinCtrlBase; + uint32_t regBasePhy; //【必要】寄存器物理基地址 + uint32_t regSize; //【必要】寄存器位宽 + uint32_t deviceNum; //【必要】设备号 + uint32_t dataWidth; //【必要】信号接收的数据位宽 + uint32_t validChannel; //【必要】有效通道 + uint32_t scanMode; //【必要】扫描模式 + uint32_t delta; + uint32_t deglitch; + uint32_t glitchSample; + uint32_t rate; //【必要】采样率 + }; + + //AdcDevice是核心层控制器结构体,其中的成员在Init函数中会被赋值 + struct AdcDevice { + const struct AdcMethod *ops; + OsalSpinlock spin; + uint32_t devNum; + uint32_t chanNum; + const struct AdcLockMethod *lockOps; + void *priv; + }; + ``` + + +- **【重要】** AdcDevice成员回调函数结构体AdcMethod的实例化,AdcLockMethod回调函数结构体本例未实现,若要实例化,可参考I2C驱动开发,其他成员在Init函数中初始化 + + ```c + static const struct AdcMethod g_method = { + .read = Hi35xxAdcRead, + .stop = Hi35xxAdcStop, + .start = Hi35xxAdcStart, + }; + ``` + +- **init函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 (下表为部分展示,如需使用其他状态,可见//drivers/framework/include/utils/hdf_base.h中HDF_STATUS 定义) + > + > |状态(值)|问题描述| + > |:-|:-:| + > |HDF_ERR_INVALID_OBJECT|控制器对象非法| + > |HDF_ERR_INVALID_PARAM |参数非法| + > |HDF_ERR_MALLOC_FAIL |内存分配失败| + > |HDF_ERR_IO |I/O 错误| + > |HDF_SUCCESS |传输成功| + > |HDF_FAILURE |传输失败| + > + > **函数说明:** + > 初始化自定义结构体对象,初始化AdcDevice成员,调用核心层AdcDeviceAdd函数。 + + ```c + static int32_t Hi35xxAdcInit(struct HdfDeviceObject *device) + { + int32_t ret; + struct DeviceResourceNode *childNode = NULL; + ... + //遍历、解析adc_config.hcs中的所有配置节点,并分别进行初始化,需要调用 Hi35xxAdcParseInit函数 + DEV_RES_NODE_FOR_EACH_CHILD_NODE(device->property, childNode) { + ret = Hi35xxAdcParseInit(device, childNode);//函数定义见下 + ... + } + return ret; + } + + static int32_t Hi35xxAdcParseInit(struct HdfDeviceObject *device, struct DeviceResourceNode *node) + { + int32_t ret; + struct Hi35xxAdcDevice *hi35xx = NULL; //【必要】自定义结构体对象 + (void)device; + + hi35xx = (struct Hi35xxAdcDevice *)OsalMemCalloc(sizeof(*hi35xx)); //【必要】内存分配 + ... + ret = Hi35xxAdcReadDrs(hi35xx, node); //【必要】将adc_config文件的默认值填充到结构体中 + ... + hi35xx->regBase = OsalIoRemap(hi35xx->regBasePhy, hi35xx->regSize);//【必要】地址映射 + ... + hi35xx->pinCtrlBase = OsalIoRemap(HI35XX_ADC_IO_CONFIG_BASE, HI35XX_ADC_IO_CONFIG_SIZE); + ... + Hi35xxAdcDeviceInit(hi35xx); //【必要】ADC设备的初始化 + hi35xx->device.priv = (void *)node; //【必要】存储设备属性 + hi35xx->device.devNum = hi35xx->deviceNum;//【必要】初始化AdcDevice成员 + hi35xx->device.ops = &g_method; //【必要】AdcMethod的实例化对象的挂载 + ret = AdcDeviceAdd(&hi35xx->device); //【必要且重要】调用此函数填充核心层结构体,返回成功信号后驱动才完全接入平台核心层 + ... + return HDF_SUCCESS; + + __ERR__: + if (hi35xx != NULL) { //不成功的话,需要反向执行初始化相关函数 + if (hi35xx->regBase != NULL) { + OsalIoUnmap((void *)hi35xx->regBase); + hi35xx->regBase = NULL; + } + AdcDeviceRemove(&hi35xx->device); + OsalMemFree(hi35xx); + } + return ret; + } + ``` + +- **Release 函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > 无 + > + > **函数说明:** + > 释放内存和删除控制器,该函数需要在驱动入口结构体中赋值给 Release 接口, 当HDF框架调用Init函数初始化驱动失败时,可以调用 Release 释放驱动资源。所有强制转换获取相应对象的操作**前提**是在Init函数中具备对应赋值的操作。 + + ```c + static void Hi35xxAdcRelease(struct HdfDeviceObject *device) + { + const struct DeviceResourceNode *childNode = NULL; + ... + //遍历、解析adc_config.hcs中的所有配置节点,并分别进行release操作 + DEV_RES_NODE_FOR_EACH_CHILD_NODE(device->property, childNode) { + Hi35xxAdcRemoveByNode(childNode);//函数定义见下 + } + } + + static void Hi35xxAdcRemoveByNode(const struct DeviceResourceNode *node) + { + int32_t ret; + int32_t deviceNum; + struct AdcDevice *device = NULL; + struct Hi35xxAdcDevice *hi35xx = NULL; + struct DeviceResourceIface *drsOps = NULL; + + drsOps = DeviceResourceGetIfaceInstance(HDF_CONFIG_SOURCE); + ... + ret = drsOps->GetUint32(node, "deviceNum", (uint32_t *)&deviceNum, 0); + ... + //可以调用AdcDeviceGet函数通过设备的deviceNum获取AdcDevice对象, 以及调用AdcDeviceRemove函数来释放AdcDevice对象的内容 + device = AdcDeviceGet(deviceNum); + if (device != NULL && device->priv == node) { + AdcDevicePut(device); + AdcDeviceRemove(device); //【必要】主要是从管理器驱动那边移除AdcDevice对象 + hi35xx = (struct Hi35xxAdcDevice *)device;//【必要】通过强制转换获取自定义的对象并进行release操作 + OsalIoUnmap((void *)hi35xx->regBase); + OsalMemFree(hi35xx); + } + return; + } + ``` \ No newline at end of file diff --git a/zh-cn/device-dev/driver/driver-platform-develop.md b/zh-cn/device-dev/driver/driver-platform-develop.md new file mode 100755 index 0000000000..53685a6ada --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-develop.md @@ -0,0 +1,12 @@ +# 平台驱动开发 + - **[GPIO](driver-platform-gpio-develop.md)** + - **[I2C](driver-platform-i2c-develop.md)** + - **[RTC](driver-platform-rtc-develop.md)** + - **[SDIO](driver-platform-sdio-develop.md)** + - **[SPI](driver-platform-spi-develop.md)** + - **[UART](driver-platform-uart-develop.md)** + - **[WATCHDOG](driver-platform-watchdog-develop.md)** + - **[MIPI_DSI](driver-platform-mipidsi-develop.md)** + - **[MMC](driver-platform-mmc-develop.md)** + - **[PWM](driver-platform-pwm-develop.md)** + - **[ADC](driver-platform-adc-develop.md)** diff --git a/zh-cn/device-dev/driver/driver-platform-gpio-develop.md b/zh-cn/device-dev/driver/driver-platform-gpio-develop.md new file mode 100755 index 0000000000..7689a94b1b --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-gpio-develop.md @@ -0,0 +1,285 @@ +# GPIO + +- [概述](#1) +- [开发步骤](#2) +- [开发实例](#3) + +## 概述 + +GPIO(General-purpose input/output)即通用型输入输出,在HDF框架中, +GPIO的接口适配模式采用无服务模式,用于不需要在用户态提供API的设备类型,或者没有用户态和内核区分的OS系统,其关联方式是DevHandle直接指向设备对象内核态地址(DevHandle是一个void类型指针)。 + +图 1 无服务模式结构图 +![image1](figure/无服务模式.png) + +## 开发步骤 + +GPIO模块适配的三个环节是配置属性文件,实例化驱动入口,以及实例化核心层接口函数。GPIO控制器分组管理所有管脚,相关参数会在属性文件中有所体现;驱动入口和接口函数的实例化环节是厂商驱动接入HDF的核心环节。 + +1. **实例化驱动入口:** + - 实例化HdfDriverEntry结构体成员。 + - 调用HDF_INIT将HdfDriverEntry实例化对象注册到HDF框架中。 + +2. **配置属性文件:** + + - 在device_info.hcs文件中添加deviceNode描述。 + - 【可选】添加gpio_config.hcs器件属性文件。 + +3. **实例化GPIO控制器对象:** + + - 初始化GpioCntlr成员。 + - 实例化GpioCntlr成员GpioMethod,其定义和成员**说明**见下 + +4. **驱动调试:** + - 【可选】针对新增驱动程序,建议验证驱动基本功能,例如GPIO控制状态,中断响应情况等。 + +> ![](../public_sys-resources/icon-note.gif) **说明:** +> +> GpioMethod定义 +> +> ```c +> struct GpioMethod { +> int32_t (*request)(struct GpioCntlr *cntlr, uint16_t local);// 【可选】 +> int32_t (*release)(struct GpioCntlr *cntlr, uint16_t local);// 【可选】 +> int32_t (*write)(struct GpioCntlr *cntlr, uint16_t local, uint16_t val); +> int32_t (*read)(struct GpioCntlr *cntlr, uint16_t local, uint16_t *val); +> int32_t (*setDir)(struct GpioCntlr *cntlr, uint16_t local, uint16_t dir); +> int32_t (*getDir)(struct GpioCntlr *cntlr, uint16_t local, uint16_t *dir); +> int32_t (*toIrq)(struct GpioCntlr *cntlr, uint16_t local, uint16_t *irq);// 【可选】 +> int32_t (*setIrq)(struct GpioCntlr *cntlr, uint16_t local, uint16_t mode, GpioIrqFunc func, void *arg); +> int32_t (*unsetIrq)(struct GpioCntlr *cntlr, uint16_t local); +> int32_t (*enableIrq)(struct GpioCntlr *cntlr, uint16_t local); +> int32_t (*disableIrq)(struct GpioCntlr *cntlr, uint16_t local); +> }; +> ``` +> +> 表1 GpioMethod结构体成员的回调函数功能说明 +> +> |函数成员|入参|出参|返回值|功能| +> |-|-|-|-|-| +> |write |**cntlr**:结构体指针,核心层GPIO控制器;
**local**:uint16_t,GPIO端口标识号 ;
**val**:uint16_t,电平传入值; |无| HDF_STATUS相关状态 | GPIO引脚写入电平值 | +> |read |**cntlr**:结构体指针,核心层GPIO控制器;
**local**:uint16_t,GPIO端口标识; |**val**:uint16_t 指针,
用于传出电平值 ;| HDF_STATUS相关状态 | GPIO引脚读取电平值 | +> |setDir |**cntlr**:结构体指针,核心层GPIO控制器;
**local**:uint16_t,GPIO端口标识号 ;
**dir**:uint16_t,管脚方向传入值; |无| HDF_STATUS相关状态 | 设置GPIO引脚输入/输出方向 | +> |getDir |**cntlr**:结构体指针,核心层GPIO控制器;
**local**:uint16_t,GPIO端口标识号 ; |**dir**:uint16_t 指针,
用于传出管脚方向值 ;| HDF_STATUS相关状态 | 读GPIO引脚输入/输出方向 | +> |setIrq |**cntlr**:结构体指针,核心层GPIO控制器;
**local**:uint16_t,GPIO端口标识号;
**mode**:uint16_t,表示触发模式(边沿或电平);
**func**:函数指针,中断服务程序 ;
**arg**:void指针,中断服务程序入参;|无| HDF_STATUS相关状态 |将GPIO引脚设置为中断模式 | +> |unsetIrq |**cntlr**:结构体指针,核心层GPIO控制器;
**local**:uint16_t,GPIO端口标识号 ; |无| HDF_STATUS相关状态 |取消GPIO中断设置 | +> |enableIrq |**cntlr**:结构体指针,核心层GPIO控制器;
**local**:uint16_t,GPIO端口标识号; |无| HDF_STATUS相关状态 |使能GPIO管脚中断 | +> |disableIrq|**cntlr**:结构体指针,核心层GPIO控制器;
**local**:uint16_t,GPIO端口标识号; |无| HDF_STATUS相关状态 |禁止GPIO管脚中断 | + + + + +## 开发实例 +下方将以gpio_hi35xx.c为示例,展示需要厂商提供哪些内容来完整实现设备功能。 + +1. 驱动开发首先需要实例化驱动入口,驱动入口必须为HdfDriverEntry(在 hdf_device_desc.h 中定义)类型的全局变量,且moduleName要和device_info.hcs中保持一致。HDF框架会将所有加载的驱动的HdfDriverEntry对象首地址汇总,形成一个类似数组的段地址空间,方便上层调用。 + + 一般在加载驱动时HDF会先调用Bind函数,再调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。 + +- GPIO 驱动入口参考 + + ```c + struct HdfDriverEntry g_gpioDriverEntry = { + .moduleVersion = 1, + .Bind = Pl061GpioBind, //GPIO不需要实现Bind,本例是一个空实现,厂商可根据自身需要添加相关操作 + .Init = Pl061GpioInit, //见Init参考 + .Release = Pl061GpioRelease, //见Release参考 + .moduleName = "hisi_pl061_driver",//【必要且需要与HCS文件中里面的moduleName匹配】 + }; + //调用HDF_INIT将驱动入口注册到HDF框架中 + HDF_INIT(g_gpioDriverEntry); + ``` + +2. 完成驱动入口注册之后,下一步请在device_info.hcs文件中添加deviceNode信息,并在 gpio_config.hcs 中配置器件属性。deviceNode信息与驱动入口注册相关,器件属性值与核心层GpioCntlr 成员的默认值或限制范围有密切关系。 + + **本例只有一个GPIO控制器,如有多个器件信息,则需要在device_info文件增加deviceNode信息,以及在gpio_config文件中增加对应的器件属性**。 + +- device_info.hcs 配置参考 + + ```c + root { + device_info { + platform :: host { + hostName = "platform_host"; + priority = 50; + device_gpio :: device { + device0 :: deviceNode { + policy = 0; // 等于0,不需要发布服务 + priority = 10; // 驱动启动优先级 + permission = 0644; // 驱动创建设备节点权限 + moduleName = "hisi_pl061_driver"; //【必要】用于指定驱动名称,需要与期望的驱动Entry中的moduleName一致; + deviceMatchAttr = "hisilicon_hi35xx_pl061";//【必要】用于配置控制器私有数据,要与 gpio_config.hcs 中 + //对应控制器保持一致,其他控制器信息也在文件中 + } + } + } + } + } + ``` + +- gpio_config.hcs 配置参考 + + ```c + root { + platform { + gpio_config { + controller_0x120d0000 { + match_attr = "hisilicon_hi35xx_pl061"; //【必要】必须和device_info.hcs中的deviceMatchAttr值一致 + groupNum = 12; //【必要】GPIO组索引 需要根据设备情况填写 + bitNum = 8; //【必要】每组GPIO管脚数 + regBase = 0x120d0000;//【必要】物理及地址 + regStep = 0x1000; //【必要】寄存器偏移步进 + irqStart = 48; //【必要】开启中断 + irqShare = 0; //【必要】共享中断 + } + } + } + } + ``` + +3. 完成驱动入口注册之后,最后一步就是以核心层GpioCntlr对象的初始化为核心,包括厂商自定义结构体(传递参数和数据),实例化GpioCntlr成员GpioMethod(让用户可以通过接口来调用驱动底层函数),实现HdfDriverEntry成员函数(Bind,Init,Release) + + +- 自定义结构体参考 + + > 从驱动的角度看,自定义结构体是参数和数据的载体,而且gpio_config.hcs文件中的数值会被HDF读入通过DeviceResourceIface来初始化结构体成员,其中一些重要数值也会传递给核心层GpioCntlr对象,例如索引、管脚数等。 + + ```c + struct Pl061GpioCntlr { + struct GpioCntlr cntlr;//【必要】 是核心层控制对象,其成员定义见下面 + volatile unsigned char *regBase; //【必要】寄存器基地址 + uint32_t phyBase; //【必要】 物理基址 + uint32_t regStep; //【必要】 寄存器偏移步进 + uint32_t irqStart; //【必要】 中断开启 + uint16_t groupNum; //【必要】 用于描述厂商的GPIO端口号的参数 + uint16_t bitNum; //【必要】 用于描述厂商的GPIO端口号的参数 + uint8_t irqShare; //【必要】 共享中断 + struct Pl061GpioGroup *groups; //【可选】 根据厂商需要设置 + }; + struct Pl061GpioGroup { //包括寄存器地址,中断号,中断函数和和锁 + volatile unsigned char *regBase; + unsigned int index; + unsigned int irq; + OsalIRQHandle irqFunc; + OsalSpinlock lock; + }; + + // GpioCntlr是核心层控制器结构体,其中的成员在Init函数中会被赋值 + struct GpioCntlr { + struct IDeviceIoService service; + struct HdfDeviceObject *device; + struct GpioMethod *ops; + struct DListHead list; + OsalSpinlock spin; + uint16_t start; + uint16_t count; + struct GpioInfo *ginfos; + void *priv; + }; + ``` + +- **【重要】** GpioCntlr成员回调函数结构体GpioMethod的实例化,其他成员在Init函数中初始化 + + ```c + //GpioMethod结构体成员都是回调函数,厂商需要根据表1完成相应的函数功能。 + static struct GpioMethod g_method = { + .request = NULL, + .release = NULL, + .write = Pl061GpioWrite, //写管脚 + .read = Pl061GpioRead, //读管脚 + .setDir = Pl061GpioSetDir, //设置管脚方向 + .getDir = Pl061GpioGetDir, //获取管脚方向 + .toIrq = NULL, + .setIrq = Pl061GpioSetIrq, //设置管脚中断,如不具备此能力可忽略 + .unsetIrq = Pl061GpioUnsetIrq, //取消管脚中断设置,如不具备此能力可忽略 + .enableIrq = Pl061GpioEnableIrq, //使能管脚中断,如不具备此能力可忽略 + .disableIrq = Pl061GpioDisableIrq,//禁止管脚中断,如不具备此能力可忽略 + }; + ``` + + +- **Init函数参考** + + > **入参:** + > HdfDeviceObject这个是整个驱动对外暴露的接口参数,具备HCS配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态(下表为部分展示,如需使用其他状态,可见//drivers/framework/include/utils/hdf_base.h中HDF_STATUS 定义) + > + > |状态(值)|问题描述| + > |:-|:-:| + > |HDF_ERR_INVALID_OBJECT|控制器对象非法| + > |HDF_ERR_MALLOC_FAIL |内存分配失败| + > |HDF_ERR_INVALID_PARAM |参数非法| + > |HDF_ERR_IO |I/O 错误| + > |HDF_SUCCESS |初始化成功| + > |HDF_FAILURE |初始化失败| + > + > **函数说明:** + > 初始化自定义结构体对象,初始化GpioCntlr成员,调用核心层GpioCntlrAdd函数,【可选】接入VFS + + ```c + static int32_t Pl061GpioInit(struct HdfDeviceObject *device) + { + ... + struct Pl061GpioCntlr *pl061 = &g_pl061;//利用静态全局变量完成初始化 + //static struct Pl061GpioCntlr g_pl061 = { + // .groups = NULL, + // .groupNum = PL061_GROUP_MAX, + // .bitNum = PL061_BIT_MAX, + //}; + ret = Pl061GpioReadDrs(pl061, device->property);//利用从gpio_config.HCS文件读取的属性值来初始化自定义结构体对象成员 + ... + pl061->regBase = OsalIoRemap(pl061->phyBase, pl061->groupNum * pl061->regStep);//地址映射 + ... + ret = Pl061GpioInitCntlrMem(pl061); // 内存分配 + ... + pl061->cntlr.count = pl061->groupNum * pl061->bitNum;//【必要】管脚数量计算 + pl061->cntlr.priv = (void *)device->property; //【必要】存储设备属性 + pl061->cntlr.ops = &g_method; // 【必要】GpioMethod的实例化对象的挂载 + pl061->cntlr.device = device; // 【必要】使HdfDeviceObject与GpioCntlr可以相互转化的前提 + ret = GpioCntlrAdd(&pl061->cntlr); // 【必要】调用此函数填充核心层结构体,返回成功信号后驱动才完全接入平台核心层 + ... + Pl061GpioDebugCntlr(pl061); + #ifdef PL061_GPIO_USER_SUPPORT //【可选】若支持用户级的虚拟文件系统,则接入 + if (GpioAddVfs(pl061->bitNum) != HDF_SUCCESS) { + HDF_LOGE("%s: add vfs fail!", __func__); + } + #endif + ... + } + ``` + +- **Release 函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > 无 + > + > **函数说明:** + > 释放内存和删除控制器,该函数需要在驱动入口结构体中赋值给 Release 接口, 当HDF框架调用Init函数初始化驱动失败时,可以调用 Release 释放驱动资源。所有强制转换获取相应对象的操作**前提**是在Init函数中具备对应赋值的操作。 + + ```c + static void Pl061GpioRelease(struct HdfDeviceObject *device) + { + struct GpioCntlr *cntlr = NULL; + struct Pl061GpioCntlr *pl061 = NULL; + ... + cntlr = GpioCntlrFromDevice(device);//【必要】通过强制转换获取核心层控制对象 + //return (device == NULL) ? NULL : (struct GpioCntlr *)device->service; + ... + #ifdef PL061_GPIO_USER_SUPPORT + GpioRemoveVfs();//与Init中GpioAddVfs相反 + #endif + GpioCntlrRemove(cntlr); //【必要】取消设备信息、服务等内容在核心层上的挂载 + pl061 = ToPl061GpioCntlr(cntlr); //return (struct Pl061GpioCntlr *)cntlr; + Pl061GpioRleaseCntlrMem(pl061); //【必要】锁和内存的释放 + OsalIoUnmap((void *)pl061->regBase);//【必要】解除地址映射 + pl061->regBase = NULL; + } + ``` + + + diff --git a/zh-cn/device-dev/driver/driver-platform-i2c-develop.md b/zh-cn/device-dev/driver/driver-platform-i2c-develop.md new file mode 100755 index 0000000000..b62b1a3d37 --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-i2c-develop.md @@ -0,0 +1,309 @@ +# I2C + +- [概述](#1) +- [开发步骤](#2) +- [开发实例](#3) + + +## 概述 + +I2C(Inter Integrated Circuit)总线是由Philips公司开发的一种简单、双向二线制同步串行总线,在HDF框架中,I2C模块接口适配模式采用统一服务模式,这需要一个设备服务来作为I2C模块的管理器,统一处理外部访问,这会在配置文件中有所体现。统一服务模式适合于同类型设备对象较多的情况,如I2C可能同时具备十几个控制器,采用独立服务模式需要配置更多的设备节点,且服务会占据内存资源。 + +图 1 统一服务模式结构图 +![image1](figure/统一服务模式.png) + +## 开发步骤 + +I2C模块适配的三个环节是配置属性文件,实例化驱动入口,以及实例化核心层接口函数。 + +1. **实例化驱动入口:** + + - 实例化HdfDriverEntry结构体成员。 + - 调用HDF_INIT将HdfDriverEntry实例化对象注册到HDF框架中。 + +2. **配置属性文件:** + + - 在device_info.hcs文件中添加deviceNode描述。 + - 【可选】添加i2c_config.hcs器件属性文件。 + +3. **实例化I2C控制器对象:** + + - 初始化I2cCntlr成员。 + - 实例化I2cCntlr成员I2cMethod和I2cLockMethod,其定义和成员说明见下 + +4. **驱动调试:** + - 【可选】针对新增驱动程序,建议验证驱动基本功能,例如挂载后的信息反馈,消息传输的成功与否等。 + +> ![](../public_sys-resources/icon-note.gif) **说明:** +> I2cMethod和I2cLockMethod定义 +> +> ```c +> struct I2cMethod { +> int32_t (*transfer)(struct I2cCntlr *cntlr, struct I2cMsg *msgs, int16_t count); +> }; +> struct I2cLockMethod {//锁机制操作结构体 +> int32_t (*lock)(struct I2cCntlr *cntlr);//加锁 +> void (*unlock)(struct I2cCntlr *cntlr); //解锁 +> }; +> ``` +> 表1 I2cMethod结构体成员的回调函数功能说明 +> |函数成员|入参|出参|返回值|功能| +> |-|-|-|-|-| +> |transfer |**cntlr**:结构体指针,核心层I2C控制器;
**msgs**:结构体指针,用户消息 ;
**count**:uint16_t,消息数量 |无|HDF_STATUS相关状态| 传递用户消息| + +## 开发实例 + +下方将以i2c_hi35xx.c为示例,展示需要厂商提供哪些内容来完整实现设备功能。 + +1. 驱动开发首先需要实例化驱动入口,驱动入口必须为HdfDriverEntry(在 hdf_device_desc.h 中定义)类型的全局变量,且moduleName要和device_info.hcs中保持一致。HDF框架会将所有加载的驱动的HdfDriverEntry对象首地址汇总,形成一个类似数组的段地址空间,方便上层调用。 + + 一般在加载驱动时HDF会先调用Bind函数,再调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。 + +- I2C驱动入口参考 + + > I2C模块这种类型的控制器会出现很多个设备挂接的情况,因而在HDF框架中首先会为这类型的设备创建一个管理器对象,并同时对外发布一个管理器服务来统一处理外部访问。这样,用户需要打开某个设备时,会先获取到管理器服务,然后管理器服务根据用户指定参数查找到指定设备。 + > + > I2C管理器服务的驱动由核心层实现,**厂商不需要关注这部分内容的实现,这个但在实现Init函数的时候需要调用核心层的I2cCntlrAdd函数,它会实现相应功能。** + + + ```c + struct HdfDriverEntry g_i2cDriverEntry = { + .moduleVersion = 1, + .Init = Hi35xxI2cInit, + .Release = Hi35xxI2cRelease, + .moduleName = "hi35xx_i2c_driver",//【必要且与config.hcs文件里面匹配】 + }; + HDF_INIT(g_i2cDriverEntry); //调用HDF_INIT将驱动入口注册到HDF框架中 + + //核心层i2c_core.c 管理器服务的驱动入口 + struct HdfDriverEntry g_i2cManagerEntry = { + .moduleVersion = 1, + .Bind = I2cManagerBind, + .Init = I2cManagerInit, + .Release = I2cManagerRelease, + .moduleName = "HDF_PLATFORM_I2C_MANAGER",//这与device_info文件中device0对应 + }; + HDF_INIT(g_i2cManagerEntry); + ``` + +2. 完成驱动入口注册之后,下一步请在device_info.hcs文件中添加deviceNode信息,并在i2c_config.hcs中配置器件属性。deviceNode信息与驱动入口注册相关,器件属性值对于厂商驱动的实现以及核心层I2cCntlr相关成员的默认值或限制范围有密切关系。 + + **统一服务模式**的特点是device_info文件中第一个设备节点必须为I2C管理器,其各项参数必须如下设置: + + |成员名|值| + |-|-| + |moduleName | 固定为 HDF_PLATFORM_I2C_MANAGER| + |serviceName| 固定为 HDF_PLATFORM_I2C_MANAGER| + |policy| 具体配置为1或2取决于是否对用户态可见| + |deviceMatchAttr| 没有使用,可忽略| + + **从第二个节点开始配置具体I2C控制器信息**,此节点并不表示某一路I2C控制器,而是代表一个资源性质设备,用于描述一类I2C控制器的信息。多个控制器之间相互区分的参数是busID和reg_pbase,这在i2c_config文件中有所体现。 + +- device_info.hcs 配置参考 + + ```c + root { + device_info { + match_attr = "hdf_manager"; + device_i2c :: device { + device0 :: deviceNode { + policy = 2; + priority = 50; + permission = 0644; + moduleName = "HDF_PLATFORM_I2C_MANAGER"; + serviceName = "HDF_PLATFORM_I2C_MANAGER"; + deviceMatchAttr = "hdf_platform_i2c_manager"; + } + device1 :: deviceNode { + policy = 0; // 等于0,不需要发布服务 + priority = 55; // 驱动启动优先级 + permission = 0644; // 驱动创建设备节点权限 + moduleName = "hi35xx_i2c_driver"; //【必要】用于指定驱动名称,需要与期望的驱动Entry中的moduleName一致; + serviceName = "HI35XX_I2C_DRIVER"; //【必要】驱动对外发布服务的名称,必须唯一 + deviceMatchAttr = "hisilicon_hi35xx_i2c";//【必要】用于配置控制器私有数据,要与i2c_config.hcs中对应控制器保持一致 + // 具体的控制器信息在 i2c_config.hcs 中 + } + } + } + } + ``` + +- i2c_config.hcs 配置参考 + + ```c + root { + platform { + i2c_config { + match_attr = "hisilicon_hi35xx_i2c";//【必要】需要和device_info.hcs中的deviceMatchAttr值一致 + template i2c_controller { //模板公共参数,继承该模板的节点如果使用模板中的默认值,则节点字段可以缺省 + bus = 0; //【必要】i2c 识别号 + reg_pbase = 0x120b0000; //【必要】物理基地址 + reg_size = 0xd1; //【必要】寄存器位宽 + irq = 0; //【可选】根据厂商需要来使用 + freq = 400000; //【可选】根据厂商需要来使用 + clk = 50000000; //【可选】根据厂商需要来使用 + } + controller_0x120b0000 :: i2c_controller { + bus = 0; + } + controller_0x120b1000 :: i2c_controller { + bus = 1; + reg_pbase = 0x120b1000; + } + ... + } + } + } + ``` + +3. 完成驱动入口注册之后,最后一步就是以核心层I2cCntlr对象的初始化为核心,包括厂商自定义结构体(传递参数和数据),实例化I2cCntlr成员I2cMethod(让用户可以通过接口来调用驱动底层函数),实现HdfDriverEntry成员函数(Bind,Init,Release) + + +- 自定义结构体参考 + + > 从驱动的角度看,自定义结构体是参数和数据的载体,而且i2c_config.hcs文件中的数值会被HDF读入通过DeviceResourceIface来初始化结构体成员,其中一些重要数值也会传递给核心层I2cCntlr对象,例如设备号、总线号等。 + + ```c + // 厂商自定义功能结构体 + struct Hi35xxI2cCntlr { + struct I2cCntlr cntlr; //【必要】是核心层控制对象,具体描述见下面 + OsalSpinlock spin; //【必要】厂商需要基于此锁变量对各个 i2c 操作函数实现对应的加锁解锁 + volatile unsigned char *regBase; //【必要】寄存器基地址 + uint16_t regSize; //【必要】寄存器位宽 + int16_t bus; //【必要】i2c_config.hcs 文件中可读取具体值 + uint32_t clk; //【可选】厂商自定义 + uint32_t freq; //【可选】厂商自定义 + uint32_t irq; //【可选】厂商自定义 + uint32_t regBasePhy; //【必要】寄存器物理基地址 + }; + + // I2cCntlr是核心层控制器结构体,其中的成员在Init函数中会被赋值 + struct I2cCntlr { + struct OsalMutex lock; + void *owner; + int16_t busId; + void *priv; + const struct I2cMethod *ops; + const struct I2cLockMethod *lockOps; + }; + ``` + +- **【重要】** I2cCntlr成员回调函数结构体I2cMethod的实例化,和锁机制回调函数结构体I2cLockMethod实例化,其他成员在Init函数中初始化 + + ```c + // i2c_hi35xx.c 中的示例 + static const struct I2cMethod g_method = { + .transfer = Hi35xxI2cTransfer, + }; + + static const struct I2cLockMethod g_lockOps = { + .lock = Hi35xxI2cLock, //加锁函数 + .unlock = Hi35xxI2cUnlock,//解锁函数 + }; + ``` + +- **init函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 (下表为部分展示,如需使用其他状态,可见//drivers/framework/include/utils/hdf_base.h中HDF_STATUS 定义) + > + > |状态(值)|问题描述| + > |:-|:-:| + > |HDF_ERR_INVALID_OBJECT|控制器对象非法| + > |HDF_ERR_INVALID_PARAM |参数非法| + > |HDF_ERR_MALLOC_FAIL |内存分配失败| + > |HDF_ERR_IO |I/O 错误| + > |HDF_SUCCESS |传输成功| + > |HDF_FAILURE |传输失败| + > + > **函数说明:** + > 初始化自定义结构体对象,初始化I2cCntlr成员,调用核心层I2cCntlrAdd函数,【可选】接入VFS + + ```c + static int32_t Hi35xxI2cInit(struct HdfDeviceObject *device) + { + ... + //遍历、解析 i2c_config.hcs 中的所有配置节点,并分别进行初始化,需要调用Hi35xxI2cParseAndInit函数 + DEV_RES_NODE_FOR_EACH_CHILD_NODE(device->property, childNode) { + ret = Hi35xxI2cParseAndInit(device, childNode);//函数定义见下 + ... + } + ... + } + + static int32_t Hi35xxI2cParseAndInit(struct HdfDeviceObject *device, const struct DeviceResourceNode *node) + { + struct Hi35xxI2cCntlr *hi35xx = NULL; + ... + hi35xx = (struct Hi35xxI2cCntlr *)OsalMemCalloc(sizeof(*hi35xx)); // 内存分配 + ... + hi35xx->regBase = OsalIoRemap(hi35xx->regBasePhy, hi35xx->regSize); // 地址映射 + ... + Hi35xxI2cCntlrInit(hi35xx); // 【必要】i2c设备的初始化 + + hi35xx->cntlr.priv = (void *)node; //【必要】存储设备属性 + hi35xx->cntlr.busId = hi35xx->bus; //【必要】初始化I2cCntlr成员busId + hi35xx->cntlr.ops = &g_method; //【必要】I2cMethod的实例化对象的挂载 + hi35xx->cntlr.lockOps = &g_lockOps; //【必要】I2cLockMethod的实例化对象的挂载 + (void)OsalSpinInit(&hi35xx->spin); //【必要】锁的初始化 + ret = I2cCntlrAdd(&hi35xx->cntlr); //【必要】调用此函数填充核心层结构体,返回成功信号后驱动才完全接入平台核心层 + ... + #ifdef USER_VFS_SUPPORT + (void)I2cAddVfsById(hi35xx->cntlr.busId);//【可选】若支持用户级的虚拟文件系统,则接入 + #endif + return HDF_SUCCESS; + __ERR__: //不成功的话,需要反向执行初始化相关函数 + if (hi35xx != NULL) { + if (hi35xx->regBase != NULL) { + OsalIoUnmap((void *)hi35xx->regBase); + hi35xx->regBase = NULL; + } + OsalMemFree(hi35xx); + hi35xx = NULL; + } + return ret; + } + ``` + +- **Release 函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > 无 + > + > **函数说明:** + > 释放内存和删除控制器,该函数需要在驱动入口结构体中赋值给 Release 接口, 当HDF框架调用Init函数初始化驱动失败时,可以调用 Release 释放驱动资源。 + + ```c + static void Hi35xxI2cRelease(struct HdfDeviceObject *device) + { + ... + //与Hi35xxI2cInit一样,需要将对每个节点分别进行释放 + DEV_RES_NODE_FOR_EACH_CHILD_NODE(device->property, childNode) { + Hi35xxI2cRemoveByNode(childNode);//函数定义见下 + } + } + + static void Hi35xxI2cRemoveByNode(const struct DeviceResourceNode *node) + { + ... + //【必要】可以调用 I2cCntlrGet 函数通过设备的 busid 获取 I2cCntlr 对象, 以及调用 I2cCntlrRemove 函数来释放 I2cCntlr 对象的内容 + cntlr = I2cCntlrGet(bus); + if (cntlr != NULL && cntlr->priv == node) { + ... + I2cCntlrRemove(cntlr); + //【必要】解除地址映射,锁和内存的释放 + hi35xx = (struct Hi35xxI2cCntlr *)cntlr; + OsalIoUnmap((void *)hi35xx->regBase); + (void)OsalSpinDestroy(&hi35xx->spin); + OsalMemFree(hi35xx); + } + return; + } + ``` + diff --git a/zh-cn/device-dev/driver/driver-platform-mipidsi-develop.md b/zh-cn/device-dev/driver/driver-platform-mipidsi-develop.md new file mode 100755 index 0000000000..636950be2d --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-mipidsi-develop.md @@ -0,0 +1,235 @@ +# MIPI-DSI + +- [概述](#1) +- [开发步骤](#2) +- [开发实例](#3) + +## 概述 + +DSI(Display Serial Interface)是由移动行业处理器接口联盟(Mobile Industry Processor Interface \(MIPI\) Alliance)制定的规范。在HDF框架中, +MIPI-DSI的接口适配模式采用无服务模式,用于不需要在用户态提供API的设备类型,或者没有用户态和内核区分的OS系统,其关联方式是DevHandle直接指向设备对象内核态地址(DevHandle是一个void类型指针)。 + +图 1 无服务模式结构图 +![image1](figure/无服务模式.png) + +## 开发步骤 + +MIPI-DSI模块适配的三个环节是配置属性文件,实例化驱动入口,以及实例化核心层接口函数。 + +1. **实例化驱动入口:** + + - 实例化HdfDriverEntry结构体成员。 + - 调用HDF_INIT将HdfDriverEntry实例化对象注册到HDF框架中。 + +2. **配置属性文件:** + + - 在device_info.hcs文件中添加deviceNode描述。 + - 【可选】添加mipidsi_config.hcs器件属性文件。 + +3. **实例化MIPIDSI控制器对象:** + + - 初始化MipiDsiCntlr成员。 + - 实例化MipiDsiCntlr成员MipiDsiCntlrMethod,其定义和成员**说明**见下 + +4. **驱动调试:** + - 【可选】针对新增驱动程序,建议验证驱动基本功能,例如挂载后的信息反馈,数据传输的成功与否等。 + +> ![](../public_sys-resources/icon-note.gif) **说明:** +> +> MipiDsiCntlrMethod定义 +> +> ```c +> struct MipiDsiCntlrMethod { // 核心层结构体的成员函数 +> int32_t (*setCntlrCfg)(struct MipiDsiCntlr *cntlr); +> int32_t (*setCmd)(struct MipiDsiCntlr *cntlr, struct DsiCmdDesc *cmd); +> int32_t (*getCmd)(struct MipiDsiCntlr *cntlr, struct DsiCmdDesc *cmd, uint32_t readLen, uint8_t *out); +> void (*toHs)(struct MipiDsiCntlr *cntlr); +> void (*toLp)(struct MipiDsiCntlr *cntlr); +> void (*enterUlps)(struct MipiDsiCntlr *cntlr);//【可选】进入超低功耗模式 +> void (*exitUlps)(struct MipiDsiCntlr *cntlr); //【可选】退出超低功耗模式 +> int32_t (*powerControl)(struct MipiDsiCntlr *cntlr, uint8_t enable);//【可选】使能/去使能功耗控制 +> int32_t (*attach)(struct MipiDsiCntlr *cntlr);//【可选】将一个DSI设备连接上host +> }; +> ``` +> +> 表1 MipiDsiCntlrMethod成员的回调函数功能说明 +> |成员函数|入参|出参|返回状态|功能| +> |-|-|-|-|-| +> |setCntlrCfg | **cntlr**: 结构体指针,MipiDsi控制器 ; | 无 |HDF_STATUS相关状态 | 设置控制器参数 | +> |setCmd | **cntlr**: 结构体指针,MipiDsi控制器 ;
**cmd**: 结构体指针,指令传入值|无|HDF_STATUS相关状态| 向显示设备发送指令 | +> |getCmd | **cntlr**: 结构体指针,MipiDsi控制器 ; |**cmd**: 结构体指针,
用于传出指令值;|HDF_STATUS相关状态|从显示设备读取信息指令 | +> |toHs | **cntlr**: 结构体指针,MipiDsi控制器 ; | 无 |HDF_STATUS相关状态 |设置为高速模式| +> |toLp | **cntlr**: 结构体指针,MipiDsi控制器 ; | 无 |HDF_STATUS相关状态 |设置为低电模式 | + +## 开发实例 + +下方将以mipi_tx_hi35xx.c为示例,展示需要厂商提供哪些内容来完整实现设备功能。 + + +1. 一般来说,驱动开发首先需要在 xx_config.hcs 中配置器件属性,并在device_info.hcs文件中添加deviceNode描述。器件属性值与核心层MipiDsiCntlr 成员的默认值或限制范围有密切关系,deviceNode信息与驱动入口注册相关。 + + **但本例中MIPI控制器无需配置额外属性,如有厂商需要,则需要在device_info文件的deviceNode增加deviceMatchAttr信息,以及增加mipidsi_config文件**。 + +- device_info.hcs 配置参考 + + ```c + root { + device_info { + match_attr = "hdf_manager"; + platform :: host { + hostName = "platform_host"; + priority = 50; + device_mipi_dsi:: device { + device0 :: deviceNode { + policy = 0; + priority = 150; + permission = 0644; + moduleName = "HDF_MIPI_TX"; //【必要】用于指定驱动名称,需要与期望的驱动Entry中的moduleName一致; + serviceName = "HDF_MIPI_TX"; // 【必要且唯一】驱动对外发布服务的名称 + } + } + } + } + } + ``` + +2. 完成器件属性文件的配置之后,下一步请实例化驱动入口,驱动入口必须为HdfDriverEntry(在 hdf_device_desc.h 中定义)类型的全局变量,且moduleName要和device_info.hcs中保持一致。HdfDriverEntry结构体的函数指针成员会被厂商操作函数填充,HDF框架会将所有加载的驱动的HdfDriverEntry对象首地址汇总,形成一个类似数组,方便调用。 + + 一般在加载驱动时HDF框架会先调用Bind函数,再调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。 + +- MIPI-DSI驱动入口参考 + + ```c + struct HdfDriverEntry g_mipiTxDriverEntry = { + .moduleVersion = 1, + .Init = Hi35xxMipiTxInit, //见Init参考 + .Release = Hi35xxMipiTxRelease,//见Release参考 + .moduleName = "HDF_MIPI_TX", //【必要】需要与device_info.hcs 中保持一致。 + }; + HDF_INIT(g_mipiTxDriverEntry); //调用HDF_INIT将驱动入口注册到HDF框架中 + ``` + +3. 完成驱动入口注册之后,最后一步就是以核心层MipiDsiCntlr对象的初始化为核心,包括厂商自定义结构体(传递参数和数据),实例化MipiDsiCntlr成员MipiDsiCntlrMethod(让用户可以通过接口来调用驱动底层函数),实现HdfDriverEntry成员函数(Bind,Init,Release) + +- 自定义结构体参考 + + > 从驱动的角度看,自定义结构体是参数和数据的载体,一般来说,config文件中的数值也会用来初始化结构体成员,但本例的mipidsi无器件属性文件,故基本成员结构与MipiDsiCntlr无太大差异。 + + ```c + typedef struct { + unsigned int devno; // 设备号 + short laneId[LANE_MAX_NUM]; // lane号 + OutPutModeTag outputMode; // 输出模式选择:刷新模式,命令行模式和视频流模式 + VideoModeTag videoMode; // 显示设备的同步模式 + OutputFormatTag outputFormat; // 输出DSI图像数据格式:RGB or YUV + SyncInfoTag syncInfo; // 时序相关的设置 + unsigned int phyDataRate; // mbps + unsigned int pixelClk; // KHz + } ComboDevCfgTag; + + // MipiDsiCntlr是核心层控制器结构体,其中的成员在Init函数中会被赋值 + struct MipiDsiCntlr { + struct IDeviceIoService service; + struct HdfDeviceObject *device; + unsigned int devNo; // 设备号 + struct MipiCfg cfg; + struct MipiDsiCntlrMethod *ops; + struct OsalMutex lock; + void *priv; + }; + ``` + +- **【重要】** MipiDsiCntlr成员回调函数结构体MipiDsiCntlrMethod的实例化,其他成员在Init函数中初始化 + + ```c + static struct MipiDsiCntlrMethod g_method = { + .setCntlrCfg = Hi35xxSetCntlrCfg, + .setCmd = Hi35xxSetCmd, + .getCmd = Hi35xxGetCmd, + .toHs = Hi35xxToHs, + .toLp = Hi35xxToLp, + }; + ``` + +- **Init函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 (下表为部分展示,如需使用其他状态,可见//drivers/framework/include/utils/hdf_base.h中HDF_STATUS 定义) + > + > |状态(值)|问题描述| + > |:-|:-:| + > |HDF_ERR_INVALID_OBJECT | 无效对象 | + > |HDF_ERR_MALLOC_FAIL |内存分配失败| + > |HDF_ERR_INVALID_PARAM |无效参数| + > |HDF_ERR_IO |I/O 错误| + > |HDF_SUCCESS |执行成功| + > |HDF_FAILURE |执行失败| + > + > **函数说明:** + > MipiDsiCntlrMethod的实例化对象的挂载,调用MipiDsiRegisterCntlr,以及其他厂商自定义初始化操作 + + ```c + static int32_t Hi35xxMipiTxInit(struct HdfDeviceObject *device) + { + int32_t ret; + g_mipiTx.priv = NULL; //g_mipiTx是定义的全局变量 + //static struct MipiDsiCntlr g_mipiTx { + // .devNo=0 + //}; + g_mipiTx.ops = &g_method;//MipiDsiCntlrMethod的实例化对象的挂载 + ret = MipiDsiRegisterCntlr(&g_mipiTx, device);//【必要】调用核心层函数和g_mipiTx初始化核心层全局变量 + ... + return MipiTxDrvInit(0); //【必要】厂商对设备的初始化,形式不限 + } + + //mipi_dsi_core.c核心层 + int32_t MipiDsiRegisterCntlr(struct MipiDsiCntlr *cntlr, struct HdfDeviceObject *device) + { + ... + //定义的全局变量:static struct MipiDsiHandle g_mipiDsihandle[MAX_CNTLR_CNT]; + if (g_mipiDsihandle[cntlr->devNo].cntlr == NULL) { + (void)OsalMutexInit(&g_mipiDsihandle[cntlr->devNo].lock); + (void)OsalMutexInit(&(cntlr->lock)); + + g_mipiDsihandle[cntlr->devNo].cntlr = cntlr;//初始化MipiDsiHandle成员 + g_mipiDsihandle[cntlr->devNo].priv = NULL; + cntlr->device = device; //使HdfDeviceObject与MmcCntlr可以相互转化的前提 + device->service = &(cntlr->service); //使HdfDeviceObject与MmcCntlr可以相互转化的前提 + cntlr->priv = NULL; + ... + return HDF_SUCCESS; + } + ... + return HDF_FAILURE; + } + ``` + +- **Release函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > 无 + > + > **函数说明:** + > 该函数需要在驱动入口结构体中赋值给 Release 接口, 当 HDF 框架调用 Init 函数初始化驱动失败时,可以调用 Release 释放驱动资源, 该函数中需包含释放内存和删除控制器等操作。所有强制转换获取相应对象的操作**前提**是在Init函数中具备对应赋值的操作。 + + ```c + static void Hi35xxMipiTxRelease(struct HdfDeviceObject *device) + { + struct MipiDsiCntlr *cntlr = NULL; + ... + cntlr = MipiDsiCntlrFromDevice(device);//这里有HdfDeviceObject到MipiDsiCntlr的强制转化 + //return (device == NULL) ? NULL : (struct MipiDsiCntlr *)device->service; + ... + MipiTxDrvExit(); //【必要】对厂商设备所占资源的释放 + MipiDsiUnregisterCntlr(&g_mipiTx); //空函数 + g_mipiTx.priv = NULL; + HDF_LOGI("%s: unload mipi_tx driver 1212!", __func__); + } + ``` + diff --git a/zh-cn/device-dev/driver/driver-platform-mmc-develop.md b/zh-cn/device-dev/driver/driver-platform-mmc-develop.md new file mode 100755 index 0000000000..a8c18fd7bc --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-mmc-develop.md @@ -0,0 +1,379 @@ +# MMC + +- [概述](#1) +- [开发步骤](#2) +- [开发实例](#3) + +## 概述 + +MMC(MultiMedia Card),即多媒体卡,在HDF框架中,MMC的接口适配模式采用独立服务模式,在这种模式下,每一个设备对象会独立发布一个设备服务来处理外部访问,设备管理器收到API的访问请求之后,通过提取该请求的参数,达到调用实际设备对象的相应内部方法的目的。 + +独立服务模式可以直接借助HDFDeviceManager的服务管理能力,但需要为每个设备单独配置设备节点,增加内存占用。 + + +图 1 独立服务模式结构图 +![image1](figure/独立服务模式.png) + + +## 开发步骤 + +MMC模块适配的三个环节是配置属性文件,实例化驱动入口,以及实例化核心层接口函数。 + +1. **实例化驱动入口:** + - 实例化HdfDriverEntry结构体成员。 + - 调用HDF_INIT将HdfDriverEntry实例化对象注册到HDF框架中。 + +2. **配置属性文件:** + + - 在device_info.hcs文件中添加deviceNode描述。 + - 【可选】添加mmc_config.hcs器件属性文件。 + +3. **实例化MMC控制器对象:** + - 初始化MmcCntlr成员。 + - 实例化MmcCntlr成员MmcCntlrOps,其定义和成员说明见下 + +4. **驱动调试:** + - 【可选】针对新增驱动程序,建议验证驱动基本功能,例如挂载后的信息反馈,设备启动是否成功等。 + +> ![](../public_sys-resources/icon-note.gif) **说明:** +> +> MmcCntlrOps定义 +> ```c +> struct MmcCntlrOps { +> int32_t (*request)(struct MmcCntlr *cntlr, struct MmcCmd *cmd); +> int32_t (*setClock)(struct MmcCntlr *cntlr, uint32_t clock); +> int32_t (*setPowerMode)(struct MmcCntlr *cntlr, enum MmcPowerMode mode); +> int32_t (*setBusWidth)(struct MmcCntlr *cntlr, enum MmcBusWidth width); +> int32_t (*setBusTiming)(struct MmcCntlr *cntlr, enum MmcBusTiming timing); +> int32_t (*setSdioIrq)(struct MmcCntlr *cntlr, bool enable); +> int32_t (*hardwareReset)(struct MmcCntlr *cntlr); +> int32_t (*systemInit)(struct MmcCntlr *cntlr); +> int32_t (*setEnhanceSrobe)(struct MmcCntlr *cntlr, bool enable); +> int32_t (*switchVoltage)(struct MmcCntlr *cntlr, enum MmcVolt volt); +> bool (*devReadOnly)(struct MmcCntlr *cntlr); +> bool (*devPluged)(struct MmcCntlr *cntlr); +> bool (*devBusy)(struct MmcCntlr *cntlr); +> int32_t (*tune)(struct MmcCntlr *cntlr, uint32_t cmdCode); +> int32_t (*rescanSdioDev)(struct MmcCntlr *cntlr); +> }; +> ``` +> +> 表1 MmcCntlrOps结构体成员的回调函数功能说明 +> |成员函数|入参|返回值|功能| +> |-|-|-|-| +> |doRequest |**cntlr**: 核心层结构体指针,mmc控制器 ;
**cmd**: 结构体指针,传入命令值 |HDF_STATUS相关状态|request相应处理| +> |setClock |**cntlr**: 核心层结构体指针,mmc控制器 ;
**clock**: 时钟传入值 |HDF_STATUS相关状态|设置时钟频率| +> |setPowerMode |**cntlr**: 核心层结构体指针,mmc控制器 ;
**mode**: 枚举值(见MmcPowerMode定义),功耗模式 |HDF_STATUS相关状态|设置功耗模式| +> |setBusWidth |**cntlr**: 核心层结构体指针,mmc控制器 ;
**width**: 枚举值(见MmcBusWidth定义),总线带宽 |HDF_STATUS相关状态|设置总线带宽| +> |setBusTiming |**cntlr**: 核心层结构体指针,mmc控制器 ;
**timing**: 枚举值(见MmcBusTiming定义),总线时序 |HDF_STATUS相关状态|设置总线时序| +> |setSdioIrq |**cntlr**: 核心层结构体指针,mmc控制器 ;
**enable**: 布尔值,控制中断|HDF_STATUS相关状态|使能/去使能SDIO中断| +> |hardwareReset |**cntlr**: 核心层结构体指针,mmc控制器 ; |HDF_STATUS相关状态|复位硬件 | +> |systemInit |**cntlr**: 核心层结构体指针,mmc控制器 ; |HDF_STATUS相关状态|系统初始化 | +> |setEnhanceSrobe |**cntlr**: 核心层结构体指针,mmc控制器 ;
**enable**: 布尔值,设置功能 |HDF_STATUS相关状态|设置增强选通 | +> |switchVoltage |**cntlr**: 核心层结构体指针,mmc控制器 ;
**volt**: 枚举值,电压值(3.3,1.8,1.2V); |HDF_STATUS相关状态|设置电压值 | +> |devReadOnly |**cntlr**: 核心层结构体指针,mmc控制器 ; |布尔值 |检验设备是否只读 | +> |cardPluged |**cntlr**: 核心层结构体指针,mmc控制器 ; |布尔值 |检验设备是否拔出| +> |devBusy |**cntlr**: 核心层结构体指针,mmc控制器 ; |布尔值 |检验设备是否忙碌| +> |tune |**cntlr**: 核心层结构体指针,mmc控制器 ;
**cmdCode**: uint32_t,命令代码; |HDF_STATUS相关状态|调谐 | +> |rescanSdioDev |**cntlr**: 核心层结构体指针,mmc控制器 ; |HDF_STATUS相关状态|扫描并添加SDIO设备,| + + +## 开发实例 + +下方将以himci.c为示例,展示需要厂商提供哪些内容来完整实现设备功能。 + +1. 驱动开发首先需要实例化驱动入口,驱动入口必须为HdfDriverEntry(在 hdf_device_desc.h 中定义)类型的全局变量,且moduleName要和device_info.hcs中保持一致。HDF框架会将所有加载的驱动的HdfDriverEntry对象首地址汇总,形成一个类似数组的段地址空间,方便上层调用。 + + 一般在加载驱动时HDF会先调用Bind函数,再调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。 + +- MMC驱动入口参考 + + ```c + struct HdfDriverEntry g_mmcDriverEntry = { + .moduleVersion = 1, + .Bind = HimciMmcBind, //见Bind参考 + .Init = HimciMmcInit, //见Init参考 + .Release = HimciMmcRelease, //见Release参考 + .moduleName = "hi3516_mmc_driver",//【必要且与HCS文件中里面的moduleName匹配】 + }; + HDF_INIT(g_mmcDriverEntry); //调用HDF_INIT将驱动入口注册到HDF框架中 + ``` + +2. 完成驱动入口注册之后,下一步请在device_info.hcs文件中添加deviceNode信息,并在mmc_config.hcs中配置器件属性。deviceNode信息与驱动入口注册相关,器件属性值与核心层MmcCntlr成员的默认值或限制范围有密切关系。 + + **如有多个器件信息,则需要在device_info文件增加deviceNode信息,以及在mmc_config文件中增加对应的器件属性。**。 + + +- device_info.hcs 配置参考 + + ```c + root { + device_info { + match_attr = "hdf_manager"; + platform :: host { + hostName = "platform_host"; + priority = 50; + device_mmc:: device { + device0 :: deviceNode { + policy = 2; + priority = 10; + permission = 0644; + moduleName = "hi3516_mmc_driver"; //【必要】用于指定驱动名称,需要与驱动Entry中的moduleName一致; + serviceName = "HDF_PLATFORM_MMC_0"; //【必要】驱动对外发布服务的名称,必须唯一 + deviceMatchAttr = "hi3516_mmc_emmc";//【必要】用于配置控制器私有数据,要与 mmc_config.hcs 中对应控制器保持一致 + } + device1 :: deviceNode { + policy = 1; + priority = 20; + permission = 0644; + moduleName = "hi3516_mmc_driver"; + serviceName = "HDF_PLATFORM_MMC_1"; + deviceMatchAttr = "hi3516_mmc_sd"; //SD类型 + } + device2 :: deviceNode { + policy = 1; + priority = 30; + permission = 0644; + moduleName = "hi3516_mmc_driver"; + serviceName = "HDF_PLATFORM_MMC_2"; + deviceMatchAttr = "hi3516_mmc_sdio";//SDIO类型 + } + } + } + } + } + ``` + +- mmc_config.hcs 配置参考 + + ```c + root { + platform { + mmc_config { + template mmc_controller {//模板公共参数,继承该模板的节点如果使用模板中的默认值,则节点字段可以缺省 + match_attr = ""; + voltDef = 0; // 3.3V + freqMin = 50000; //【必要】最小频率值 + freqMax = 100000000; //【必要】最大频率值 + freqDef = 400000; //【必要】默认频率值 + maxBlkNum = 2048; //【必要】最大的block号 + maxBlkSize = 512; //【必要】最大的block个数 + ocrDef = 0x300000; //【必要】工作电压设置相关 + caps2 = 0; //【必要】属性寄存器相关,见mmc_caps.h 中 MmcCaps2 定义 + regSize = 0x118; //【必要】寄存器位宽 + hostId = 0; //【必要】主机号 + regBasePhy = 0x10020000;//【必要】寄存器物理基地址 + irqNum = 63; //【必要】中断号 + devType = 2; //【必要】模式选择:emmc, SD, SDIO ,COMBO + caps = 0x0001e045; //【必要】属性寄存器相关,见mmc_caps.h 中 MmcCaps 定义 + } + controller_0x10100000 :: mmc_controller { + match_attr = "hi3516_mmc_emmc";//【必要】需要和device_info.hcs中的deviceMatchAttr值一致 + hostId = 0; + regBasePhy = 0x10100000; + irqNum = 96; + devType = 0; // emmc类型 + caps = 0xd001e045; + caps2 = 0x60; + } + controller_0x100f0000 :: mmc_controller { + match_attr = "hi3516_mmc_sd"; + hostId = 1; + regBasePhy = 0x100f0000; + irqNum = 62; + devType = 1; // sd类型 + caps = 0xd001e005; + } + controller_0x10020000 :: mmc_controller { + match_attr = "hi3516_mmc_sdio"; + hostId = 2; + regBasePhy = 0x10020000; + irqNum = 63; + devType = 2; // sdio类型 + caps = 0x0001e04d; + } + } + } + } + ``` + +3. 完成驱动入口注册之后,最后一步就是以核心层MmcCntlr对象的初始化为核心,包括厂商自定义结构体(传递参数和数据),实例化MmcCntlr成员MmcCntlrOps(让用户可以通过接口来调用驱动底层函数),实现HdfDriverEntry成员函数(Bind,Init,Release) + +- 自定义结构体参考 + + > 从驱动的角度看,自定义结构体是参数和数据的载体,而且mmc_config.hcs文件中的数值会被HDF读入通过DeviceResourceIface来初始化结构体成员 ,一些重要数值也会传递给核心层对象。 + + ```c + struct HimciHost { + struct MmcCntlr *mmc;//【必要】核心层结构体 + struct MmcCmd *cmd; //【必要】核心层结构体,传递命令的,相关命令见枚举量 MmcCmdCode + //【可选】根据厂商驱动需要添加 + void *base; + enum HimciPowerStatus powerStatus; + uint8_t *alignedBuff; + uint32_t buffLen; + struct scatterlist dmaSg; + struct scatterlist *sg; + uint32_t dmaSgNum; + DMA_ADDR_T dmaPaddr; + uint32_t *dmaVaddr; + uint32_t irqNum; + bool isTuning; + uint32_t id; + struct OsalMutex mutex; + bool waitForEvent; + HIMCI_EVENT himciEvent; + }; + //MmcCntlr是核心层控制器结构体,其中的成员在bind函数中会被赋值 + struct MmcCntlr { + struct IDeviceIoService service; + struct HdfDeviceObject *hdfDevObj; + struct PlatformDevice device; + struct OsalMutex mutex; + struct OsalSem released; + uint32_t devType; + struct MmcDevice *curDev; + struct MmcCntlrOps *ops; + struct PlatformQueue *msgQueue; + uint16_t index; + uint16_t voltDef; + uint32_t vddBit; + uint32_t freqMin; + uint32_t freqMax; + uint32_t freqDef; + union MmcOcr ocrDef; + union MmcCaps caps; + union MmcCaps2 caps2; + uint32_t maxBlkNum; + uint32_t maxBlkSize; + uint32_t maxReqSize; + bool devPluged; + bool detecting; + void *priv; + }; + ``` + + +- **【重要】** MmcCntlr成员回调函数结构体MmcCntlrOps的实例化,其他成员在Bind函数中初始化 + + ```c + static struct MmcCntlrOps g_himciHostOps = { + .request = HimciDoRequest, + .setClock = HimciSetClock, + .setPowerMode = HimciSetPowerMode, + .setBusWidth = HimciSetBusWidth, + .setBusTiming = HimciSetBusTiming, + .setSdioIrq = HimciSetSdioIrq, + .hardwareReset = HimciHardwareReset, + .systemInit = HimciSystemInit, + .setEnhanceSrobe= HimciSetEnhanceSrobe, + .switchVoltage = HimciSwitchVoltage, + .devReadOnly = HimciDevReadOnly, + .devPluged = HimciCardPluged, + .devBusy = HimciDevBusy, + .tune = HimciTune, + .rescanSdioDev = HimciRescanSdioDev, + }; + ``` + +- **Bind函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 (下表为部分展示,如需使用其他状态,可见//drivers/framework/include/utils/hdf_base.h中HDF_STATUS 定义) + > + > |状态(值)|问题描述| + > |:-|:-:| + > |HDF_ERR_INVALID_OBJECT|控制器对象非法| + > |HDF_ERR_MALLOC_FAIL |内存分配失败| + > |HDF_ERR_INVALID_PARAM |参数非法| + > |HDF_ERR_IO |I/O 错误| + > |HDF_SUCCESS |初始化成功| + > |HDF_FAILURE |初始化失败| + > + > **函数说明:** + > MmcCntlr,HimciHost,HdfDeviceObject之间互相赋值,方便其他函数可以相互转化,初始化自定义结构体HimciHost对象,初始化MmcCntlr成员,调用核心层MmcCntlrAdd函数。 + + ```c + static int32_t HimciMmcBind(struct HdfDeviceObject *obj) + { + struct MmcCntlr *cntlr = NULL; + struct HimciHost *host = NULL; + int32_t ret; + cntlr = (struct MmcCntlr *)OsalMemCalloc(sizeof(struct MmcCntlr)); + host = (struct HimciHost *)OsalMemCalloc(sizeof(struct HimciHost)); + + host->mmc = cntlr; //【必要】使HimciHost与MmcCntlr可以相互转化的前提 + cntlr->priv = (void *)host; //【必要】使HimciHost与MmcCntlr可以相互转化的前提 + cntlr->ops = &g_himciHostOps; //【必要】MmcCntlrOps的实例化对象的挂载 + cntlr->hdfDevObj = obj; //【必要】使HdfDeviceObject与MmcCntlr可以相互转化的前提 + obj->service = &cntlr->service; //【必要】使HdfDeviceObject与MmcCntlr可以相互转化的前提 + ret = MmcCntlrParse(cntlr, obj); //【必要】 初始化 cntlr. 失败就 goto _ERR; + ... + ret = HimciHostParse(host, obj); //【必要】 初始化 host对象的相关属性,失败就 goto _ERR; + ... + ret = HimciHostInit(host, cntlr);//厂商自定义的初始化,失败就 goto _ERR; + ... + ret = MmcCntlrAdd(cntlr); //调用核心层函数 失败就 goto _ERR; + ... + (void)MmcCntlrAddDetectMsgToQueue(cntlr);//将卡检测消息添加到队列中。 + HDF_LOGD("HimciMmcBind: success."); + return HDF_SUCCESS; + _ERR: + HimciDeleteHost(host); + HDF_LOGD("HimciMmcBind: fail, err = %d.", ret); + return ret; + } + ``` + + +- **Init函数参考** + + > **入参:** + > HdfDeviceObject是整个驱动对外暴露的接口参数,具备HCS配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 + > + > **函数说明:** + > 实现ProcMciInit + + ```c + static int32_t HimciMmcInit(struct HdfDeviceObject *obj) + { + static bool procInit = false; + (void)obj; + if (procInit == false) { + if (ProcMciInit() == HDF_SUCCESS) { + procInit = true; + HDF_LOGD("HimciMmcInit: proc init success."); + } + } + HDF_LOGD("HimciMmcInit: success."); + return HDF_SUCCESS; + } + ``` + +- **Release函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > 无 + > + > **函数说明:** + > 释放内存和删除控制器等操作,该函数需要在驱动入口结构体中赋值给Release接口,当HDF框架调用Init函数初始化驱动失败时,可以调用 Release释放驱动资源。所有强制转换获取相应对象的操作**前提**是在Init函数中具备对应赋值的操作。 + + ```c + static void HimciMmcRelease(struct HdfDeviceObject *obj) + { + struct MmcCntlr *cntlr = NULL; + ... + cntlr = (struct MmcCntlr *)obj->service;//这里有HdfDeviceObject到MmcCntlr的强制转化,通过service成员,赋值见Bind函数 + ... + HimciDeleteHost((struct HimciHost *)cntlr->priv);//厂商自定义的内存释放函数,这里有MmcCntlr到HimciHost的强制转化 + } + ``` diff --git a/zh-cn/device-dev/driver/driver-platform-pwm-develop.md b/zh-cn/device-dev/driver/driver-platform-pwm-develop.md new file mode 100755 index 0000000000..12ccdece68 --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-pwm-develop.md @@ -0,0 +1,275 @@ +# PWM + +- [概述](#1) +- [开发步骤](#2) +- [开发实例](#3) + +## 概述 + +PWM(Pulse Width Modulator)即脉冲宽度调节器,在HDF框架中,PWM的接口适配模式采用独立服务模式,在这种模式下,每一个设备对象会独立发布一个设备服务来处理外部访问,设备管理器收到API的访问请求之后,通过提取该请求的参数,达到调用实际设备对象的相应内部方法的目的。 + +独立服务模式可以直接借助HDFDeviceManager的服务管理能力,但需要为每个设备单独配置设备节点,增加内存占用。 + +图 1 独立服务模式结构图 +![image1](figure/独立服务模式.png) + +## 开发步骤 + +PWM模块适配HDF框架的三个环节是配置属性文件,实例化驱动入口,以及填充核心层接口函数。 + +1. **实例化驱动入口:** + - 实例化HdfDriverEntry结构体成员。 + - 调用HDF_INIT将HdfDriverEntry实例化对象注册到HDF框架中。 + +2. **配置属性文件:** + + - 在device_info.hcs文件中添加deviceNode描述。 + - 【可选】添加pwm_config.hcs器件属性文件。 + +3. **实例化PWM控制器对象:** + + - 初始化PwmDev成员。 + - 实例化PwmDev成员PwmMethod,其定义和成员说明见下 + +4. **驱动调试:** + - 【可选】针对新增驱动程序,建议验证驱动基本功能,例如PWM控制状态,中断响应情况等。 + +> ![](../public_sys-resources/icon-note.gif) **说明:** +> +> PwmMethod定义 +> ```c +> struct PwmMethod { +> int32_t (*setConfig)(struct PwmDev *pwm, struct PwmConfig *config); +> int32_t (*open)(struct PwmDev *pwm); +> int32_t (*close)(struct PwmDev *pwm); +> }; +> ``` +> +> 表1 PwmMethod结构体成员的回调函数功能说明 +> |成员函数|入参|返回值|功能| +> |-|-|-|-| +> |setConfig|**pwm**: 结构体指针,核心层PWM控制器;
**config**: 结构体指针,属性传入值;|HDF_STATUS相关状态|配置属性| +> |open |**pwm**: 结构体指针,核心层PWM控制器;|HDF_STATUS相关状态|打开设备| +> |close|**pwm**: 结构体指针,核心层PWM控制器;|HDF_STATUS相关状态|关闭设备| + +## 开发实例 + +下方将以pwm_hi35xx.c为示例,展示需要厂商提供哪些内容来完整实现设备功能。 + +1. 驱动开发首先需要实例化驱动入口,驱动入口必须为HdfDriverEntry(在 hdf_device_desc.h 中定义)类型的全局变量,且moduleName要和device_info.hcs中保持一致。HDF框架会将所有加载的驱动的HdfDriverEntry对象首地址汇总,形成一个类似数组的段地址空间,方便上层调用。 + + 一般在加载驱动时HDF会先调用Bind函数,再调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。 + +- PWM驱动入口参考 + + ```c + struct HdfDriverEntry g_hdfPwm = { + .moduleVersion = 1, + .moduleName = "HDF_PLATFORM_PWM",//【必要 且与 HCS文件中里面的moduleName匹配】 + .Bind = HdfPwmBind, + .Init = HdfPwmInit, + .Release = HdfPwmRelease, + }; + //调用HDF_INIT将驱动入口注册到HDF框架中 + HDF_INIT(g_hdfPwm); + ``` + +2. 完成驱动入口注册之后,下一步请在device_info.hcs文件中添加deviceNode信息,并在 pwm_config.hcs 中配置器件属性。deviceNode信息与驱动入口注册相关,器件属性值与核心层PwmDev成员的默认值或限制范围有密切关系。 + + **如有更多个器件信息,则需要在device_info文件增加deviceNode信息,以及在pwm_config文件中增加对应的器件属性**。 + +- device_info.hcs 配置参考 + + ```c + root { + device_info { + platform :: host { + hostName = "platform_host"; + priority = 50; + device_pwm :: device {//为每一个 pwm 控制器配置一个HDF设备节点,存在多个时【必须】添加,否则不用 + device0 :: deviceNode { + policy = 1; // 等于1,向内核态发布服务 + priority = 80; // 驱动启动优先级 + permission = 0644;// 驱动创建设备节点权限 + moduleName = "HDF_PLATFORM_PWM"; //【必要】用于指定驱动名称,需要与期望的驱动Entry中的moduleName一致; + serviceName = "HDF_PLATFORM_PWM_0";//【必要且唯一】驱动对外发布服务的名称 + deviceMatchAttr = "hisilicon_hi35xx_pwm_0";//【必要】用于配置控制器私有数据,要与 pwm_config.hcs 中对应 + // 控制器保持一致,具体的控制器信息在 pwm_config.hcs 中 + } + device1 :: deviceNode { + policy = 1; + priority = 80; + permission = 0644; + moduleName = "HDF_PLATFORM_PWM"; + serviceName = "HDF_PLATFORM_PWM_1"; + deviceMatchAttr = "hisilicon_hi35xx_pwm_1"; + } + } + } + } + } + ``` + +- pwm_config.hcs 配置参考 + + ```c + root { + platform { + pwm_config { + template pwm_device { //【必要】模板配置,继承该模板的节点如果使用模板中的默认值,则节点字段可以缺省 + serviceName = ""; + match_attr = ""; + num = 0; //【必要】设备号 + base = 0x12070000; //【必要】地址映射需要 + } + device_0x12070000 :: pwm_device { + match_attr = "hisilicon_hi35xx_pwm_0";//【必要】需要和device_info.hcs中的deviceMatchAttr值一致 + } + device_0x12070020 :: pwm_device { //存在多个设备时【必须】添加,否则不用 + match_attr = "hisilicon_hi35xx_pwm_1"; + num = 1; + base = 0x12070020; //【必要】地址映射需要 + } + } + } + } + ``` + +3. 完成驱动入口注册之后,最后一步就是以核心层PwmDev对象的初始化为核心,包括厂商自定义结构体(传递参数和数据),实例化PwmDev成员PwmMethod(让用户可以通过接口来调用驱动底层函数),实现HdfDriverEntry成员函数(Bind,Init,Release) + +- 自定义结构体参考 + + > 从驱动的角度看,自定义结构体是参数和数据的载体,而且pwm_config.hcs文件中的数值会被HDF读入通过DeviceResourceIface来初始化结构体成员,一些重要数值也会传递给核心层对象,例如设备号等。 + + ```c + struct HiPwm { + struct PwmDev dev; //【必要】 核心层结构体 + volatile unsigned char *base; + struct HiPwmRegs *reg; // 设备属性结构体,可自定义 + bool supportPolarity; + }; + + // PwmDev是核心层控制器结构体,其中的成员在Init函数中会被赋值 + struct PwmDev { + struct IDeviceIoService service; + struct HdfDeviceObject *device; + struct PwmConfig cfg; //属性结构体,相关定义见下 + struct PwmMethod *method; //钩子函数模板 + bool busy; + uint32_t num; //设备号 + OsalSpinlock lock; + void *priv; //私有数据,一般存储自定义结构体首地址,方便调用 + }; + struct PwmConfig { + uint32_t duty; // 占空时间 nanoseconds + uint32_t period; // pwm 周期 nanoseconds + uint32_t number; // pwm 连续个数 + uint8_t polarity; // Polarity + // ------------------- | -------------- + // PWM_NORMAL_POLARITY | Normal polarity + // PWM_INVERTED_POLARITY | Inverted polarity + // + uint8_t status; // 运行状态 + // ------------------ | ----------------- + // PWM_DISABLE_STATUS | Disabled + // PWM_ENABLE_STATUS | Enabled + }; + ``` + + +- **【重要】** PwmDev成员回调函数结构体PwmMethod的实例化,其他成员在Init函数中初始化 + + ```c + // pwm_hi35xx.c 中的示例:钩子函数的填充 + struct PwmMethod g_pwmOps = { + .setConfig = HiPwmSetConfig,//配置属性 + }; + ``` + +- **Init函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 (下表为部分展示,如需使用其他状态,可见//drivers/framework/include/utils/hdf_base.h中HDF_STATUS 定义) + > + > |状态(值)|问题描述| + > |:-|:-:| + > |HDF_ERR_INVALID_OBJECT|控制器对象非法| + > |HDF_ERR_MALLOC_FAIL |内存分配失败| + > |HDF_ERR_INVALID_PARAM |参数非法| + > |HDF_ERR_IO |I/O 错误| + > |HDF_SUCCESS |初始化成功| + > |HDF_FAILURE |初始化失败| + > + > **函数说明:** + > 初始化自定义结构体对象,初始化PwmDev成员,调用核心层PwmDeviceAdd函数。 + + ```c + //此处bind函数为空函数,可与init函数结合,也可根据厂商需要实现相关操作 + static int32_t HdfPwmBind(struct HdfDeviceObject *obj) + { + (void)obj; + return HDF_SUCCESS; + } + //挂载init的 + static int32_t HdfPwmInit(struct HdfDeviceObject *obj) + { + int ret; + struct HiPwm *hp = NULL; + ... + hp = (struct HiPwm *)OsalMemCalloc(sizeof(*hp)); + ... + ret = HiPwmProbe(hp, obj); //【必要】实现见下 + ... + return ret; + } + + static int32_t HiPwmProbe(struct HiPwm *hp, struct HdfDeviceObject *obj) + { + uint32_t tmp; + struct DeviceResourceIface *iface = NULL; + + iface = DeviceResourceGetIfaceInstance(HDF_CONFIG_SOURCE);//初始化自定义结构体HiPwm + ... + + hp->reg = (struct HiPwmRegs *)hp->base; //初始化自定义结构体HiPwm + hp->supportPolarity = false; //初始化自定义结构体HiPwm + hp->dev.method = &g_pwmOps; //PwmMethod的实例化对象的挂载 + hp->dev.cfg.duty = PWM_DEFAULT_DUTY_CYCLE; //初始化PwmDev + hp->dev.cfg.period = PWM_DEFAULT_PERIOD; //初始化PwmDev + hp->dev.cfg.polarity = PWM_DEFAULT_POLARITY; //初始化PwmDev + hp->dev.cfg.status = PWM_DISABLE_STATUS; //初始化PwmDev + hp->dev.cfg.number = 0; //初始化PwmDev + hp->dev.busy = false; //初始化PwmDev + if (PwmDeviceAdd(obj, &(hp->dev)) != HDF_SUCCESS) {//【重要】调用核心层函数,初始化hp->dev 的设备和服务 + OsalIoUnmap((void *)hp->base); + return HDF_FAILURE; + } + return HDF_SUCCESS; + } + ``` + +- **Release 函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > 无 + > + > **函数说明:** + > 释放内存和删除控制器,该函数需要在驱动入口结构体中赋值给 Release 接口, 当HDF框架调用Init函数初始化驱动失败时,可以调用 Release 释放驱动资源。 + + ```c + static void HdfPwmRelease(struct HdfDeviceObject *obj) + { + struct HiPwm *hp = NULL; + ... + hp = (struct HiPwm *)obj->service;//这里有HdfDeviceObject到HiPwm的强制转化 + ... + PwmDeviceRemove(obj, &(hp->dev));//【必要】调用核心层函数,释放PwmDev的设备和服务,这里有HiPwm到PwmDev的强制转化 + HiPwmRemove(hp); //释放HiPwm + } + ``` \ No newline at end of file diff --git a/zh-cn/device-dev/driver/driver-platform-rtc-develop.md b/zh-cn/device-dev/driver/driver-platform-rtc-develop.md new file mode 100755 index 0000000000..4e31ceff43 --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-rtc-develop.md @@ -0,0 +1,294 @@ +# RTC + +- [概述](#1) +- [开发步骤](#2) +- [开发实例](#3) + +## 概述 + +RTC(real-time clock)为操作系统中的实时时钟设备,在HDF框架中,RTC的接口适配模式采用独立服务模式,在这种模式下,每一个设备对象会独立发布一个设备服务来处理外部访问,设备管理器收到API的访问请求之后,通过提取该请求的参数,达到调用实际设备对象的相应内部方法的目的。 + +独立服务模式可以直接借助HDFDeviceManager的服务管理能力,但需要为每个设备单独配置设备节点,增加内存占用。 + +图 1 独立服务模式结构图 +![image1](figure/独立服务模式.png) + +## 开发步骤 + +RTC模块适配HDF框架的三个环节是配置属性文件,实例化驱动入口,以及填充核心层接口函数。 + + +1. **实例化驱动入口:** + - 实例化HdfDriverEntry结构体成员。 + - 调用HDF_INIT将HdfDriverEntry实例化对象注册到HDF框架中。 + +2. **配置属性文件:** + + - 在device_info.hcs文件中添加deviceNode描述。 + - 【可选】添加rtc_config.hcs器件属性文件。 + +3. **实例化RTC控制器对象:** + + - 初始化RtcHost成员。 + - 实例化RtcHost成员RtcMethod,其定义和成员说明见下 + +4. **驱动调试:** + - 【可选】针对新增驱动程序,建议验证驱动基本功能,例如RTC控制状态,中断响应情况等。 + +> ![](../public_sys-resources/icon-note.gif) **说明:** +> +> RtcMethod定义 +> +> ```c +> struct RtcMethod { +> int32_t (*ReadTime)(struct RtcHost *host, struct RtcTime *time); +> int32_t (*WriteTime)(struct RtcHost *host, const struct RtcTime *time); +> int32_t (*ReadAlarm)(struct RtcHost *host, enum RtcAlarmIndex alarmIndex, struct RtcTime *time); +> int32_t (*WriteAlarm)(struct RtcHost *host, enum RtcAlarmIndex alarmIndex, const struct RtcTime *time); +> int32_t (*RegisterAlarmCallback)(struct RtcHost *host, enum RtcAlarmIndex alarmIndex, RtcAlarmCallback cb); +> int32_t (*AlarmInterruptEnable)(struct RtcHost *host, enum RtcAlarmIndex alarmIndex, uint8_t enable); +> int32_t (*GetFreq)(struct RtcHost *host, uint32_t *freq); +> int32_t (*SetFreq)(struct RtcHost *host, uint32_t freq); +> int32_t (*Reset)(struct RtcHost *host); +> int32_t (*ReadReg)(struct RtcHost *host, uint8_t usrDefIndex, uint8_t *value); +> int32_t (*WriteReg)(struct RtcHost *host, uint8_t usrDefIndex, uint8_t value); +> }; +> ``` +> +> 表1 RtcMethod结构体成员的回调函数功能说明 +> +> |函数|入参|出参|返回值|功能| +> |-|-|-|-|-| +> |ReadTime |**host**: 结构体指针,核心层RTC控制器 ; |**time**: 结构体指针,
传出的时间值; |HDF_STATUS相关状态| 读RTC时间信息[^2] | +> |WriteTime |**host**: 结构体指针,核心层RTC控制器 ;
**time**: 结构体指针,时间传入值; |无 |HDF_STATUS相关状态| 写RTC时间信息(包括毫秒~年) | +> |ReadAlarm |**host**: 结构体指针,核心层RTC控制器 ;
**alarmIndex**: 枚举值,闹钟报警索引 ;|**time**: 结构体指针,
传出的时间值;|HDF_STATUS相关状态| 读RTC报警时间信息 | +> |WriteAlarm |**host**: 结构体指针,核心层RTC控制器 ;
**alarmIndex**: 枚举值,闹钟报警索引 ;
**time**: 结构体指针,时间传入值;|无|HDF_STATUS相关状态| 写RTC报警时间信息 | +> |RegisterAlarmCallback|**host**: 结构体指针,核心层RTC控制器 ;
**alarmIndex**: 枚举值,闹钟报警索引 ;
**cb**:函数指针,回调函数; |无|HDF_STATUS相关状态| 注册报警超时回调函数| +> |AlarmInterruptEnable |**host**: 结构体指针,核心层RTC控制器 ;
**alarmIndex**: 枚举值,闹钟报警索引 ;
**enable**: 布尔值,控制报警; |无|HDF_STATUS相关状态| 使能/去使能RTC报警中断 | +> |GetFreq |**host**: 结构体指针,核心层RTC控制器 ; |**freq**: uint32_t指针,
传出的频率值;|HDF_STATUS相关状态| 读RTC外接晶振频率 | +> |SetFreq |**host**: 结构体指针,核心层RTC控制器 ;
**freq**: uint32_t,频率传入值; |无|HDF_STATUS相关状态| 配置RTC外接晶振频率 | +> |Reset |**host**: 结构体指针,核心层RTC控制器 ; |无|HDF_STATUS相关状态| RTC复位 | +> |ReadReg |**host**: 结构体指针,核心层RTC控制器 ;
**usrDefIndex**: 结构体,用户自定义寄存器索引; |**value**: uint8_t指针,
传出的寄存器值;|HDF_STATUS相关状态| 按照用户定义的寄存器索引,
读取对应的寄存器配置,
一个索引对应一字节的配置值 | +> |WriteReg |**host**: 结构体指针,核心层RTC控制器 ;
**usrDefIndex**: 结构体,用户自定义寄存器索引;
**value**: uint8_t,寄存器传入值; |无|HDF_STATUS相关状态| 按照用户定义的寄存器索引,
设置对应的寄存器配置,
一个索引对应一字节的配置值| + +## 开发实例 + +下方将以rtc_hi35xx.c为示例,展示需要厂商提供哪些内容来完整实现设备功能。 + +1. 驱动开发首先需要实例化驱动入口,驱动入口必须为HdfDriverEntry(在 hdf_device_desc.h 中定义)类型的全局变量,且moduleName要和device_info.hcs中保持一致。HDF框架会将所有加载的驱动的HdfDriverEntry对象首地址汇总,形成一个类似数组的段地址空间,方便上层调用。 + + 一般在加载驱动时HDF会先调用Bind函数,再调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。 + +- RTC驱动入口参考 + + ```c + struct HdfDriverEntry g_rtcDriverEntry = { + .moduleVersion = 1, + .Bind = HiRtcBind, //见Bind参考 + .Init = HiRtcInit, //见Init参考 + .Release = HiRtcRelease, //见Release参考 + .moduleName = "HDF_PLATFORM_RTC",//【必要】且与 HCS 里面的名字匹配 + }; + //调用HDF_INIT将驱动入口注册到HDF框架中 + HDF_INIT(g_rtcDriverEntry); + ``` + +2. 完成驱动入口注册之后,下一步请在device_info.hcs文件中添加deviceNode信息,并在 rtc_config.hcs 中配置器件属性。deviceNode信息与驱动入口注册相关,器件属性值与核心层RtcHost成员的默认值或限制范围有密切关系。 + + **本例只有一个RTC控制器,如有多个器件信息,则需要在device_info文件增加deviceNode信息,以及在rtc_config文件中增加对应的器件属性**。 + +- device_info.hcs 配置参考 + + ```c + root { + device_info { + platform :: host { + device_rtc :: device { + device0 :: deviceNode { + policy = 1; //2:用户态可见,1:内核态可见,0:不需要发布服务 + priority = 30; //优先级越大,值越小 + permission = 0644; //驱动创建设备节点权限 + moduleName = "HDF_PLATFORM_RTC"; //【必要】驱动注册名字 + serviceName = "HDF_PLATFORM_RTC"; //【必要】驱动对外发布服务的名称,必须唯一 + deviceMatchAttr = "hisilicon_hi35xx_rtc";//【必要】需要与设备hcs文件中的 match_attr 匹配 + } + } + } + } + } + ``` + +- rtc_config.hcs 配置参考 + + ```c + root { + platform { + rtc_config { + controller_0x12080000 { + match_attr = "hisilicon_hi35xx_rtc";//【必要】需要和device_info.hcs中的deviceMatchAttr值一致 + rtcSpiBaseAddr = 0x12080000; //地址映射相关 + regAddrLength = 0x100; //地址映射相关 + irq = 37; //中断号 + supportAnaCtrl = false; + supportLock = false; + anaCtrlAddr = 0xff; + lock0Addr = 0xff; + lock1Addr = 0xff; + lock2Addr = 0xff; + lock3Addr = 0xff; + } + } + } + } + ``` + +3. 完成驱动入口注册之后,最后一步就是以核心层RtcHost对象的初始化为核心,包括厂商自定义结构体(传递参数和数据),实例化RtcHost成员RtcMethod(让用户可以通过接口来调用驱动底层函数),实现HdfDriverEntry成员函数(Bind,Init,Release) + +- 自定义结构体参考 + + > 从驱动的角度看,自定义结构体是参数和数据的载体,而且rtc_config.hcs文件中的数值会被HDF读入通过DeviceResourceIface来初始化结构体成员。 + + ```c + struct RtcConfigInfo { + uint32_t spiBaseAddr; //地址映射相关 + volatile void *remapBaseAddr; //地址映射相关 + uint16_t regAddrLength; //地址映射相关 + uint8_t supportAnaCtrl; //是否支持anactrl + uint8_t supportLock; //是否支持锁 + uint8_t irq; //中断号 + uint8_t alarmIndex; //闹钟索引 + uint8_t anaCtrlAddr; //anactrl地址 + struct RtcLockAddr lockAddr; //锁地址 + RtcAlarmCallback cb; //回调函数 + struct OsalMutex mutex; //互斥锁 + }; + + // RtcHost是核心层控制器结构体,其中的成员在Init函数中会被赋值 + struct RtcHost { + struct IDeviceIoService service; + struct HdfDeviceObject *device; + struct RtcMethod *method; + void *data; + }; + ``` + +- **【重要】** RtcHost成员回调函数结构体RtcMethod的实例化,其他成员在Init函数中初始化 + + ```c + // rtc_hi35xx.c 中的示例:钩子函数的填充 + static struct RtcMethod g_method = { + .ReadTime = HiRtcReadTime, + .WriteTime = HiRtcWriteTime, + .ReadAlarm = HiReadAlarm, + .WriteAlarm = HiWriteAlarm, + .RegisterAlarmCallback = HiRegisterAlarmCallback, + .AlarmInterruptEnable = HiAlarmInterruptEnable, + .GetFreq = HiGetFreq, + .SetFreq = HiSetFreq, + .Reset = HiReset, + .ReadReg = HiReadReg, + .WriteReg = HiWriteReg, + }; + ``` + + +- **Bind 函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 (下表为部分展示,如需使用其他状态,可见//drivers/framework/include/utils/hdf_base.h中HDF_STATUS 定义) + > + > |状态(值)|问题描述| + > |:-|:-:| + > |HDF_ERR_INVALID_OBJECT|控制器对象非法| + > |HDF_ERR_MALLOC_FAIL |内存分配失败| + > |HDF_ERR_INVALID_PARAM |参数非法| + > |HDF_ERR_IO |I/O 错误| + > |HDF_SUCCESS |初始化成功| + > |HDF_FAILURE |初始化失败| + > **函数说明:** + > 链接HdfDeviceObject对象和RtcHost + + ```c + static int32_t HiRtcBind(struct HdfDeviceObject *device) + { + struct RtcHost *host = NULL; + host = RtcHostCreate(device); //实际是申请内存并挂接device: host->device = device; + //使HdfDeviceObject与RtcHost可以相互转化的前提 + ... + device->service = &host->service;//使HdfDeviceObject与RtcHost可以相互转化的前提 + //方便后续通过调用RtcHostFromDevice 实现全局性质的host 使用 + return HDF_SUCCESS; + } + ``` + + +- **Init函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 + > + > **函数说明:** + > 初始化自定义结构体对象,初始化RtcHost成员。 + + ```c + static int32_t HiRtcInit(struct HdfDeviceObject *device) + { + struct RtcHost *host = NULL; + struct RtcConfigInfo *rtcInfo = NULL; + ... + host = RtcHostFromDevice(device);//这里有HdfDeviceObject到RtcHost的强制转化 + rtcInfo = OsalMemCalloc(sizeof(*rtcInfo)); + ... + //HiRtcConfigData 会从设备配置树中读取属性填充rtcInfo 的supportAnaCtrl, supportLock, spiBaseAddr, regAddrLength, irq + //为HiRtcSwInit 和HiRtcSwInit 提供参数,...函数内部处理失败后内存释放等操作 + if (HiRtcConfigData(rtcInfo, device->property) != 0) { + ... + } + if (HiRtcSwInit(rtcInfo) != 0) {//地址映射以及中断注册相关 + ... + } + if (HiRtcHwInit(rtcInfo) != 0) {//初始化anaCtrl 和 lockAddr 相关内容 + ... + } + + host->method = &g_method;//RtcMethod的实例化对象的挂载 + host->data = rtcInfo; //使RtcConfigInfo与RtcHost可以相互转化的前提 + HDF_LOGI("Hdf dev service:%s init success!", HdfDeviceGetServiceName(device)); + return HDF_SUCCESS; + } + ``` + +- **Release 函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > 无 + > + > **函数说明:** + > 释放内存和删除控制器,该函数需要在驱动入口结构体中赋值给 Release 接口, 当HDF框架调用Init函数初始化驱动失败时,可以调用 Release 释放驱动资源。所有强制转换获取相应对象的操作**前提**是在Init或Bind函数中具备对应赋值的操作。 + + ```c + static void HiRtcRelease(struct HdfDeviceObject *device) + { + struct RtcHost *host = NULL; + struct RtcConfigInfo *rtcInfo = NULL; + ... + host = RtcHostFromDevice(device); //这里有HdfDeviceObject到RtcHost的强制转化 + rtcInfo = (struct RtcConfigInfo *)host->data;//这里有RtcHost到RtcConfigInfo的强制转化 + if (rtcInfo != NULL) { + HiRtcSwExit(rtcInfo); + OsalMemFree(rtcInfo); //释放RtcConfigInfo + host->data = NULL; + } + RtcHostDestroy(host); //释放RtcHost + } + ``` + diff --git a/zh-cn/device-dev/driver/driver-platform-sdio-develop.md b/zh-cn/device-dev/driver/driver-platform-sdio-develop.md new file mode 100755 index 0000000000..4254bec8b1 --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-sdio-develop.md @@ -0,0 +1,309 @@ +# SDIO + +- [概述](#1) +- [开发步骤](#2) +- [开发实例](#3) + +## 概述 + +SDIO由SD卡发展而来,被统称为mmc(MultiMediaCard),相关技术差别不大,在HDF框架中, +SDIO的接口适配模式采用独立服务模式,在这种模式下,每一个设备对象会独立发布一个设备服务来处理外部访问,设备管理器收到API的访问请求之后,通过提取该请求的参数,达到调用实际设备对象的相应内部方法的目的。 + +独立服务模式可以直接借助HDFDeviceManager的服务管理能力,但需要为每个设备单独配置设备节点,增加内存占用。 + +图 1 独立服务模式结构图 +![image1](figure/独立服务模式.png) + +## 开发步骤 + +SDIO模块适配HDF框架的三个环节是配置属性文件,实例化驱动入口,以及填充核心层接口函数。 + + +1. **实例化驱动入口:** + - 实例化HdfDriverEntry结构体成员。 + - 调用HDF_INIT将HdfDriverEntry实例化对象注册到HDF框架中。 + +2. **配置属性文件:** + + - 在device_info.hcs文件中添加deviceNode描述。 + - 【可选】添加sdio_config.hcs器件属性文件。 + +3. **实例化SDIO控制器对象:** + + - 初始化SdioDevice成员。 + - 实例化SdioDevice成员SdioDeviceOps,其定义和成员说明见下 + +4. **驱动调试:** + - 【可选】针对新增驱动程序,建议验证驱动基本功能,例如SDIO控制状态,中断响应情况等。 + +> ![](../public_sys-resources/icon-note.gif) **说明:** +> +> SdioDeviceOps定义 +> +> ```c +> // 函数模板 +> struct SdioDeviceOps { +> int32_t (*incrAddrReadBytes)(struct SdioDevice *dev, uint8_t *data, uint32_t addr, uint32_t size); +> int32_t (*incrAddrWriteBytes)(struct SdioDevice *dev, uint8_t *data, uint32_t addr, uint32_t size); +> int32_t (*fixedAddrReadBytes)(struct SdioDevice *dev, uint8_t *data, uint32_t addr, uint32_t size, uint32_t scatterLen); +> int32_t (*fixedAddrWriteBytes)(struct SdioDevice *dev, uint8_t *data, uint32_t addr, uint32_t size, uint32_t scatterLen); +> int32_t (*func0ReadBytes)(struct SdioDevice *dev, uint8_t *data, uint32_t addr, uint32_t size); +> int32_t (*func0WriteBytes)(struct SdioDevice *dev, uint8_t *data, uint32_t addr, uint32_t size); +> int32_t (*setBlockSize)(struct SdioDevice *dev, uint32_t blockSize); +> int32_t (*getCommonInfo)(struct SdioDevice *dev, SdioCommonInfo *info, uint32_t infoType); +> int32_t (*setCommonInfo)(struct SdioDevice *dev, SdioCommonInfo *info, uint32_t infoType); +> int32_t (*flushData)(struct SdioDevice *dev); +> int32_t (*enableFunc)(struct SdioDevice *dev); +> int32_t (*disableFunc)(struct SdioDevice *dev); +> int32_t (*claimIrq)(struct SdioDevice *dev, SdioIrqHandler *irqHandler); +> int32_t (*releaseIrq)(struct SdioDevice *dev); +> int32_t (*findFunc)(struct SdioDevice *dev, struct SdioFunctionConfig *configData); +> int32_t (*claimHost)(struct SdioDevice *dev); +> int32_t (*releaseHost)(struct SdioDevice *dev); +> }; +> ``` +> +> 表1 SdioDeviceOps结构体成员的回调函数功能说明 +> +> |函数|入参|出参|返回值|功能| +> |-|-|-|-|-| +> |incrAddrReadBytes, |**dev**: 结构体指针,SDIO设备控制器;
**addr**: uint32_t,地址值;
**size**: uint32_t,大小 |**data**: uint8_t指针,传出值;|HDF_STATUS相关状态| 从指定的SDIO地址增量读取给定长度的数据 | +> |incrAddrWriteBytes, |**dev**: 结构体指针,SDIO设备控制器;
**data**: uint8_t指针,传入值;
**addr**: uint32_t,地址值;
**size**: uint32_t,大小 |无|HDF_STATUS相关状态| 将给定长度的数据增量写入指定的SDIO地址 | +> |fixedAddrReadBytes, |**dev**: 结构体指针,SDIO设备控制器;
**addr**: uint32_t,地址值;
**size**: uint32_t,大小;
**scatterLen**: uint32_t,数据长度;|**data**: uint8_t指针,传出值;|HDF_STATUS相关状态| 从固定SDIO地址读取给定长度的数据。 | +> |fixedAddrWriteBytes,|**dev**: 结构体指针,SDIO设备控制器;
**data**: uint8_t指针,传入值;
**addr**: uint32_t,地址值;
**size**: uint32_t,大小;
**scatterLen**: uint32_t,数据长度;|无|HDF_STATUS相关状态| 将给定长度的数据写入固定SDIO地址 | +> |func0ReadBytes, |**dev**: 结构体指针,SDIO设备控制器;
**addr**: uint32_t,地址值;
**size**: uint32_t,大小; |**data**: uint8_t指针,传出值;|HDF_STATUS相关状态| 从SDIO函数0的地址空间读取给定长度的数据。 | +> |func0WriteBytes, |**dev**: 结构体指针,SDIO设备控制器;
**data**: uint8_t指针,传入值;
**addr**: uint32_t,地址值;
**size**: uint32_t,大小; |无|HDF_STATUS相关状态| 将给定长度的数据写入SDIO函数0的地址空间。 | +> |setBlockSize, |**dev**: 结构体指针,SDIO设备控制器;
**blockSize**: uint32_t,Block大小 |无|HDF_STATUS相关状态| 设置block大小| +> |getCommonInfo, |**dev**: 联合体指针,SDIO设备控制器;
**infoType**: uint32_t,info类型; |**info**: 结构体指针,传出SdioFuncInfo信息;|HDF_STATUS相关状态| 获取CommonInfo,说明见下 | +> |setCommonInfo, |**dev**: 结构体指针,SDIO设备控制器;
**info**: 联合体指针,SdioFuncInfo信息传入;
**infoType**: uint32_t,info类型; |无|HDF_STATUS相关状态| 设置CommonInfo,说明见下 | +> |flushData, |**dev**: 结构体指针,SDIO设备控制器; |无|HDF_STATUS相关状态|当SDIO需要重新初始化或发生意外错误时调用的函数| +> |enableFunc, |**dev**: 结构体指针,SDIO设备控制器; |无|HDF_STATUS相关状态|使能SDIO设备 | +> |disableFunc, |**dev**: 结构体指针,SDIO设备控制器; |无|HDF_STATUS相关状态|去使能SDIO设备 | +> |claimIrq, |**dev**: 结构体指针,SDIO设备控制器;
**irqHandler**: void函数指针; |无|HDF_STATUS相关状态|注册SDIO中断 | +> |releaseIrq, |**dev**: 结构体指针,SDIO设备控制器; |无|HDF_STATUS相关状态|释放SDIO中断| +> |findFunc, |**dev**: 结构体指针,SDIO设备控制器;
**configData**: 结构体指针, SDIO函数关键信息 |无|HDF_STATUS相关状态|寻找匹配的funcNum| +> |claimHost, |**dev**: 结构体指针,SDIO设备控制器; |无|HDF_STATUS相关状态|独占HOST | +> |releaseHost, |**dev**: 结构体指针,SDIO设备控制器; |无|HDF_STATUS相关状态|释放HOST | +> +> >![](../public_sys-resources/icon-note.gif) **CommonInfo说明:** +> > 包括maxBlockNum(单个request中最大block数), maxBlockSize(单个block最大字节数), maxRequestSize(单个Request最大字节数), enTimeout(最大超时时间,毫秒), funcNum(功能编号1~7), irqCap(IRQ capabilities), (void \*)data. + +## 开发实例 + +下方将以sdio_adapter.c为示例,展示需要厂商提供哪些内容来完整实现设备功能。 + +1. 驱动开发首先需要实例化驱动入口,驱动入口必须为HdfDriverEntry(在 hdf_device_desc.h 中定义)类型的全局变量,且moduleName要和device_info.hcs中保持一致。HDF框架会将所有加载的驱动的HdfDriverEntry对象首地址汇总,形成一个类似数组的段地址空间,方便上层调用。 + + 一般在加载驱动时HDF会先调用Bind函数,再调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。 + +- SDIO 驱动入口参考 + + ```c + struct HdfDriverEntry g_sdioDriverEntry = { + .moduleVersion = 1, + .Bind = Hi35xxLinuxSdioBind, //见Bind参考 + .Init = Hi35xxLinuxSdioInit, //见Init参考 + .Release = Hi35xxLinuxSdioRelease,//见Release参考 + .moduleName = "HDF_PLATFORM_SDIO",//【必要 且与 HCS文件中里面的moduleName匹配】 + }; + //调用HDF_INIT将驱动入口注册到HDF框架中 + HDF_INIT(g_sdioDriverEntry); + ``` + +2. 完成驱动入口注册之后,下一步请在device_info.hcs文件中添加deviceNode信息,并在 sdio_config.hcs 中配置器件属性。deviceNode信息与驱动入口注册相关,器件属性值与核心层SdioDevice成员的默认值或限制范围有密切关系。 + + **本例只有一个SDIO控制器,如有多个器件信息,则需要在device_info文件增加deviceNode信息,以及在sdio_config文件中增加对应的器件属性**。 + +- device_info.hcs 配置参考 + + ```c + root { + device_info { + match_attr = "hdf_manager"; + platform :: host { + hostName = "platform_host"; + priority = 50; + device_sdio :: device { + device0 :: deviceNode { + policy = 1; + priority = 70; + permission = 0644; + moduleName = "HDF_PLATFORM_SDIO"; //【必要】用于指定驱动名称,需要与驱动Entry中的moduleName一致; + serviceName = "HDF_PLATFORM_MMC_2"; //【必要】驱动对外发布服务的名称,必须唯一 + deviceMatchAttr = "hisilicon_hi35xx_sdio_0";//【必要】用于配置控制器私有数据,要与sdio_config.hcs中对应控制器保持一致 + } + } + } + } + } + ``` + +- sdio_config.hcs 配置参考 + + ```c + root { + platform { + sdio_config { + template sdio_controller { + match_attr = ""; + hostId = 2; //【必要】模式固定为2,在mmc_config.hcs有介绍 + devType = 2; //【必要】模式固定为2,在mmc_config.hcs有介绍 + } + controller_0x2dd1 :: sdio_controller { + match_attr = "hisilicon_hi35xx_sdio_0";//【必要】需要和device_info.hcs中的deviceMatchAttr值一致 + } + } + } + ``` + +3. 完成驱动入口注册之后,最后一步就是以核心层SdioDevice对象的初始化为核心,包括厂商自定义结构体(传递参数和数据),实例化SdioDevice成员SdioDeviceOps(让用户可以通过接口来调用驱动底层函数),实现HdfDriverEntry成员函数(Bind,Init,Release) + +- 自定义结构体参考 + + > 从驱动的角度看,自定义结构体是参数和数据的载体,而且sdio_config.hcs文件中的数值会被HDF读入通过DeviceResourceIface来初始化结构体成员,一些重要数值也会传递给核心层对象。 + + ```c + typedef struct { + uint32_t maxBlockNum; // 单个request最大的block个数 + uint32_t maxBlockSize; // 单个block最大的字节数1~2048 + uint32_t maxRequestSize; // 单个request最大的字节数 1~2048 + uint32_t enTimeout; // 最大超时时间,单位毫秒,且不能超过一秒 + uint32_t funcNum; // 函数编号1~7 + uint32_t irqCap; // 中断能力 + void *data; // 私有数据 + } SdioFuncInfo; + + //SdioDevice是核心层控制器结构体,其中的成员在Bind函数中会被赋值 + struct SdioDevice { + struct SdDevice sd; + struct SdioDeviceOps *sdioOps; + struct SdioRegister sdioReg; + uint32_t functions; + struct SdioFunction *sdioFunc[SDIO_MAX_FUNCTION_NUMBER]; + struct SdioFunction *curFunction; + struct OsalThread thread; /* irq thread */ + struct OsalSem sem; + bool irqPending; + bool threadRunning; + }; + ``` + +- **【重要】** SdioDevice成员回调函数结构体SdioDeviceOps的实例化,其他成员在Init函数中初始化 + + ```c + static struct SdioDeviceOps g_sdioDeviceOps = { + .incrAddrReadBytes = Hi35xxLinuxSdioIncrAddrReadBytes, + .incrAddrWriteBytes = Hi35xxLinuxSdioIncrAddrWriteBytes, + .fixedAddrReadBytes = Hi35xxLinuxSdioFixedAddrReadBytes, + .fixedAddrWriteBytes = Hi35xxLinuxSdioFixedAddrWriteBytes, + .func0ReadBytes = Hi35xxLinuxSdioFunc0ReadBytes, + .func0WriteBytes = Hi35xxLinuxSdioFunc0WriteBytes, + .setBlockSize = Hi35xxLinuxSdioSetBlockSize, + .getCommonInfo = Hi35xxLinuxSdioGetCommonInfo, + .setCommonInfo = Hi35xxLinuxSdioSetCommonInfo, + .flushData = Hi35xxLinuxSdioFlushData, + .enableFunc = Hi35xxLinuxSdioEnableFunc, + .disableFunc = Hi35xxLinuxSdioDisableFunc, + .claimIrq = Hi35xxLinuxSdioClaimIrq, + .releaseIrq = Hi35xxLinuxSdioReleaseIrq, + .findFunc = Hi35xxLinuxSdioFindFunc, + .claimHost = Hi35xxLinuxSdioClaimHost, + .releaseHost = Hi35xxLinuxSdioReleaseHost, + }; + ``` + +- **Bind函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 (下表为部分展示,如需使用其他状态,可见//drivers/framework/include/utils/hdf_base.h中HDF_STATUS 定义) + > + > |状态(值)|问题描述| + > |:-|:-:| + > |HDF_ERR_INVALID_OBJECT|控制器对象非法| + > |HDF_ERR_MALLOC_FAIL |内存分配失败| + > |HDF_ERR_INVALID_PARAM |参数非法| + > |HDF_ERR_IO |I/O 错误| + > |HDF_SUCCESS |初始化成功| + > |HDF_FAILURE |初始化失败| + > + > **函数说明:** + > 将 MmcCntlr 对象同 HdfDeviceObject 进行了关联 + > + > **函数说明:** + > 初始化自定义结构体对象,初始化SdioCntlr成员,调用核心层SdioCntlrAdd函数,以及其他厂商自定义初始化操作 + + ```c + static int32_t Hi35xxLinuxSdioBind(struct HdfDeviceObject *obj) + { + struct MmcCntlr *cntlr = NULL; + int32_t ret; + ... + cntlr = (struct MmcCntlr *)OsalMemCalloc(sizeof(struct MmcCntlr));// 分配内存 + ... + cntlr->ops = &g_sdioCntlrOps; //【必要】struct MmcCntlrOps g_sdioCntlrOps={ + // .rescanSdioDev = Hi35xxLinuxSdioRescan,}; + cntlr->hdfDevObj = obj; //【必要】使HdfDeviceObject与MmcCntlr可以相互转化的前提 + obj->service = &cntlr->service;//【必要】使HdfDeviceObject与MmcCntlr可以相互转化的前提 + ret = Hi35xxLinuxSdioCntlrParse(cntlr, obj);//【必要】初始化cntlr 的 index, devType, 失败则 goto _ERR; + ... + ret = MmcCntlrAdd(cntlr); //【必要】调用核心层mmc_core.c的函数, 失败则 goto _ERR; + ... + ret = MmcCntlrAllocDev(cntlr, (enum MmcDevType)cntlr->devType);//【必要】调用核心层mmc_core.c的函数, 失败则 goto _ERR; + ... + + MmcDeviceAddOps(cntlr->curDev, &g_sdioDeviceOps);//【必要】调用核心层mmc_core.c的函数, 钩子函数挂载 + HDF_LOGD("Hi35xxLinuxSdioBind: Success!"); + return HDF_SUCCESS; + + _ERR: + Hi35xxLinuxSdioDeleteCntlr(cntlr); + HDF_LOGE("Hi35xxLinuxSdioBind: Fail!"); + return HDF_FAILURE; + } + ``` + + +- **Init函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 + > + > **函数说明:** + > 无操作,可根据厂商需要添加 + + ```c + static int32_t Hi35xxLinuxSdioInit(struct HdfDeviceObject *obj) + { + (void)obj;//无操作,可根据厂商需要添加 + HDF_LOGD("Hi35xxLinuxSdioInit: Success!"); + return HDF_SUCCESS; + } + ``` + +- **Release函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > 无 + > + > **函数说明:** + > 释放内存和删除控制器,该函数需要在驱动入口结构体中赋值给 Release 接口, 当HDF框架调用Init函数初始化驱动失败时,可以调用 Release 释放驱动资源。所有强制转换获取相应对象的操作**前提**是在Bnit函数中具备对应赋值的操作。 + + ```c + static void Hi35xxLinuxSdioRelease(struct HdfDeviceObject *obj) + { + if (obj == NULL) { + return; + } + Hi35xxLinuxSdioDeleteCntlr((struct MmcCntlr *)obj->service);//【必要】自定义的内存释放函数,这里有HdfDeviceObject到MmcCntlr的强制转化 + } + ``` + diff --git a/zh-cn/device-dev/driver/driver-platform-spi-develop.md b/zh-cn/device-dev/driver/driver-platform-spi-develop.md new file mode 100755 index 0000000000..7b172fe66a --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-spi-develop.md @@ -0,0 +1,355 @@ +# SPI + +- [概述](#1) +- [开发步骤](#2) +- [开发实例](#3) + +## 概述 + +SPI是串行外设接口(Serial Peripheral Interface)的缩写,在HDF框架中,SPI的接口适配模式采用独立服务模式,在这种模式下,每一个设备对象会独立发布一个设备服务来处理外部访问,设备管理器收到API的访问请求之后,通过提取该请求的参数,达到调用实际设备对象的相应内部方法的目的。 + +独立服务模式可以直接借助HDFDeviceManager的服务管理能力,但需要为每个设备单独配置设备节点,增加内存占用。 + +图 1 独立服务模式结构图 +![image1](figure/独立服务模式.png) + +## 开发步骤 + +SPI模块适配HDF框架的三个环节是配置属性文件,实例化驱动入口,以及填充核心层接口函数。 + +1. **实例化驱动入口:** + - 实例化HdfDriverEntry结构体成员。 + - 调用HDF_INIT将HdfDriverEntry实例化对象注册到HDF框架中。 + +2. **配置属性文件:** + + - 在device_info.hcs文件中添加deviceNode描述。 + - 【可选】添加spi_config.hcs器件属性文件。 + +3. **实例化SPI控制器对象:** + + - 初始化SpiCntlr成员。 + - 实例化SpiCntlr成员SpiCntlrMethod,其定义和成员说明见下 + +4. **驱动调试:** + - 【可选】针对新增驱动程序,建议验证驱动基本功能,例如spi控制状态,中断响应情况等。 + +> ![](../public_sys-resources/icon-note.gif) **说明:** +> +> SpiCntlrMethod定义 +> ```c +> struct SpiCntlrMethod { +> int32_t (*GetCfg)(struct SpiCntlr *cntlr, struct SpiCfg *cfg); +> int32_t (*SetCfg)(struct SpiCntlr *cntlr, struct SpiCfg *cfg); +> int32_t (*Transfer)(struct SpiCntlr *cntlr, struct SpiMsg *msg, uint32_t count); +> int32_t (*Open)(struct SpiCntlr *cntlr); +> int32_t (*Close)(struct SpiCntlr *cntlr); +> }; +> ``` +> 表1 SpiCntlrMethod结构体成员的回调函数功能说明 +> +> |成员函数|入参|返回值|功能| +> |-|-|-|-| +> |Transfer |**cntlr**: 结构体指针,核心层spi控制器 ;
**msg**: 结构体指针,Spi消息;
**count**: uint32_t,消息个数 |HDF_STATUS相关状态|传输消息| +> |SetCfg |**cntlr**: 结构体指针,核心层spi控制器 ;
**cfg**: 结构体指针,Spi属性 |HDF_STATUS相关状态|设置控制器属性 | +> |GetCfg |**cntlr**: 结构体指针,核心层spi控制器 ;
**cfg**: 结构体指针,Spi属性 |HDF_STATUS相关状态|获取控制器属性 | +> |Open |**cntlr**: 结构体指针,核心层spi控制器 ; |HDF_STATUS相关状态|打开SPI | +> |Close |**cntlr**: 结构体指针,核心层spi控制器 ; |HDF_STATUS相关状态|关闭SPI | + +## 开发实例 + +下方将以spi_hi35xx.c为示例,展示需要厂商提供哪些内容来完整实现设备功能。 + +1. 驱动开发首先需要实例化驱动入口,驱动入口必须为HdfDriverEntry(在 hdf_device_desc.h 中定义)类型的全局变量,且moduleName要和device_info.hcs中保持一致。HDF框架会将所有加载的驱动的HdfDriverEntry对象首地址汇总,形成一个类似数组的段地址空间,方便上层调用。 + + 一般在加载驱动时HDF会先调用Bind函数,再调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。 + +- SPI驱动入口参考 + + ```c + struct HdfDriverEntry g_hdfSpiDevice = { + .moduleVersion = 1, + .moduleName = "HDF_PLATFORM_SPI",//【必要 且与 HCS文件中里面的moduleName匹配】 + .Bind = HdfSpiDeviceBind, //见Bind参考 + .Init = HdfSpiDeviceInit, //见Init参考 + .Release = HdfSpiDeviceRelease, //见Release参考 + }; + //调用HDF_INIT将驱动入口注册到HDF框架中 + HDF_INIT(g_hdfSpiDevice); + ``` + +2. 完成驱动入口注册之后,下一步请在device_info.hcs文件中添加deviceNode信息,并在 spi_config.hcs 中配置器件属性。deviceNode信息与驱动入口注册相关,器件属性值与核心层SpiCntlr 成员的默认值或限制范围有密切关系。 +**本例只有一个SPI控制器,如有多个器件信息,则需要在device_info文件增加deviceNode信息,以及在spi_config文件中增加对应的器件属性**。 + +- device_info.hcs 配置参考 + + ```c + root { + device_info { + match_attr = "hdf_manager"; + platform :: host { + hostName = "platform_host"; + priority = 50; + device_spi :: device { //为每一个 SPI 控制器配置一个HDF设备节点 + device0 :: deviceNode { + policy = 1; + priority = 60; + permission = 0644; + moduleName = "HDF_PLATFORM_SPI"; + serviceName = "HDF_PLATFORM_SPI_0"; + deviceMatchAttr = "hisilicon_hi35xx_spi_0"; + } + device1 :: deviceNode { + policy = 1; + priority = 60; + permission = 0644; + moduleName = "HDF_PLATFORM_SPI"; // 【必要】用于指定驱动名称,该字段的值必须和驱动入口结构的moduleName值一致 + serviceName = "HDF_PLATFORM_SPI_1"; // 【必要且唯一】驱动对外发布服务的名称 + deviceMatchAttr = "hisilicon_hi35xx_spi_1";// 需要与设备hcs文件中的 match_attr 匹配 + } + ... + } + } + } + } + ``` + +- spi_config.hcs 配置参考 + + ```c + root { + platform { + spi_config {//每一个SPI控制器配置私有数据 + template spi_controller {//模板公共参数, 继承该模板的节点如果使用模板中的默认值, 则节点字段可以缺省 + serviceName = ""; + match_attr = ""; + transferMode = 0; // 数据传输模式:中断传输(0),流控传输(1),DMA传输(2) + busNum = 0; // 总线号 + clkRate = 100000000; + bitsPerWord = 8; // 传输位宽 + mode = 19; // SPI 数据的输入输出模式 + maxSpeedHz = 0; // 最大时钟频率 + minSpeedHz = 0; // 最小时钟频率 + speed = 2000000; // 当前消息传输速度 + fifoSize = 256; // FIFO大小 + numCs = 1; // 片选号 + regBase = 0x120c0000; // 地址映射需要 + irqNum = 100; // 中断号 + REG_CRG_SPI = 0x120100e4; // CRG_REG_BASE(0x12010000) + 0x0e4 + CRG_SPI_CKEN = 0; + CRG_SPI_RST = 0; + REG_MISC_CTRL_SPI = 0x12030024; // MISC_REG_BASE(0x12030000) + 0x24 + MISC_CTRL_SPI_CS = 0; + MISC_CTRL_SPI_CS_SHIFT = 0; + } + controller_0x120c0000 :: spi_controller { + busNum = 0; //【必要】总线号 + CRG_SPI_CKEN = 0x10000; // (0x1 << 16) 0:close clk, 1:open clk + CRG_SPI_RST = 0x1; // (0x1 << 0) 0:cancel reset, 1:reset + match_attr = "hisilicon_hi35xx_spi_0";//【必要】需要和device_info.hcs中的deviceMatchAttr值一致 + } + controller_0x120c1000 :: spi_controller { + busNum = 1; + CRG_SPI_CKEN = 0x20000; // (0x1 << 17) 0:close clk, 1:open clk + CRG_SPI_RST = 0x2; // (0x1 << 1) 0:cancel reset, 1:reset + match_attr = "hisilicon_hi35xx_spi_1"; + regBase = 0x120c1000; //【必要】地址映射需要 + irqNum = 101; //【必要】中断号 + } + ... + // 【可选】可新增,但需要在 device_info.hcs 添加对应的节点 + } + } + } + ``` + +3. 完成驱动入口注册之后,最后一步就是以核心层SpiCntlr对象的初始化为核心,包括厂商自定义结构体(传递参数和数据),实例化SpiCntlr成员SpiCntlrMethod(让用户可以通过接口来调用驱动底层函数),实现HdfDriverEntry成员函数(Bind,Init,Release) + +- 自定义结构体参考 + + > 从驱动的角度看,自定义结构体是参数和数据的载体,而且spi_config.hcs文件中的数值会被HDF读入通过DeviceResourceIface来初始化结构体成员,一些重要数值也会传递给核心层对象,例如设备号、总线号等。 + + ```c + struct Pl022 {//对应于hcs中的参数 + struct SpiCntlr *cntlr; + struct DListHead deviceList; + struct OsalSem sem; + volatile unsigned char *phyBase; + volatile unsigned char *regBase; + uint32_t irqNum; + uint32_t busNum; + uint32_t numCs; + uint32_t curCs; + uint32_t speed; + uint32_t fifoSize; + uint32_t clkRate; + uint32_t maxSpeedHz; + uint32_t minSpeedHz; + uint32_t regCrg; + uint32_t clkEnBit; + uint32_t clkRstBit; + uint32_t regMiscCtrl; + uint32_t miscCtrlCsShift; + uint32_t miscCtrlCs; + uint16_t mode; + uint8_t bitsPerWord; + uint8_t transferMode; + }; + + //SpiCntlr是核心层控制器结构体,其中的成员在Init函数中会被赋值 + struct SpiCntlr { + struct IDeviceIoService service; + struct HdfDeviceObject *device; + uint32_t busNum; + uint32_t numCs; + uint32_t curCs; + struct OsalMutex lock; + struct SpiCntlrMethod *method; + struct DListHead list; + void *priv; + }; + ``` + + +- **【重要】** SpiCntlr成员回调函数结构体SpiCntlrMethod的实例化,其他成员在Init函数中初始化 + + ```c + // spi_hi35xx.c 中的示例:钩子函数的填充 + struct SpiCntlrMethod g_method = { + .Transfer = Pl022Transfer, + .SetCfg = Pl022SetCfg, + .GetCfg = Pl022GetCfg, + .Open = Pl022Open, + .Close = Pl022Close, + }; + ``` + + +- **Bind 函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 + > + > **函数说明:** + > 将 SpiCntlr 对象同 HdfDeviceObject 进行了关联 + + ```c + static int32_t HdfSpiDeviceBind(struct HdfDeviceObject *device) + { + ... + return (SpiCntlrCreate(device) == NULL) ? HDF_FAILURE : HDF_SUCCESS; + } + + struct SpiCntlr *SpiCntlrCreate(struct HdfDeviceObject *device) + { + struct SpiCntlr *cntlr = NULL; //创建核心层 SpiCntlr 对象 + ... + cntlr = (struct SpiCntlr *)OsalMemCalloc(sizeof(*cntlr));//非配内存 + ... + cntlr->device = device; //使HdfDeviceObject与SpiCntlr可以相互转化的前提 + device->service = &(cntlr->service);//使HdfDeviceObject与SpiCntlr可以相互转化的前提 + (void)OsalMutexInit(&cntlr->lock); //锁初始化 + DListHeadInit(&cntlr->list); //添加对应的节点 + cntlr->priv = NULL; + return cntlr; + } + ``` + + +- **Init函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 (下表为部分展示,如需使用其他状态,可见//drivers/framework/include/utils/hdf_base.h中HDF_STATUS 定义) + > + > |状态(值)|问题描述| + > |:-|:-:| + > |HDF_ERR_INVALID_OBJECT|控制器对象非法| + > |HDF_ERR_MALLOC_FAIL |内存分配失败| + > |HDF_ERR_INVALID_PARAM |参数非法| + > |HDF_ERR_IO |I/O 错误| + > |HDF_SUCCESS |初始化成功| + > |HDF_FAILURE |初始化失败| + > + > **函数说明:** + > 初始化自定义结构体对象,初始化SpiCntlr成员。 + + ```c + //挂载init的 + static int32_t HdfSpiDeviceInit(struct HdfDeviceObject *device) + { + int32_t ret; + struct SpiCntlr *cntlr = NULL; + ... + cntlr = SpiCntlrFromDevice(device);//这里有HdfDeviceObject到SpiCntlr的强制转化,通过service成员,赋值见Bind函数 + //return (device == NULL) ? NULL : (struct SpiCntlr *)device->service; + ... + ret = Pl022Init(cntlr, device);//【必要】填充厂商自定义操作对象,示例见下 + ... + ret = Pl022Probe(cntlr->priv); + ... + return ret; + } + + static int32_t Pl022Init(struct SpiCntlr *cntlr, const struct HdfDeviceObject *device) + { + int32_t ret; + struct Pl022 *pl022 = NULL; + ... + pl022 = (struct Pl022 *)OsalMemCalloc(sizeof(*pl022));//申请内存 + ... + ret = SpiGetBaseCfgFromHcs(pl022, device->property); //填充busNum, numCs, speed, fifoSize, clkRate, + //mode, bitsPerWord, transferMode参数值 + ... + ret = SpiGetRegCfgFromHcs(pl022, device->property); //填充regBase, phyBase, irqNum, regCrg, clkEnBit, + //clkRstBit, regMiscCtrl, regMiscCtrl, miscCtrlCs, + //miscCtrlCsShift参数值 + ... + //计算最大,最小速度对应的频率 + pl022->maxSpeedHz = (pl022->clkRate) / ((SCR_MIN + 1) * CPSDVSR_MIN); + pl022->minSpeedHz = (pl022->clkRate) / ((SCR_MAX + 1) * CPSDVSR_MAX); + DListHeadInit(&pl022->deviceList);//初始化DList链表 + pl022->cntlr = cntlr; //使Pl022与SpiCntlr可以相互转化的前提 + cntlr->priv = pl022; //使Pl022与SpiCntlr可以相互转化的前提 + cntlr->busNum = pl022->busNum; //挂载总线号 + cntlr->method = &g_method; //SpiCntlrMethod的实例化对象的挂载 + ... + ret = Pl022CreatAndInitDevice(pl022); + if (ret != 0) { + Pl022Release(pl022); //初始化失败就释放Pl022对象 + return ret; + } + return 0; + } + ``` + +- **Release函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > 无 + > + > **函数说明:** + > 释放内存和删除控制器,该函数需要在驱动入口结构体中赋值给 Release 接口, 当HDF框架调用Init函数初始化驱动失败时,可以调用 Release 释放驱动资源。所有强制转换获取相应对象的操作**前提**是在Init函数中具备对应赋值的操作。 + + ```c + static void HdfSpiDeviceRelease(struct HdfDeviceObject *device) + { + struct SpiCntlr *cntlr = NULL; + ... + cntlr = SpiCntlrFromDevice(device);//这里有HdfDeviceObject到SpiCntlr的强制转化,通过service成员,赋值见Bind函数 + // return (device==NULL) ?NULL:(struct SpiCntlr *)device->service; + ... + if (cntlr->priv != NULL) { + Pl022Remove((struct Pl022 *)cntlr->priv);//这里有SpiCntlr到Pl022的强制转化 + } + SpiCntlrDestroy(cntlr); //释放Pl022对象 + } + ``` diff --git a/zh-cn/device-dev/driver/driver-platform-uart-develop.md b/zh-cn/device-dev/driver/driver-platform-uart-develop.md new file mode 100755 index 0000000000..800c16dc47 --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-uart-develop.md @@ -0,0 +1,391 @@ +# UART + +- [概述](#1) +- [开发步骤](#2) +- [开发实例](#3) + +## 概述 + +UART是通用异步收发传输器(Universal Asynchronous Receiver/Transmitter)的缩写,在HDF框架中,uart的接口适配模式采用独立服务模式。在这种模式下,每一个设备对象会独立发布一个设备服务来处理外部访问,设备管理器收到API的访问请求之后,通过提取该请求的参数,达到调用实际设备对象的相应内部方法的目的。 + +独立服务模式可以直接借助HDFDeviceManager的服务管理能力,但需要为每个设备单独配置设备节点,增加内存占用。 + +图 1 独立服务模式结构图 +![image1](figure/独立服务模式.png) + +## 开发步骤 + +uart模块适配HDF框架的三个环节是配置属性文件,实例化驱动入口,以及填充核心层接口函数。 + +1. **实例化驱动入口:** + - 实例化HdfDriverEntry结构体成员。 + - 调用HDF_INIT将HdfDriverEntry实例化对象注册到HDF框架中。 + +2. **配置属性文件:** + + - 在device_info.hcs文件中添加deviceNode描述。 + - 【可选】添加uart_config.hcs器件属性文件。 + +3. **实例化UART控制器对象:** + + - 初始化UartHost成员。 + - 实例化UartHost成员UartHostMethod,其定义和成员说明见下 + +4. **驱动调试:** + - 【可选】针对新增驱动程序,建议验证驱动基本功能,例如UART控制状态,中断响应情况等。 + +> ![](../public_sys-resources/icon-note.gif) **说明:** +> UartHostMethod定义 +> +> ```c +> struct UartHostMethod { +> int32_t (*Init)(struct UartHost *host); +> int32_t (*Deinit)(struct UartHost *host); +> int32_t (*Read)(struct UartHost *host, uint8_t *data, uint32_t size); +> int32_t (*Write)(struct UartHost *host, uint8_t *data, uint32_t size); +> int32_t (*GetBaud)(struct UartHost *host, uint32_t *baudRate); +> int32_t (*SetBaud)(struct UartHost *host, uint32_t baudRate); +> int32_t (*GetAttribute)(struct UartHost *host, struct UartAttribute *attribute); +> int32_t (*SetAttribute)(struct UartHost *host, struct UartAttribute *attribute); +> int32_t (*SetTransMode)(struct UartHost *host, enum UartTransMode mode); +> int32_t (*pollEvent)(struct UartHost *host, void *filep, void *table); +> }; +> ``` +> +> 表1 UartHostMethod结构体成员的回调函数功能说明 +> +> |函数|入参|出参|返回值|功能| +> |-|-|-|-|-| +> |Init |**host**: 结构体指针,核心层uart控制器; |无|HDF_STATUS相关状态|初始化Uart设备| +> |Deinit |**host**: 结构体指针,核心层uart控制器; |无|HDF_STATUS相关状态| 去初始化Uart设备 | +> |Read |**host**: 结构体指针,核心层uart控制器;
**size**:uint32_t,数据大小; |**data**: uint8_t指针,传出的数据 |HDF_STATUS相关状态|接收数据 RX | +> |Write |**host**: 结构体指针,核心层uart控制器;
**data**:uint8_t指针,传入数据;
**size**:uint32_t,数据大小; |无|HDF_STATUS相关状态|发送数据 TX | +> |SetBaud |**host**: 结构体指针,核心层uart控制器;
**baudRate**: uint32_t指针,波特率传入值; |无|HDF_STATUS相关状态| 设置波特率 | +> |GetBaud |**host**: 结构体指针,核心层uart控制器; |**baudRate**: uint32_t指针,传出的波特率; |HDF_STATUS相关状态|获取当前设置的波特率| +> |GetAttribute |**host**: 结构体指针,核心层uart控制器; |**attribute**: 结构体指针,传出的属性值
(见uart_if.h中UartAttribute定义)|HDF_STATUS相关状态| 获取设备uart相关属性| +> |SetAttribute |**host**: 结构体指针,核心层uart控制器;
**attribute**: 结构体指针,属性传入值; |无|HDF_STATUS相关状态| 设置设备uart相关属性 | +> |SetTransMode |**host**: 结构体指针,核心层uart控制器;
**mode**: 枚举值(见uart_if.h中
UartTransMode定义),传输模式 |无|HDF_STATUS相关状态| 设置传输模式 | +> |PollEvent |**host**: 结构体指针,核心层uart控制器;
**filep**: void 指针,file ;
**table**: void 指针,poll_table ;|无|HDF_STATUS相关状态|poll机制| + + +## 开发实例 + +下方将以uart_hi35xx.c为示例,展示需要厂商提供哪些内容来完整实现设备功能。 + +1. 驱动开发首先需要实例化驱动入口,驱动入口必须为HdfDriverEntry(在 hdf_device_desc.h 中定义)类型的全局变量,且moduleName要和device_info.hcs中保持一致。HDF框架会将所有加载的驱动的HdfDriverEntry对象首地址汇总,形成一个类似数组的段地址空间,方便上层调用。 + + 一般在加载驱动时HDF会先调用Bind函数,再调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。 + +- UART驱动入口参考 + + ```c + struct HdfDriverEntry g_hdfUartDevice = { + .moduleVersion = 1, + .moduleName = "HDF_PLATFORM_UART",//【必要且与 HCS 里面的名字匹配】 + .Bind = HdfUartDeviceBind, //见Bind参考 + .Init = HdfUartDeviceInit, //见Init参考 + .Release = HdfUartDeviceRelease, //见Release参考 + }; + //调用HDF_INIT将驱动入口注册到HDF框架中 + HDF_INIT(g_hdfUartDevice); + ``` + +2. 完成驱动入口注册之后,下一步请在device_info.hcs文件中添加deviceNode信息,并在 uart_config.hcs 中配置器件属性。deviceNode信息与驱动入口注册相关,器件属性值与核心层UartHost成员的默认值或限制范围有密切关系。 + + **本例只有一个UART控制器,如有多个器件信息,则需要在device_info文件增加deviceNode信息,以及在uart_config文件中增加对应的器件属性**。 + +- device_info.hcs 配置参考 + + ```c + root { + device_info { + match_attr = "hdf_manager"; + platform :: host { + hostName = "platform_host"; + priority = 50; + device_uart :: device { + device0 :: deviceNode { + policy = 1; //驱动服务发布的策略,policy大于等于1(用户态可见为2,仅内核态可见为1); + priority = 40; //驱动启动优先级 + permission = 0644; //驱动创建设备节点权限 + moduleName = "HDF_PLATFORM_UART"; //驱动名称,该字段的值必须和驱动入口结构的moduleName值一致 + serviceName = "HDF_PLATFORM_UART_0";//驱动对外发布服务的名称,必须唯一,必须要按照HDF_PLATFORM_UART_X的格式,X为UART控制器编号 + deviceMatchAttr = "hisilicon_hi35xx_uart_0";//驱动私有数据匹配的关键字,必须和驱动私有数据配置表中的match_attr值一致 + } + device1 :: deviceNode { + policy = 2; + permission = 0644; + priority = 40; + moduleName = "HDF_PLATFORM_UART"; + serviceName = "HDF_PLATFORM_UART_1"; + deviceMatchAttr = "hisilicon_hi35xx_uart_1"; + } + ... + } + } + } + } + ``` + +- uart_config.hcs 配置参考 + + ```c + root { + platform { + template uart_controller {//模板公共参数, 继承该模板的节点如果使用模板中的默认值, 则节点字段可以缺省 + match_attr = ""; + num = 0; //【必要】设备号 + baudrate = 115200; //【必要】波特率,数值可按需填写 + fifoRxEn = 1; //【必要】使能接收FIFO + fifoTxEn = 1; //【必要】使能发送FIFO + flags = 4; //【必要】标志信号 + regPbase = 0x120a0000; //【必要】地址映射需要 + interrupt = 38; //【必要】中断号 + iomemCount = 0x48; //【必要】地址映射需要 + } + controller_0x120a0000 :: uart_controller { + match_attr = "hisilicon_hi35xx_uart_0";//【必要】必须和device_info.hcs中对应的设备的deviceMatchAttr值一致 + } + controller_0x120a1000 :: uart_controller { + num = 1; + baudrate = 9600; + regPbase = 0x120a1000; + interrupt = 39; + match_attr = "hisilicon_hi35xx_uart_1"; + } + ... + // 【可选】可新增,但需要在 device_info.hcs 添加对应的节点 + } + } + ``` + +3. 完成驱动入口注册之后,最后一步就是以核心层UartHost对象的初始化为核心,包括厂商自定义结构体(传递参数和数据),实例化UartHost成员UartHostMethod(让用户可以通过接口来调用驱动底层函数),实现HdfDriverEntry成员函数(Bind,Init,Release) + +- 自定义结构体参考 + + > 从驱动的角度看,自定义结构体是参数和数据的载体,而且uart_config.hcs文件中的数值会被HDF读入通过DeviceResourceIface来初始化结构体成员,一些重要数值也会传递给核心层对象,例如设备号等。 + + ```c + struct UartPl011Port { //接口相关的结构体 + int32_t enable; + unsigned long physBase; //物理地址 + uint32_t irqNum; //中断号 + uint32_t defaultBaudrate;//默认波特率 + uint32_t flags; //标志信号,下面三个宏与之相关 + #define PL011_FLG_IRQ_REQUESTED (1 << 0) + #define PL011_FLG_DMA_RX_REQUESTED (1 << 1) + #define PL011_FLG_DMA_TX_REQUESTED (1 << 2) + struct UartDmaTransfer *rxUdt; //DMA传输相关 + struct UartDriverData *udd; //见下 + }; + struct UartDriverData { //数据传输相关的结构体 + uint32_t num; + uint32_t baudrate; //波特率(可设置) + struct UartAttribute attr; //数据位、停止位等传输属性相关 + struct UartTransfer *rxTransfer; //缓冲区相关,可理解为FIFO结构 + wait_queue_head_t wait; //条件变量相关的排队等待信号 + int32_t count; //数据数量 + int32_t state; //uart控制器状态 + #define UART_STATE_NOT_OPENED 0 + #define UART_STATE_OPENING 1 + #define UART_STATE_USEABLE 2 + #define UART_STATE_SUSPENED 3 + uint32_t flags; //状态标志 + #define UART_FLG_DMA_RX (1 << 0) + #define UART_FLG_DMA_TX (1 << 1) + #define UART_FLG_RD_BLOCK (1 << 2) + RecvNotify recv; //函数指针类型,指向串口数据接收函数 + struct UartOps *ops; //自定义函数指针结构体,详情见device/hisilicon/drivers/uart/uart_pl011.c + void *private; //一般用来存储UartPl011Port首地址,方便调用 + }; + + // UartHost是核心层控制器结构体,其中的成员在Init函数中会被赋值 + struct UartHost { + struct IDeviceIoService service; + struct HdfDeviceObject *device; + uint32_t num; + OsalAtomic atom; + void *priv; //一般存储厂商自定义结构体首地址,方便后者被调用 + struct UartHostMethod *method; //核心层钩子函数,厂商需要实现其成员函数功能并填充 + }; + ``` + +- **【重要】** UartHost成员回调函数结构体UartHostMethod的实例化,其他成员在Bind函数中初始化 + + ```c + // uart_hi35xx.c 中的示例:钩子函数的填充 + struct UartHostMethod g_uartHostMethod = { + .Init = Hi35xxInit, + .Deinit = Hi35xxDeinit, + .Read = Hi35xxRead, + .Write = Hi35xxWrite, + .SetBaud = Hi35xxSetBaud, + .GetBaud = Hi35xxGetBaud, + .SetAttribute = Hi35xxSetAttribute, + .GetAttribute = Hi35xxGetAttribute, + .SetTransMode = Hi35xxSetTransMode, + .pollEvent = Hi35xxPollEvent, + }; + ``` + + +- **Bind函数参考** + + > **入参:** + > HdfDeviceObject 这个是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 (下表为部分展示,如需使用其他状态,可见//drivers/framework/include/utils/hdf_base.h中HDF_STATUS 定义) + > + > |状态(值)|问题描述| + > |:-|:-:| + > |HDF_ERR_INVALID_OBJECT|控制器对象非法| + > |HDF_ERR_MALLOC_FAIL |内存分配失败| + > |HDF_ERR_INVALID_PARAM |参数非法| + > |HDF_ERR_IO |I/O 错误| + > |HDF_SUCCESS |初始化成功| + > |HDF_FAILURE |初始化失败| + > + > **函数说明:** + > 初始化自定义结构体对象,初始化UartHost成员 + + ```c + //uart_hi35xx.c + static int32_t HdfUartDeviceBind(struct HdfDeviceObject *device) + { + ... + return (UartHostCreate(device) == NULL) ? HDF_FAILURE : HDF_SUCCESS;//【必须做】调用核心层函数 UartHostCreate + } + //uart_core.c 核心层 UartHostCreate 函数说明 + struct UartHost *UartHostCreate(struct HdfDeviceObject *device) + { + struct UartHost *host = NULL; //新建 UartHost + ... + host = (struct UartHost *)OsalMemCalloc(sizeof(*host));//分配内存 + ... + host->device = device; //【必要】使HdfDeviceObject与UartHost可以相互转化的前提 + device->service = &(host->service);//【必要】使HdfDeviceObject与UartHost可以相互转化的前提 + host->device->service->Dispatch = UartIoDispatch;//为 service 成员的 Dispatch 方法赋值 + OsalAtomicSet(&host->atom, 0); //原子量初始化或者原子量设置 + host->priv = NULL; + host->method = NULL; + return host; + } + ``` + + +- **Init函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > HDF_STATUS相关状态 + > + > **函数说明:** + > 初始化自定义结构体对象,初始化UartHost成员,调用核心层UartAddDev函数,接入VFS + + ```c + //挂载init的 + int32_t HdfUartDeviceInit(struct HdfDeviceObject *device) + { + int32_t ret; + struct UartHost *host = NULL; + HDF_LOGI("%s: entry", __func__); + ... + host = UartHostFromDevice(device);//通过service成员后强制转为UartHost,赋值是在Bind函数中 + ... + ret = Hi35xxAttach(host, device); //完成UartHost对象的初始化,见下 + ... + host->method = &g_uartHostMethod; //UartHostMethod的实例化对象的挂载 + return ret; + } + //完成 UartHost 对象的初始化 + static int32_t Hi35xxAttach(struct UartHost *host, struct HdfDeviceObject *device) + { + int32_t ret; + //udd 和 port 对象是厂商自定义的结构体对象,可根据需要实现相关功能 + struct UartDriverData *udd = NULL; + struct UartPl011Port *port = NULL; + ... + // 【必要相关功能】步骤【1】~【7】主要实现对 udd 对象的填充赋值,然后赋值给核心层UartHost对象上 + udd = (struct UartDriverData *)OsalMemCalloc(sizeof(*udd));//【1】 + ... + port = (struct UartPl011Port *)OsalMemCalloc(sizeof(struct UartPl011Port));//【2】 + ... + udd->ops = Pl011GetOps();//【3】设备开启、关闭、属性设置、发送操作等函数挂载 + udd->recv = PL011UartRecvNotify;//【4】数据接收通知函数(条件锁机制)挂载 + udd->count = 0; //【5】 + port->udd = udd; //【6】使UartPl011Port与UartDriverData可以相互转化的前提 + ret = UartGetConfigFromHcs(port, device->property);//【必要】 此步骤是将 HdfDeviceObject 的属性传递给厂商自定义结构体 + // 用于相关操作,示例代码见下 + ... + udd->private = port; //【7】 + + host->priv = udd; //【必要】使UartHost与UartDriverData可以相互转化的前提 + host->num = udd->num;//【必要】uart 设备号 + UartAddDev(host); //【必要】核心层uart_dev.c 中的函数,作用:注册了一个字符设备节点到vfs, 这样从用户态可以通过这个虚拟文件节点访问uart + return HDF_SUCCESS; + } + + static int32_t UartGetConfigFromHcs(struct UartPl011Port *port, const struct DeviceResourceNode *node) + { + uint32_t tmp, regPbase, iomemCount; + struct UartDriverData *udd = port->udd; + struct DeviceResourceIface *iface = DeviceResourceGetIfaceInstance(HDF_CONFIG_SOURCE); + ... + //通过请求参数提取相应的值,并赋值给厂商自定义的结构体 + if (iface->GetUint32(node, "num", &udd->num, 0) != HDF_SUCCESS) { + HDF_LOGE("%s: read busNum fail", __func__); + return HDF_FAILURE; + } + ... + return 0; + } + ``` + +- **Release函数参考** + + > **入参:** + > HdfDeviceObject 是整个驱动对外暴露的接口参数,具备 HCS 配置文件的信息 + > + > **返回值:** + > 无 + > + > **函数说明:** + > 该函数需要在驱动入口结构体中赋值给 Release 接口, 当HDF框架调用Init函数初始化驱动失败时,可以调用 Release 释放驱动资源, 该函数中需包含释放内存和删除控制器等操作。所有强制转换获取相应对象的操作**前提**是在Init函数中具备对应赋值的操作。 + + ```c + void HdfUartDeviceRelease(struct HdfDeviceObject *device) + { + struct UartHost *host = NULL; + ... + host = UartHostFromDevice(device);//这里有HdfDeviceObject到UartHost的强制转化,通过service成员,赋值见Bind函数 + ... + if (host->priv != NULL) { + Hi35xxDetach(host); //厂商自定义的内存释放函数,见下 + } + UartHostDestroy(host); //调用核心层函数释放host + } + + static void Hi35xxDetach(struct UartHost *host) + { + struct UartDriverData *udd = NULL; + struct UartPl011Port *port = NULL; + ... + udd = host->priv; //这里有UartHost到UartDriverData的转化 + ... + UartRemoveDev(host);//VFS注销 + port = udd->private;//这里有UartDriverData到UartPl011Port的转化 + if (port != NULL) { + if (port->physBase != 0) { + OsalIoUnmap((void *)port->physBase);//地址反映射 + } + (void)OsalMemFree(port); + udd->private = NULL; + } + (void)OsalMemFree(udd);//释放UartDriverData + host->priv = NULL; + } + ``` + diff --git a/zh-cn/device-dev/driver/driver-platform-watchdog-develop.md b/zh-cn/device-dev/driver/driver-platform-watchdog-develop.md new file mode 100755 index 0000000000..1bb85b3e1f --- /dev/null +++ b/zh-cn/device-dev/driver/driver-platform-watchdog-develop.md @@ -0,0 +1,254 @@ +# WatchDog + +- [概述](#1) +- [开发步骤](#2) +- [开发实例](#3) + +## 概述 + +看门狗(watchdog),又叫看门狗计时器(watchdog timer),是一种硬件的计时设备,在HDF框架中,watchdog接口适配模式采用独立服务模式,在这种模式下,每一个设备对象会独立发布一个设备服务来处理外部访问,设备管理器收到API的访问请求之后,通过提取该请求的参数,达到调用实际设备对象的相应内部方法的目的。 + +独立服务模式可以直接借助HDFDeviceManager的服务管理能力,但需要为每个设备单独配置设备节点,增加内存占用。 + +图 1 独立服务模式结构图 +![image1](figure/独立服务模式.png) + +## 开发步骤 + +watchdog模块适配HDF框架的三个环节是配置属性文件,实例化驱动入口,以及填充核心层接口函数。 + +1. **实例化驱动入口:** + - 实例化HdfDriverEntry结构体成员。 + - 调用HDF_INIT将HdfDriverEntry实例化对象注册到HDF框架中。 + +2. **配置属性文件:** + + - 在device_info.hcs文件中添加deviceNode描述。 + - 【可选】添加watchdog_config.hcs器件属性文件。 + +3. **实例化Watchdog控制器对象:** + + - 初始化WatchdogCntlr成员。 + - 实例化WatchdogCntlr成员WatchdogMethod,其定义和成员**说明**见下 + +4. **驱动调试:** + - 【可选】针对新增驱动程序,建议验证驱动基本功能,例如挂载后的信息反馈,超时时间设置的成功与否等。 + +> ![](../public_sys-resources/icon-note.gif) **说明:** +> WatchdogMethod定义 +> +> ```c +> struct WatchdogMethod { +> int32_t (*getStatus)(struct WatchdogCntlr *wdt, int32_t *status); +> int32_t (*setTimeout)(struct WatchdogCntlr *wdt, uint32_t seconds); +> int32_t (*getTimeout)(struct WatchdogCntlr *wdt, uint32_t *seconds); +> int32_t (*start)(struct WatchdogCntlr *wdt); +> int32_t (*stop)(struct WatchdogCntlr *wdt); +> int32_t (*feed)(struct WatchdogCntlr *wdt); +> int32_t (*getPriv)(struct WatchdogCntlr *wdt); //【可选】如果WatchdogCntlr 中的priv成员存在,则按需填充 +> void (*releasePriv)(struct WatchdogCntlr *wdt);//【可选】 +> }; +> ``` +> +> 表1 WatchdogMethod成员的回调函数功能说明 +> |成员函数|入参|出参|返回值|功能| +> |-|-|-|-|-| +> |getStatus |**wdt**: 结构体指针,核心层WDG控制器; |**status**: int32_t指针,
表示狗的状态(打开或关闭); |HDF_STATUS相关状态|获取看门狗所处的状态| +> |start |**wdt**: 结构体指针,核心层WDG控制器; |无 |HDF_STATUS相关状态|打开开门狗 | +> |stop |**wdt**: 结构体指针,核心层WDG控制器; |无 |HDF_STATUS相关状态|关闭开门狗 | +> |setTimeout|**wdt**: 结构体指针,核心层WDG控制器;
**seconds**: uint32_t,时间传入值;|无|HDF_STATUS相关状态|设置超时时间值,
需要与设置的时间相对应,
与厂商看门狗的时钟周期相关 | +> |getTimeout|**wdt**: 结构体指针,核心层WDG控制器; |**seconds**: uint32_t,
传出的时间值|HDF_STATUS相关状态|回读设置的超时时间值 | +> |feed |**wdt**: 结构体指针,核心层WDG控制器; |无 |HDF_STATUS相关状态|喂狗 | + + +## 开发实例 + +下方将以watchdog_hi35xx.c为示例,展示需要厂商提供哪些内容来完整实现设备功能。 + +1. 驱动开发首先需要实例化驱动入口,驱动入口必须为HdfDriverEntry(在 hdf_device_desc.h 中定义)类型的全局变量,且moduleName要和device_info.hcs中保持一致。HDF框架会将所有加载的驱动的HdfDriverEntry对象首地址汇总,形成一个类似数组的段地址空间,方便上层调用。 + + 一般在加载驱动时HDF会先调用Bind函数,再调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。 + +- watchdog驱动入口参考 + + ```c + struct HdfDriverEntry g_watchdogDriverEntry = { + .moduleVersion = 1, + .Bind = Hi35xxWatchdogBind, //见Bind参考 + .Init = Hi35xxWatchdogInit, //见Init参考 + .Release = Hi35xxWatchdogRelease, //见Release参考 + .moduleName = "HDF_PLATFORM_WATCHDOG",//【必要且与HCS文件中里面的moduleName匹配】 + }; + HDF_INIT(g_watchdogDriverEntry);//调用HDF_INIT将驱动入口注册到HDF框架中 + ``` + +2. 完成驱动入口注册之后,下一步请在device_info.hcs文件中添加deviceNode信息,并在 watchdog_config.hcs 中配置器件属性。deviceNode信息与驱动入口注册相关,器件属性值与核心层WatchdogCntlr 成员的默认值或限制范围有密切关系。 + + **本例只有一个Watchdog控制器,如有多个器件信息,则需要在device_info文件增加deviceNode信息,以及在watchdog_config文件中增加对应的器件属性**。 + +- device_info.hcs 配置参考 + + ```c + root { + device_info { + match_attr = "hdf_manager"; + device_watchdog :: device { // 设备节点 + device0 :: deviceNode { // 驱动的DeviceNode节点 + policy = 1; // policy字段是驱动服务发布的策略,如果需要面向用户态,则为2 + priority = 20; // 驱动启动优先级 + permission = 0644; // 驱动创建设备节点权限 + moduleName = "HDF_PLATFORM_WATCHDOG"; + // 【必要】驱动名称,该字段的值必须和驱动入口结构的moduleName值一致 + serviceName = "HDF_PLATFORM_WATCHDOG_0"; + // 【必要且唯一】驱动对外发布服务的名称 + deviceMatchAttr = "hisilicon_hi35xx_watchdog_0"; + // 【必要】驱动私有数据匹配的关键字,必须和驱动私有数据配置表中的match_attr值相等 + } + } + } + } + ``` + +- watchdog_config.hcs 配置参考 + + ```c + root { + platform { + template watchdog_controller {//【必要】模板配置,继承该模板的节点如果使用模板中的默认值,则节点字段可以缺省 + id = 0; + match_attr = ""; + regBase = 0x12050000; //【必要】地址映射需要 + regStep = 0x1000; //【必要】地址映射需要 + } + controller_0x12050000 :: watchdog_controller {//【必要】是作为设备驱动私有数据匹配的关键字 + match_attr = "hisilicon_hi35xx_watchdog_0"; //【必要】必须和device_info.hcs中的deviceMatchAttr值一致 + } + //存在多个 watchdog 时【必须】添加,否则不用 + ... + } + } + ``` + +3. 完成驱动入口注册之后,最后一步就是以核心层WatchdogCntlr对象的初始化为核心,包括厂商自定义结构体(传递参数和数据),实例化WatchdogCntlr成员WatchdogMethod(让用户可以通过接口来调用驱动底层函数),实现HdfDriverEntry成员函数(Bind,Init,Release) + +- 自定义结构体参考 + + > 从驱动的角度看,自定义结构体是参数和数据的载体,而且watchdog_config.hcs文件中的数值会被HDF读入通过DeviceResourceIface来初始化结构体成员,其中一些重要数值也会传递给核心层WatchdogCntlr对象,例如索引、管脚数等。 + + ```c + struct Hi35xxWatchdog { + struct WatchdogCntlr wdt; //【必要】是链接上下层的载体,具体描述见下面 + OsalSpinlock lock; //【可选】可挂载到 WatchdogCntlr 的lock成员上,两个是相同的定义 + volatile unsigned char *regBase;//【必要】地址映射需要 + uint32_t phyBase; //【必要】地址映射需要 + uint32_t regStep; //【必要】地址映射需要 + }; + //WatchdogCntlr是核心层控制器结构体,其中的成员在Init函数中会被赋值 + struct WatchdogCntlr { + struct IDeviceIoService service;//驱动服务,【无需挂载】 + struct HdfDeviceObject *device; //驱动设备,需要挂载 bind 函数的入参:struct HdfDeviceObject *device + OsalSpinlock lock; //在HDF核心层调用时从系统代码实现了一个自旋锁的机制,挂载的变量需要是相同的变量,不建议挂载 + struct WatchdogMethod *ops; //接口回调函数 + int16_t wdtId; //WDG设备的识别id + void *priv; //存储指针 + }; + ``` + +- **【重要】** WatchdogCntlr成员回调函数结构体WatchdogMethod的实例化,其他成员在Init和Bind函数中初始化 + + ```c + static struct WatchdogMethod g_method = { + .getStatus = Hi35xxWatchdogGetStatus, + .start = Hi35xxWatchdogStart, + .stop = Hi35xxWatchdogStop, + .setTimeout = Hi35xxWatchdogSetTimeout, + .getTimeout = Hi35xxWatchdogGetTimeout, + .feed = Hi35xxWatchdogFeed, + }; + ``` + + +- **Init函数和Bind函数参考** + + > **入参:** + > HdfDeviceObject :HDF框架给每一个驱动创建的设备对象,用来保存设备相关的私有数据和服务接口 + > + > **返回值:** + > HDF_STATUS相关状态 (下表为部分展示,如需使用其他状态,可见//drivers/framework/include/utils/hdf_base.h中HDF_STATUS 定义) + > + > |状态(值)|问题描述| + > |:-|:-:| + > |HDF_ERR_INVALID_OBJECT|找不到 WDG 设备| + > |HDF_ERR_MALLOC_FAIL |内存分配失败| + > |HDF_ERR_IO |I/O 错误| + > |HDF_SUCCESS |初始化成功| + > |HDF_FAILURE |初始化失败| + > + > **函数说明:** + > 初始化自定义结构体对象,初始化WatchdogCntlr成员,调用核心层WatchdogCntlrAdd函数。 + + ```c + //一般而言,初始化函数需要根据传入设备的属性实现 Hi35xxWatchdog 结构的填充, + //但此示例中的这部分集成在 bind 函数中 + static int32_t Hi35xxWatchdogInit(struct HdfDeviceObject *device) + { + (void)device; + return HDF_SUCCESS; + } + + static int32_t Hi35xxWatchdogBind(struct HdfDeviceObject *device) + { + int32_t ret; + struct Hi35xxWatchdog *hwdt = NULL; + ... + hwdt = (struct Hi35xxWatchdog *)OsalMemCalloc(sizeof(*hwdt));//Hi35xxWatchdog 结构体的内存申请 + ... + hwdt->regBase = OsalIoRemap(hwdt->phyBase, hwdt->regStep); //地址映射 + ... + //最重要的是这个挂载的过程 + hwdt->wdt.priv = (void *)device->property;//【可选】此处填充的是设备属性,但后续没有调用 priv 成员, + // 如果需要用到 priv 成员,需要实现对应的钩子函数填充 WatchdogMethod + // 结构体的 getPriv 和 releasePriv 成员函数 + hwdt->wdt.ops = &g_method; //【必要】WatchdogMethod的实例化对象的挂载 + hwdt->wdt.device = device; //【必要】使HdfDeviceObject与WatchdogcCntlr可以相互转化的前提 + ret = WatchdogCntlrAdd(&hwdt->wdt); //【必要】调用此函数填充核心层结构体,返回成功信号后驱动才完全接入平台核心层 + if (ret != HDF_SUCCESS) { //不成功的话,需要反向执行初始化相关函数 + OsalIoUnmap((void *)hwdt->regBase); + OsalMemFree(hwdt); + return ret; + } + return HDF_SUCCESS; + } + ``` + +- **Release函数参考** + + > **入参:** + > HdfDeviceObject :HDF框架给每一个驱动创建的设备对象,用来保存设备相关的私有数据和服务接口 + > + > **返回值:** + > 无 + > + > **函数说明:** + > 该函数需要在驱动入口结构体中赋值给Release,当HDF框架调用Init函数初始化驱动失败时,可以调用Release释放驱动资源。该函数中需包含释放内存和删除控制器等操作。所有强制转换获取相应对象的操作**前提**是在Init函数中具备对应赋值的操作。 + + ```c + static void Hi35xxWatchdogRelease(struct HdfDeviceObject *device) + { + struct WatchdogCntlr *wdt = NULL; + struct Hi35xxWatchdog *hwdt = NULL; + ... + wdt = WatchdogCntlrFromDevice(device);//这里有HdfDeviceObject到MmcCntlr的强制转化,通过service成员(第一个成员),赋值见Bind函数 + //return (device == NULL) ? NULL : (struct WatchdogCntlr *)device->service; + if (wdt == NULL) { + return; + } + WatchdogCntlrRemove(wdt); //核心层函数,实际执行wdt->device->service = NULL以及cntlr->lock的释放 + hwdt = (struct Hi35xxWatchdog *)wdt; //这里有MmcCntlr到HimciHost的强制转化 + if (hwdt->regBase != NULL) { //地址反映射 + OsalIoUnmap((void *)hwdt->regBase); + hwdt->regBase = NULL; + } + OsalMemFree(hwdt); //厂商自定义对象的内存释放 + } + ``` diff --git a/zh-cn/device-dev/driver/driver-platform.md b/zh-cn/device-dev/driver/driver-platform.md index a05dec1767..8c34624b9f 100644 --- a/zh-cn/device-dev/driver/driver-platform.md +++ b/zh-cn/device-dev/driver/driver-platform.md @@ -1,4 +1,4 @@ -# 平台驱动 +# 平台驱动使用 - **[GPIO](driver-platform-gpio-des.md)** diff --git "a/zh-cn/device-dev/driver/figure/\346\227\240\346\234\215\345\212\241\346\250\241\345\274\217.png" "b/zh-cn/device-dev/driver/figure/\346\227\240\346\234\215\345\212\241\346\250\241\345\274\217.png" new file mode 100755 index 0000000000000000000000000000000000000000..833ff6cc89e49ed4210dd68a502e4b304ac1c273 GIT binary patch literal 40274 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%0D)hUo|!Ldv`f6693s!R-FcDi4A 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 r!G?VP)I-okgO^PuigJ&E-w(y literal 0 HcmV?d00001 diff --git "a/zh-cn/device-dev/driver/figure/\347\213\254\347\253\213\346\234\215\345\212\241\346\250\241\345\274\217.png" "b/zh-cn/device-dev/driver/figure/\347\213\254\347\253\213\346\234\215\345\212\241\346\250\241\345\274\217.png" new file mode 100755 index 0000000000000000000000000000000000000000..23324872566e5affac8baa186a30b64b3257f673 GIT binary patch literal 52253 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 &eNi{TXCioUfA`+i!=jc>6w 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 FtUOuHR>G462RTc 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 Aj{pDw literal 0 HcmV?d00001 -- GitLab