鸿蒙内核源码分析(中断管理篇) | 江湖从此不再怕中断

关于中断部分系列篇将用三篇详细说明整个过程.

  • 中断概念篇 中断概念很多,比如中断控制器,中断源,中断向量,中断共享,中断处理程序等等.本篇做一次整理.先了解透概念才好理解中断过程.用海公公打比方说明白中断各个概念.可前往查看.

  • 中断管理篇(本篇) 从中断初始化HalIrqInit开始,到注册中断的LOS_HwiCreate函数,到消费中断函数的 HalIrqHandler,剖析鸿蒙内核实现中断的过程,很像设计模式中的观察者模式.

  • 中断切换篇 用自下而上的方式,从中断源头纯汇编代码往上跟踪代码细节.说清楚保存和恢复中断现场TaskIrqContext过程.

编译开关

//....
#define LOSCFG_ARCH_ARM_VER "armv7-a"
#define LOSCFG_ARCH_CPU "cortex-a7"
#define LOSCFG_PLATFORM "hi3516dv300"
#define LOSCFG_PLATFORM_BSP_GIC_V2 1
#define LOSCFG_PLATFORM_ROOTFS 1
#define LOSCFG_KERNEL_CPPSUPPORT 1
#define LOSCFG_HW_RANDOM_ENABLE 1
#define LOSCFG_ARCH_CORTEX_A7 1
#define LOSCFG_DRIVERS_HDF_PLATFORM_RTC 1
#define LOSCFG_DRIVERS_HDF_PLATFORM_UART 1

中断初始化

hi3516dv300 中断控制器选择了 LOSCFG_PLATFORM_BSP_GIC_V2 ,对应代码为 gic_v2.c
GIC(Generic Interrupt Controller)是ARM公司提供的一个通用的中断控制器.
看这种代码因为涉及硬件部分,需要对照 ARM中断控制器 gic_v2.pdf 文档看.可前往地址下载查看.

//硬件中断初始化
VOID HalIrqInit(VOID)
{UINT32 i;/* set externel interrupts to be level triggered, active low. */	//将外部中断设置为电平触发,低电平激活for (i = 32; i < OS_HWI_MAX_NUM; i += 16) {GIC_REG_32(GICD_ICFGR(i / 16)) = 0;}/* set externel interrupts to CPU 0 */	//将外部中断设置为CPU 0for (i = 32; i < OS_HWI_MAX_NUM; i += 4) {GIC_REG_32(GICD_ITARGETSR(i / 4)) = 0x01010101;}/* set priority on all interrupts */	//设置所有中断的优先级for (i = 0; i < OS_HWI_MAX_NUM; i += 4) {GIC_REG_32(GICD_IPRIORITYR(i / 4)) = GICD_INT_DEF_PRI_X4;}/* disable all interrupts. */			//禁用所有中断。for (i = 0; i < OS_HWI_MAX_NUM; i += 32) {GIC_REG_32(GICD_ICENABLER(i / 32)) = ~0;}HalIrqInitPercpu();//初始化当前CPU中断信息/* enable gic distributor control */GIC_REG_32(GICD_CTLR) = 1; //使能分发中断寄存器,该寄存器作用是允许给CPU发送中断信号#if (LOSCFG_KERNEL_SMP == YES)/* register inter-processor interrupt *///注册核间中断,啥意思?就是CPU各核直接可以发送中断信号//处理器间中断允许一个CPU向系统其他的CPU发送中断信号,处理器间中断(IPI)不是通过IRQ线传输的,而是作为信号直接放在连接所有CPU本地APIC的总线上。LOS_HwiCreate(LOS_MP_IPI_WAKEUP, 0xa0, 0, OsMpWakeHandler, 0);//注册唤醒CPU的中断处理函数LOS_HwiCreate(LOS_MP_IPI_SCHEDULE, 0xa0, 0, OsMpScheduleHandler, 0);//注册调度CPU的中断处理函数LOS_HwiCreate(LOS_MP_IPI_HALT, 0xa0, 0, OsMpScheduleHandler, 0);//注册停止CPU的中断处理函数
#endif
}
//给每个CPU core初始化硬件中断
VOID HalIrqInitPercpu(VOID)
{/* unmask interrupts */	//取消中断屏蔽GIC_REG_32(GICC_PMR) = 0xFF;/* enable gic cpu interface */	//启用gic cpu接口GIC_REG_32(GICC_CTLR) = 1;
}

