Harmony鸿蒙南向驱动开发-SDIO接口使用

功能简介

SDIO是安全数字输入输出接口(Secure Digital Input and Output)的缩写,是从SD内存卡接口的基础上演化出来的一种外设接口。SDIO接口兼容以前的SD卡,并且可以连接支持SDIO接口的其他设备。

SDIO接口定义了操作SDIO的通用方法集合,包括:

  • 打开/关闭SDIO控制器

  • 独占/释放HOST

  • 使能/去使能设备

  • 申请/释放中断

  • 读写、获取/设置公共信息

运作机制

在HDF框架中,SDIO的接口适配模式采用独立服务模式。在这种模式下,每一个设备对象会独立发布一个设备服务来处理外部访问,设备管理器收到API的访问请求之后,通过提取该请求的参数,达到调用实际设备对象的相应内部方法的目的。独立服务模式可以直接借助HDFDeviceManager的服务管理能力,但需要为每个设备单独配置设备节点,若设备过多可能增加内存占用。

独立服务模式下,核心层不会统一发布一个服务供上层使用,因此这种模式下驱动要为每个控制器发布一个服务,具体表现为:

  • 驱动适配者需要实现HdfDriverEntry的Bind钩子函数以绑定服务。

  • device_info.hcs文件中deviceNode的policy字段为1或2,不能为0。

SDIO模块各分层作用:

  • 接口层提供打开SDIO设备、设置块的大小、读取数据、写数据、设置公共信息、获取公共信息、刷新数据、独占HOST、释放Host、使能SDIO功能设备、去使能SDIO功能设备、申请中断、释放中断关闭SDIO设备的接口。

  • 核心层主要提供SDIO控制器的添加、移除及管理的能力,通过钩子函数与适配层交互。

  • 适配层主要是将钩子函数的功能实例化,实现具体的功能。

图 1 SDIO独立服务模式结构图

SDIO独立服务模式结构图

SDIO总线有两端,其中一端是主机端(HOST),另一端是设备端(DEVICE)。所有的通信都是由HOST端发出命令开始的,在DEVICE端只要能解析HOST的命令,就可以同HOST进行通信了。SDIO的HOST可以连接多个DEVICE,如图2所示:

  • CLK信号:HOST给DEVICE的时钟信号。

  • VDD信号:电源信号。

  • VSS信号:Ground信号。

  • D0-3信号:4条数据线,其中,DAT1信号线复用为中断线,在1BIT模式下DAT0用来传输数据,在4BIT模式下DAT0-DAT3用来传输数据。

  • CMD信号:用于HOST发送命令和DEVICE回复响应。

图 2 SDIO的HOST-DEVICE连接示意图

SDIO的HOST-DEVICE连接示意图

约束与限制

SDIO模块API当前仅支持内核态调用。

使用指导

场景介绍

SDIO的应用比较广泛,目前,有许多手机都支持SDIO功能,并且很多SDIO外设也被开发出来,使得手机外接外设更加容易。常见的SDIO外设有WLAN、GPS、CAMERA、蓝牙等。

接口说明

SDIO模块提供的主要接口如表1所示,具体API详见//drivers/hdf_core/framework/include/platform/sdio_if.h。

表 1 SDIO驱动API接口功能介绍

接口名接口描述
DevHandle SdioOpen(int16_t mmcBusNum, struct SdioFunctionConfig *config)打开指定总线号的SDIO控制器
void SdioClose(DevHandle handle)关闭SDIO控制器
int32_t SdioReadBytes(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size)从指定地址开始,增量读取指定长度的数据
int32_t SdioWriteBytes(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size)从指定地址开始,增量写入指定长度的数据
int32_t SdioReadBytesFromFixedAddr(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size, uint32_t scatterLen)从固定地址读取指定长度的数据
int32_t SdioWriteBytesToFixedAddr(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size, uint32_t scatterLen)向固定地址写入指定长度的数据
int32_t SdioReadBytesFromFunc0(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size)从SDIO function 0的指定地址空间读取指定长度的数据
int32_t SdioWriteBytesToFunc0(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size)向SDIO function 0的指定地址空间写入指定长度的数据
int32_t SdioSetBlockSize(DevHandle handle, uint32_t blockSize)设置块的大小
int32_t SdioGetCommonInfo(DevHandle handle, SdioCommonInfo *info, SdioCommonInfoType infoType)获取公共信息
int32_t SdioSetCommonInfo(DevHandle handle, SdioCommonInfo *info, SdioCommonInfoType infoType)设置公共信息
int32_t SdioFlushData(DevHandle handle)刷新数据
void SdioClaimHost(DevHandle handle)独占Host
void SdioReleaseHost(DevHandle handle)释放Host
int32_t SdioEnableFunc(DevHandle handle)使能SDIO功能设备
int32_t SdioDisableFunc(DevHandle handle)去使能SDIO功能设备
int32_t SdioClaimIrq(DevHandle handle, SdioIrqHandler *irqHandler)申请中断
int32_t SdioReleaseIrq(DevHandle handle)释放中断

