vxWorks中系统任务tIsr的实现及使用

2024-05-13 04:38

本文主要是介绍vxWorks中系统任务tIsr的实现及使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

tIsr作为一个系统任务,看其注释,知道其有两种使用方式:

This module operates in two distinct modes depending on the module's
configuration.  If the mode is configured for "per-CPU" deferral
tasks, this module creates (as needed) a single deferral task on each
CPU in the system.  This single queue handles all deferral operations
performed by device drivers that are servicing their interrupts on
that CPU.

If the mode is configured for "per-ISR" deferral tasks, this module
creates a unique deferral task for each requester, and sets the CPU
affinity for the created task to the requested CPU index.

这个任务具体做什么任务呢?
举一个例子:
ns16550vxInt串口中断处理函数中使用到了。
当这个串口处理函数去读取这个串口中的值时,如果出现了超时没有读取完成的情况,就会把这个中断函数延后执行。

LOCAL void ns16550vxbInt(VXB_DEVICE_ID pDev){FAST NS16550VXB_CHAN * pChan = (NS16550VXB_CHAN *)(pDev->pDrvCtrl);    /* pointer to channel */FAST volatile char        intStatus;UINT8   iirValue, lsrValue;while ( pChan != NULL ){VXB_NS16550_ISR_SET(pChan);switch (intStatus){case IIR_RLS:REG_GET(LSR, pChan, lsrValue);/*read LSR to reset interrupt */intStatus = (char)lsrValue;break;case IIR_RDA:           /* received data available */case IIR_TIMEOUT:{pChan->ier &= ~(RxFIFO_BIT); /* indicate to disable Rx Int */REG_SET(IER, pChan, pChan->ier);VXB_NS16550_ISR_CLEAR(pChan);/*把这个任务延后,继续处理下一个,剩下的任务交给tIsr系统任务执行*/isrDeferJobAdd (pChan->queueId, &pChan->isrDefRd);goto nextChan;}........
}

这相当于就是把一个中断上下文转换为任务上下文来继续执行。
初始化过程

usrRoot--->usrIosCoreInit--->usrIsrDeferInit--->isrDeferLibInit

STATUS isrDeferLibInit(int		mode			/* global deferral queue mode */){
/*记录要使用的模式*/isrDeferLibMode = mode;
/*如果是ISR_DEFER_MODE_PER_CPU模式,就为每一个cpu创建一个队列,来处理这个cpu上所有的延后函数*/if (isrDeferLibMode == ISR_DEFER_MODE_PER_CPU){pCpuQueueId = (ISR_DEFER_QUEUE_ID *) calloc (vxCpuConfiguredGet (),sizeof(*pCpuQueueId));if (pCpuQueueId == NULL)return ERROR;}return OK;}
/*创建处理延后队列函数*/
LOCAL ISR_DEFER_QUEUE * isrDeferQueueCreate(VXB_DEVICE_ID	pInst,	        /* VxBus device id of requester */int			intIdx,	        /* interrupt source index */int			logicalCpuIndex	/* logical CPU index for deferral task */){char taskName[32];char *pT;cpuset_t affinity;ISR_DEFER_QUEUE *pQueue = malloc(sizeof(*pQueue));lstInit (&pQueue->list);semBInit (&pQueue->syncSem, SEM_Q_PRIORITY, SEM_EMPTY);SPIN_LOCK_ISR_INIT (&pQueue->lock, 0);/*根据不同的cpu号,设置一个独一无二的名字,cpu0---tIsr0,cpu1---tIsr1*/strcpy (taskName, "tIsr");pT = taskName + 4;if (pInst == NULL)  /* use CPU index only */{*pT++ = '0' + logicalCpuIndex % 10;*pT   = '\0';}else  /* use device unit number and interrupt index */{*pT++ = '0' + pInst->unitNumber % 10;*pT++ = '0' + intIdx % 10;strcpy (pT, pInst->pName);}/*创建对应处理的系统任务*/pQueue->tid = taskSpawn (taskName,ISR_DEFER_TASK_PRIO,ISR_DEFER_TASK_OPTS,8192, (FUNCPTR) isrDeferTask,(int) pQueue, 0, 0, 0, 0, 0, 0, 0, 0, 0);.....}
/*具体的处理函数*/
LOCAL void isrDeferTask (ISR_DEFER_QUEUE_ID queueId	/* queue for incoming deferred work */){ISR_DEFER_JOB *pJob;
/*死循环,一直等待任务的到来*/FOREVER{semBTake (&queueId->syncSem, WAIT_FOREVER);
/*获取任务,并执行*/FOREVER{SPIN_LOCK_ISR_TAKE (&queueId->lock);pJob = (ISR_DEFER_JOB *) lstGet  (&queueId->list);SPIN_LOCK_ISR_GIVE (&queueId->lock);pJob->func (pJob->pData);}}}


/*把一个任务添加到延迟队列上来*/
void isrDeferJobAdd(ISR_DEFER_QUEUE_ID queueId,	/* queue allocated by isrDeferQueueGet */ISR_DEFER_JOB *    pJob	/* job to enqueue */){SPIN_LOCK_ISR_TAKE (&queueId->lock);lstAdd (&queueId->list, &pJob->node);SPIN_LOCK_ISR_GIVE (&queueId->lock);/*添加完成后,通知isrDeferTask有任务到来了*/semGive (&queueId->syncSem);}/*从队列获取一个任务*/
ISR_DEFER_QUEUE_ID isrDeferQueueGet(VXB_DEVICE_ID  pInst,	           /* VxBus device ID of requester */int		   intIdx,	            /* interrupt source index */int		   logicalCpuIndex, /* logical CPU index for deferral task */int		   mode	            /* deferral queue mode(for future use) */){
/*如果是ISR_DEFER_MODE_PER_CPU模式,就返回对应cpu的那个队列*/if (isrDeferLibMode == ISR_DEFER_MODE_PER_CPU){if (pCpuQueueId [logicalCpuIndex] == NULL)pCpuQueueId [logicalCpuIndex] = isrDeferQueueCreate (0, 0, logicalCpuIndex);return pCpuQueueId [logicalCpuIndex];}
/*如果是另一种模式,就创建这个设备对应的队列*/return isrDeferQueueCreate (pInst, intIdx, logicalCpuIndex);}



这篇关于vxWorks中系统任务tIsr的实现及使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/984706

相关文章

postgresql使用UUID函数的方法

《postgresql使用UUID函数的方法》本文给大家介绍postgresql使用UUID函数的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录PostgreSQL有两种生成uuid的方法。可以先通过sql查看是否已安装扩展函数,和可以安装的扩展函数

Python实现终端清屏的几种方式详解

《Python实现终端清屏的几种方式详解》在使用Python进行终端交互式编程时,我们经常需要清空当前终端屏幕的内容,本文为大家整理了几种常见的实现方法,有需要的小伙伴可以参考下... 目录方法一:使用 `os` 模块调用系统命令方法二:使用 `subprocess` 模块执行命令方法三:打印多个换行符模拟

SpringBoot+EasyPOI轻松实现Excel和Word导出PDF

《SpringBoot+EasyPOI轻松实现Excel和Word导出PDF》在企业级开发中,将Excel和Word文档导出为PDF是常见需求,本文将结合​​EasyPOI和​​Aspose系列工具实... 目录一、环境准备与依赖配置1.1 方案选型1.2 依赖配置(商业库方案)二、Excel 导出 PDF

Python实现MQTT通信的示例代码

《Python实现MQTT通信的示例代码》本文主要介绍了Python实现MQTT通信的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 安装paho-mqtt库‌2. 搭建MQTT代理服务器(Broker)‌‌3. pytho

如何使用Lombok进行spring 注入

《如何使用Lombok进行spring注入》本文介绍如何用Lombok简化Spring注入,推荐优先使用setter注入,通过注解自动生成getter/setter及构造器,减少冗余代码,提升开发效... Lombok为了开发环境简化代码,好处不用多说。spring 注入方式为2种,构造器注入和setter

MySQL中比较运算符的具体使用

《MySQL中比较运算符的具体使用》本文介绍了SQL中常用的符号类型和非符号类型运算符,符号类型运算符包括等于(=)、安全等于(=)、不等于(/!=)、大小比较(,=,,=)等,感兴趣的可以了解一下... 目录符号类型运算符1. 等于运算符=2. 安全等于运算符<=>3. 不等于运算符<>或!=4. 小于运

使用zip4j实现Java中的ZIP文件加密压缩的操作方法

《使用zip4j实现Java中的ZIP文件加密压缩的操作方法》本文介绍如何通过Maven集成zip4j1.3.2库创建带密码保护的ZIP文件,涵盖依赖配置、代码示例及加密原理,确保数据安全性,感兴趣的... 目录1. zip4j库介绍和版本1.1 zip4j库概述1.2 zip4j的版本演变1.3 zip4

Python 字典 (Dictionary)使用详解

《Python字典(Dictionary)使用详解》字典是python中最重要,最常用的数据结构之一,它提供了高效的键值对存储和查找能力,:本文主要介绍Python字典(Dictionary)... 目录字典1.基本特性2.创建字典3.访问元素4.修改字典5.删除元素6.字典遍历7.字典的高级特性默认字典

使用Python构建一个高效的日志处理系统

《使用Python构建一个高效的日志处理系统》这篇文章主要为大家详细讲解了如何使用Python开发一个专业的日志分析工具,能够自动化处理、分析和可视化各类日志文件,大幅提升运维效率,需要的可以了解下... 目录环境准备工具功能概述完整代码实现代码深度解析1. 类设计与初始化2. 日志解析核心逻辑3. 文件处

python生成随机唯一id的几种实现方法

《python生成随机唯一id的几种实现方法》在Python中生成随机唯一ID有多种方法,根据不同的需求场景可以选择最适合的方案,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习... 目录方法 1:使用 UUID 模块(推荐)方法 2:使用 Secrets 模块(安全敏感场景)方法