Download presentation
Presentation is loading. Please wait.
1
(Dr. Öğr. Üyesi Deniz Dal)
BM-305 Mikrodenetleyiciler Güz 2018 (8. Sunu) (Dr. Öğr. Üyesi Deniz Dal)
2
Kesmeler (Interrupts)
3
Are we there yet? Yoklama (Polling) "Are we there yet?" No…
Yoklama (Polling) "Are we there yet?" No… Kesme (Interrupt) "Read a book, I’ll tell you when we are there"
4
Yoklama (Polling) Sabah 6’da işine gitmek isteyen birinin geceyi uyumadan geçirip, acaba saat 6 oldu mu diye sürekli saatini kontrol etmesi Kesme (Interrupt) Sabah 6’da işine gitmek isteyen birinin saatinin alarmını kurduktan sonra uyuması ve alarm çalınca uyanması
5
Yoklama (Polling) Bir iş mülakatı için önemli bir telefon bekleyen birinin her saniye telefonun ahizesini kaldırıp acaba aradılar mı diye telefonu kontrol etmesi Kesme (Interrupt) Bir iş mülakatı için önemli bir telefon bekleyen birinin telefonun zili çalıncaya kadar mülakatta çıkabilecek sorulara çalışması veya kahvesini içmesi
6
Interrupt On a very basic level, an interrupt is a signal that interrupts the current processor activity. It may be triggered by an external event (change in pin state) or an internal event (a timer or a software signal). Once triggered, an interrupt pauses the current activity and causes the program to execute a different function. This function is called an Interrupt Handler or an Interrupt Service Routine (ISR). Once the function is completed, the program returns to what it was doing before the interrupt was triggered. Interrupts are asynchronous. An asynchronous event is something that occurs outside of the regular flow of our program – it can happen at any time, no matter what our code is crunching on at the moment. This means that rather than manually checking whether your desired event has happened, you can let your AVR do the checking for you.
7
A Real World Example Imagine you are sitting on your couch, enjoying a frosty brew and watching a movie after a long day. Life is good. There is only one problem: you are waiting for an incredibly important package to arrive, and you need it as soon as possible. If you were a normal AVR program or Arduino sketch, you would have to repeatedly stop your movie, get up, and go check the mailbox every 5 minutes to make sure you knew when the package was there. Instead, imagine if the package was sent Fedex or UPS with delivery confirmation. Now, the delivery man will go to your front door and ring the doorbell as soon as he arrives. That is your interrupt trigger. Once you get the trigger, you can pause your movie and go deal with the package. That is your Interrupt Service Routine. As soon as you are done, you can pick up the film where you left off, with no extra time wasted. That is the power of interrupts.
8
Why are Interrupts Important?
The AVR chips used in most Arduinos are not capable of parallel processing, i.e. they can’t do multiple things at once. Using asynchronous processing via interrupts enables us to maximize the efficiency of our code, so we don’t waste any precious clock cycles on polling loops or waiting for things to occur. Interrupts are also good for applications that require precise timing, because we know we’ll catch our event the moment it occurs, and won’t accidentally miss anything.
9
Polling vs Interrupt Interrupt Polling
instruction k instruction k+1 normal_execution interrupt signal interrupt_ service_ routine runs Interrupt while(1) { get_device_status; if(service_required) service_routine; normal_execution; } Using polling, the CPU must continually check the device’s status. Using interrupt: A device will send an interrupt signal when needed. In response, the CPU will perform an Interrupt Service Routine (ISR), and then resume its normal execution. Compared to polling, interrupt is a more efficient approach for the CPU to handle peripheral devices. E.g. serial port, external switches, timers, PWM and ADC.
10
Interrupt Execution Sequence
1. A device issues an interrupt request 2. CPU finishes the current instruction 3. CPU acknowledges the interrupt 4. CPU saves its states and PC onto stack 5. CPU loads the address of ISR onto PC 6. CPU executes the ISR 7. CPU retrieves its states and PC from stack 8. Normal execution resumes
11
Kesme Denetleyicisi (Interrupt Controller)
12
Kesme Denetleyicisi (Interrupt Controller)
13
Timer Interrupts Really handy to have timed action, despite whatever loop() is doing could check for serial input on a regular basis could read analog signal for regular sampling could produce custom signal at specific frequency Idea is to set up timer so when it reaches a specified count, it creates an interrupt and also resets counter to zero so cycle begins a new turn Interrupt Service Routine (ISR) must be as short and fast as possible. If your sketch uses multiple ISRs, only one can run at a time, other interrupts will be ignored (turned off) until the current one is finished. Since delay() and millis() both rely on interrupts, they will not work while an ISR is running.
14
CAUTION Messing with timer configurations can compromise other timer-based functions like PWM outputs: analogWrite() (diff. pins diff. timers) delay() (uses timer 0, depends on counter wrap) millis() and micros() (uses timer 0, depends on wrap) Servo library (uses timer 1) tone() (uses timer 2)
15
TIMSKn (Timer Interrupt MaSK) Register
TIMSK1 controls what generates interrupts for Timer 1 TOIE1: Timer Overflow Interrupt Enable (When TCNT1 wraps) OCIE1A: Output Compare Match Interrupt Enable for Channel A of Timer 1 (When the TCNT1 reaches OCR1A) OCIE1B: Output Compare Match Interrupt Enable for Channel B of Timer 1 (When the TCNT1 reaches OCR1B) ICIE1: Input Capture Interrupt Enable for Timer 1
16
Timer Overflow Interrupt
Timer overflow means the timer has reached is limit value. When a timer overflow interrupt occurs, the timer overflow bit TOVx will be set in the interrupt flag register TIFRx. When the timer overflow interrupt enable bit TOIEx in the interrupt mask register TIMSKx is set, the timer overflow interrupt service routine ISR(TIMERx_OVF_vect) will be called.
17
Output Compare Match Interrupt
When an output compare match interrupt occurs, the OCFxy flag will be set in the interrupt flag register TIFRx. When the output compare interrupt enable bit OCIExy in the interrupt mask register TIMSKx is set, the output compare match interrupt service routine ISR(TIMERx_COMPy_vect) will be called. When an interrupt occurs, a flag in the interrupt flag register TIFRx is set. This interrupt will be automatically cleared when entering the ISR or by manually clearing the bit in the interrupt flag register.
18
What Do We Do with this Power?
Let’s set up a compare match timer interrupt to change the state of an LED every second Need TIMER 1 to reach beyond 16 micro seconds prescale by 1024, so frequency is ticks/sec Set up registers: DDRB: set Digital Pin 13 TCCR1B: set WGM12 (for CTC) TCCR1B: set CS12, CS10 (for prescale by 1024) OCR1A: set (OCR1AH = 61, OCR1AL = 9) TIMSK1: set OCIE1A (Enable the Associated Interrupt Source) Set I in SREG (Enable CPU to Serve for IRQs) Make ISR function: ISR(TIMER1_COMPA_vect){}
19
Zamanlayıcı 1 ile CTC Modunda Kesme Sinyali Üretilmesi İçin…
SREG saklayıcısı içerisindeki I isimli global kesme bayrağı aktif hale getirilmeli TIMSK1 saklayıcısı içerisindeki OCIE1A bayrağı aktif hale getirilmeli TIFR1 saklayıcısı içerisindeki OCF1A bayrağının aktif hale gelmesi beklenmeli
20
Example: Compare Match Interrupt-Driven LED Blink
#include <avr/io.h> #include <avr/interrupt.h> // cli and sei functions volatile int state=0; // it is volatile since it will be used in ISR int main() { cli(); // turn off (clear) global interrupt enable flag DDRB |= (1<<5); // set up on-board LED for OUTPUT TCCR1B |= (1 << WGM12); // set bit for CTC mode TCCR1B |= (1 << CS12); // set bit 2 of prescaler for 1024x TCCR1B |= (1 << CS10); // set bit 0 of prescaler for 1024x OCR1A = 15625; // set L & H bytes to (1 sec) TIMSK1 |= (1 << OCIE1A); // enable interrupt on OCR1A sei(); // turn on (set) global interrupt enable flag for(;;) } return 0; ISR(TIMER1_COMPA_vect) state++; state %= 2; PORTB = state ? PORTB|(1<<5) : PORTB&(~(1<<5)); volatile prevents compiler from optimizing code Variables shared between ISR functions and normal functions should be declared volatile. This tells the compiler that such variables might change at any time, and thus the compiler must reload the variable whenever you reference it, rather than relying upon a copy it might have in a processor register. The reason to disable the interrupts prior to updating the 16 bit registers (e.g. TCNT1) is to ensure that nothing else updates the registers in between the two instructions (updating the low and high bytes). Generally speaking it would be a good idea to disable interrupts prior to writing to the 16 bit registers. Inside the ISR, delay() won't work and the value returned by millis() will not increment.
21
A Custom PWM When time is up:
ISR(TIMER1_COMPA_vect) { if(state) OCR1A = 31250; // two seconds for OFF else OCR1A = 15625; // one second for ON state++; state %= 2; PORTB = state ? PORTB|(1<<5) : PORTB&(~(1<<5)); } When time is up: if state == 1 (LED ON), set compare register to 2 seconds otherwise (LED OFF), set compare register to 1 second In this way, you can customize a PWM-like signal arbitrarily
22
Interrupt Vectors in ATMega328P
By placing a jump instruction at each interrupt's address in the table, we can make the AVR processor jump to our ISR which lies elsewhere. .org INT0addr rjmp int0_handle INT0addr ? An interrupt with a lower VectorNo has a higher priority. E.g. INT0 has a higher priority then INT1.
23
Timer Interrupt in AVR Assembly
;To do a 16-bit write, the high byte must be written before the low byte. ;For a 16-bit read, the low byte must be read before the high byte. .INCLUDE "m328pdef.inc";Header for ATMEGA328P state:.BYTE 2;Reserve Two Bytes (int Data Type) in Data Segment .DSEG;Data SEGment RJMP setup .org 0x0000;Address of the "RESET" Vector .CSEG;Code SEGment RJMP ISR_TIMER1_COMPA;Address of the "Timer/Counter 1 Compare Match Channel A" Vector (CTC Mode) .org 0x0016 setup: ;STEP 1 - Initialize the "state" Variable to 0 (zero) STS LOW(state),R16 STS HIGH(state),R16 LDI R16,0x00 ;STEP 2 - Initialize Stack Pointer LDI R16,LOW(RAMEND) OUT SPH,R16;Stack Pointer High LDI R16,HIGH(RAMEND) OUT SPL,R16;Stack Pointer Low CLI ;STEP 3 - Turn off (Clear) Global Interrupt Enable Flag (I in SREG) IN R16,DDRB ;STEP 4 - Set PORTB.5 (On-board LED) as OUTPUT ORI R16,(1<<5) OUT DDRB,R16 LDS R16,TCCR1B ;STEP 5 - Set Bit for the CTC Mode ORI R16,(1<<WGM12) ;STEP 6 - Set Timer 1 Prescaler as Fcpu/1024 ORI R16,(1<<CS10) ORI R16,(1<<CS12) STS TCCR1B,R16 ;STEP 7 - Update TCCR1B LDI R16,HIGH(15625);Load 0x3D to R16 ;STEP 8 - Set Low & High Bytes of OCR1A to (1 sec) STS OCR1AH,R16;STore direct to data Space STS OCR1AL,R16;STore direct to data Space LDI R16,LOW(15625);Load 0x09 to R16 LDS R16,TIMSK1 ;STEP 9 - Enable Interrupt on CTC Mode for Channel A STS TIMSK1,R16 ORI R16,(1<<OCIE1A) SEI ;STEP 10 - Turn on (Set) Global Interrupt Enable Flag (I in SREG) RJMP loop;Relative JuMP loop: LDS R30,LOW(state) ;state++ ISR_TIMER1_COMPA: ADIW ZH:ZL,1;ADd Immediate to Word LDS R31,HIGH(state) LDI R31,0x00 ;state %= 2; ANDI R30,0x01;Check Whether the LSB of Low Byte of "state" is 1 (Odd Number) ;PORTB = state ? PORTB|(1<<5) : PORTB&(~(1<<5)); LDI R16,0x00;LoaD Immeadiate value to R16 ifLabel: BRNE elseLabel;BRanch Not Equal OUT PORTB,R16;Turn LED off LDI R16,0x20;LoaD Immeadiate value to R16 elseLabel: RJMP ifEndLabel;Relative JuMP ifEndLabel: OUT PORTB,R16;Turn led on STS HIGH(state),ZH ;Update "state" Variable STS LOW(state),ZL RETI;RETurn Interrupt
24
External Interrupts On Arduino Uno (ATMega328P), there are 2 external interrupts available. These are located on Digital Pin 2 (0 - INT0) and Digital Pin 3 (1 - INT1) Let us use a pushbutton to trigger an external interrupt on Digital Pin 2 that controls (changes the state of) the on-board LED (Digital Pin 13). (The Pin 2 is supposed to be OFF while the pushbutton is not pressed and ON when it is pressed.)
25
Example: External Interrupt-Driven LED Blink
Global olarak kesme devreye alınmamış görünüyor??? Arduino tarafından ISR(INT0_vect) şekline dönüştürülür. Inside the attached function, delay() won't work and the value returned by millis() will not increment.
26
attachInterrupt Function
The attachInterrupt function requires 3 parameters. attachInterrupt(param1, param2, param3) These are; param1 = Which interrupt to listen for. This is the Interrupt Number not the Digital Pin number. param2 = Which function to call, this must be a method that takes no parameters and returns no value. param3 = Which condition to watch for.
27
attachInterrupt Conditions (Trigger Modes)
The Arduino can listen for 4 types of condition changes. These are; LOW = The input is at a LOW state RISING = The input state changes from LOW to HIGH FALLING = the input state changes from HIGH to LOW CHANGE = The input state changes from HIGH to LOW or LOW to HIGH, i.e. changes its state
28
detachInterrupt Function
External interrupts can also be removed by using the detachInterrupt(interrupt_number) method.
29
interrupts and noInterrupts Functions
The Arduino also has the ability to temporarily ignore all the interrupts. You may want to do this if you have some sensitive code that must be executed without interruption. In this case you would issue a noInterrupts() call. Once your sensitive code block has completed, interrupts can be restarted by calling interrupts().
30
External Interrupt in AVR C
#include <avr/io.h> #include <avr/interrupt.h> int main() { //STEP 1 - Set Digital Pin 13 as Output DDRB |= (1<<5); //STEP 2 - Turn off the LED connected to Digital Pin 13 PORTB &= ~(1<<5); //STEP 3 - Set Digital Pin 2 as Input DDRD &= ~(1<<2); //STEP 4 - Set INT0 to be Triggered on the RISING Edge in Digital Pin 2 EICRA |= (1 << ISC00); EICRA |= (1 << ISC01); //STEP 5 - Enable INT0 Interrupt EIMSK |= (1 << INT0); //STEP 6 - Turn on (Set) Global Interrupt Enable Flag (I in SREG) sei(); while(1) } ISR (INT0_vect) PORTB ^= (1<<5);//Toggle the Output
31
External Interrupt in AVR Assembly
.INCLUDE "m328pdef.inc";Header for ATMEGA328P .org 0x0000;Address of the "RESET" Vector RJMP setup .org 0x0002 RJMP ISR_INT0;Address of the "INT0" Vector setup: ;STEP 1 - Initialize Stack Pointer LDI R16,HIGH(RAMEND) LDI R16,LOW(RAMEND) OUT SPH,R16;Stack Pointer High OUT SPL,R16;Stack Pointer Low ;STEP 2 - Set Digital Pin 13 as Output ORI R16,(1<<5) IN R16,DDRB OUT DDRB,R16 ;STEP 3 - Turn off the LED connected to Digital Pin 13 ANDI R16,~(1<<5) IN R16,PORTB OUT PORTB,R16 ;STEP 4 - Set Digital Pin 2 as Input ANDI R16,~(1<<2) IN R16,DDRD OUT DDRD,R16 ;STEP 5 - Set INT0 to be Triggered on the RISING Edge in Digital Pin 2 ORI R16,(1<<ISC00) LDS R16,EICRA ORI R16,(1<<ISC01) STS EICRA,R16 IN R16,EIMSK ;STEP 6 - Enable INT0 Interrupt ORI R16,(1<<INT0) OUT EIMSK,R16 ;STEP 7 - Turn on (Set) Global Interrupt Enable Flag (I in SREG) SEI loop: RJMP loop;Relative JuMP ISR_INT0: ;Toggle the Output LDI R17,(1<<5) EOR R16,R17 RETI;RETurn Interrupt
Similar presentations
© 2025 SlidePlayer.com. Inc.
All rights reserved.