解读

  • 上来四个循环,是对中断控制器寄存器组的初始化,也就是驱动程序,驱动程序是配置硬件寄存器的过程.寄存器分通用和专用寄存器.下图为 gic_v2 的寄存器功能 ,这里对照代码和datasheet重点说下中断配置寄存器(GICD_ICFGRn)

  • 以下是GICD_ICFGRn的介绍

    The GICD_ICFGRs provide a 2-bit Int_config field for each interrupt supported by the GIC.
    This field identifies whether the corresponding interrupt is edge-triggered or level-sensitive

    GICD_ICFGRs为GIC支持的每个中断提供一个2位配置字段。此字段标识相应的中断是边缘触发的还是电平触发的

    0xC00 - 0xCFC GICD_ICFGRn RW IMPLEMENTATION DEFINED Interrupt Configuration Registers#define GICD_ICFGR(n)                   (GICD_OFFSET + 0xc00 + (n) * 4) /* Interrupt Configuration Registers */		//中断配置寄存器

如此一个32位寄存器可以记录16个中断的信息,这也是代码中出现 GIC_REG_32(GICD_ICFGR(i / 16))的原因.

  • GIC-v2支持三种类型的中断

    • PPI:私有外设中断(Private Peripheral Interrupt),是每个CPU私有的中断。最多支持16个PPI中断,硬件中断号从ID16~ID31。PPI通常会送达到指定的CPU上,应用场景有CPU本地时钟。
    • SPI:公用外设中断(Shared Peripheral Interrupt),最多可以支持988个外设中断,硬件中断号从ID32~ID1019。
    • SGI:软件触发中断(Software Generated Interrupt)通常用于多核间通讯,最多支持16个SGI中断,硬件中断号从ID0~ID15。SGI通常在内核中被用作 IPI 中断(inter-processor interrupts),并会送达到系统指定的CPU上,函数的最后就注册了三个核间中断的函数.
        typedef enum {//核间中断LOS_MP_IPI_WAKEUP,	//唤醒CPULOS_MP_IPI_SCHEDULE,//调度CPULOS_MP_IPI_HALT,	//停止CPU} MP_IPI_TYPE;

中断相关的结构体

size_t g_intCount[LOSCFG_KERNEL_CORE_NUM] = {0};//记录每个CPUcore的中断数量 
HwiHandleForm g_hwiForm[OS_HWI_MAX_NUM];//中断注册表 @note_why 用 form 来表示?有种写 HTML的感觉 :P
STATIC CHAR *g_hwiFormName[OS_HWI_MAX_NUM] = {0};//记录每个硬中断的名称 
STATIC UINT32 g_hwiFormCnt[OS_HWI_MAX_NUM] = {0};//记录每个硬中断的总数量
STATIC UINT32 g_curIrqNum = 0; //记录当前中断号
typedef VOID (*HWI_PROC_FUNC)(VOID); //中断函数指针
typedef struct tagHwiHandleForm {	HWI_PROC_FUNC pfnHook;	//中断处理函数HWI_ARG_T uwParam;		//中断处理函数参数struct tagHwiHandleForm *pstNext;	//节点,指向下一个中断,用于共享中断的情况
} HwiHandleForm;typedef struct tagIrqParam {	//中断参数int swIrq;		//	软件中断VOID *pDevId;	//	设备IDconst CHAR *pName;	//名称
} HwiIrqParam;

注册硬中断

/******************************************************************************创建一个硬中断中断创建,注册中断号、中断触发模式、中断优先级、中断处理程序。中断被触发时,handleIrq会调用该中断处理程序
******************************************************************************/
LITE_OS_SEC_TEXT_INIT UINT32 LOS_HwiCreate(HWI_HANDLE_T hwiNum,	//硬中断句柄编号 默认范围[0-127]HWI_PRIOR_T hwiPrio,		//硬中断优先级	HWI_MODE_T hwiMode,		//硬中断模式 共享和非共享HWI_PROC_FUNC hwiHandler,//硬中断处理函数HwiIrqParam *irqParam)	//硬中断处理函数参数
{UINT32 ret;(VOID)hwiPrio;if (hwiHandler == NULL) {//中断处理函数不能为NULLreturn OS_ERRNO_HWI_PROC_FUNC_NULL;}if ((hwiNum > OS_USER_HWI_MAX) || ((INT32)hwiNum < OS_USER_HWI_MIN)) {//中断数区间限制 [32,96]return OS_ERRNO_HWI_NUM_INVALID;}#ifdef LOSCFG_NO_SHARED_IRQ	//不支持共享中断ret = OsHwiCreateNoShared(hwiNum, hwiMode, hwiHandler, irqParam);
#elseret = OsHwiCreateShared(hwiNum, hwiMode, hwiHandler, irqParam);
#endifreturn ret;
}
//创建一个共享硬件中断,共享中断就是一个中断能触发多个响应函数
STATIC UINT32 OsHwiCreateShared(HWI_HANDLE_T hwiNum, HWI_MODE_T hwiMode,HWI_PROC_FUNC hwiHandler, const HwiIrqParam *irqParam)
{UINT32 intSave;HwiHandleForm *hwiFormNode = NULL;HwiHandleForm *hwiForm = NULL;HwiIrqParam *hwiParam = NULL;HWI_MODE_T modeResult = hwiMode & IRQF_SHARED;if (modeResult && ((irqParam == NULL) || (irqParam->pDevId == NULL))) {return OS_ERRNO_HWI_SHARED_ERROR;}HWI_LOCK(intSave);//中断自旋锁hwiForm = &g_hwiForm[hwiNum];//获取中断处理项if ((hwiForm->pstNext != NULL) && ((modeResult == 0) || (!(hwiForm->uwParam & IRQF_SHARED)))) {HWI_UNLOCK(intSave);return OS_ERRNO_HWI_SHARED_ERROR;}while (hwiForm->pstNext != NULL) {//pstNext指向 共享中断的各处理函数节点,此处一直撸到最后一个hwiForm = hwiForm->pstNext;//找下一个中断hwiParam = (HwiIrqParam *)(hwiForm->uwParam);//获取中断参数,用于检测该设备ID是否已经有中断处理函数if (hwiParam->pDevId == irqParam->pDevId) {//设备ID一致时,说明设备对应的中断处理函数已经存在了.HWI_UNLOCK(intSave);return OS_ERRNO_HWI_ALREADY_CREATED;}}hwiFormNode = (HwiHandleForm *)LOS_MemAlloc(m_aucSysMem0, sizeof(HwiHandleForm));//创建一个中断处理节点if (hwiFormNode == NULL) {HWI_UNLOCK(intSave);return OS_ERRNO_HWI_NO_MEMORY;}hwiFormNode->uwParam = OsHwiCpIrqParam(irqParam);//获取中断处理函数的参数if (hwiFormNode->uwParam == LOS_NOK) {HWI_UNLOCK(intSave);(VOID)LOS_MemFree(m_aucSysMem0, hwiFormNode);return OS_ERRNO_HWI_NO_MEMORY;}hwiFormNode->pfnHook = hwiHandler;//绑定中断处理函数hwiFormNode->pstNext = (struct tagHwiHandleForm *)NULL;//指定下一个中断为NULL,用于后续遍历找到最后一个中断项(见于以上 while (hwiForm->pstNext != NULL)处)hwiForm->pstNext = hwiFormNode;//共享中断if ((irqParam != NULL) && (irqParam->pName != NULL)) {g_hwiFormName[hwiNum] = (CHAR *)irqParam->pName;}g_hwiForm[hwiNum].uwParam = modeResult;HWI_UNLOCK(intSave);return LOS_OK;
}

解读

  • 内核将硬中断进行编号,如:
      #define NUM_HAL_INTERRUPT_TIMER0        33#define NUM_HAL_INTERRUPT_TIMER1        33#define NUM_HAL_INTERRUPT_TIMER2        34#define NUM_HAL_INTERRUPT_TIMER3        34#define NUM_HAL_INTERRUPT_TIMER4        35#define NUM_HAL_INTERRUPT_TIMER5        35#define NUM_HAL_INTERRUPT_TIMER6        36#define NUM_HAL_INTERRUPT_TIMER7        36#define NUM_HAL_INTERRUPT_DMAC          60#define NUM_HAL_INTERRUPT_UART0         38#define NUM_HAL_INTERRUPT_UART1         39#define NUM_HAL_INTERRUPT_UART2         40#define NUM_HAL_INTERRUPT_UART3         41#define NUM_HAL_INTERRUPT_UART4         42#define NUM_HAL_INTERRUPT_TIMER         NUM_HAL_INTERRUPT_TIMER4

例如:时钟节拍处理函数 OsTickHandler 就是在 HalClockInit中注册的

    //硬时钟初始化VOID HalClockInit(VOID){// ...(void)LOS_HwiCreate(NUM_HAL_INTERRUPT_TIMER, 0xa0, 0, OsTickHandler, 0);//注册OsTickHandler到中断向量表 }//节拍中断处理函数 ,鸿蒙默认10ms触发一次LITE_OS_SEC_TEXT VOID OsTickHandler(VOID){UINT32 intSave;TICK_LOCK(intSave);//tick自旋锁g_tickCount[ArchCurrCpuid()]++;// 累加当前CPU核tick数TICK_UNLOCK(intSave);OsTimesliceCheck();//时间片检查OsTaskScan(); /* task timeout scan *///扫描超时任务 例如:delay(300)#if (LOSCFG_BASE_CORE_SWTMR == YES)OsSwtmrScan();//扫描定时器,查看是否有超时定时器,加入队列#endif} 
  • 鸿蒙是支持中断共享的,在OsHwiCreateShared中,将函数注册到g_hwiForm中.中断向量完成注册后,就是如何触发和回调的问题.触发在中断切换篇中已经讲清楚,触发是从底层汇编向上调用,调用的C函数就是HalIrqHandler

中断怎么触发的?

分两种情况:

  • 通过硬件触发,比如按键,USB的插拔这些中断源向中断控制器发送电信号(高低电平触发或是上升/下降沿触发),中断控制器经过过滤后将信号发给对应的CPU处理,通过硬件改变PC和CPSR寄存值,直接跳转到中断向量(固定地址)执行.
      b   reset_vector            @开机代码b   _osExceptUndefInstrHdl 	@异常处理之CPU碰到不认识的指令b   _osExceptSwiHdl			@异常处理之:软中断b   _osExceptPrefetchAbortHdl	@异常处理之:取指异常b   _osExceptDataAbortHdl		@异常处理之:数据异常b   _osExceptAddrAbortHdl		@异常处理之:地址异常b   OsIrqHandler				@异常处理之:硬中断b   _osExceptFiqHdl				@异常处理之:快中断
  • 通过软件触发,常见于核间中断的情况, 核间中断指的是几个CPU之间相互通讯的过程.以下为某一个CPU向其他CPU(可以是多个)发起让这些CPU重新调度LOS_MpSchedule的中断请求信号.最终是写了中断控制器的GICD_SGIR寄存器,这是一个由软件触发中断的寄存器.中断控制器会将请求分发给对应的CPU处理中断,即触发了OsIrqHandler.
    //给参数CPU发送调度信号VOID LOS_MpSchedule(UINT32 target)//target每位对应CPU core {UINT32 cpuid = ArchCurrCpuid();target &= ~(1U << cpuid);//获取除了自身之外的其他CPUHalIrqSendIpi(target, LOS_MP_IPI_SCHEDULE);//向目标CPU发送调度信号,核间中断(Inter-Processor Interrupts),IPI}//SGI软件触发中断(Software Generated Interrupt)通常用于多核间通讯STATIC VOID GicWriteSgi(UINT32 vector, UINT32 cpuMask, UINT32 filter){UINT32 val = ((filter & 0x3) << 24) | ((cpuMask & 0xFF) << 16) |(vector & 0xF);GIC_REG_32(GICD_SGIR) = val;//写SGI寄存器}//向指定核发送核间中断VOID HalIrqSendIpi(UINT32 target, UINT32 ipi){GicWriteSgi(ipi, target, 0);}

中断统一处理入口函数 HalIrqHandler

//硬中断统一处理函数,这里由硬件触发,调用见于 ..\arch\arm\arm\src\los_dispatch.S
VOID HalIrqHandler(VOID)
{UINT32 iar = GIC_REG_32(GICC_IAR);//从中断确认寄存器获取中断ID号UINT32 vector = iar & 0x3FFU;//计算中断向量号/** invalid irq number, mainly the spurious interrupts 0x3ff,* gicv2 valid irq ranges from 0~1019, we use OS_HWI_MAX_NUM* to do the checking.*/if (vector >= OS_HWI_MAX_NUM) {return;}g_curIrqNum = vector;//记录当前中断ID号OsInterrupt(vector);//调用上层中断处理函数/* use orignal iar to do the EOI */GIC_REG_32(GICC_EOIR) = iar;//更新中断结束寄存器
}
VOID OsInterrupt(UINT32 intNum)//中断实际处理函数
{HwiHandleForm *hwiForm = NULL;UINT32 *intCnt = NULL;intCnt = &g_intCount[ArchCurrCpuid()];//当前CPU的中断总数量 ++*intCnt = *intCnt + 1;//@note_why 这里没看明白为什么要 +1#ifdef LOSCFG_CPUP_INCLUDE_IRQ //开启查询系统CPU的占用率的中断OsCpupIrqStart();//记录本次中断处理开始时间
#endif#ifdef LOSCFG_KERNEL_TICKLESSOsTicklessUpdate(intNum);
#endifhwiForm = (&g_hwiForm[intNum]);//获取对应中断的实体
#ifndef LOSCFG_NO_SHARED_IRQ	//如果没有定义不共享中断 ,意思就是如果是共享中断while (hwiForm->pstNext != NULL) { //一直撸到最后hwiForm = hwiForm->pstNext;//下一个继续撸
#endifif (hwiForm->uwParam) {//有参数的情况HWI_PROC_FUNC2 func = (HWI_PROC_FUNC2)hwiForm->pfnHook;//获取回调函数if (func != NULL) {UINTPTR *param = (UINTPTR *)(hwiForm->uwParam);func((INT32)(*param), (VOID *)(*(param + 1)));//运行带参数的回调函数}} else {//木有参数的情况HWI_PROC_FUNC0 func = (HWI_PROC_FUNC0)hwiForm->pfnHook;//获取回调函数if (func != NULL) {func();//运行回调函数}}
#ifndef LOSCFG_NO_SHARED_IRQ}
#endif++g_hwiFormCnt[intNum];//中断号计数器总数累加*intCnt = *intCnt - 1;	//@note_why 这里没看明白为什么要 -1 
#ifdef LOSCFG_CPUP_INCLUDE_IRQ	//开启查询系统CPU的占用率的中断OsCpupIrqEnd(intNum);//记录中断处理时间完成时间
#endif
}

解读
统一中断处理函数是一个通过一个中断号去找到注册函数的过程,分四步走:

  • 第一步:取号,这号是由中断控制器的 GICC_IAR寄存器提供的,这是一个专门保存当前中断号的寄存器.
  • 第二步:从注册表g_hwiForm中查询注册函数,同时取出参数.
  • 第三步:执行函数,也就是回调注册函数,分有参和无参两种情况 func(...),在中断共享的情况,注册函数会指向 next 注册函数pstNext,依次执行回调函数,这是中断共享的实现细节.
    typedef struct tagHwiHandleForm {	HWI_PROC_FUNC pfnHook;	//中断处理函数HWI_ARG_T uwParam;		//中断处理函数参数struct tagHwiHandleForm *pstNext;	//节点,指向next中断,用于共享中断的情况} HwiHandleForm;
  • 第四步:销号,本次中断完成了就需要消除记录,中断控制器也有专门的销号寄存器GICC_EOIR
  • 另外的是一些统一数据,每次中断号处理内核都会记录次数,和耗时,以便定位/跟踪/诊断问题.

鸿蒙全栈开发全新学习指南

也为了积极培养鸿蒙生态人才,让大家都能学习到鸿蒙开发最新的技术,针对一些在职人员、0基础小白、应届生/计算机专业、鸿蒙爱好者等人群,整理了一套纯血版鸿蒙(HarmonyOS Next)全栈开发技术的学习路线【包含了大厂APP实战项目开发】

本路线共分为四个阶段:

第一阶段:鸿蒙初中级开发必备技能

第二阶段:鸿蒙南北双向高工技能基础:gitee.com/MNxiaona/733GH

第三阶段:应用开发中高级就业技术

第四阶段:全网首发-工业级南向设备开发就业技术:https://gitee.com/MNxiaona/733GH

《鸿蒙 (Harmony OS)开发学习手册》(共计892页)

如何快速入门?

1.基本概念
2.构建第一个ArkTS应用
3.……

开发基础知识:gitee.com/MNxiaona/733GH

1.应用基础知识
2.配置文件
3.应用数据管理
4.应用安全管理
5.应用隐私保护
6.三方应用调用管控机制
7.资源分类与访问
8.学习ArkTS语言
9.……

基于ArkTS 开发

1.Ability开发
2.UI开发
3.公共事件与通知
4.窗口管理
5.媒体
6.安全
7.网络与链接
8.电话服务
9.数据管理
10.后台任务(Background Task)管理
11.设备管理
12.设备使用信息统计
13.DFX
14.国际化开发
15.折叠屏系列
16.……

鸿蒙开发面试真题(含参考答案):gitee.com/MNxiaona/733GH

鸿蒙入门教学视频:

美团APP实战开发教学:gitee.com/MNxiaona/733GH

写在最后

  • 如果你觉得这篇内容对你还蛮有帮助,我想邀请你帮我三个小忙:
  • 点赞,转发,有你们的 『点赞和评论』,才是我创造的动力。
  • 关注小编,同时可以期待后续文章ing🚀,不定期分享原创知识。
  • 想要获取更多完整鸿蒙最新学习资源,请移步前往小编:gitee.com/MNxiaona/733GH

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

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

相关文章

一堆自定义C#代码片段,让你开发效率飞涨

SharpBoxes 是一款用于 Visual Studio 的扩展&#xff0c;作者为本人&#xff1b; 该扩展旨在提高开发效率。它为开发人员提供了一组日常使用频率较高的代码片段&#xff0c;让你在编写代码时能够更快地插入常用的代码段。通过安装这个扩展&#xff0c;你可以使用快捷键轻松插…

HarmonyOS开发案例:【电子相册】

介绍 如何实现一个简单的电子相册应用的开发&#xff0c;主要功能包括&#xff1a; 实现首页顶部的轮播效果。 实现页面跳转时共享元素的转场动画效果。 实现通过手势控制图片的放大、缩小、左右滑动查看细节等效果。 相关概念 [Swiper]&#xff1a;滑块视图容器&#x…

W801学习笔记二十二:英语背单词学习应用——下

续上篇&#xff1a; W801学习笔记二十一&#xff1a;英语背单词学习应用——上 五、处理用户交互 由于英语也是采用了和唐诗一样的《三分钟限时挑战》《五十题竞速挑战》《零错误闯关挑战》&#xff0c;所以用户交互的逻辑和唐诗是一样的。所以&#xff0c;我们抽一个基类&a…

Leetcode—138. 随机链表的复制【中等】

2024每日刷题&#xff08;129&#xff09; Leetcode—138. 随机链表的复制 实现代码 /* // Definition for a Node. class Node { public:int val;Node* next;Node* random;Node(int _val) {val _val;next NULL;random NULL;} }; */class Solution { public:Node* copyRan…

海洋行业工业气体检测传感器的重要性

海洋行业是一个广阔而复杂的领域&#xff0c;涉及多个分支和应用&#xff0c;包括浮式生产、储存和卸载&#xff08;FPSO&#xff09;装置、渡轮和潜艇等。这些船舶和设施在执行任务时&#xff0c;都可能遇到各种潜在的气体危害。因此&#xff0c;对于海洋行业来说&#xff0c;…

STM32接入CH340芯片的初始化进入升级模式(死机)问题处理

目录 1. 问题描述2. 问题分析2.1 CH340G/K 的初始化波形2.2 第1种USB升级电路2.3 第2种USB升级电路2.4 第3种USB升级电路2.5 第4种USB升级电路 3. 总结 1. 问题描述 我所用的CH340G&#xff08;CH340K也用过&#xff09;接在MCU的电路中&#xff0c;在插入CH340G/K 的接插件&a…

Unity EventSystem入门

概述 相信在学习Unity中&#xff0c;一定有被UI事件困扰的时候把&#xff0c;当添加UICanvas的时候&#xff0c;Unity会为我们自动添加EventSystem&#xff0c;这个是为什么呢&#xff0c;Unity的UI事件是如何处理的呢&#xff0c;在使用各个UI组件的时候&#xff0c;一定有不…

Redis(Redis配置和订阅发布)

文章目录 1.Redis配置1.网络配置1.配置文件位置 /etc/redis.conf2.bind&#xff08;注销支持远程访问&#xff09;1.默认情况bind 127.0.0.1 只能接受本机的访问2.首先编辑配置文件3.进入命令模式输入/bind定位&#xff0c;输入n查找下一个&#xff0c;shift n查找上一个&…

OpenHarmony 实战开发—— refreshlayout 组件开发学习指南~

1. RefreshLayout_harmonyos 功能介绍 1.1. 组件介绍&#xff1a; RefreshLayout_harmonyos 是一款下拉刷新组件 1.2. 手机模拟器上运行效果&#xff1a; 2. RefreshLayout_harmonyos 使用方法 2.1 在目录 build.gradle 下 implementation project(":refreshlayout_ha…

新能源汽车动力电池热管理-液冷方案应用原理与应用前景简介

前言 动力电池是新能源汽车的核心部件之一&#xff0c;其性能和寿命直接影响着车辆的续航里程和使用成本。液冷方案作为一种常见的动力电池温控解决方案&#xff0c;被广泛应用于新能源汽车领域。本文将详细介绍液冷方案的原理、发展方向以及市场前景。 一、液冷方案的原理 …

Jmeter 中 CSV 如何参数化测试数据并实现自动断言

当我们使用Jmeter工具进行接口测试&#xff0c;可利用CSV Data Set Config配置元件&#xff0c;对测试数据进行参数化&#xff0c;循环读取csv文档中每一行测试用例数据&#xff0c;来实现接口自动化。此种情况下&#xff0c;很多测试工程师只会人工地查看响应结果来判断用例是…

Springboot+Vue项目-基于Java+MySQL的影院订票系统(附源码+演示视频+LW)

大家好&#xff01;我是程序猿老A&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f49e;当前专栏&#xff1a;Java毕业设计 精彩专栏推荐&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; &#x1f380; Python毕业设计 &…

生成gitee公钥

1、打开设置 2、设置SSH公钥 3、生成公钥 4、复制终端输出的公钥&#xff0c;放到这里&#xff0c;标题随便取。 5、测试 ssh -T gitgitee.com 最后用这个测试

帆软报表实现填报报表

我们拿emp表举例 登记信息表 设计一个报表实现对emp表员工的登记 &#xff08;emp表为ORACLE自带用户scott下的一个表&#xff09; 首先&#xff0c;我们设计好填报界面&#xff0c;新建一个普通报表&#xff0c;将emp表中需要的输入一一回应填写进表中。 如下图所示&#xf…

区块链扩容:水平扩展 vs.垂直扩展

1. 引言 随着Rollups 的兴起&#xff0c;区块链扩容一直集中在模块化&#xff08;modular&#xff09;vs. 整体式&#xff08;monolithic&#xff09;之争。 如今&#xff0c;模块化与整体式这种一分为二的心理模型&#xff0c;已不适合于当前的扩容场景。本文&#xff0c;将展…

ASP.NET MVC企业级程序设计 (入住退房,删除)

目录 效果图 实现过程 控制器代码 DAL BLL Index 效果图 实现过程 控制器代码 using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Mvc;namespace MvcApplication1.Controllers {public class HomeController …

【linux-IMX6ULL中断配置流程】

目录 1. Cortex-A7和GIC中断概述1. 1 Cortex-A7中断系统&#xff1a;1. 2 GIC中断控制器简介&#xff1a; 2. 中断配置概述3. 底层中断文件配置3.1 对启动文件.s的配置思路3.2 对中断函数配置思路 4. 上层中断配置流程 1. Cortex-A7和GIC中断概述 学习IMX6UL的中断处理系统&…

Git同步代码

Git中5个区&#xff0c;和具体操作&#xff1f; 代码提交和同步代码 代码撤销和撤销同步 平时是怎么提交代码的&#xff1f; 第零步: 工作区与仓库保持一致第一步: 文件增删改&#xff0c;变为已修改状态第二步: git add &#xff0c;变为已暂存状态 $ git status $ git a…

1天搞定SpringBoot+Vue全栈开发 (8)前端路由VueRouter(进行组件切换)

1.VueRouter安装与使用 2.参数传递 创建路由组件 在项目中定义Discover.vue、Friends.vue、My.vue三个组件&#xff0c;将来要使用vue-router来控制它们的展示与切换&#xff1a; Discover.vue <template><div><h1>发现音乐</h1></div> <…

智能实训-wheeltec小车-抓取(源代码)

语言 :C 源代码&#xff1a; #include <ros/ros.h> #include <image_transport/image_transport.h> #include <cv_bridge/cv_bridge.h> #include <sensor_msgs/image_encodings.h> #include <sensor_msgs/JointState.h> #include <geometry…