WinCE--中断服务线程IST(详解)

2024-02-05 10:30

本文主要是介绍WinCE--中断服务线程IST(详解),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

WinCE--中断服务线程IST(详解)

 
转载
转自:http://blog.sina.com.cn/s/blog_62714d6a0100m7tl.html

中断服务线程负责处理中断大部分的工作。Wince5.0下IST为用户态线程,而在wince6.0下,IST的加载空间会和加载驱动程序的载体相同,也就是说载体为内核态IST就为内核态。

   对于IST的大体描述是:

      IST的大部分时间都是在等待某一个事件,这个事件与相对应的逻辑中断号相关联。当有中断发生时,操作系统就会引发与该逻辑中断号相关联的事件,那么IST将会得到调度执行。

编写处理中断驱动程序的两个步骤:

一、进行中断初始化。

二、 编写中断服务线程。

 

下面我们来具体介绍这两步的细节:

 

一、进行中断初始化。

1、  创建事件

2、  获取IRQ的系统中断号

3、  创建挂起的中断服务线程IST

4、  调用InterruptInitialize以创建IRQ与事件之间的关联。(创建未挂起的中断服务线程有可能导致InterruptInitialize函数调用失败,因为该事件已经处于等待状态)

5、  调用CeSetThreadPriority函数设置IST的优先级

6、  启动IST线程

void InitialInterrupt( void )

