diff options
| author | Nick Kledzik <kledzik@apple.com> | 2010-07-27 06:24:32 +0000 |
|---|---|---|
| committer | Nick Kledzik <kledzik@apple.com> | 2010-07-27 06:24:32 +0000 |
| commit | cceb1f2e677ae46ed830072d04777b082e88190a (patch) | |
| tree | 8877fdf51bc9bf23e72d4b881ed3aa6309fa0c4a | |
| parent | 2c35bc1232d899c7dad2ca805f2986b4b79db350 (diff) | |
| download | bcm5719-llvm-cceb1f2e677ae46ed830072d04777b082e88190a.tar.gz bcm5719-llvm-cceb1f2e677ae46ed830072d04777b082e88190a.zip | |
add assembly implementation of modsi3 so compiler does not have to special case a - (a / b) * b optimization
llvm-svn: 109492
| -rw-r--r-- | compiler-rt/lib/arm/modsi3.S | 36 |
1 files changed, 36 insertions, 0 deletions
diff --git a/compiler-rt/lib/arm/modsi3.S b/compiler-rt/lib/arm/modsi3.S new file mode 100644 index 00000000000..64a3afd9aa0 --- /dev/null +++ b/compiler-rt/lib/arm/modsi3.S @@ -0,0 +1,36 @@ +//===-------- modsi3.S - Implement modsi3 ---------------------------------===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#include "../assembly.h" + +// +// extern int32_t __modsi3(int32_t a, int32_t b); +// +// Returns the remainder when dividing two 32-bit signed integers. +// Conceptually, the function is: { return a - (a / b) * b; } +// But if you write that in C, llvm compiles it to a call to __modsi3... +// + .align 2 +DEFINE_COMPILERRT_FUNCTION(__modsi3) + push {r4, r5, r7, lr} + add r7, sp, #8 // set stack frame + mov r5, r0 // save a + mov r4, r1 // save b + bl ___divsi3 // compute a/b +#if __ARM_ARCH_7A__ + mls r0, r4, r0, r5 // mulitple result * b and subtract from a +#else + // before armv7, does not have "mls" instruction + mul r3, r0, r4 // multiple result * b + sub r0, r5, r3 // a - result +#endif + pop {r4, r5, r7, pc} + + + |