使用流程

使用SDIO的一般流程如图3所示。

图 3 SDIO使用流程图

SDIO使用流程图

打开SDIO控制器

在使用SDIO进行通信前,首先要调用SdioOpen获取SDIO控制器的设备句柄,该函数会返回指定总线号的SDIO控制器的设备句柄。

DevHandle SdioOpen(int16_t mmcBusNum, struct SdioFunctionConfig *config);

表 2 SdioOpen函数的参数和返回值描述

参数参数描述
mmcBusNumint16_t类型,总线号
config结构体指针,SDIO功能配置信息
返回值返回值描述
NULL获取SDIO控制器的设备句柄失败
设备句柄SDIO控制器的设备句柄

打开SDIO控制器的示例如下:

DevHandle handle = NULL;
struct SdioFunctionConfig config;
config.funcNr = 1;
config.vendorId = 0x123;
config.deviceId = 0x456;
// 打开总线号为1的SDIO控制器
handle = SdioOpen(1, &config);
if (handle == NULL) {HDF_LOGE("SdioOpen: open sdio fail!\n");return NULL;
}
独占HOST

获取到SDIO控制器的设备句柄之后,需要先独占HOST才能进行SDIO后续的一系列操作,独占HOST函数如下所示:

void SdioClaimHost(DevHandle handle);

表 3 SdioClaimHost函数的参数描述

参数参数描述
handleDevHandle类型,SDIO控制器的设备句柄

独占HOST示例如下:

SdioClaimHost(handle); // 独占HOST
使能SDIO设备

在访问寄存器之前,需要先使能SDIO设备,使能SDIO设备的函数如下所示:

int32_t SdioEnableFunc(DevHandle handle);

表 4 SdioEnableFunc函数的参数和返回值描述

参数参数描述
handleDevHandle类型,SDIO控制器的设备句柄
返回值返回值描述
HDF_SUCCESSSDIO使能成功
负数SDIO使能失败

使能SDIO设备的示例如下:

int32_t ret;
// 使能SDIO设备
ret = SdioEnableFunc(handle);
if (ret != HDF_SUCCESS) {HDF_LOGE("SdioEnableFunc: sdio enable func fail, ret:%d\n", ret);return ret;
}
注册SDIO中断

在通信之前,还需要注册SDIO中断,注册SDIO中断函数如下所示:

int32_t SdioClaimIrq(DevHandle handle, SdioIrqHandler *handler);

表 5 SdioClaimIrq函数的参数和返回值描述

参数参数描述
handleDevHandle类型,SDIO控制器的设备句柄
handler函数指针,中断服务函数
返回值返回值描述
HDF_SUCCESS注册SDIO中断成功
负数注册SDIO中断失败

注册SDIO中的示例如下:

// 中断服务函数需要根据各自平台的情况去实现
static void SdioIrqFunc(void *data)
{if (data == NULL) {HDF_LOGE("SdioIrqFunc: data is NULL.\n");return;}// 需要开发者自行添加具体实现
}int32_t ret;
// 注册SDIO中断
ret = SdioClaimIrq(handle, SdioIrqFunc);
if (ret != HDF_SUCCESS) {HDF_LOGE("SdioClaimIrq: sdio claim irq fail, ret:%d\n", ret);return ret;
}
进行SDIO通信
  • 向SDIO设备增量写入指定长度的数据

    对应的接口函数如下所示:

    int32_t SdioWriteBytes(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size);

    表 6 SdioWriteBytes函数的参数和返回值描述

    参数参数描述
    handleDevHandle类型,SDIO控制器的设备句柄
    datauint8_t类型指针,待写入数据
    addruint32_t类型,待写入数据的起始地址
    sizeuint32_t类型,待写入数据的长度
    返回值返回值描述
    HDF_SUCCESSSDIO写数据成功
    负数SDIO写数据失败

    向SDIO设备增量写入指定长度的数据的示例如下:

    int32_t ret;
    uint8_t wbuff[] = {1,2,3,4,5};
    uint32_t addr = 0x100 + 0x09;
    // 向SDIO设备起始地址0x109,增量写入5个字节的数据
    ret = SdioWriteBytes(handle, wbuff, addr, sizeof(wbuff) / sizeof(wbuff[0]));
    if (ret != HDF_SUCCESS) {HDF_LOGE("SdioWriteBytes: sdio write bytes fail, ret:%d\n", ret);return ret;
    }
  • 从SDIO设备增量读取指定长度的数据

    对应的接口函数如下所示:

    int32_t SdioReadBytes(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size);

    表 7 SdioReadBytes函数的参数和返回值描述

    参数参数描述
    handleDevHandle类型,SDIO控制器的设备句柄
    datauint8_t类型指针,接收读取数据
    addruint32_t类型,待读取数据的起始地址
    sizeuint32_t类型,待读取数据的长度
    返回值返回值描述
    HDF_SUCCESSSDIO读数据成功
    负数SDIO读数据失败

    从SDIO设备增量读取指定长度的数据的示例如下:

    int32_t ret;
    uint8_t rbuff[5] = {0};
    uint32_t addr = 0x100 + 0x09;
    // 从SDIO设备起始地址0x109,增量读取5个字节的数据
    ret = SdioReadBytes(handle, rbuff, addr, 5);
    if (ret != HDF_SUCCESS) {HDF_LOGE("SdioReadBytes: sdio read bytes fail, ret:%d\n", ret);return ret;
    }
  • 向SDIO设备的固定地址写入指定长度的数据

    对应的接口函数如下所示:

    int32_t SdioWriteBytesToFixedAddr(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size, uint32_t scatterLen);

    表 8 SdioWriteBytesToFixedAddr函数的参数和返回值描述

    参数参数描述
    handleDevHandle类型,SDIO控制器的设备句柄
    datauint8_t类型指针,待写入数据
    addruint32_t类型,待写入数据的固定地址
    sizeuint32_t类型,待写入数据的长度
    scatterLenuint32_t类型,集散表的长度。如果该字段不为0,则data为集散表类型。
    返回值返回值描述
    HDF_SUCCESSSDIO写数据成功
    负数SDIO写数据失败

    向SDIO设备的固定地址写入指定长度的数据的示例如下:

    int32_t ret;
    uint8_t wbuff[] = {1, 2, 3, 4, 5};
    uint32_t addr = 0x100 + 0x09;
    // 向SDIO设备固定地址0x109写入5个字节的数据
    ret = SdioWriteBytesToFixedAddr(handle, wbuff, addr, sizeof(wbuff) / sizeof(wbuff[0]), 0);
    if (ret != HDF_SUCCESS) {HDF_LOGE("SdioWriteBytesToFixedAddr: sdio write bytes to fixed addr fail, ret:%d\n", ret);return ret;
    }
  • 从SDIO设备的固定地址读取指定长度的数据

    对应的接口函数如下所示:

    int32_t SdioReadBytesFromFixedAddr(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size, uint32_t scatterLen);

    表 9 SdioReadBytesFromFixedAddr函数的参数和返回值描述

    参数参数描述
    handleDevHandle类型,SDIO控制器的设备句柄
    datauint8_t类型指针,接收读取数据
    addruint32_t类型,待读取数据的起始地址
    sizeuint32_t类型,待读取数据的长度
    scatterLenuint32_t类型,集散表的长度。如果该字段不为0,则data为集散表类型。
    返回值返回值描述
    HDF_SUCCESSSDIO读数据成功
    负数SDIO读数据失败

    从SDIO设备的固定地址读取指定长度的数据的示例如下:

    int32_t ret;
    uint8_t rbuff[5] = {0};
    uint32_t addr = 0x100 + 0x09;
    // 从SDIO设备固定地址0x109中读取5个字节的数据 
    ret = SdioReadBytesFromFixedAddr(handle, rbuff, addr, 5, 0);
    if (ret != HDF_SUCCESS) {HDF_LOGE("SdioReadBytesFromFixedAddr: sdio read bytes from fixed addr fail, ret:%d\n", ret);return ret;
    }
  • 向SDIO function 0的指定地址空间写入指定长度的数据

    当前只支持写入一个字节的数据,对应的接口函数如下所示:

    int32_t SdioWriteBytesToFunc0(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size);

    表 10 SdioWriteBytesToFunc0函数的参数和返回值描述

    参数参数描述
    handleDevHandle类型,SDIO控制器的设备句柄
    datauint8_t类型指针,待写入数据
    addruint32_t类型,待写入数据的起始地址
    sizeuint32_t类型,待写入数据的长度
    返回值返回值描述
    HDF_SUCCESSSDIO写数据成功
    负数SDIO写数据失败

    向SDIO function 0的指定地址空间写入指定长度的数据的示例如下:

    int32_t ret;
    uint8_t wbuff = 1;
    // 向SDIO function 0地址0x2中写入1字节的数据
    ret = SdioWriteBytesToFunc0(handle, &wbuff, 0x2, 1);
    if (ret != HDF_SUCCESS) {HDF_LOGE("SdioWriteBytesToFunc0: sdio write bytes to func0 fail, ret:%d\n", ret);return ret;
    }
  • 从SDIO function 0的指定地址空间读取指定长度的数据

    当前只支持读取一个字节的数据,对应的接口函数如下所示:

    int32_t SdioReadBytesFromFunc0(DevHandle handle, uint8_t *data, uint32_t addr, uint32_t size);

    表 11 SdioReadBytesFromFunc0函数的参数和返回值描述

    参数参数描述
    handleDevHandle类型,SDIO控制器的设备句柄
    datauint8_t类型指针,接收读取数据
    addruint32_t类型,待读取数据的起始地址
    sizeuint32_t类型,待读取数据的长度
    返回值返回值描述
    HDF_SUCCESSSDIO读数据成功
    负数SDIO读数据失败

    从SDIO function 0的指定地址空间读取指定长度的数据的示例如下:

    int32_t ret;
    uint8_t rbuff;
    /* 从SDIO function 0设备地址0x2中读取1字节的数据 */
    ret = SdioReadBytesFromFunc0(handle, &rbuff, 0x2, 1);
    if (ret != 0) {HDF_LOGE("SdioReadBytesFromFunc0: sdio read bytes from func0 fail, ret:%d\n", ret);return ret;
    }