{

// Create an event 创建事件

    g_hevInterrupt = CreateEvent(NULL, FALSE, FALSE, NULL);

    if (g_hevInterrupt == NULL)

    {

        RETAILMSG(1, (TEXT("DEMO: Event creation failed!!!\r\n")));

        return;

    }

 

// Have the OAL Translate the IRQ to a system irq 获取IRQ的系统中断号

    fRetVal = KernelIoControl( IOCTL_HAL_TRANSLATE_IRQ, &dwIrq, sizeof( dwIrq ),

                               &g_dwSysInt, sizeof( g_dwSysInt ), NULL );

 

// Create a thread that waits for signaling 创建挂起的中断服务线程

    g_fRun  = TRUE;

    g_htIST   = CreateThread(NULL,  // Security

                             0,     // No Stack Size

                             ThreadIST,//Interrupt Thread                    

                             NULL,     // No  Parameters    

                             CREATE_SUSPENDED, // Create Suspended

                             &dwThreadID  // Thread Id  );

 

// Set the thread priority – arbitrarily 5   设置线程优先级

    m_nISTPriority = 5;

    if( !CeSetThreadPriority( g_htIST, m_nISTPriority ))

    {

        RETAILMSG(1,(TEXT("DEMO: Failed setting Thread Priority.\r\n")));

        return;

    }

 

// Initialize the interrupt 创建IRQ与事件之间的关联

    if ( !InterruptInitialize(g_dwSysInt,g_hevInterrupt,NULL,0) )

    {

        RETAILMSG (1, (TEXT("DEMO: InterruptInitialize failed!!!\r\n")));

        return;

    }

 

// Get the thread started 启动IST线程

    ResumeThread( g_htIST );

}

 

需要注意的是,对 InterruptInitialize 的调用仅采用 SYSINTR 值和事件作为

参数。内核不知道或者不关心将要等待该事件的线程。这样,就可以建立多种应用程序和驱动程序体系结构。应用程序的简单主循环可以初始化中断,然后立即

等待该事件。中断只能与一个事件关联,并且该事件不能用于对 

WaitForMultipleObjects 的调用中。我们将观察一个简单的为中断提供服务的

线程。这是大多数实现中的标准解决方案。 

 

二、编写中断服务线程

1、  等待中断事件

2、  确认得到来自操作系统的事件通知

3、  执行必要的中断处理

4、  在尽可能短的时间内完成中断的处理

5、  创建CELOGDATA,供内核调试工具Kernel Tracker查看

6、  调用InterruptDone通知内核中断完成(系统收到信号后会将该优先级中断恢复)

7、  再次等待中断事件

DWORD  WINAPI  ThreadIST( LPVOID lpvParam )

{

  DWORD  dwStatus;

  BOOL  fState = TRUE;

  // Always chec the running flag   检查运行标志

  while( g_fRun )

{

       dwStatus  = WaitForSingleObject(g_hevInterrupt, INFINITE);

 // Check to see if we are finished 确认是否需要关闭IST

        if(!g_fRun )   

          return 0;

 // Make sure we have the object  确认是否成功获得事件

      if( dwStatus == WAIT_OBJECT_0 )

     {

// Do all interrupt processing to complete the interaction做关于中断的相应处理,用户实现

// with the board so we can receive another interrupt.

          if (!( READ_REGISTER_ULONG(g_pBoard Register) & INTR_MASK))

         {

          RETAILMSG(1, (TEXT("DEMO: Interrupt...")));

          g_dwInterruptCount ++;

          }

          // Finish the interrupt   中断完成

          InterruptDone( g_dwSysInt );

      }

  }

 return 0;

}

未理解下面一段话的真正含义!!!????????????????

该示例读取一个 ULONG 寄存器以确定中断状态。您只需用您的代码替换该代码

段。非常关键的一点是,要使 IST 处理尽可能地简单。如果将来需要处理来自

该设备的数据:  

* 在 IST 中尽可能快速地从该设备获取数据。  

* 创建一个事件,以通知某个优先级较低的线程完成该工作。   

* 通过 InterruptDone 从该 IST 中立即返回。  

* 让优先级较低的线程进一步处理数据。  

* 在 IST 与优先级较低的线程之间放置 FIFO 以处理溢出。 

这篇关于WinCE--中断服务线程IST(详解)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Linux系统性能检测命令详解

《Linux系统性能检测命令详解》本文介绍了Linux系统常用的监控命令(如top、vmstat、iostat、htop等)及其参数功能,涵盖进程状态、内存使用、磁盘I/O、系统负载等多维度资源监控,... 目录toppsuptimevmstatIOStatiotopslabtophtopdstatnmon

java使用protobuf-maven-plugin的插件编译proto文件详解

《java使用protobuf-maven-plugin的插件编译proto文件详解》:本文主要介绍java使用protobuf-maven-plugin的插件编译proto文件,具有很好的参考价... 目录protobuf文件作为数据传输和存储的协议主要介绍在Java使用maven编译proto文件的插件

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

Java中的数组与集合基本用法详解

《Java中的数组与集合基本用法详解》本文介绍了Java数组和集合框架的基础知识,数组部分涵盖了一维、二维及多维数组的声明、初始化、访问与遍历方法,以及Arrays类的常用操作,对Java数组与集合相... 目录一、Java数组基础1.1 数组结构概述1.2 一维数组1.2.1 声明与初始化1.2.2 访问

Javaee多线程之进程和线程之间的区别和联系(最新整理)

《Javaee多线程之进程和线程之间的区别和联系(最新整理)》进程是资源分配单位,线程是调度执行单位,共享资源更高效,创建线程五种方式:继承Thread、Runnable接口、匿名类、lambda,r... 目录进程和线程进程线程进程和线程的区别创建线程的五种写法继承Thread,重写run实现Runnab

SpringBoot线程池配置使用示例详解

《SpringBoot线程池配置使用示例详解》SpringBoot集成@Async注解,支持线程池参数配置(核心数、队列容量、拒绝策略等)及生命周期管理,结合监控与任务装饰器,提升异步处理效率与系统... 目录一、核心特性二、添加依赖三、参数详解四、配置线程池五、应用实践代码说明拒绝策略(Rejected

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注

C#读写文本文件的多种方式详解

《C#读写文本文件的多种方式详解》这篇文章主要为大家详细介绍了C#中各种常用的文件读写方式,包括文本文件,二进制文件、CSV文件、JSON文件等,有需要的小伙伴可以参考一下... 目录一、文本文件读写1. 使用 File 类的静态方法2. 使用 StreamReader 和 StreamWriter二、二进

Conda与Python venv虚拟环境的区别与使用方法详解

《Conda与Pythonvenv虚拟环境的区别与使用方法详解》随着Python社区的成长,虚拟环境的概念和技术也在不断发展,:本文主要介绍Conda与Pythonvenv虚拟环境的区别与使用... 目录前言一、Conda 与 python venv 的核心区别1. Conda 的特点2. Python v

Spring Boot中WebSocket常用使用方法详解

《SpringBoot中WebSocket常用使用方法详解》本文从WebSocket的基础概念出发,详细介绍了SpringBoot集成WebSocket的步骤,并重点讲解了常用的使用方法,包括简单消... 目录一、WebSocket基础概念1.1 什么是WebSocket1.2 WebSocket与HTTP