mirror of
https://github.com/RPCS3/llvm.git
synced 2025-05-16 18:35:53 +00:00

Summary: Made it convert from register to stack based instructions, and removed the registers. Fixes to related code that was expecting register based instructions. Added the correct testing flag to all tests, depending on what the format they were expecting so far. Translated one test to stack format as example: reg-stackify-stack.ll tested: llvm-lit -v `find test -name WebAssembly` unittests/MC/* Reviewers: dschuff, sunfish Subscribers: sbc100, jgravelle-google, eraman, aheejin, llvm-commits, jfb Differential Revision: https://reviews.llvm.org/D51241 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@340750 91177308-0d34-0410-b5e6-96231b3b80d8
389 lines
10 KiB
LLVM
389 lines
10 KiB
LLVM
; RUN: llc < %s -mattr=+atomics -asm-verbose=false -disable-wasm-fallthrough-return-opt -wasm-disable-explicit-locals -wasm-keep-registers | FileCheck %s
|
|
|
|
; Test loads and stores with custom alignment values.
|
|
|
|
target datalayout = "e-m:e-p:32:32-i64:64-n32:64-S128"
|
|
target triple = "wasm32-unknown-unknown"
|
|
|
|
;===----------------------------------------------------------------------------
|
|
; Loads
|
|
;===----------------------------------------------------------------------------
|
|
|
|
; CHECK-LABEL: ldi64_a1:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load $push[[NUM:[0-9]+]]=, 0($0):p2align=0{{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi64_a1(i64 *%p) {
|
|
%v = load i64, i64* %p, align 1
|
|
ret i64 %v
|
|
}
|
|
|
|
; CHECK-LABEL: ldi64_a2:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load $push[[NUM:[0-9]+]]=, 0($0):p2align=1{{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi64_a2(i64 *%p) {
|
|
%v = load i64, i64* %p, align 2
|
|
ret i64 %v
|
|
}
|
|
|
|
; CHECK-LABEL: ldi64_a4:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load $push[[NUM:[0-9]+]]=, 0($0):p2align=2{{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi64_a4(i64 *%p) {
|
|
%v = load i64, i64* %p, align 4
|
|
ret i64 %v
|
|
}
|
|
|
|
; 8 is the default alignment for i64 so no attribute is needed.
|
|
|
|
; CHECK-LABEL: ldi64_a8:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load $push[[NUM:[0-9]+]]=, 0($0){{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi64_a8(i64 *%p) {
|
|
%v = load i64, i64* %p, align 8
|
|
ret i64 %v
|
|
}
|
|
|
|
; The default alignment in LLVM is the same as the defualt alignment in wasm.
|
|
|
|
; CHECK-LABEL: ldi64:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load $push[[NUM:[0-9]+]]=, 0($0){{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi64(i64 *%p) {
|
|
%v = load i64, i64* %p
|
|
ret i64 %v
|
|
}
|
|
|
|
; 16 is greater than the default alignment so it is ignored.
|
|
|
|
; CHECK-LABEL: ldi64_a16:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load $push[[NUM:[0-9]+]]=, 0($0){{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi64_a16(i64 *%p) {
|
|
%v = load i64, i64* %p, align 16
|
|
ret i64 %v
|
|
}
|
|
|
|
;===----------------------------------------------------------------------------
|
|
; Extending loads
|
|
;===----------------------------------------------------------------------------
|
|
|
|
; CHECK-LABEL: ldi8_a1:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load8_u $push[[NUM:[0-9]+]]=, 0($0){{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi8_a1(i8 *%p) {
|
|
%v = load i8, i8* %p, align 1
|
|
%w = zext i8 %v to i64
|
|
ret i64 %w
|
|
}
|
|
|
|
; CHECK-LABEL: ldi8_a2:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load8_u $push[[NUM:[0-9]+]]=, 0($0){{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi8_a2(i8 *%p) {
|
|
%v = load i8, i8* %p, align 2
|
|
%w = zext i8 %v to i64
|
|
ret i64 %w
|
|
}
|
|
|
|
; CHECK-LABEL: ldi16_a1:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load16_u $push[[NUM:[0-9]+]]=, 0($0):p2align=0{{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi16_a1(i16 *%p) {
|
|
%v = load i16, i16* %p, align 1
|
|
%w = zext i16 %v to i64
|
|
ret i64 %w
|
|
}
|
|
|
|
; CHECK-LABEL: ldi16_a2:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load16_u $push[[NUM:[0-9]+]]=, 0($0){{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi16_a2(i16 *%p) {
|
|
%v = load i16, i16* %p, align 2
|
|
%w = zext i16 %v to i64
|
|
ret i64 %w
|
|
}
|
|
|
|
; CHECK-LABEL: ldi16_a4:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load16_u $push[[NUM:[0-9]+]]=, 0($0){{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi16_a4(i16 *%p) {
|
|
%v = load i16, i16* %p, align 4
|
|
%w = zext i16 %v to i64
|
|
ret i64 %w
|
|
}
|
|
|
|
; CHECK-LABEL: ldi32_a1:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load32_u $push[[NUM:[0-9]+]]=, 0($0):p2align=0{{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi32_a1(i32 *%p) {
|
|
%v = load i32, i32* %p, align 1
|
|
%w = zext i32 %v to i64
|
|
ret i64 %w
|
|
}
|
|
|
|
; CHECK-LABEL: ldi32_a2:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load32_u $push[[NUM:[0-9]+]]=, 0($0):p2align=1{{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi32_a2(i32 *%p) {
|
|
%v = load i32, i32* %p, align 2
|
|
%w = zext i32 %v to i64
|
|
ret i64 %w
|
|
}
|
|
|
|
; CHECK-LABEL: ldi32_a4:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load32_u $push[[NUM:[0-9]+]]=, 0($0){{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi32_a4(i32 *%p) {
|
|
%v = load i32, i32* %p, align 4
|
|
%w = zext i32 %v to i64
|
|
ret i64 %w
|
|
}
|
|
|
|
; CHECK-LABEL: ldi32_a8:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.load32_u $push[[NUM:[0-9]+]]=, 0($0){{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi32_a8(i32 *%p) {
|
|
%v = load i32, i32* %p, align 8
|
|
%w = zext i32 %v to i64
|
|
ret i64 %w
|
|
}
|
|
|
|
;===----------------------------------------------------------------------------
|
|
; Stores
|
|
;===----------------------------------------------------------------------------
|
|
|
|
; CHECK-LABEL: sti64_a1:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store 0($0):p2align=0, $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti64_a1(i64 *%p, i64 %v) {
|
|
store i64 %v, i64* %p, align 1
|
|
ret void
|
|
}
|
|
|
|
; CHECK-LABEL: sti64_a2:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store 0($0):p2align=1, $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti64_a2(i64 *%p, i64 %v) {
|
|
store i64 %v, i64* %p, align 2
|
|
ret void
|
|
}
|
|
|
|
; CHECK-LABEL: sti64_a4:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store 0($0):p2align=2, $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti64_a4(i64 *%p, i64 %v) {
|
|
store i64 %v, i64* %p, align 4
|
|
ret void
|
|
}
|
|
|
|
; 8 is the default alignment for i32 so no attribute is needed.
|
|
|
|
; CHECK-LABEL: sti64_a8:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store 0($0), $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti64_a8(i64 *%p, i64 %v) {
|
|
store i64 %v, i64* %p, align 8
|
|
ret void
|
|
}
|
|
|
|
; The default alignment in LLVM is the same as the defualt alignment in wasm.
|
|
|
|
; CHECK-LABEL: sti64:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store 0($0), $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti64(i64 *%p, i64 %v) {
|
|
store i64 %v, i64* %p
|
|
ret void
|
|
}
|
|
|
|
; CHECK-LABEL: sti64_a16:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store 0($0), $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti64_a16(i64 *%p, i64 %v) {
|
|
store i64 %v, i64* %p, align 16
|
|
ret void
|
|
}
|
|
|
|
;===----------------------------------------------------------------------------
|
|
; Truncating stores
|
|
;===----------------------------------------------------------------------------
|
|
|
|
; CHECK-LABEL: sti8_a1:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store8 0($0), $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti8_a1(i8 *%p, i64 %w) {
|
|
%v = trunc i64 %w to i8
|
|
store i8 %v, i8* %p, align 1
|
|
ret void
|
|
}
|
|
|
|
; CHECK-LABEL: sti8_a2:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store8 0($0), $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti8_a2(i8 *%p, i64 %w) {
|
|
%v = trunc i64 %w to i8
|
|
store i8 %v, i8* %p, align 2
|
|
ret void
|
|
}
|
|
|
|
; CHECK-LABEL: sti16_a1:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store16 0($0):p2align=0, $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti16_a1(i16 *%p, i64 %w) {
|
|
%v = trunc i64 %w to i16
|
|
store i16 %v, i16* %p, align 1
|
|
ret void
|
|
}
|
|
|
|
; CHECK-LABEL: sti16_a2:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store16 0($0), $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti16_a2(i16 *%p, i64 %w) {
|
|
%v = trunc i64 %w to i16
|
|
store i16 %v, i16* %p, align 2
|
|
ret void
|
|
}
|
|
|
|
; CHECK-LABEL: sti16_a4:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store16 0($0), $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti16_a4(i16 *%p, i64 %w) {
|
|
%v = trunc i64 %w to i16
|
|
store i16 %v, i16* %p, align 4
|
|
ret void
|
|
}
|
|
|
|
; CHECK-LABEL: sti32_a1:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store32 0($0):p2align=0, $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti32_a1(i32 *%p, i64 %w) {
|
|
%v = trunc i64 %w to i32
|
|
store i32 %v, i32* %p, align 1
|
|
ret void
|
|
}
|
|
|
|
; CHECK-LABEL: sti32_a2:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store32 0($0):p2align=1, $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti32_a2(i32 *%p, i64 %w) {
|
|
%v = trunc i64 %w to i32
|
|
store i32 %v, i32* %p, align 2
|
|
ret void
|
|
}
|
|
|
|
; CHECK-LABEL: sti32_a4:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store32 0($0), $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti32_a4(i32 *%p, i64 %w) {
|
|
%v = trunc i64 %w to i32
|
|
store i32 %v, i32* %p, align 4
|
|
ret void
|
|
}
|
|
|
|
; CHECK-LABEL: sti32_a8:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.store32 0($0), $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti32_a8(i32 *%p, i64 %w) {
|
|
%v = trunc i64 %w to i32
|
|
store i32 %v, i32* %p, align 8
|
|
ret void
|
|
}
|
|
|
|
;===----------------------------------------------------------------------------
|
|
; Atomic loads
|
|
;===----------------------------------------------------------------------------
|
|
|
|
; Wasm atomics have the alignment field, but it must always have the type's
|
|
; natural alignment.
|
|
|
|
; CHECK-LABEL: ldi64_atomic_a8:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.atomic.load $push[[NUM:[0-9]+]]=, 0($0){{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi64_atomic_a8(i64 *%p) {
|
|
%v = load atomic i64, i64* %p seq_cst, align 8
|
|
ret i64 %v
|
|
}
|
|
|
|
; 16 is greater than the default alignment so it is ignored.
|
|
|
|
; CHECK-LABEL: ldi64_atomic_a16:
|
|
; CHECK-NEXT: .param i32{{$}}
|
|
; CHECK-NEXT: .result i64{{$}}
|
|
; CHECK-NEXT: i64.atomic.load $push[[NUM:[0-9]+]]=, 0($0){{$}}
|
|
; CHECK-NEXT: return $pop[[NUM]]{{$}}
|
|
define i64 @ldi64_atomic_a16(i64 *%p) {
|
|
%v = load atomic i64, i64* %p seq_cst, align 16
|
|
ret i64 %v
|
|
}
|
|
|
|
;===----------------------------------------------------------------------------
|
|
; Atomic stores
|
|
;===----------------------------------------------------------------------------
|
|
|
|
; CHECK-LABEL: sti64_atomic_a4:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.atomic.store 0($0), $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti64_atomic_a4(i64 *%p, i64 %v) {
|
|
store atomic i64 %v, i64* %p seq_cst, align 8
|
|
ret void
|
|
}
|
|
|
|
; 16 is greater than the default alignment so it is ignored.
|
|
|
|
; CHECK-LABEL: sti64_atomic_a8:
|
|
; CHECK-NEXT: .param i32, i64{{$}}
|
|
; CHECK-NEXT: i64.atomic.store 0($0), $1{{$}}
|
|
; CHECK-NEXT: return{{$}}
|
|
define void @sti64_atomic_a8(i64 *%p, i64 %v) {
|
|
store atomic i64 %v, i64* %p seq_cst, align 16
|
|
ret void
|
|
}
|