| ; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py |
| ; RUN: llc -mtriple=arm-eabi -mcpu=cortex-a9 < %s | FileCheck %s |
| |
| define i8 @atomicrmw_usub_cond_i8(ptr %ptr, i8 %val) { |
| ; CHECK-LABEL: atomicrmw_usub_cond_i8: |
| ; CHECK: @ %bb.0: |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: .LBB0_1: @ %atomicrmw.start |
| ; CHECK-NEXT: @ =>This Inner Loop Header: Depth=1 |
| ; CHECK-NEXT: ldrexb r12, [r0] |
| ; CHECK-NEXT: uxtb r3, r1 |
| ; CHECK-NEXT: cmp r12, r3 |
| ; CHECK-NEXT: mov r3, r12 |
| ; CHECK-NEXT: subhs r3, r3, r1 |
| ; CHECK-NEXT: strexb r2, r3, [r0] |
| ; CHECK-NEXT: cmp r2, #0 |
| ; CHECK-NEXT: bne .LBB0_1 |
| ; CHECK-NEXT: @ %bb.2: @ %atomicrmw.end |
| ; CHECK-NEXT: mov r0, r12 |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: bx lr |
| %result = atomicrmw usub_cond ptr %ptr, i8 %val seq_cst |
| ret i8 %result |
| } |
| |
| define i16 @atomicrmw_usub_cond_i16(ptr %ptr, i16 %val) { |
| ; CHECK-LABEL: atomicrmw_usub_cond_i16: |
| ; CHECK: @ %bb.0: |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: .LBB1_1: @ %atomicrmw.start |
| ; CHECK-NEXT: @ =>This Inner Loop Header: Depth=1 |
| ; CHECK-NEXT: ldrexh r12, [r0] |
| ; CHECK-NEXT: uxth r3, r1 |
| ; CHECK-NEXT: cmp r12, r3 |
| ; CHECK-NEXT: mov r3, r12 |
| ; CHECK-NEXT: subhs r3, r3, r1 |
| ; CHECK-NEXT: strexh r2, r3, [r0] |
| ; CHECK-NEXT: cmp r2, #0 |
| ; CHECK-NEXT: bne .LBB1_1 |
| ; CHECK-NEXT: @ %bb.2: @ %atomicrmw.end |
| ; CHECK-NEXT: mov r0, r12 |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: bx lr |
| %result = atomicrmw usub_cond ptr %ptr, i16 %val seq_cst |
| ret i16 %result |
| } |
| |
| define i32 @atomicrmw_usub_cond_i32(ptr %ptr, i32 %val) { |
| ; CHECK-LABEL: atomicrmw_usub_cond_i32: |
| ; CHECK: @ %bb.0: |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: .LBB2_1: @ %atomicrmw.start |
| ; CHECK-NEXT: @ =>This Inner Loop Header: Depth=1 |
| ; CHECK-NEXT: ldrex r12, [r0] |
| ; CHECK-NEXT: subs r3, r12, r1 |
| ; CHECK-NEXT: movlo r3, r12 |
| ; CHECK-NEXT: strex r2, r3, [r0] |
| ; CHECK-NEXT: cmp r2, #0 |
| ; CHECK-NEXT: bne .LBB2_1 |
| ; CHECK-NEXT: @ %bb.2: @ %atomicrmw.end |
| ; CHECK-NEXT: mov r0, r12 |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: bx lr |
| %result = atomicrmw usub_cond ptr %ptr, i32 %val seq_cst |
| ret i32 %result |
| } |
| |
| define i64 @atomicrmw_usub_cond_i64(ptr %ptr, i64 %val) { |
| ; CHECK-LABEL: atomicrmw_usub_cond_i64: |
| ; CHECK: @ %bb.0: |
| ; CHECK-NEXT: .save {r4, r5, r11, lr} |
| ; CHECK-NEXT: push {r4, r5, r11, lr} |
| ; CHECK-NEXT: mov r12, r0 |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: .LBB3_1: @ %atomicrmw.start |
| ; CHECK-NEXT: @ =>This Inner Loop Header: Depth=1 |
| ; CHECK-NEXT: ldrexd r0, r1, [r12] |
| ; CHECK-NEXT: subs r4, r0, r2 |
| ; CHECK-NEXT: sbcs r5, r1, r3 |
| ; CHECK-NEXT: movlo r5, r1 |
| ; CHECK-NEXT: movlo r4, r0 |
| ; CHECK-NEXT: strexd lr, r4, r5, [r12] |
| ; CHECK-NEXT: cmp lr, #0 |
| ; CHECK-NEXT: bne .LBB3_1 |
| ; CHECK-NEXT: @ %bb.2: @ %atomicrmw.end |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: pop {r4, r5, r11, pc} |
| %result = atomicrmw usub_cond ptr %ptr, i64 %val seq_cst |
| ret i64 %result |
| } |
| |
| define i8 @atomicrmw_usub_sat_i8(ptr %ptr, i8 %val) { |
| ; CHECK-LABEL: atomicrmw_usub_sat_i8: |
| ; CHECK: @ %bb.0: |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: .LBB4_1: @ %atomicrmw.start |
| ; CHECK-NEXT: @ =>This Inner Loop Header: Depth=1 |
| ; CHECK-NEXT: ldrexb r12, [r0] |
| ; CHECK-NEXT: uqsub8 r3, r12, r1 |
| ; CHECK-NEXT: strexb r2, r3, [r0] |
| ; CHECK-NEXT: cmp r2, #0 |
| ; CHECK-NEXT: bne .LBB4_1 |
| ; CHECK-NEXT: @ %bb.2: @ %atomicrmw.end |
| ; CHECK-NEXT: mov r0, r12 |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: bx lr |
| %result = atomicrmw usub_sat ptr %ptr, i8 %val seq_cst |
| ret i8 %result |
| } |
| |
| define i16 @atomicrmw_usub_sat_i16(ptr %ptr, i16 %val) { |
| ; CHECK-LABEL: atomicrmw_usub_sat_i16: |
| ; CHECK: @ %bb.0: |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: .LBB5_1: @ %atomicrmw.start |
| ; CHECK-NEXT: @ =>This Inner Loop Header: Depth=1 |
| ; CHECK-NEXT: ldrexh r12, [r0] |
| ; CHECK-NEXT: uqsub16 r3, r12, r1 |
| ; CHECK-NEXT: strexh r2, r3, [r0] |
| ; CHECK-NEXT: cmp r2, #0 |
| ; CHECK-NEXT: bne .LBB5_1 |
| ; CHECK-NEXT: @ %bb.2: @ %atomicrmw.end |
| ; CHECK-NEXT: mov r0, r12 |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: bx lr |
| %result = atomicrmw usub_sat ptr %ptr, i16 %val seq_cst |
| ret i16 %result |
| } |
| |
| define i32 @atomicrmw_usub_sat_i32(ptr %ptr, i32 %val) { |
| ; CHECK-LABEL: atomicrmw_usub_sat_i32: |
| ; CHECK: @ %bb.0: |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: .LBB6_1: @ %atomicrmw.start |
| ; CHECK-NEXT: @ =>This Inner Loop Header: Depth=1 |
| ; CHECK-NEXT: ldrex r12, [r0] |
| ; CHECK-NEXT: subs r3, r12, r1 |
| ; CHECK-NEXT: movlo r3, #0 |
| ; CHECK-NEXT: strex r2, r3, [r0] |
| ; CHECK-NEXT: cmp r2, #0 |
| ; CHECK-NEXT: bne .LBB6_1 |
| ; CHECK-NEXT: @ %bb.2: @ %atomicrmw.end |
| ; CHECK-NEXT: mov r0, r12 |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: bx lr |
| %result = atomicrmw usub_sat ptr %ptr, i32 %val seq_cst |
| ret i32 %result |
| } |
| |
| define i64 @atomicrmw_usub_sat_i64(ptr %ptr, i64 %val) { |
| ; CHECK-LABEL: atomicrmw_usub_sat_i64: |
| ; CHECK: @ %bb.0: |
| ; CHECK-NEXT: .save {r4, r5, r6, r7, r11, lr} |
| ; CHECK-NEXT: push {r4, r5, r6, r7, r11, lr} |
| ; CHECK-NEXT: mov r12, #0 |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: .LBB7_1: @ %atomicrmw.start |
| ; CHECK-NEXT: @ =>This Inner Loop Header: Depth=1 |
| ; CHECK-NEXT: ldrexd r4, r5, [r0] |
| ; CHECK-NEXT: subs r6, r4, r2 |
| ; CHECK-NEXT: sbcs r7, r5, r3 |
| ; CHECK-NEXT: adc r1, r12, #0 |
| ; CHECK-NEXT: teq r1, #1 |
| ; CHECK-NEXT: movwne r7, #0 |
| ; CHECK-NEXT: movwne r6, #0 |
| ; CHECK-NEXT: strexd r1, r6, r7, [r0] |
| ; CHECK-NEXT: cmp r1, #0 |
| ; CHECK-NEXT: bne .LBB7_1 |
| ; CHECK-NEXT: @ %bb.2: @ %atomicrmw.end |
| ; CHECK-NEXT: mov r0, r4 |
| ; CHECK-NEXT: mov r1, r5 |
| ; CHECK-NEXT: dmb ish |
| ; CHECK-NEXT: pop {r4, r5, r6, r7, r11, pc} |
| %result = atomicrmw usub_sat ptr %ptr, i64 %val seq_cst |
| ret i64 %result |
| } |