释放SDIO中断

通信完成之后,需要释放SDIO中断,函数如下所示:

int32_t SdioReleaseIrq(DevHandle handle);

表 12 SdioReleaseIrq函数的参数和返回值描述

参数参数描述
handleDevHandle类型,SDIO控制器的设备句柄
返回值返回值描述
HDF_SUCCESS释放SDIO中断成功
负数释放SDIO中断失败

释放SDIO中断的示例如下:

int32_t ret;
// 释放SDIO中断
ret = SdioReleaseIrq(handle);
if (ret != HDF_SUCCESS) {HDF_LOGE("SdioReleaseIrq: sdio release irq fail, ret:%d\n", ret);return ret;
}
去使能SDIO设备

通信完成之后,还需要去使能SDIO设备,函数如下所示:

int32_t SdioDisableFunc(DevHandle handle);

表 13 SdioDisableFunc函数的参数和返回值描述

参数参数描述
handleDevHandle类型,SDIO控制器的设备句柄
返回值返回值描述
HDF_SUCCESS去使能SDIO设备成功
负数去使能SDIO设备失败

去使能SDIO设备的示例如下:

int32_t ret;
// 去使能SDIO设备
ret = SdioDisableFunc(handle);
if (ret != HDF_SUCCESS) {HDF_LOGE("SdioDisableFunc: sdio disable func fail, ret:%d\n", ret);return ret;
}
释放HOST

