ARM Linux 如何--注册和触发--软中断

简介: 1. 注册软中断当然是通过open_softirq 例子如下: [cpp] view plain copy   void __init init_timers(void)   {       int err = timer_cpu_notify(&timers_nb,...

1. 注册软中断当然是通过open_softirq

例子如下:

[cpp]  view plain  copy
 
  1. void __init init_timers(void)  
  2. {  
  3.     int err = timer_cpu_notify(&timers_nb, (unsigned long)CPU_UP_PREPARE,  
  4.                 (void *)(long)smp_processor_id());  
  5.   
  6.     init_timer_stats();  
  7.   
  8.     BUG_ON(err == NOTIFY_BAD);  
  9.     register_cpu_notifier(&timers_nb);  
  10.     open_softirq(TIMER_SOFTIRQ, run_timer_softirq);  
  11. }  
  12.   
  13. void open_softirq(int nr, void (*action)(struct softirq_action *))  
  14. {  
  15.     softirq_vec[nr].action = action;  
  16. }  

软中断TIMER_SOFTIRQ的中断处理函数为:run_timer_softirq

之所以成为softirq,是因为这些中断是由硬件中断来间接触发的,如何间接触发的呢:
硬件中断处理函数-->对软中断的相应位置位-->唤醒ksoftirqd线程-->执行软中断的中断处理函数

 

2. 硬件中断如何通过置位唤醒ksoftirqd线程

timer interrupt handler->
timer_tick->
update_process_times->
run_local_timers->
hrtimer_run_queues()和raise_softirq(TIMER_SOFTIRQ)->
raise_softirq_irqoff->
__raise_softirq_irqoff { or_softirq_pending(1UL << (nr)); }
即(local_softirq_pending() |= (x))

 

3. 如何执行软中断的action<中断处理函数>

对于TIMER_SOFTIRQ来说,每次system clock产生中断时,即一个tick 到来时,在system clock的中断处理函数中会调用run_local_timers来设置TIMER_SOFTIRQ触发条件;也就是当前CPU对应的irq_cpustat_t结构体中的__softirq_pending成员的第TIMER_SOFTIRQ个BIT被置为1。 而当这个条件满足时,ksoftirqd线程(入口函数run_ksoftirqd,cpu_callback:kthread_create(run_ksoftirqd, hcpu, "ksoftirqd/%d", hotcpu);)会被唤醒,然后按照下面的流程调用TIMER_SOFTIRQ在数组softirq_vec中注册的action,即run_timer_softirq。
run_ksoftirqd--->do_softirq--->__do_softirq--->softirq_vec[TIMER_SOFTIRQ].action


 

[cpp]  view plain  copy
 
  1. static int run_ksoftirqd(void * __bind_cpu)  
  2. {  
  3.     set_current_state(TASK_INTERRUPTIBLE);  
  4.   
  5.     while (!kthread_should_stop()) {  
  6.         preempt_disable();  
  7.         if (!local_softirq_pending()) {  
  8.             preempt_enable_no_resched();  
  9.             schedule();  
  10.             preempt_disable();  
  11.         }  
  12.   
  13.         __set_current_state(TASK_RUNNING);  
  14.   
  15.         while (local_softirq_pending()) {  
  16.             /* Preempt disable stops cpu going offline. 
  17.                If already offline, we'll be on wrong CPU: 
  18.                don't process */  
  19.             if (cpu_is_offline((long)__bind_cpu))  
  20.                 goto wait_to_die;  
  21.             do_softirq();  
  22.             preempt_enable_no_resched();  
  23.             cond_resched();  
  24.             preempt_disable();  
  25.             rcu_sched_qs((long)__bind_cpu);  
  26.         }  
  27.         preempt_enable();  
  28.         set_current_state(TASK_INTERRUPTIBLE);  
  29.     }  
  30.     __set_current_state(TASK_RUNNING);  
  31.     return 0;  
  32.   
  33. wait_to_die:  
  34.     preempt_enable();  
  35.     /* Wait for kthread_stop */  
  36.     set_current_state(TASK_INTERRUPTIBLE);  
  37.     while (!kthread_should_stop()) {  
  38.         schedule();  
  39.         set_current_state(TASK_INTERRUPTIBLE);  
  40.     }  
  41.     __set_current_state(TASK_RUNNING);  
  42.     return 0;  
  43. }  


 

 

目录
相关文章
|
Linux
Linux(1)arm64根目录扩容
Linux(1)arm64根目录扩容
250 0
|
传感器 Linux 数据处理
ARM Linux摄像头传感器数据处理全景视野:从板端编码视频到高级应用(二)
ARM Linux摄像头传感器数据处理全景视野:从板端编码视频到高级应用
273 1
|
Ubuntu Linux
查看Linux系统架构的命令,查看linux系统是哪种架构:AMD、ARM、x86、x86_64、pcc 或 查看Ubuntu的版本号
查看Linux系统架构的命令,查看linux系统是哪种架构:AMD、ARM、x86、x86_64、pcc 或 查看Ubuntu的版本号
3800 4
|
10月前
|
Linux
Linux编程: 在业务线程中注册和处理Linux信号
通过本文,您可以了解如何在业务线程中注册和处理Linux信号。正确处理信号可以提高程序的健壮性和稳定性。希望这些内容能帮助您更好地理解和应用Linux信号处理机制。
209 26
|
10月前
|
Linux
Linux编程: 在业务线程中注册和处理Linux信号
本文详细介绍了如何在Linux中通过在业务线程中注册和处理信号。我们讨论了信号的基本概念,并通过完整的代码示例展示了在业务线程中注册和处理信号的方法。通过正确地使用信号处理机制,可以提高程序的健壮性和响应能力。希望本文能帮助您更好地理解和应用Linux信号处理,提高开发效率和代码质量。
210 17
|
Linux 数据安全/隐私保护 索引
linux inode索引节点使用率100% 解决+hustoj忘记密码+最新MDK注册方法
linux inode索引节点使用率100% 解决+hustoj忘记密码+最新MDK注册方法
186 1
|
Linux 网络安全 开发工具
内核实验(二):自定义一个迷你Linux ARM系统,基于Kernel v5.15.102, Busybox,Qemu
本文介绍了如何基于Linux Kernel 5.15.102版本和BusyBox创建一个自定义的迷你Linux ARM系统,并使用QEMU进行启动和调试,包括内核和BusyBox的编译配置、根文件系统的制作以及运行QEMU时的命令和参数设置。
1329 0
内核实验(二):自定义一个迷你Linux ARM系统,基于Kernel v5.15.102, Busybox,Qemu
|
编解码 安全 Linux
基于arm64架构国产操作系统|Linux下的RTMP|RTSP低延时直播播放器开发探究
这段内容讲述了国产操作系统背景下,大牛直播SDK针对国产操作系统与Linux平台发布的RTMP/RTSP直播播放SDK。此SDK支持arm64架构,基于X协议输出视频,采用PulseAudio和Alsa Lib处理音频,具备实时静音、快照、缓冲时间设定等功能,并支持H.265编码格式。此外,提供了示例代码展示如何实现多实例播放器的创建与管理,包括窗口布局调整、事件监听、视频分辨率变化和实时快照回调等关键功能。这一技术实现有助于提高直播服务的稳定性和响应速度,适应国产操作系统在各行业中的应用需求。
434 3
|
SQL 自然语言处理 网络协议
【Linux开发实战指南】基于TCP、进程数据结构与SQL数据库:构建在线云词典系统(含注册、登录、查询、历史记录管理功能及源码分享)
TCP(Transmission Control Protocol)连接是互联网上最常用的一种面向连接、可靠的、基于字节流的传输层通信协议。建立TCP连接需要经过著名的“三次握手”过程: 1. SYN(同步序列编号):客户端发送一个SYN包给服务器,并进入SYN_SEND状态,等待服务器确认。 2. SYN-ACK:服务器收到SYN包后,回应一个SYN-ACK(SYN+ACKnowledgment)包,告诉客户端其接收到了请求,并同意建立连接,此时服务器进入SYN_RECV状态。 3. ACK(确认字符):客户端收到服务器的SYN-ACK包后,发送一个ACK包给服务器,确认收到了服务器的确
366 1
|
存储 Ubuntu Linux
xenomai3+linux构建linux实时操作系统-基于X86_64和arm
Xenomai是一个实时性解决方案,通过在Linux上添加实时内核Cobalt来增强实时性能。它有三个主要部分:libcobalt(用户空间实时库)、Cobalt(内核空间实时内核)和硬件架构特定层(ipipe-core或dovetail)。ipipe-core适用于Linux 5.4以下版本,而dovetail用于5.4及以上版本。本文介绍了在X86 Ubuntu环境下,如何编译Xenomai内核,搭建应用环境,包括配置、编译、安装和实时性测试。对于其他硬件架构,如ARM和ARM64,步骤类似。文章还提到了Xenomai与Linux内核版本的兼容性和实时性测试结果。
1793 0
xenomai3+linux构建linux实时操作系统-基于X86_64和arm

热门文章

最新文章