; RUN: llc -mtriple=arm-eabi %s -o - | FileCheck %s --check-prefix=ARM ; RUN: llc -mtriple=arm-eabi -mcpu=arm1156t2-s -mattr=+thumb2 %s -o - \ ; RUN: | FileCheck %s --check-prefix=ARMT2 ; RUN: llc -mtriple=thumb-eabi -mcpu=cortex-m0 %s -o - \ ; RUN: | FileCheck %s --check-prefix=THUMB1 ; RUN: llc -mtriple=thumb-eabi -mcpu=arm1156t2-s -mattr=+thumb2 %s -o - \ ; RUN: | FileCheck %s --check-prefix=THUMB2 ; RUN: llc -mtriple=thumbv8m.base-eabi %s -o - \ ; RUN: | FileCheck %s --check-prefix=V8MBASE define i32 @t1(i32 %c) nounwind readnone { entry: ; ARM-LABEL: t1: ; ARM: mov [[R1:r[0-9]+]], #101 ; ARM: orr [[R1b:r[0-9]+]], [[R1]], #256 ; ARM: movgt {{r[0-1]}}, #123 ; ARMT2-LABEL: t1: ; ARMT2: movw [[R:r[0-1]]], #357 ; ARMT2: movwgt [[R]], #123 ; THUMB1-LABEL: t1: ; THUMB1: mov r1, r0 ; THUMB1: movs r2, #255 ; THUMB1: adds r2, #102 ; THUMB1: movs r0, #123 ; THUMB1: cmp r1, #1 ; THUMB1: bgt ; THUMB2-LABEL: t1: ; THUMB2: movw [[R:r[0-1]]], #357 ; THUMB2: movgt [[R]], #123 %0 = icmp sgt i32 %c, 1 %1 = select i1 %0, i32 123, i32 357 ret i32 %1 } define i32 @t2(i32 %c) nounwind readnone { entry: ; ARM-LABEL: t2: ; ARM: mov [[R:r[0-9]+]], #101 ; ARM: orr [[R]], [[R]], #256 ; ARM: movle [[R]], #123 ; ARMT2-LABEL: t2: ; ARMT2: mov [[R:r[0-1]]], #123 ; ARMT2: movwgt [[R]], #357 ; THUMB1-LABEL: t2: ; THUMB1: cmp r{{[0-9]+}}, #1 ; THUMB1: bgt ; THUMB2-LABEL: t2: ; THUMB2: mov{{(s|\.w)}} [[R:r[0-1]]], #123 ; THUMB2: movwgt [[R]], #357 %0 = icmp sgt i32 %c, 1 %1 = select i1 %0, i32 357, i32 123 ret i32 %1 } define i32 @t3(i32 %a) nounwind readnone { entry: ; ARM-LABEL: t3: ; ARM: rsbs r1, r0, #0 ; ARM: adc r0, r0, r1 ; ARMT2-LABEL: t3: ; ARMT2: clz r0, r0 ; ARMT2: lsr r0, r0, #5 ; THUMB1-LABEL: t3: ; THUMB1: movs r1, #0 ; THUMB1: subs r1, r1, r0 ; THUMB1: adcs r0, r1 ; THUMB2-LABEL: t3: ; THUMB2: clz r0, r0 ; THUMB2: lsrs r0, r0, #5 %0 = icmp eq i32 %a, 160 %1 = zext i1 %0 to i32 ret i32 %1 } define i32 @t4(i32 %a, i32 %b, i32 %x) nounwind { entry: ; ARM-LABEL: t4: ; ARM: ldr ; ARM: mov{{lt|ge}} ; ARMT2-LABEL: t4: ; ARMT2: movwlt [[R0:r[0-9]+]], #65365 ; ARMT2: movtlt [[R0]], #65365 ; THUMB1-LABEL: t4: ; THUMB1: cmp r{{[0-9]+}}, r{{[0-9]+}} ; THUMB1: b{{lt|ge}} ; THUMB2-LABEL: t4: ; THUMB2: mvnlt [[R0:r[0-9]+]], #11141290 %0 = icmp slt i32 %a, %b %1 = select i1 %0, i32 4283826005, i32 %x ret i32 %1 } ; rdar://9758317 define i32 @t5(i32 %a) nounwind { entry: ; ARM-LABEL: t5: ; ARM-NOT: mov ; ARM: sub r0, r0, #1 ; ARM-NOT: mov ; ARM: rsbs r1, r0, #0 ; ARM: adc r0, r0, r1 ; THUMB1-LABEL: t5: ; THUMB1-NOT: bne ; THUMB1: movs r0, #0 ; THUMB1: subs r0, r0, r1 ; THUMB1: adcs r0, r1 ; THUMB2-LABEL: t5: ; THUMB2-NOT: mov ; THUMB2: subs r0, #1 ; THUMB2: clz r0, r0 ; THUMB2: lsrs r0, r0, #5 %cmp = icmp eq i32 %a, 1 %conv = zext i1 %cmp to i32 ret i32 %conv } define i32 @t6(i32 %a) nounwind { entry: ; ARM-LABEL: t6: ; ARM-NOT: mov ; ARM: cmp r0, #0 ; ARM: movne r0, #1 ; THUMB1-LABEL: t6: ; THUMB1: cmp r{{[0-9]+}}, #0 ; THUMB1: bne ; THUMB2-LABEL: t6: ; THUMB2-NOT: mov ; THUMB2: cmp r0, #0 ; THUMB2: it ne ; THUMB2: movne r0, #1 %tobool = icmp ne i32 %a, 0 %lnot.ext = zext i1 %tobool to i32 ret i32 %lnot.ext } define i32 @t7(i32 %a, i32 %b) nounwind readnone { entry: ; ARM-LABEL: t7: ; ARM: subs r0, r0, r1 ; ARM: movne r0, #1 ; ARM: lsl r0, r0, #2 ; ARMT2-LABEL: t7: ; ARMT2: subs r0, r0, r1 ; ARMT2: movwne r0, #1 ; ARMT2: lsl r0, r0, #2 ; THUMB1-LABEL: t7: ; THUMB1: subs r0, r0, r1 ; THUMB1: subs r1, r0, #1 ; THUMB1: sbcs r0, r1 ; THUMB1: lsls r0, r0, #2 ; THUMB2-LABEL: t7: ; THUMB2: subs r0, r0, r1 ; THUMB2: it ne ; THUMB2: movne r0, #1 ; THUMB2: lsls r0, r0, #2 %0 = icmp ne i32 %a, %b %1 = select i1 %0, i32 4, i32 0 ret i32 %1 } define void @t8(i32 %a) { entry: ; ARM scheduler emits icmp/zext before both calls, so isn't relevant ; ARMT2-LABEL: t8: ; ARMT2: bl t7 ; ARMT2: mov r1, r0 ; ARMT2: sub r0, r4, #5 ; ARMT2: clz r0, r0 ; ARMT2: lsr r0, r0, #5 ; THUMB1-LABEL: t8: ; THUMB1: bl t7 ; THUMB1: mov r1, r0 ; THUMB1: subs r2, r4, #5 ; THUMB1: movs r0, #0 ; THUMB1: subs r0, r0, r2 ; THUMB1: adcs r0, r2 ; THUMB2-LABEL: t8: ; THUMB2: bl t7 ; THUMB2: mov r1, r0 ; THUMB2: subs r0, r4, #5 ; THUMB2: clz r0, r0 ; THUMB2: lsrs r0, r0, #5 %cmp = icmp eq i32 %a, 5 %conv = zext i1 %cmp to i32 %call = tail call i32 @t7(i32 9, i32 %a) tail call i32 @t7(i32 %conv, i32 %call) ret void } define void @t9(i8* %a, i8 %b) { entry: ; ARM scheduler emits icmp/zext before both calls, so isn't relevant ; ARMT2-LABEL: t9: ; ARMT2: bl f ; ARMT2: uxtb r0, r4 ; ARMT2: cmp r0, r0 ; ARMT2: add r1, r4, #1 ; ARMT2: mov r2, r0 ; ARMT2: add r2, r2, #1 ; ARMT2: add r1, r1, #1 ; ARMT2: uxtb r3, r2 ; ARMT2: cmp r3, r0 ; THUMB1-LABEL: t9: ; THUMB1: bl f ; THUMB1: sxtb r1, r4 ; THUMB1: uxtb r0, r1 ; THUMB1: cmp r0, r0 ; THUMB1: adds r1, r1, #1 ; THUMB1: mov r2, r0 ; THUMB1: adds r1, r1, #1 ; THUMB1: adds r2, r2, #1 ; THUMB1: uxtb r3, r2 ; THUMB1: cmp r3, r0 ; THUMB2-LABEL: t9: ; THUMB2: bl f ; THUMB2: uxtb r0, r4 ; THUMB2: cmp r0, r0 ; THUMB2: adds r1, r4, #1 ; THUMB2: mov r2, r0 ; THUMB2: adds r2, #1 ; THUMB2: adds r1, #1 ; THUMB2: uxtb r3, r2 ; THUMB2: cmp r3, r0 %0 = load i8, i8* %a %conv = sext i8 %0 to i32 %conv119 = zext i8 %0 to i32 %conv522 = and i32 %conv, 255 %cmp723 = icmp eq i32 %conv522, %conv119 tail call void @f(i1 zeroext %cmp723) br i1 %cmp723, label %while.body, label %while.end while.body: ; preds = %entry, %while.body %ref.025 = phi i8 [ %inc9, %while.body ], [ %0, %entry ] %in.024 = phi i32 [ %inc, %while.body ], [ %conv, %entry ] %inc = add i32 %in.024, 1 %inc9 = add i8 %ref.025, 1 %conv1 = zext i8 %inc9 to i32 %cmp = icmp slt i32 %conv1, %conv119 %conv5 = and i32 %inc, 255 br i1 %cmp, label %while.body, label %while.end while.end: ret void } declare void @f(i1 zeroext) define i1 @t10() { entry: %q = alloca i32 %p = alloca i32 store i32 -3, i32* %q store i32 -8, i32* %p %0 = load i32, i32* %q %1 = load i32, i32* %p %div = sdiv i32 %0, %1 %mul = mul nsw i32 %div, %1 %rem = srem i32 %0, %1 %add = add nsw i32 %mul, %rem %cmp = icmp eq i32 %add, %0 ret i1 %cmp ; ARM-LABEL: t10: ; ARM: rsbs r1, r0, #0 ; ARM: adc r0, r0, r1 ; ARMT2-LABEL: t10: ; ARMT2: clz r0, r0 ; ARMT2: lsr r0, r0, #5 ; THUMB1-LABEL: t10: ; THUMB1: movs r0, #0 ; THUMB1: subs r0, r0, r1 ; THUMB1: adcs r0, r1 ; THUMB2-LABEL: t10: ; THUMB2: clz r0, r0 ; THUMB2: lsrs r0, r0, #5 ; V8MBASE-LABEL: t10: ; V8MBASE-NOT: movs r0, #0 ; V8MBASE: movs r0, #7 } define i1 @t11() { entry: %bit = alloca i32 %load = load i32, i32* %bit %clear = and i32 %load, -4096 %set = or i32 %clear, 33 store i32 %set, i32* %bit %load1 = load i32, i32* %bit %clear2 = and i32 %load1, -33550337 %set3 = or i32 %clear2, 40960 %clear5 = and i32 %set3, 4095 %rem = srem i32 %clear5, 10 %clear9 = and i32 %set3, -4096 %set10 = or i32 %clear9, %rem store i32 %set10, i32* %bit %clear12 = and i32 %set10, 4095 %cmp = icmp eq i32 %clear12, 3 ret i1 %cmp ; ARM-LABEL: t11: ; ARM: rsbs r1, r0, #0 ; ARM: adc r0, r0, r1 ; ARMT2-LABEL: t11: ; ARMT2: clz r0, r0 ; ARMT2: lsr r0, r0, #5 ; THUMB1-LABEL: t11: ; THUMB1-NOT: movs r0, #0 ; THUMB1: movs r0, #5 ; THUMB2-LABEL: t11: ; THUMB2: clz r0, r0 ; THUMB2: lsrs r0, r0, #5 ; V8MBASE-LABEL: t11: ; V8MBASE-NOT: movs r0, #0 ; V8MBASE: movw r0, #40960 }