通信完成之后,还需要释放去HOST,函数如下所示:

void SdioReleaseHost(DevHandle handle);

表 14 SdioReleaseHost函数的参数描述

参数参数描述
handleDevHandle类型,SDIO控制器的设备句柄

释放HOST的示例如下:

SdioReleaseHost(handle); // 释放HOST
关闭SDIO控制器

SDIO通信完成之后,最后需要关闭SDIO控制器,函数如下所示:

void SdioClose(DevHandle handle);

该函数会释放掉申请的资源。

表 15 SdioClose函数的参数描述

参数参数描述
handleDevHandle类型,SDIO控制器的设备句柄

关闭SDIO控制器的示例如下:

SdioClose(handle); // 关闭SDIO控制器

使用实例

本例拟对Hi3516DV300开发板上SDIO设备进行操作。

SDIO设备完整的使用示例如下所示,首先打开总线号为1的SDIO控制器,然后独占HOST、使能设备、注册中断,接着进行SDIO通信(读写等),通信完成之后,释放中断、去使能设备、释放HOST,最后关闭SDIO控制器。

#include "hdf_log.h"
#include "sdio_if.h"#define TEST_FUNC_NUM 1              /* 本测试用例中,使用编号为1的I/O function */
#define TEST_FBR_BASE_ADDR 0x100     /* 编号为1的I/O function的FBR基地址 */
#define TEST_ADDR_OFFSET 9           /* 本测试用例中,需要读写的寄存器的地址偏移 */
#define TEST_DATA_LEN 3              /* 本测试用例中,读写数据的长度 */
#define TEST_BLOCKSIZE 2             /* 本测试用例中,数据块的大小,单位字节 *//* 中断服务函数,需要根据各自平台的情况去实现 */
static void SdioIrqFunc(void *data)
{if (data == NULL) {HDF_LOGE("SdioIrqFunc: data is NULL.\n");return;}/* 需要开发者自行添加具体的实现 */
}void SdioTestSample(void)
{int32_t ret;  DevHandle handle = NULL;uint8_t data[TEST_DATA_LEN] = {0};struct SdioFunctionConfig config = {1, 0x123, 0x456};uint8_t val;uint32_t addr;/* 打开总线号为1的SDIO设备 */handle = SdioOpen(1, &config);if (handle == NULL) {HDF_LOGE("SdioOpen: failed!\n");return;}/* 独占HOST */SdioClaimHost(handle);/* 使能SDIO设备 */ret = SdioEnableFunc(handle);if (ret != 0) {HDF_LOGE("SdioEnableFunc: failed, ret %d\n", ret);goto ENABLE_ERR;}/* 注册中断 */ret = SdioClaimIrq(handle, SdioIrqFunc);if (ret != 0) {HDF_LOGE("SdioClaimIrq: failed, ret %d\n", ret);goto CLAIM_IRQ_ERR;}/* 设置块大小为2字节 */ret = SdioSetBlockSize(handle, TEST_BLOCKSIZE);if (ret != 0) {HDF_LOGE("SdioSetBlockSize: failed, ret %d\n", ret);goto COMM_ERR;}/* 从SDIO设备增量地址读取3字节的数据 */addr = TEST_FBR_BASE_ADDR * TEST_FUNC_NUM + TEST_ADDR_OFFSET;ret = SdioReadBytes(handle, data, addr, TEST_DATA_LEN);if (ret != 0) {HDF_LOGE("SdioReadBytes: failed, ret %d\n", ret);goto COMM_ERR;}/* 向SDIO设备增量地址写入3字节的数据 */ret = SdioWriteBytes(handle, data, addr, TEST_DATA_LEN);if (ret != 0) {HDF_LOGE("SdioWriteBytes: failed, ret %d\n", ret);goto COMM_ERR;}/* 从SDIO设备读取1字节的数据 */ret = SdioReadBytes(handle, &val, addr, 1);if (ret != 0) {HDF_LOGE("SdioReadBytes: failed, ret %d\n", ret);goto COMM_ERR;}/* 向SDIO设备写入1字节的数据 */ret = SdioWriteBytes(handle, &val, addr, 1);if (ret != 0) {HDF_LOGE("SdioWriteBytes: failed, ret %d\n", ret);goto COMM_ERR;}/* 从SDIO设备固定地址读取3字节的数据 */ret = SdioReadBytesFromFixedAddr(handle, data, addr, TEST_DATA_LEN, 0);if (ret != 0) {HDF_LOGE("SdioReadBytesFromFixedAddr: failed, ret %d\n", ret);goto COMM_ERR;}/* 向SDIO设备固定地址写入1字节的数据 */ret = SdioWriteBytesToFixedAddr(handle, data, addr, 1, 0);if (ret != 0) {HDF_LOGE("SdioWriteBytesToFixedAddr: failed, ret %d\n", ret);goto COMM_ERR;}/* 从SDIO function 0读取1字节的数据 */addr = 0x02;ret = SdioReadBytesFromFunc0(handle, &val, addr, 1);if (ret != 0) {HDF_LOGE("SdioReadBytesFromFunc0: failed, ret %d\n", ret);goto COMM_ERR;}/* 向SDIO function 0写入1字节的数据 */ret = SdioWriteBytesToFunc0(handle, &val, addr, 1);if (ret != 0) {HDF_LOGE("SdioWriteBytesToFunc0: failed, ret %d\n", ret);goto COMM_ERR;}
COMM_ERR:/* 释放中断 */ret = SdioReleaseIrq(handle);if (ret != 0) {HDF_LOGE("SdioReleaseIrq: failed, ret %d\n", ret);}
CLAIM_IRQ_ERR:/* 去使能SDIO设备 */ret = SdioDisableFunc(handle);if (ret != 0) {HDF_LOGE("SdioDisableFunc: failed, ret %d\n", ret);}
ENABLE_ERR:/* 释放HOST */SdioReleaseHost(handle);/* 关闭SDIO设备 */SdioClose(handle); 
}

