| ; RUN: llc < %s -march=avr | FileCheck %s |
| |
| @count = global i8 0 |
| |
| define avr_intrcc void @interrupt_handler() { |
| ; CHECK-LABEL: interrupt_handler: |
| ; CHECK: sei |
| ; CHECK-NEXT: push r0 |
| ; CHECK-NEXT: push r1 |
| ; CHECK-NEXT: in r0, 63 |
| ; CHECK-NEXT: push r0 |
| ; CHECK-NEXT: clr r1 |
| ; CHECK: pop r0 |
| ; CHECK-NEXT: out 63, r0 |
| ; CHECK-NEXT: pop r1 |
| ; CHECK-NEXT: pop r0 |
| ; CHECK-NEXT: reti |
| ret void |
| } |
| |
| define void @interrupt_handler_via_ir_attribute() #0 { |
| ; CHECK-LABEL: interrupt_handler_via_ir_attribute: |
| ; CHECK: sei |
| ; CHECK-NEXT: push r0 |
| ; CHECK-NEXT: push r1 |
| ; CHECK-NEXT: in r0, 63 |
| ; CHECK-NEXT: push r0 |
| ; CHECK-NEXT: clr r1 |
| ; CHECK: pop r0 |
| ; CHECK-NEXT: out 63, r0 |
| ; CHECK-NEXT: pop r1 |
| ; CHECK-NEXT: pop r0 |
| ; CHECK-NEXT: reti |
| ret void |
| } |
| |
| define avr_signalcc void @signal_handler() { |
| ; CHECK-LABEL: signal_handler: |
| ; CHECK-NOT: sei |
| ; CHECK: push r0 |
| ; CHECK-NEXT: push r1 |
| ; CHECK-NEXT: in r0, 63 |
| ; CHECK-NEXT: push r0 |
| ; CHECK-NEXT: clr r1 |
| ; CHECK: pop r0 |
| ; CHECK-NEXT: out 63, r0 |
| ; CHECK-NEXT: pop r1 |
| ; CHECK-NEXT: pop r0 |
| ; CHECK-NEXT: reti |
| ret void |
| } |
| |
| define void @signal_handler_via_attribute() #1 { |
| ; CHECK-LABEL: signal_handler_via_attribute: |
| ; CHECK-NOT: sei |
| ; CHECK: push r0 |
| ; CHECK-NEXT: push r1 |
| ; CHECK-NEXT: in r0, 63 |
| ; CHECK-NEXT: push r0 |
| ; CHECK-NEXT: clr r1 |
| ; CHECK: pop r0 |
| ; CHECK-NEXT: out 63, r0 |
| ; CHECK-NEXT: pop r1 |
| ; CHECK-NEXT: pop r0 |
| ; CHECK-NEXT: reti |
| ret void |
| } |
| |
| define avr_intrcc void @interrupt_alloca() { |
| ; CHECK-LABEL: interrupt_alloca: |
| ; CHECK: sei |
| ; CHECK-NEXT: push r0 |
| ; CHECK-NEXT: push r1 |
| ; CHECK-NEXT: in r0, 63 |
| ; CHECK-NEXT: push r0 |
| ; CHECK-NEXT: clr r1 |
| ; CHECK: push r28 |
| ; CHECK-NEXT: push r29 |
| ; CHECK-NEXT: in r28, 61 |
| ; CHECK-NEXT: in r29, 62 |
| ; CHECK-NEXT: sbiw r28, 1 |
| ; CHECK-NEXT: in r0, 63 |
| ; CHECK-NEXT: cli |
| ; CHECK-NEXT: out 62, r29 |
| ; CHECK-NEXT: out 63, r0 |
| ; CHECK-NEXT: out 61, r28 |
| ; CHECK: adiw r28, 1 |
| ; CHECK-NEXT: in r0, 63 |
| ; CHECK-NEXT: cli |
| ; CHECK-NEXT: out 62, r29 |
| ; CHECK-NEXT: out 63, r0 |
| ; CHECK-NEXT: out 61, r28 |
| ; CHECK-NEXT: pop r29 |
| ; CHECK-NEXT: pop r28 |
| ; CHECK: pop r0 |
| ; CHECK-NEXT: out 63, r0 |
| ; CHECK-NEXT: pop r1 |
| ; CHECK-NEXT: pop r0 |
| ; CHECK-NEXT: reti |
| alloca i8 |
| ret void |
| } |
| |
| define void @signal_handler_with_increment() #1 { |
| ; CHECK-LABEL: signal_handler_with_increment: |
| ; CHECK: push r0 |
| ; CHECK-NEXT: push r1 |
| ; CHECK-NEXT: in r0, 63 |
| ; CHECK-NEXT: push r0 |
| ; CHECK-NEXT: clr r1 |
| ; CHECK-NEXT: push r24 |
| ; CHECK-NEXT: lds r24, count |
| ; CHECK-NEXT: inc r24 |
| ; CHECK-NEXT: sts count, r24 |
| ; CHECK-NEXT: pop r24 |
| ; CHECK-NEXT: pop r0 |
| ; CHECK-NEXT: out 63, r0 |
| ; CHECK-NEXT: pop r1 |
| ; CHECK-NEXT: pop r0 |
| ; CHECK-NEXT: reti |
| %old = load volatile i8, i8* @count |
| %new = add i8 %old, 1 |
| store volatile i8 %new, i8* @count |
| ret void |
| } |
| |
| declare void @foo() |
| |
| ; When a signal handler calls a function, it must push/pop all call clobbered |
| ; registers. |
| define void @signal_handler_with_call() #1 { |
| ; CHECK-LABEL: signal_handler_with_call: |
| ; CHECK: push r0 |
| ; CHECK-NEXT: push r1 |
| ; CHECK-NEXT: in r0, 63 |
| ; CHECK-NEXT: push r0 |
| ; CHECK-NEXT: clr r1 |
| ; CHECK-NEXT: push r18 |
| ; CHECK-NEXT: push r19 |
| ; CHECK-NEXT: push r20 |
| ; CHECK-NEXT: push r21 |
| ; CHECK-NEXT: push r22 |
| ; CHECK-NEXT: push r23 |
| ; CHECK-NEXT: push r24 |
| ; CHECK-NEXT: push r25 |
| ; CHECK-NEXT: push r26 |
| ; CHECK-NEXT: push r27 |
| ; CHECK-NEXT: push r30 |
| ; CHECK-NEXT: push r31 |
| ; CHECK-NEXT: call foo |
| ; CHECK-NEXT: pop r31 |
| ; CHECK-NEXT: pop r30 |
| ; CHECK-NEXT: pop r27 |
| ; CHECK-NEXT: pop r26 |
| ; CHECK-NEXT: pop r25 |
| ; CHECK-NEXT: pop r24 |
| ; CHECK-NEXT: pop r23 |
| ; CHECK-NEXT: pop r22 |
| ; CHECK-NEXT: pop r21 |
| ; CHECK-NEXT: pop r20 |
| ; CHECK-NEXT: pop r19 |
| ; CHECK-NEXT: pop r18 |
| ; CHECK-NEXT: pop r0 |
| ; CHECK-NEXT: out 63, r0 |
| ; CHECK-NEXT: pop r1 |
| ; CHECK-NEXT: pop r0 |
| ; CHECK-NEXT: reti |
| call void @foo() |
| ret void |
| } |
| |
| attributes #0 = { "interrupt" } |
| attributes #1 = { "signal" } |