The concept of interruptions: the CPU in the execution process, there are certain emergencies that need to be dealt with urgently, the CPU pauses the execution of the current program to deal with the emergencies
After processing, the CPU returns to the location where the original program was interrupted to continue to execute
The interruptions are categorized into: internal interruptions and external interruptions
Internal Interrupt:Interrupt source from within the CPU (software interrupt instruction, overflow, trigger error, etc.)
External interrupt:Interrupt source from outside the CPU, requested by a peripheral device
Masked interrupts and non-maskable interrupts:
Maskable interrupts:Can be masked through the masking word masked, masking the interrupt is no longer responded to
Non-Plainable interrupts: Can not be masked, masked interrupts. Cannot be masked
Vector interrupts and non-vector interrupts:
Vector interrupts: CPU usually assigns different interrupt numbers to different interrupts, when an interrupt is detected coming from a certain interrupt number, it automatically jumps to the address corresponding to that interrupt number for execution
Non-vector interrupts: Multiple interrupts **** enjoy one entry address. After entering this entry address, the software will judge the interrupt flag to identify which interrupt is specific
That is to say, vector interrupts are provided with the interrupt service program entry address by the software, and non-vector interrupts are provided with the interrupt entry address by the software
/*Typical non-vector interrupts first judge the source of the interrupt, and then call interrupt handlers for the different interrupt sources*
irq _handler()
{
...
int int_src = read_int_status();/* read hardware's interrupt-related registers*/
switch(int_src){// determine interrupt flag
case DEV_A:
dev_a_handler();
break;
case DEV_B:
dev_b_handler();
break;
...
default:
break;
}
...
}
Timer interrupt principle:
Timer in hardware also since the interrupt, PIT (Programmable Interval Timer) receives a clock input,
when the clock pulse arrives, will be the current count value incremented by 1 and compared with the count value has been set, if equal to the proof of the counting cycle is full, the generation of timer interrupt, and
reset the count value.
The following figure shows:
Linux interrupt handler architecture:
Linux interrupts are divided into: the top half (top half) and the bottom half (bottom half)
Top part: to complete the least possible urgent functions, it is often simply read the interrupt status in the register and clear the interrupt flag and then carry out
The top part: to complete as little as possible more urgent functions, it is often just simple read the interrupt status and clear the interrupt flag and then carry out
The top part: to complete the least possible more urgent functions, it is often just simple read the interrupt status and clear the interrupt flag and then carry out
"registering the interrupt" (that is, hooking the bottom half handler into the bottom half of the device's execution queue)
Features: fast response
bottom half:Most of the work of interrupt handling is done in the bottom half, which does almost all of the interrupt handler's things.
Features:Handles relatively non-urgent events
Trivia:You can get statistics on system interrupts by looking at the /proc/interrupts file in Linux.
The following figure shows:
The first column is the interrupt number and the second column is the number of times the interrupt has been generated to the CPU
After the introduction of the relevant basic concepts, let's explore Linux interrupt programming
Linux Interrupt Programming:
1. Requesting and releasing interrupts
Requesting an interrupt:
Requesting an interrupt:
int request_irq(unsigned int irq,irq_handler_t handler,
unsigned long irqflags,const char *devname,void *dev_id)
Parameter introduction:irq is the hardware interrupt number to be requested
Parameter introduction:irq is the number of the
handler is the interrupt handler registered with the system (top half), it is a callback function, when an interrupt occurs, the system calls it and passes the
dev_id parameter to it
irqflags: it is the attribute of interrupt handler, you can specify the way of interrupt triggering and handling:
Trigger Mode:IRQF_TRIGGER_RISING, IRQF_TRIGGER_FALLING, IRQF_TRIGGER_HIGH, IRQF_TRIGGER_LOW
Handling Mode:IRQF_DISABLE indicates that the interrupt handler is a fast handler, and the fast handler will block all interrupts when called. All interrupts
IRQF_SHARED indicates that multiple devices *** enjoy interrupts, dev_id will be used when interrupts *** enjoy, usually set to NULL
Return value: 0 indicates success, -EINVAL indicates that the interrupt number is invalid, and -EBUSY indicates that the interrupt has been occupied and cannot be *** enjoy
Top The top half of the handler type irq_handler_t is defined as
typedef irqreturn_t (*irq_handler_t)(int,void*);
typedef int irqreturn_t;
2. p>With request there is of course release
void free_irq(unsigned int irq,void *dev_id);
Parameter definitions are similar to request_irq
3. enable and mask interrupts
void disable_irq(int irq );//wait for the current interrupt processing to finish (best not to use it in the top board section, you know)
void disable_irq_nosync(int irq);//return immediately
void enable_irq(int irq);//
4. Masking of all interrupts within the CPU:
#define local_irq_save(flags)... // disable interrupts and save state
void local_irq_disable(void);// disable interrupts, don't save state
Below is a description of the top half and bottom half implementation mechanisms
Bottom Half Mechanisms:
Introduction:Bottom Half Mechanisms are mainly tasklet, work queue and soft interrupt
1. The bottom half is to think of one of the methods tasklet
(1) we need to define the tasklet machine processor and associate the two
Example:
void my_tasklet_func(unsigned long);/* define a processor function */
< p>DECLARE_TASKLET(my_tasklet,my_tasklet_func,data);/*The above code defines the tasklet named my_tasklet and binds the rest of the
my_tasklet_func() function, with the incoming argument being data */
(2) scheduling
tasklet_schedule(&my_tasklet);
///Use this function to schedule a run when it's time
tasklet using template:
//// Define the tasklet and the bottom half of the function and associate it* with the tasklet.
void xxx_do_tasklet(unsigned long);
DECLARE_TASKLET(xxx_tasklet,xxx_do_tasklet,0);
//*Interrupt Handling Bottom Half*
void xxx_do_ tasklet(unsigned long)
{
...
}
/*interrupt handling top half */
irqreturn_t xxx_interrupt(int irq,void *dev_id)
{
...
tasklet_schedule(&xxx_tasklet);//schedule the floor section
...
}
/*Device Driver Module Load Function**
int __init xxx_init(void)
{
...
/*Request interrupt**
result = request_irq(xxx_irq,xxx_interrupt,
IRQF_DISABLED, "xxx",NULL);
...
return IRQ_HANDLED;
}
/*Device Driver Module Uninstall Function**
void __exit xxx_exit(void)
{
...
/*free interrupt**
free_irq(xxx_irq,xxx_interrupt);
...
}
2. The bottom half of the implementation of the second method --- work queue
Used in a similar way and tasklet
Related operations:
struct work_struct my_wq;/* Define a work queue */
void my_wq_func( unsigned long);/*Define a handler function*/
This work queue can be initialized and bound to a handler function by INIT_WORK()
INIT_WORK(&my_wq,(void (*)(void *))my_wq_func,NULL) ;
/*Initialize the work queue and bind it to a handler function*/
schedule_work(&my_wq);/*Schedule the work queue to execute*/
/*Template for use of the work queue*/
/*Define the work queue and the associated function*/
struct work_struct(unsigned long);
void xxx_do_work(unsigned long);
/*Interrupt Handling Bottom Half*/
void xxx_do_work(unsigned long)
{
< p>...}
/*Interrupt handling top half */
/*Interrupt handling top half */
irqreturn_t xxx_interrupt(int irq,void *dev_id)
{
...
schedule_work(&my_wq);//schedule bottom half
...
return IRQ_HANDLED;
}
/*Device Driver Module Load Function**
int xxx_init(void)
{
...
/*Request interrupt**
result = request_irq(xxx_irq,xxx_interrupt,
IRQF_DISABLED, "xxx",NULL);
...
/*Initialize work queue**
INIT_WORK(&my_wq,(void (*)(void *))xxx_do_work,NULL);
}
//*Device Driver Module Uninstallation Function**
void xxx_exit( void)
{
...
/*free interrupt**
free_irq(xxx_irq,xxx_interrupt);
...
}