最后

有很多小伙伴不知道学习哪些鸿蒙开发技术?不知道需要重点掌握哪些鸿蒙应用开发知识点?而且学习时频繁踩坑,最终浪费大量时间。所以有一份实用的鸿蒙(HarmonyOS NEXT)资料用来跟着学习是非常有必要的。 

这份鸿蒙(HarmonyOS NEXT)资料包含了鸿蒙开发必掌握的核心知识要点,内容包含了ArkTS、ArkUI开发组件、Stage模型、多端部署、分布式应用开发、音频、视频、WebGL、OpenHarmony多媒体技术、Napi组件、OpenHarmony内核、Harmony南向开发、鸿蒙项目实战等等)鸿蒙(HarmonyOS NEXT)技术知识点。

希望这一份鸿蒙学习资料能够给大家带来帮助,有需要的小伙伴自行领取,限时开源,先到先得~无套路领取!!

获取这份完整版高清学习路线,请点击→纯血版全套鸿蒙HarmonyOS学习资料

鸿蒙(HarmonyOS NEXT)最新学习路线

  •  HarmonOS基础技能

  • HarmonOS就业必备技能 
  •  HarmonOS多媒体技术

  • 鸿蒙NaPi组件进阶

  • HarmonOS高级技能

  • 初识HarmonOS内核 
  • 实战就业级设备开发

有了路线图,怎么能没有学习资料呢,小编也准备了一份联合鸿蒙官方发布笔记整理收纳的一套系统性的鸿蒙(OpenHarmony )学习手册(共计1236页)鸿蒙(OpenHarmony )开发入门教学视频,内容包含:ArkTS、ArkUI、Web开发、应用模型、资源分类…等知识点。

获取以上完整版高清学习路线,请点击→纯血版全套鸿蒙HarmonyOS学习资料

《鸿蒙 (OpenHarmony)开发入门教学视频》

《鸿蒙生态应用开发V2.0白皮书》

图片

《鸿蒙 (OpenHarmony)开发基础到实战手册》

OpenHarmony北向、南向开发环境搭建

图片

 《鸿蒙开发基础》

  • ArkTS语言
  • 安装DevEco Studio
  • 运用你的第一个ArkTS应用
  • ArkUI声明式UI开发
  • .……

图片

 《鸿蒙开发进阶》

  • Stage模型入门
  • 网络管理
  • 数据管理
  • 电话服务
  • 分布式应用开发
  • 通知与窗口管理
  • 多媒体技术
  • 安全技能
  • 任务管理
  • WebGL
  • 国际化开发
  • 应用测试
  • DFX面向未来设计
  • 鸿蒙系统移植和裁剪定制
  • ……

图片

《鸿蒙进阶实战》

  • ArkTS实践
  • UIAbility应用
  • 网络案例
  • ……

图片

 获取以上完整鸿蒙HarmonyOS学习资料,请点击→纯血版全套鸿蒙HarmonyOS学习资料

总结

总的来说,华为鸿蒙不再兼容安卓,对中年程序员来说是一个挑战,也是一个机会。只有积极应对变化,不断学习和提升自己,他们才能在这个变革的时代中立于不败之地。 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/814016.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

21 标准错误

标准输出重定向关闭无数据 下面的代码&#xff1a; #include <stdio.h> #include <string.h> #include <stdlib.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h>int main() {close(1);i…

Xilinx Zynq UltraScale+ MPSoC无人机控制器

官方术语是无人驾驶飞行器&#xff08;UAV&#xff09;&#xff0c;这显然有点拗口&#xff0c;所以我们更喜欢说无人机。在过去的几十年里&#xff0c;无人机技术有了巨大的进步。我们为一个客户开发了一个无人机的飞行和视频控制器。 客户挑战 客户需要一种混合FPGA/CPU硬件&…

是的,本科毕业八年,我考研了

今天&#xff0c;是一篇纯分享文。 是的&#xff0c;本科毕业八年&#xff0c;我考研了。 停更10个月&#xff0c;历时296天&#xff0c;我考研上岸了。 小伙伴们&#xff0c;好久不见。 一 发今年第一篇文章的时候刚处理完后续事宜&#xff0c;就简单说了句&#xff0c;后台…

MoCo v1(CVPR 2020)原理与代码解读

paper&#xff1a;Momentum Contrast for Unsupervised Visual Representation Learning official implementation&#xff1a;https://github.com/facebookresearch/moco 背景 最近的一些研究提出使用对比损失相关的方法进行无监督视觉表征学习并取得了不错的结果。尽管是受…

【DM8】临时表空间

临时表分类 事务级临时表会话级临时表 临时表&#xff0c;可以像普通表一样插入、更新和删除数据临时表的DML操作产生较少的redo日志临时表支持创建索引&#xff0c;以提高查询性能在一个会话或事务结束之后&#xff0c;数据将自动从临时表中删除不同的用户访问相同的临时表&a…

吴恩达深度学习 (week3,4)

文章目录 一、神经网络概述二、神经网络的表示三、神经网络的输出四、多个例子的向量化五、向量化实现的解释六、深度学习激活函数七、激活函数导数八、神经网络的梯度下降法九、深度学习随机初始化十、上述学习总结1、第一题2、第二题3、第三题4、第四题5、第五题6、第六题7、…

关于Transformer的面试题

文章目录 前言1、Transformer模型1. 1 基本要点1.2 提升 2、BN和LN的区别和联系2.1 基本要点2.2 扩展 3、PreNorm和PostNorm的区别[链接](https://www.zhihu.com/question/519668254)4、Multi-head self-attention中为什么要用三个不同的矩阵 前言 Transformer大模型的一些问题…

【GEE实践应用】哨兵1号和2号数据叠加

目录 1.数据叠加代码 2.代码逐句解释 1.数据叠加代码 var geometry table; //table是我们提前导入的矢量数据 // 加载Sentinel-2影像 var sentinel2 ee.ImageCollection("COPERNICUS/S2").filterBounds(geometry) // geometry是你感兴趣区域的几何对象.filte…

Redis:发布和订阅

文章目录 一、介绍二、发布订阅命令 一、介绍 Redis的发布和订阅功能是一种消息通信模式&#xff0c;发送者&#xff08;pub&#xff09;发送消息&#xff0c;订阅者&#xff08;sub&#xff09;接收消息。这种功能使得消息发送者和接收者不需要直接建立连接&#xff0c;而是通…

商标没有去注册有哪些不好的影响!

有些商家咨询普推知产老杨&#xff0c;商标没有去注册有哪些不好的影响&#xff0c;其实对企业来说还有许多实际不利的影响&#xff0c;有时代价比注册一个商标要大很多。 想的商标名称没去注册商标&#xff0c;如果别人抢注拿下商标注册证&#xff0c;那就会涉及侵权&#xf…

工厂方法模式:解锁灵活的对象创建策略

在软件设计中&#xff0c;工厂方法模式是一种非常实用的创建型设计模式&#xff0c;它不仅提升了系统的灵活性&#xff0c;还简化了对象的创建过程。本文将详细探讨工厂方法模式的核心概念、实现方式、应用场景以及与其他设计模式的对比&#xff0c;旨在提供一份全面且实用的指…

磁悬浮鼓风机市场规模不断增长 我国行业发展面临挑战

磁悬浮鼓风机市场规模不断增长 我国行业发展面临挑战 磁悬浮鼓风机又称磁悬浮高速离心鼓风机&#xff0c;指基于磁悬浮技术制成的气体输送设备。磁悬浮鼓风机综合性能优良&#xff0c;属于高效节能磁悬浮动力装备&#xff0c;在众多领域需求旺盛。未来随着国家节能环保政策逐渐…

阿里云优惠口令2024最新

2024年阿里云域名优惠口令&#xff0c;com域名续费优惠口令“com批量注册更享优惠”&#xff0c;cn域名续费优惠口令“cn注册多个价格更优”&#xff0c;cn域名注册优惠口令“互联网上的中国标识”&#xff0c;阿里云优惠口令是域名专属的优惠码&#xff0c;可用于域名注册、续…

01—JavaScript概述

一、初识Javascript JavaScript一种直译式脚本语言&#xff0c;是一种动态类型、弱类型、基于原型的语言&#xff0c;内置支持类型。它的解释器被称为JavaScript引擎&#xff0c;为浏览器的一部分&#xff0c;广泛用于客户端的脚本语言&#xff0c;最早是在 HTML&#xff08;标…

jsoncpp 编译和使用

原文链接&#xff1a; jsoncpp的编译和使用 jsoncpp 编译出库文件 1.从github仓库下载 2.下载 cmake 工具 3.生成VS项目 4.编译得到需要的库文件 jsoncpp 的使用 查看原文

基于Springboot的自习室预订系统

基于SpringbootVue的自习室预订系统的设计与实现 开发语言&#xff1a;Java数据库&#xff1a;MySQL技术&#xff1a;SpringbootMybatis工具&#xff1a;IDEA、Maven、Navicat 系统展示 用户登录页 网站首页 公告信息 留言反馈 后台管理 学生信息管理 公告信息管理 留言…

入门:多层感知器Multiple-Layer Perceiver, MLP

本文将简单介绍多层感知器&#xff08;MLP&#xff09;的基本概念、原理和应用。MLP是一种前馈人工神经网络&#xff0c;由多层节点组成&#xff0c;每层节点通过权重和偏置与下一层节点相连。MLP在许多领域都有广泛的应用&#xff0c;如分类、回归、自然语言处理等。 本文将分…

SRNIC、选择性重传、伸缩性、连接扩展性、RoCEv2优化(六)

参考论文SRDMA&#xff08;A Scalable Architecture for RDMA NICs &#xff09;&#xff1a;https://download.csdn.net/download/zz2633105/89101822 借此&#xff0c;对论文内容总结、加以思考和额外猜想&#xff0c;如有侵权&#xff0c;请联系删除。 如有描述不当之处&…

04异常Lambda算法正则

异常 异常是什么&#xff1f; 异常是代码在编译或者执行的过程中可能出现的错误。避免异常的出现&#xff0c;同时处理可能出现的异常&#xff0c;让代码更稳健。 异常分为几类&#xff1f; 编译时异常、运行时异常。编译时异常&#xff1a;没有继承RuntimeExcpetion的异常…

Linux: 工具: tshark 抓到了收方向的ESP明文包?

根据这个描述&#xff0c;看着是正常的&#xff0c; 抓到包之后&#xff0c;可以方便的分析问题&#xff0c;省去在wireshark里解码的问题。 经过调查发现是内核将ESP解开之后&#xff0c;如果是tunnel模式&#xff0c;内核又重新将skb丢给了interface去做处理。这样tshark/tcp…