// RUN: mlir-opt %s -split-input-file -pass-pipeline='builtin.module(func.func(canonicalize{test-convergence region-simplify=aggressive}))' | FileCheck %s
//===----------------------------------------------------------------------===//
// spirv.AccessChain
//===----------------------------------------------------------------------===//
func.func @combine_full_access_chain() -> f32 {
// CHECK: %[[INDEX:.*]] = spirv.Constant 0
// CHECK-NEXT: %[[VAR:.*]] = spirv.Variable
// CHECK-NEXT: %[[PTR:.*]] = spirv.AccessChain %[[VAR]][%[[INDEX]], %[[INDEX]], %[[INDEX]]]
// CHECK-NEXT: spirv.Load "Function" %[[PTR]]
%c0 = spirv.Constant 0: i32
%0 = spirv.Variable : !spirv.ptr<!spirv.struct<(!spirv.array<4x!spirv.array<4xf32>>, !spirv.array<4xi32>)>, Function>
%1 = spirv.AccessChain %0[%c0] : !spirv.ptr<!spirv.struct<(!spirv.array<4x!spirv.array<4xf32>>, !spirv.array<4xi32>)>, Function>, i32
%2 = spirv.AccessChain %1[%c0, %c0] : !spirv.ptr<!spirv.array<4x!spirv.array<4xf32>>, Function>, i32, i32
%3 = spirv.Load "Function" %2 : f32
spirv.ReturnValue %3 : f32
}
// -----
func.func @combine_access_chain_multi_use() -> !spirv.array<4xf32> {
// CHECK: %[[INDEX:.*]] = spirv.Constant 0
// CHECK-NEXT: %[[VAR:.*]] = spirv.Variable
// CHECK-NEXT: %[[PTR_0:.*]] = spirv.AccessChain %[[VAR]][%[[INDEX]], %[[INDEX]]]
// CHECK-NEXT: %[[PTR_1:.*]] = spirv.AccessChain %[[VAR]][%[[INDEX]], %[[INDEX]], %[[INDEX]]]
// CHECK-NEXT: spirv.Load "Function" %[[PTR_0]]
// CHECK-NEXT: spirv.Load "Function" %[[PTR_1]]
%c0 = spirv.Constant 0: i32
%0 = spirv.Variable : !spirv.ptr<!spirv.struct<(!spirv.array<4x!spirv.array<4xf32>>, !spirv.array<4xi32>)>, Function>
%1 = spirv.AccessChain %0[%c0] : !spirv.ptr<!spirv.struct<(!spirv.array<4x!spirv.array<4xf32>>, !spirv.array<4xi32>)>, Function>, i32
%2 = spirv.AccessChain %1[%c0] : !spirv.ptr<!spirv.array<4x!spirv.array<4xf32>>, Function>, i32
%3 = spirv.AccessChain %2[%c0] : !spirv.ptr<!spirv.array<4xf32>, Function>, i32
%4 = spirv.Load "Function" %2 : !spirv.array<4xf32>
%5 = spirv.Load "Function" %3 : f32
spirv.ReturnValue %4: !spirv.array<4xf32>
}
// -----
func.func @dont_combine_access_chain_without_common_base() -> !spirv.array<4xi32> {
// CHECK: %[[INDEX:.*]] = spirv.Constant 1
// CHECK-NEXT: %[[VAR_0:.*]] = spirv.Variable
// CHECK-NEXT: %[[VAR_1:.*]] = spirv.Variable
// CHECK-NEXT: %[[VAR_0_PTR:.*]] = spirv.AccessChain %[[VAR_0]][%[[INDEX]]]
// CHECK-NEXT: %[[VAR_1_PTR:.*]] = spirv.AccessChain %[[VAR_1]][%[[INDEX]]]
// CHECK-NEXT: spirv.Load "Function" %[[VAR_0_PTR]]
// CHECK-NEXT: spirv.Load "Function" %[[VAR_1_PTR]]
%c1 = spirv.Constant 1: i32
%0 = spirv.Variable : !spirv.ptr<!spirv.struct<(!spirv.array<4x!spirv.array<4xf32>>, !spirv.array<4xi32>)>, Function>
%1 = spirv.Variable : !spirv.ptr<!spirv.struct<(!spirv.array<4x!spirv.array<4xf32>>, !spirv.array<4xi32>)>, Function>
%2 = spirv.AccessChain %0[%c1] : !spirv.ptr<!spirv.struct<(!spirv.array<4x!spirv.array<4xf32>>, !spirv.array<4xi32>)>, Function>, i32
%3 = spirv.AccessChain %1[%c1] : !spirv.ptr<!spirv.struct<(!spirv.array<4x!spirv.array<4xf32>>, !spirv.array<4xi32>)>, Function>, i32
%4 = spirv.Load "Function" %2 : !spirv.array<4xi32>
%5 = spirv.Load "Function" %3 : !spirv.array<4xi32>
spirv.ReturnValue %4 : !spirv.array<4xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.Bitcast
//===----------------------------------------------------------------------===//
func.func @convert_bitcast_full(%arg0 : vector<2xf32>) -> f64 {
// CHECK: %[[RESULT:.*]] = spirv.Bitcast {{%.*}} : vector<2xf32> to f64
// CHECK-NEXT: spirv.ReturnValue %[[RESULT]]
%0 = spirv.Bitcast %arg0 : vector<2xf32> to vector<2xi32>
%1 = spirv.Bitcast %0 : vector<2xi32> to i64
%2 = spirv.Bitcast %1 : i64 to f64
spirv.ReturnValue %2 : f64
}
// -----
func.func @convert_bitcast_multi_use(%arg0 : vector<2xf32>, %arg1 : !spirv.ptr<i64, Uniform>) -> f64 {
// CHECK: %[[RESULT_0:.*]] = spirv.Bitcast {{%.*}} : vector<2xf32> to i64
// CHECK-NEXT: %[[RESULT_1:.*]] = spirv.Bitcast {{%.*}} : vector<2xf32> to f64
// CHECK-NEXT: spirv.Store {{".*"}} {{%.*}}, %[[RESULT_0]]
// CHECK-NEXT: spirv.ReturnValue %[[RESULT_1]]
%0 = spirv.Bitcast %arg0 : vector<2xf32> to i64
%1 = spirv.Bitcast %0 : i64 to f64
spirv.Store "Uniform" %arg1, %0 : i64
spirv.ReturnValue %1 : f64
}
// -----
// CHECK-LABEL: @convert_bitcast_roundtip
// CHECK-SAME: %[[ARG:.+]]: i64
func.func @convert_bitcast_roundtip(%arg0 : i64) -> i64 {
// CHECK: spirv.ReturnValue %[[ARG]]
%0 = spirv.Bitcast %arg0 : i64 to f64
%1 = spirv.Bitcast %0 : f64 to i64
spirv.ReturnValue %1 : i64
}
// -----
// CHECK-LABEL: @convert_bitcast_chained_roundtip
// CHECK-SAME: %[[ARG:.+]]: i64
func.func @convert_bitcast_chained_roundtip(%arg0 : i64) -> i64 {
// CHECK: spirv.ReturnValue %[[ARG]]
%0 = spirv.Bitcast %arg0 : i64 to f64
%1 = spirv.Bitcast %0 : f64 to vector<2xi32>
%2 = spirv.Bitcast %1 : vector<2xi32> to vector<2xf32>
%3 = spirv.Bitcast %2 : vector<2xf32> to i64
spirv.ReturnValue %3 : i64
}
// -----
//===----------------------------------------------------------------------===//
// spirv.CompositeExtract
//===----------------------------------------------------------------------===//
// CHECK-LABEL: extract_vector
func.func @extract_vector() -> (i32, i32, i32) {
// CHECK-DAG: spirv.Constant 6 : i32
// CHECK-DAG: spirv.Constant -33 : i32
// CHECK-DAG: spirv.Constant 42 : i32
%0 = spirv.Constant dense<[42, -33, 6]> : vector<3xi32>
%1 = spirv.CompositeExtract %0[0 : i32] : vector<3xi32>
%2 = spirv.CompositeExtract %0[1 : i32] : vector<3xi32>
%3 = spirv.CompositeExtract %0[2 : i32] : vector<3xi32>
return %1, %2, %3 : i32, i32, i32
}
// -----
// CHECK-LABEL: extract_array_final
func.func @extract_array_final() -> (i32, i32) {
// CHECK-DAG: spirv.Constant -5 : i32
// CHECK-DAG: spirv.Constant 4 : i32
%0 = spirv.Constant [dense<[4, -5]> : vector<2xi32>] : !spirv.array<1 x vector<2xi32>>
%1 = spirv.CompositeExtract %0[0 : i32, 0 : i32] : !spirv.array<1 x vector<2 x i32>>
%2 = spirv.CompositeExtract %0[0 : i32, 1 : i32] : !spirv.array<1 x vector<2 x i32>>
return %1, %2 : i32, i32
}
// -----
// CHECK-LABEL: extract_array_interm
func.func @extract_array_interm() -> (vector<2xi32>) {
// CHECK: spirv.Constant dense<[4, -5]> : vector<2xi32>
%0 = spirv.Constant [dense<[4, -5]> : vector<2xi32>] : !spirv.array<1 x vector<2xi32>>
%1 = spirv.CompositeExtract %0[0 : i32] : !spirv.array<1 x vector<2 x i32>>
return %1 : vector<2xi32>
}
// -----
// CHECK-LABEL: extract_from_not_constant
func.func @extract_from_not_constant() -> i32 {
%0 = spirv.Variable : !spirv.ptr<vector<3xi32>, Function>
%1 = spirv.Load "Function" %0 : vector<3xi32>
// CHECK: spirv.CompositeExtract
%2 = spirv.CompositeExtract %1[0 : i32] : vector<3xi32>
spirv.ReturnValue %2 : i32
}
// -----
// CHECK-LABEL: extract_insert
// CHECK-SAME: (%[[COMP:.+]]: !spirv.array<1 x vector<2xf32>>, %[[VAL:.+]]: f32)
func.func @extract_insert(%composite: !spirv.array<1xvector<2xf32>>, %val: f32) -> (f32, f32) {
// CHECK: %[[INSERT:.+]] = spirv.CompositeInsert %[[VAL]], %[[COMP]]
%insert = spirv.CompositeInsert %val, %composite[0 : i32, 1 : i32] : f32 into !spirv.array<1xvector<2xf32>>
%1 = spirv.CompositeExtract %insert[0 : i32, 0 : i32] : !spirv.array<1xvector<2xf32>>
// CHECK: %[[S:.+]] = spirv.CompositeExtract %[[INSERT]][0 : i32, 0 : i32]
%2 = spirv.CompositeExtract %insert[0 : i32, 1 : i32] : !spirv.array<1xvector<2xf32>>
// CHECK: return %[[S]], %[[VAL]]
return %1, %2 : f32, f32
}
// -----
// CHECK-LABEL: extract_construct
// CHECK-SAME: (%[[VAL1:.+]]: vector<2xf32>, %[[VAL2:.+]]: vector<2xf32>)
func.func @extract_construct(%val1: vector<2xf32>, %val2: vector<2xf32>) -> (vector<2xf32>, vector<2xf32>) {
%construct = spirv.CompositeConstruct %val1, %val2 : (vector<2xf32>, vector<2xf32>) -> !spirv.array<2xvector<2xf32>>
%1 = spirv.CompositeExtract %construct[0 : i32] : !spirv.array<2xvector<2xf32>>
%2 = spirv.CompositeExtract %construct[1 : i32] : !spirv.array<2xvector<2xf32>>
// CHECK: return %[[VAL1]], %[[VAL2]]
return %1, %2 : vector<2xf32>, vector<2xf32>
}
// -----
// CHECK-LABEL: fold_composite_op
// CHECK-SAME: (%[[COMP:.+]]: !spirv.struct<(f32, f32)>, %[[VAL1:.+]]: f32, %[[VAL2:.+]]: f32)
func.func @fold_composite_op(%composite: !spirv.struct<(f32, f32)>, %val1: f32, %val2: f32) -> f32 {
%insert = spirv.CompositeInsert %val1, %composite[0 : i32] : f32 into !spirv.struct<(f32, f32)>
%1 = spirv.CompositeInsert %val2, %insert[1 : i32] : f32 into !spirv.struct<(f32, f32)>
%2 = spirv.CompositeExtract %1[0 : i32] : !spirv.struct<(f32, f32)>
// CHECK-NEXT: return %[[VAL1]]
return %2 : f32
}
// -----
// CHECK-LABEL: fold_composite_op
// CHECK-SAME: (%[[VAL1:.+]]: f32, %[[VAL2:.+]]: f32, %[[VAL3:.+]]: f32)
func.func @fold_composite_op(%val1: f32, %val2: f32, %val3: f32) -> f32 {
%composite = spirv.CompositeConstruct %val1, %val1, %val1 : (f32, f32, f32) -> !spirv.struct<(f32, f32, f32)>
%insert = spirv.CompositeInsert %val2, %composite[1 : i32] : f32 into !spirv.struct<(f32, f32, f32)>
%1 = spirv.CompositeInsert %val3, %insert[2 : i32] : f32 into !spirv.struct<(f32, f32, f32)>
%2 = spirv.CompositeExtract %1[0 : i32] : !spirv.struct<(f32, f32, f32)>
// CHECK-NEXT: return %[[VAL1]]
return %2 : f32
}
// -----
// Not yet implemented case
// CHECK-LABEL: extract_construct
func.func @extract_construct(%val1: vector<3xf32>, %val2: f32) -> (f32, f32) {
// CHECK: spirv.CompositeConstruct
%construct = spirv.CompositeConstruct %val1, %val2 : (vector<3xf32>, f32) -> vector<4xf32>
// CHECK: spirv.CompositeExtract
%1 = spirv.CompositeExtract %construct[0 : i32] : vector<4xf32>
// CHECK: spirv.CompositeExtract
%2 = spirv.CompositeExtract %construct[1 : i32] : vector<4xf32>
return %1, %2 : f32, f32
}
// -----
//===----------------------------------------------------------------------===//
// spirv.Constant
//===----------------------------------------------------------------------===//
// TODO: test constants in different blocks
func.func @deduplicate_scalar_constant() -> (i32, i32) {
// CHECK: %[[CST:.*]] = spirv.Constant 42 : i32
%0 = spirv.Constant 42 : i32
%1 = spirv.Constant 42 : i32
// CHECK-NEXT: return %[[CST]], %[[CST]]
return %0, %1 : i32, i32
}
// -----
func.func @deduplicate_vector_constant() -> (vector<3xi32>, vector<3xi32>) {
// CHECK: %[[CST:.*]] = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
%0 = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
%1 = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
// CHECK-NEXT: return %[[CST]], %[[CST]]
return %0, %1 : vector<3xi32>, vector<3xi32>
}
// -----
func.func @deduplicate_composite_constant() -> (!spirv.array<1 x vector<2xi32>>, !spirv.array<1 x vector<2xi32>>) {
// CHECK: %[[CST:.*]] = spirv.Constant [dense<5> : vector<2xi32>] : !spirv.array<1 x vector<2xi32>>
%0 = spirv.Constant [dense<5> : vector<2xi32>] : !spirv.array<1 x vector<2xi32>>
%1 = spirv.Constant [dense<5> : vector<2xi32>] : !spirv.array<1 x vector<2xi32>>
// CHECK-NEXT: return %[[CST]], %[[CST]]
return %0, %1 : !spirv.array<1 x vector<2xi32>>, !spirv.array<1 x vector<2xi32>>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.IAdd
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @iadd_zero
// CHECK-SAME: (%[[ARG:.*]]: i32)
func.func @iadd_zero(%arg0: i32) -> (i32, i32) {
%zero = spirv.Constant 0 : i32
%0 = spirv.IAdd %arg0, %zero : i32
%1 = spirv.IAdd %zero, %arg0 : i32
// CHECK: return %[[ARG]], %[[ARG]]
return %0, %1: i32, i32
}
// CHECK-LABEL: @const_fold_scalar_iadd_normal
func.func @const_fold_scalar_iadd_normal() -> (i32, i32, i32) {
%c5 = spirv.Constant 5 : i32
%cn8 = spirv.Constant -8 : i32
// CHECK-DAG: spirv.Constant -3
// CHECK-DAG: spirv.Constant -16
// CHECK-DAG: spirv.Constant 10
%0 = spirv.IAdd %c5, %c5 : i32
%1 = spirv.IAdd %cn8, %cn8 : i32
%2 = spirv.IAdd %c5, %cn8 : i32
return %0, %1, %2: i32, i32, i32
}
// CHECK-LABEL: @const_fold_scalar_iadd_flow
func.func @const_fold_scalar_iadd_flow() -> (i32, i32, i32, i32) {
%c1 = spirv.Constant 1 : i32
%c2 = spirv.Constant 2 : i32
%c3 = spirv.Constant 4294967295 : i32 // 2^32 - 1: 0xffff ffff
%c4 = spirv.Constant -2147483648 : i32 // -2^31 : 0x8000 0000
%c5 = spirv.Constant -1 : i32 // : 0xffff ffff
%c6 = spirv.Constant -2 : i32 // : 0xffff fffe
// 0x8000 0000 + 0xffff fffe = 0x1 7fff fffe -> 0x7fff fffe
// CHECK-DAG: spirv.Constant 2147483646
// 0x8000 0000 + 0xffff ffff = 0x1 7fff ffff -> 0x7fff ffff
// CHECK-DAG: spirv.Constant 2147483647
// 0x0000 0002 + 0xffff ffff = 0x1 0000 0001 -> 0x0000 0001
// CHECK-DAG: spirv.Constant 1
// 0x0000 0001 + 0xffff ffff = 0x1 0000 0000 -> 0x0000 0000
// CHECK-DAG: spirv.Constant 0
%0 = spirv.IAdd %c1, %c3 : i32
%1 = spirv.IAdd %c2, %c3 : i32
%2 = spirv.IAdd %c4, %c5 : i32
%3 = spirv.IAdd %c4, %c6 : i32
return %0, %1, %2, %3: i32, i32, i32, i32
}
// CHECK-LABEL: @const_fold_vector_iadd
func.func @const_fold_vector_iadd() -> vector<3xi32> {
%vc1 = spirv.Constant dense<[42, -55, 127]> : vector<3xi32>
%vc2 = spirv.Constant dense<[-3, -15, 28]> : vector<3xi32>
// CHECK: spirv.Constant dense<[39, -70, 155]>
%0 = spirv.IAdd %vc1, %vc2 : vector<3xi32>
return %0: vector<3xi32>
}
// CHECK-LABEL: @iadd_poison
// CHECK: %[[P:.*]] = ub.poison : i32
// CHECK: return %[[P]]
func.func @iadd_poison(%arg0: i32) -> i32 {
%0 = ub.poison : i32
%1 = spirv.IAdd %arg0, %0 : i32
return %1: i32
}
// -----
//===----------------------------------------------------------------------===//
// spirv.IAddCarry
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @iaddcarry_x_0
func.func @iaddcarry_x_0(%arg0 : i32) -> !spirv.struct<(i32, i32)> {
// CHECK: %[[RET:.*]] = spirv.CompositeConstruct
%c0 = spirv.Constant 0 : i32
%0 = spirv.IAddCarry %arg0, %c0 : !spirv.struct<(i32, i32)>
// CHECK: return %[[RET]]
return %0 : !spirv.struct<(i32, i32)>
}
// CHECK-LABEL: @const_fold_scalar_iaddcarry
func.func @const_fold_scalar_iaddcarry() -> (!spirv.struct<(i32, i32)>, !spirv.struct<(i32, i32)>) {
%c5 = spirv.Constant 5 : i32
%cn5 = spirv.Constant -5 : i32
%cn8 = spirv.Constant -8 : i32
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[CN3:.*]] = spirv.Constant -3
// CHECK-DAG: %[[UNDEF1:.*]] = spirv.Undef
// CHECK-DAG: %[[INTER1:.*]] = spirv.CompositeInsert %[[CN3]], %[[UNDEF1]][0 : i32]
// CHECK-DAG: %[[CC_CN3_C0:.*]] = spirv.CompositeInsert %[[C0]], %[[INTER1]][1 : i32]
// CHECK-DAG: %[[C1:.*]] = spirv.Constant 1
// CHECK-DAG: %[[CN13:.*]] = spirv.Constant -13
// CHECK-DAG: %[[UNDEF2:.*]] = spirv.Undef
// CHECK-DAG: %[[INTER2:.*]] = spirv.CompositeInsert %[[CN13]], %[[UNDEF2]][0 : i32]
// CHECK-DAG: %[[CC_CN13_C1:.*]] = spirv.CompositeInsert %[[C1]], %[[INTER2]][1 : i32]
%0 = spirv.IAddCarry %c5, %cn8 : !spirv.struct<(i32, i32)>
%1 = spirv.IAddCarry %cn5, %cn8 : !spirv.struct<(i32, i32)>
// CHECK: return %[[CC_CN3_C0]], %[[CC_CN13_C1]]
return %0, %1 : !spirv.struct<(i32, i32)>, !spirv.struct<(i32, i32)>
}
// CHECK-LABEL: @const_fold_vector_iaddcarry
func.func @const_fold_vector_iaddcarry() -> !spirv.struct<(vector<3xi32>, vector<3xi32>)> {
%v0 = spirv.Constant dense<[5, -3, -1]> : vector<3xi32>
%v1 = spirv.Constant dense<[-8, -8, 1]> : vector<3xi32>
// CHECK-DAG: %[[CV1:.*]] = spirv.Constant dense<[-3, -11, 0]>
// CHECK-DAG: %[[CV2:.*]] = spirv.Constant dense<[0, 1, 1]>
// CHECK-DAG: %[[UNDEF:.*]] = spirv.Undef
// CHECK-DAG: %[[INTER:.*]] = spirv.CompositeInsert %[[CV1]], %[[UNDEF]][0 : i32]
// CHECK-DAG: %[[CC_CV1_CV2:.*]] = spirv.CompositeInsert %[[CV2]], %[[INTER]][1 : i32]
%0 = spirv.IAddCarry %v0, %v1 : !spirv.struct<(vector<3xi32>, vector<3xi32>)>
// CHECK: return %[[CC_CV1_CV2]]
return %0 : !spirv.struct<(vector<3xi32>, vector<3xi32>)>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.IMul
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @imul_zero_one
// CHECK-SAME: (%[[ARG:.*]]: i32)
func.func @imul_zero_one(%arg0: i32) -> (i32, i32) {
// CHECK: %[[ZERO:.*]] = spirv.Constant 0
%zero = spirv.Constant 0 : i32
%one = spirv.Constant 1: i32
%0 = spirv.IMul %arg0, %zero : i32
%1 = spirv.IMul %one, %arg0 : i32
// CHECK: return %[[ZERO]], %[[ARG]]
return %0, %1: i32, i32
}
// CHECK-LABEL: @const_fold_scalar_imul_normal
func.func @const_fold_scalar_imul_normal() -> (i32, i32, i32) {
%c5 = spirv.Constant 5 : i32
%cn8 = spirv.Constant -8 : i32
%c7 = spirv.Constant 7 : i32
// CHECK-DAG: spirv.Constant -56
// CHECK-DAG: spirv.Constant -40
// CHECK-DAG: spirv.Constant 35
%0 = spirv.IMul %c7, %c5 : i32
%1 = spirv.IMul %c5, %cn8 : i32
%2 = spirv.IMul %cn8, %c7 : i32
return %0, %1, %2: i32, i32, i32
}
// CHECK-LABEL: @const_fold_scalar_imul_flow
func.func @const_fold_scalar_imul_flow() -> (i32, i32, i32) {
%c1 = spirv.Constant 2 : i32
%c2 = spirv.Constant 4 : i32
%c3 = spirv.Constant 4294967295 : i32 // 2^32 - 1 : 0xffff ffff
%c4 = spirv.Constant 2147483647 : i32 // 2^31 - 1 : 0x7fff ffff
// (0x7fff ffff << 2) = 0x1 ffff fffc -> 0xffff fffc
// CHECK-DAG: %[[CST4:.*]] = spirv.Constant -4
// (0xffff ffff << 1) = 0x1 ffff fffe -> 0xffff fffe
// CHECK-DAG: %[[CST2:.*]] = spirv.Constant -2
%0 = spirv.IMul %c1, %c3 : i32
// (0x7fff ffff << 1) = 0x0 ffff fffe -> 0xffff fffe
%1 = spirv.IMul %c1, %c4 : i32
%2 = spirv.IMul %c4, %c2 : i32
// CHECK: return %[[CST2]], %[[CST2]], %[[CST4]]
return %0, %1, %2: i32, i32, i32
}
// CHECK-LABEL: @const_fold_vector_imul
func.func @const_fold_vector_imul() -> vector<3xi32> {
%vc1 = spirv.Constant dense<[42, -55, 127]> : vector<3xi32>
%vc2 = spirv.Constant dense<[-3, -15, 28]> : vector<3xi32>
// CHECK: spirv.Constant dense<[-126, 825, 3556]>
%0 = spirv.IMul %vc1, %vc2 : vector<3xi32>
return %0: vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.SMulExtended
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @smulextended_x_0
func.func @smulextended_x_0(%arg0 : i32) -> !spirv.struct<(i32, i32)> {
// CHECK: %[[C0:.*]] = spirv.Constant 0
// CHECK: %[[RET:.*]] = spirv.CompositeConstruct %[[C0]], %[[C0]]
%c0 = spirv.Constant 0 : i32
%0 = spirv.SMulExtended %arg0, %c0 : !spirv.struct<(i32, i32)>
// CHECK: return %[[RET]]
return %0 : !spirv.struct<(i32, i32)>
}
// CHECK-LABEL: @const_fold_scalar_smulextended
func.func @const_fold_scalar_smulextended() -> (!spirv.struct<(i32, i32)>, !spirv.struct<(i32, i32)>) {
%c5 = spirv.Constant 5 : i32
%cn5 = spirv.Constant -5 : i32
%cn8 = spirv.Constant -8 : i32
// CHECK-DAG: %[[CN40:.*]] = spirv.Constant -40
// CHECK-DAG: %[[CN1:.*]] = spirv.Constant -1
// CHECK-DAG: %[[UNDEF1:.*]] = spirv.Undef
// CHECK-DAG: %[[INTER1:.*]] = spirv.CompositeInsert %[[CN40]], %[[UNDEF1]][0 : i32]
// CHECK-DAG: %[[CC_CN40_CN1:.*]] = spirv.CompositeInsert %[[CN1]], %[[INTER1]]
// CHECK-DAG: %[[C40:.*]] = spirv.Constant 40
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[UNDEF2:.*]] = spirv.Undef
// CHECK-DAG: %[[INTER2:.*]] = spirv.CompositeInsert %[[C40]], %[[UNDEF2]][0 : i32]
// CHECK-DAG: %[[CC_C40_C0:.*]] = spirv.CompositeInsert %[[C0]], %[[INTER2]][1 : i32]
%0 = spirv.SMulExtended %c5, %cn8 : !spirv.struct<(i32, i32)>
%1 = spirv.SMulExtended %cn5, %cn8 : !spirv.struct<(i32, i32)>
// CHECK: return %[[CC_CN40_CN1]], %[[CC_C40_C0]]
return %0, %1 : !spirv.struct<(i32, i32)>, !spirv.struct<(i32, i32)>
}
// CHECK-LABEL: @const_fold_vector_smulextended
func.func @const_fold_vector_smulextended() -> !spirv.struct<(vector<3xi32>, vector<3xi32>)> {
%v0 = spirv.Constant dense<[2147483647, -5, -1]> : vector<3xi32>
%v1 = spirv.Constant dense<[5, -8, 1]> : vector<3xi32>
// CHECK-DAG: %[[CV1:.*]] = spirv.Constant dense<[2147483643, 40, -1]>
// CHECK-DAG: %[[CV2:.*]] = spirv.Constant dense<[2, 0, -1]>
// CHECK-DAG: %[[UNDEF:.*]] = spirv.Undef
// CHECK-DAG: %[[INTER:.*]] = spirv.CompositeInsert %[[CV1]], %[[UNDEF]][0 : i32]
// CHECK-DAG: %[[CC_CV1_CV2:.*]] = spirv.CompositeInsert %[[CV2]], %[[INTER]][1 : i32]
%0 = spirv.SMulExtended %v0, %v1 : !spirv.struct<(vector<3xi32>, vector<3xi32>)>
// CHECK: return %[[CC_CV1_CV2]]
return %0 : !spirv.struct<(vector<3xi32>, vector<3xi32>)>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.UMulExtended
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @umulextended_x_0
func.func @umulextended_x_0(%arg0 : i32) -> !spirv.struct<(i32, i32)> {
// CHECK: %[[C0:.*]] = spirv.Constant 0
// CHECK: %[[RET:.*]] = spirv.CompositeConstruct %[[C0]], %[[C0]]
%c0 = spirv.Constant 0 : i32
%0 = spirv.UMulExtended %arg0, %c0 : !spirv.struct<(i32, i32)>
// CHECK: return %[[RET]]
return %0 : !spirv.struct<(i32, i32)>
}
// CHECK-LABEL: @umulextended_x_1
// CHECK-SAME: (%[[ARG:.*]]: i32)
func.func @umulextended_x_1(%arg0 : i32) -> !spirv.struct<(i32, i32)> {
// CHECK: %[[C0:.*]] = spirv.Constant 0
// CHECK: %[[RET:.*]] = spirv.CompositeConstruct %[[ARG]], %[[C0]]
%c0 = spirv.Constant 1 : i32
%0 = spirv.UMulExtended %arg0, %c0 : !spirv.struct<(i32, i32)>
// CHECK: return %[[RET]]
return %0 : !spirv.struct<(i32, i32)>
}
// CHECK-LABEL: @const_fold_scalar_umulextended
func.func @const_fold_scalar_umulextended() -> (!spirv.struct<(i32, i32)>, !spirv.struct<(i32, i32)>) {
%c5 = spirv.Constant 5 : i32
%cn5 = spirv.Constant -5 : i32
%cn8 = spirv.Constant -8 : i32
// CHECK-DAG: %[[C40:.*]] = spirv.Constant 40
// CHECK-DAG: %[[CN13:.*]] = spirv.Constant -13
// CHECK-DAG: %[[CN40:.*]] = spirv.Constant -40
// CHECK-DAG: %[[C4:.*]] = spirv.Constant 4
// CHECK-DAG: %[[UNDEF1:.*]] = spirv.Undef
// CHECK-DAG: %[[INTER1:.*]] = spirv.CompositeInsert %[[CN40]], %[[UNDEF1]][0 : i32]
// CHECK-DAG: %[[CC_CN40_C4:.*]] = spirv.CompositeInsert %[[C4]], %[[INTER1]][1 : i32]
// CHECK-DAG: %[[UNDEF2:.*]] = spirv.Undef
// CHECK-DAG: %[[INTER2:.*]] = spirv.CompositeInsert %[[C40]], %[[UNDEF2]][0 : i32]
// CHECK-DAG: %[[CC_C40_CN13:.*]] = spirv.CompositeInsert %[[CN13]], %[[INTER2]][1 : i32]
%0 = spirv.UMulExtended %c5, %cn8 : !spirv.struct<(i32, i32)>
%1 = spirv.UMulExtended %cn5, %cn8 : !spirv.struct<(i32, i32)>
// CHECK: return %[[CC_CN40_C4]], %[[CC_C40_CN13]]
return %0, %1 : !spirv.struct<(i32, i32)>, !spirv.struct<(i32, i32)>
}
// CHECK-LABEL: @const_fold_vector_umulextended
func.func @const_fold_vector_umulextended() -> !spirv.struct<(vector<3xi32>, vector<3xi32>)> {
%v0 = spirv.Constant dense<[2147483647, -5, -1]> : vector<3xi32>
%v1 = spirv.Constant dense<[5, -8, 1]> : vector<3xi32>
// CHECK-DAG: %[[CV1:.*]] = spirv.Constant dense<[2147483643, 40, -1]>
// CHECK-DAG: %[[CV2:.*]] = spirv.Constant dense<[2, -13, 0]>
// CHECK-DAG: %[[UNDEF:.*]] = spirv.Undef
// CHECK-DAG: %[[INTER:.*]] = spirv.CompositeInsert %[[CV1]], %[[UNDEF]]
// CHECK-DAG: %[[CC_CV1_CV2:.*]] = spirv.CompositeInsert %[[CV2]], %[[INTER]]
%0 = spirv.UMulExtended %v0, %v1 : !spirv.struct<(vector<3xi32>, vector<3xi32>)>
// CHECK: return %[[CC_CV1_CV2]]
return %0 : !spirv.struct<(vector<3xi32>, vector<3xi32>)>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.ISub
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @isub_x_x
func.func @isub_x_x(%arg0: i32) -> i32 {
// CHECK: spirv.Constant 0
%0 = spirv.ISub %arg0, %arg0: i32
return %0: i32
}
// CHECK-LABEL: @const_fold_scalar_isub_normal
func.func @const_fold_scalar_isub_normal() -> (i32, i32, i32) {
%c5 = spirv.Constant 5 : i32
%cn8 = spirv.Constant -8 : i32
%c7 = spirv.Constant 7 : i32
// CHECK-DAG: spirv.Constant -15
// CHECK-DAG: spirv.Constant 13
// CHECK-DAG: spirv.Constant 2
%0 = spirv.ISub %c7, %c5 : i32
%1 = spirv.ISub %c5, %cn8 : i32
%2 = spirv.ISub %cn8, %c7 : i32
return %0, %1, %2: i32, i32, i32
}
// CHECK-LABEL: @const_fold_scalar_isub_flow
func.func @const_fold_scalar_isub_flow() -> (i32, i32, i32, i32) {
%c1 = spirv.Constant 0 : i32
%c2 = spirv.Constant 1 : i32
%c3 = spirv.Constant 4294967295 : i32 // 2^32 - 1 : 0xffff ffff
%c4 = spirv.Constant 2147483647 : i32 // 2^31 : 0x7fff ffff
%c5 = spirv.Constant -1 : i32 // : 0xffff ffff
%c6 = spirv.Constant -2 : i32 // : 0xffff fffe
// 0xffff ffff - 0x7fff ffff -> 0xffff ffff + 0x8000 0001 = 0x1 8000 0000
// CHECK-DAG: spirv.Constant -2147483648
// 0x0000 0001 - 0xffff ffff -> 0x0000 0001 + 0x0000 0001 = 0x0000 0002
// CHECK-DAG: spirv.Constant 2 :
// 0x0000 0000 - 0xffff ffff -> 0x0000 0000 + 0x0000 0001 = 0x0000 0001
// CHECK-DAG: spirv.Constant 1 :
// 0xffff fffe - 0x7fff ffff -> 0xffff fffe + 0x8000 0001 = 0x1 7fff ffff
// CHECK-DAG: spirv.Constant 2147483647
%0 = spirv.ISub %c1, %c3 : i32
%1 = spirv.ISub %c2, %c3 : i32
%2 = spirv.ISub %c5, %c4 : i32
%3 = spirv.ISub %c6, %c4 : i32
return %0, %1, %2, %3: i32, i32, i32, i32
}
// CHECK-LABEL: @const_fold_vector_isub
func.func @const_fold_vector_isub() -> vector<3xi32> {
%vc1 = spirv.Constant dense<[42, -55, 127]> : vector<3xi32>
%vc2 = spirv.Constant dense<[-3, -15, 28]> : vector<3xi32>
// CHECK: spirv.Constant dense<[45, -40, 99]>
%0 = spirv.ISub %vc1, %vc2 : vector<3xi32>
return %0: vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.SDiv
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @sdiv_x_1
func.func @sdiv_x_1(%arg0 : i32) -> i32 {
// CHECK-NEXT: return %arg0 : i32
%c1 = spirv.Constant 1 : i32
%2 = spirv.SDiv %arg0, %c1: i32
return %2 : i32
}
// CHECK-LABEL: @sdiv_div_0_or_overflow
func.func @sdiv_div_0_or_overflow() -> (i32, i32) {
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[CN1:.*]] = spirv.Constant -1
// CHECK-DAG: %[[CNMIN:.*]] = spirv.Constant -2147483648
%c0 = spirv.Constant 0 : i32
%cn1 = spirv.Constant -1 : i32
%min_i32 = spirv.Constant -2147483648 : i32
// CHECK: %0 = spirv.SDiv %[[CN1]], %[[C0]]
// CHECK: %1 = spirv.SDiv %[[CNMIN]], %[[CN1]]
%0 = spirv.SDiv %cn1, %c0 : i32
%1 = spirv.SDiv %min_i32, %cn1 : i32
return %0, %1 : i32, i32
}
// CHECK-LABEL: @const_fold_scalar_sdiv
func.func @const_fold_scalar_sdiv() -> (i32, i32, i32, i32) {
%c56 = spirv.Constant 56 : i32
%c7 = spirv.Constant 7 : i32
%cn8 = spirv.Constant -8 : i32
%c3 = spirv.Constant 3 : i32
%cn3 = spirv.Constant -3 : i32
// CHECK-DAG: %[[CN18:.*]] = spirv.Constant -18
// CHECK-DAG: %[[CN2:.*]] = spirv.Constant -2
// CHECK-DAG: %[[CN7:.*]] = spirv.Constant -7
// CHECK-DAG: %[[C8:.*]] = spirv.Constant 8
%0 = spirv.SDiv %c56, %c7 : i32
%1 = spirv.SDiv %c56, %cn8 : i32
%2 = spirv.SDiv %cn8, %c3 : i32
%3 = spirv.SDiv %c56, %cn3 : i32
// CHECK: return %[[C8]], %[[CN7]], %[[CN2]], %[[CN18]]
return %0, %1, %2, %3: i32, i32, i32, i32
}
// CHECK-LABEL: @const_fold_vector_sdiv
func.func @const_fold_vector_sdiv() -> vector<3xi32> {
// CHECK: %[[CVEC:.*]] = spirv.Constant dense<[0, -1, -3]>
%cv_num = spirv.Constant dense<[42, 24, -16]> : vector<3xi32>
%cv_denom = spirv.Constant dense<[76, -24, 5]> : vector<3xi32>
%0 = spirv.SDiv %cv_num, %cv_denom : vector<3xi32>
// CHECK: return %[[CVEC]]
return %0 : vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.SMod
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @smod_x_1
func.func @smod_x_1(%arg0: i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[CVEC0:.*]] = spirv.Constant dense<0>
%c1 = spirv.Constant 1 : i32
%cv1 = spirv.Constant dense<1> : vector<3xi32>
%0 = spirv.SMod %arg0, %c1: i32
%1 = spirv.SMod %arg1, %cv1: vector<3xi32>
// CHECK: return %[[C0]], %[[CVEC0]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @smod_div_0_or_overflow
func.func @smod_div_0_or_overflow() -> (i32, i32) {
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[CN1:.*]] = spirv.Constant -1
// CHECK-DAG: %[[CNMIN:.*]] = spirv.Constant -2147483648
%c0 = spirv.Constant 0 : i32
%cn1 = spirv.Constant -1 : i32
%min_i32 = spirv.Constant -2147483648 : i32
// CHECK: %0 = spirv.SMod %[[CN1]], %[[C0]]
// CHECK: %1 = spirv.SMod %[[CNMIN]], %[[CN1]]
%0 = spirv.SMod %cn1, %c0 : i32
%1 = spirv.SMod %min_i32, %cn1 : i32
return %0, %1 : i32, i32
}
// CHECK-LABEL: @const_fold_scalar_smod
func.func @const_fold_scalar_smod() -> (i32, i32, i32, i32, i32, i32, i32, i32) {
%c56 = spirv.Constant 56 : i32
%cn56 = spirv.Constant -56 : i32
%c59 = spirv.Constant 59 : i32
%cn59 = spirv.Constant -59 : i32
%c7 = spirv.Constant 7 : i32
%cn8 = spirv.Constant -8 : i32
%c3 = spirv.Constant 3 : i32
%cn3 = spirv.Constant -3 : i32
// CHECK-DAG: %[[ZERO:.*]] = spirv.Constant 0 : i32
// CHECK-DAG: %[[TWO:.*]] = spirv.Constant 2 : i32
// CHECK-DAG: %[[FIFTYTHREE:.*]] = spirv.Constant 53 : i32
// CHECK-DAG: %[[NFIFTYTHREE:.*]] = spirv.Constant -53 : i32
// CHECK-DAG: %[[THREE:.*]] = spirv.Constant 3 : i32
// CHECK-DAG: %[[NTHREE:.*]] = spirv.Constant -3 : i32
%0 = spirv.SMod %c56, %c7 : i32
%1 = spirv.SMod %c56, %cn8 : i32
%2 = spirv.SMod %c56, %c3 : i32
%3 = spirv.SMod %cn3, %c56 : i32
%4 = spirv.SMod %cn3, %cn56 : i32
%5 = spirv.SMod %c59, %c56 : i32
%6 = spirv.SMod %c59, %cn56 : i32
%7 = spirv.SMod %cn59, %cn56 : i32
// CHECK: return %[[ZERO]], %[[ZERO]], %[[TWO]], %[[FIFTYTHREE]], %[[NTHREE]], %[[THREE]], %[[NFIFTYTHREE]], %[[NTHREE]]
return %0, %1, %2, %3, %4, %5, %6, %7 : i32, i32, i32, i32, i32, i32, i32, i32
}
// CHECK-LABEL: @const_fold_vector_smod
func.func @const_fold_vector_smod() -> vector<3xi32> {
// CHECK: %[[CVEC:.*]] = spirv.Constant dense<[42, -4, 4]>
%cv = spirv.Constant dense<[42, 24, -16]> : vector<3xi32>
%cv_mod = spirv.Constant dense<[76, -7, 5]> : vector<3xi32>
%0 = spirv.SMod %cv, %cv_mod : vector<3xi32>
// CHECK: return %[[CVEC]]
return %0 : vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.SRem
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @srem_x_1
func.func @srem_x_1(%arg0: i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[CVEC0:.*]] = spirv.Constant dense<0>
%c1 = spirv.Constant 1 : i32
%cv1 = spirv.Constant dense<1> : vector<3xi32>
%0 = spirv.SRem %arg0, %c1: i32
%1 = spirv.SRem %arg1, %cv1: vector<3xi32>
// CHECK: return %[[C0]], %[[CVEC0]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @srem_div_0_or_overflow
func.func @srem_div_0_or_overflow() -> (i32, i32) {
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[CN1:.*]] = spirv.Constant -1
// CHECK-DAG: %[[CNMIN:.*]] = spirv.Constant -2147483648
%c0 = spirv.Constant 0 : i32
%cn1 = spirv.Constant -1 : i32
%min_i32 = spirv.Constant -2147483648 : i32
// CHECK: %0 = spirv.SRem %[[CN1]], %[[C0]]
// CHECK: %1 = spirv.SRem %[[CNMIN]], %[[CN1]]
%0 = spirv.SRem %cn1, %c0 : i32
%1 = spirv.SRem %min_i32, %cn1 : i32
return %0, %1 : i32, i32
}
// CHECK-LABEL: @const_fold_scalar_srem
func.func @const_fold_scalar_srem() -> (i32, i32, i32, i32, i32) {
%c56 = spirv.Constant 56 : i32
%c7 = spirv.Constant 7 : i32
%cn8 = spirv.Constant -8 : i32
%c3 = spirv.Constant 3 : i32
%cn3 = spirv.Constant -3 : i32
// CHECK-DAG: %[[ONE:.*]] = spirv.Constant 1 : i32
// CHECK-DAG: %[[NTHREE:.*]] = spirv.Constant -3 : i32
// CHECK-DAG: %[[TWO:.*]] = spirv.Constant 2 : i32
// CHECK-DAG: %[[ZERO:.*]] = spirv.Constant 0 : i32
%0 = spirv.SRem %c56, %c7 : i32
%1 = spirv.SRem %c56, %cn8 : i32
%2 = spirv.SRem %c56, %c3 : i32
%3 = spirv.SRem %cn3, %c56 : i32
%4 = spirv.SRem %c7, %cn3 : i32
// CHECK: return %[[ZERO]], %[[ZERO]], %[[TWO]], %[[NTHREE]], %[[ONE]]
return %0, %1, %2, %3, %4 : i32, i32, i32, i32, i32
}
// -----
//===----------------------------------------------------------------------===//
// spirv.UDiv
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @udiv_x_1
func.func @udiv_x_1(%arg0 : i32) -> i32 {
// CHECK-NEXT: return %arg0 : i32
%c1 = spirv.Constant 1 : i32
%2 = spirv.UDiv %arg0, %c1: i32
return %2 : i32
}
// CHECK-LABEL: @udiv_div_0
func.func @udiv_div_0() -> i32 {
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[CN1:.*]] = spirv.Constant -1
%c0 = spirv.Constant 0 : i32
%cn1 = spirv.Constant -1 : i32
// CHECK: %0 = spirv.UDiv %[[CN1]], %[[C0]]
%0 = spirv.UDiv %cn1, %c0 : i32
return %0 : i32
}
// CHECK-LABEL: @const_fold_scalar_udiv
func.func @const_fold_scalar_udiv() -> (i32, i32, i32) {
%c56 = spirv.Constant 56 : i32
%c7 = spirv.Constant 7 : i32
%cn8 = spirv.Constant -8 : i32
%c3 = spirv.Constant 3 : i32
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[CBIG:.*]] = spirv.Constant 1431655762
// CHECK-DAG: %[[C8:.*]] = spirv.Constant 8
%0 = spirv.UDiv %c56, %c7 : i32
%1 = spirv.UDiv %cn8, %c3 : i32
%2 = spirv.UDiv %c56, %cn8 : i32
// CHECK: return %[[C8]], %[[CBIG]], %[[C0]]
return %0, %1, %2 : i32, i32, i32
}
// -----
//===----------------------------------------------------------------------===//
// spirv.UMod
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @umod_x_1
func.func @umod_x_1(%arg0: i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[CVEC0:.*]] = spirv.Constant dense<0>
%c1 = spirv.Constant 1 : i32
%cv1 = spirv.Constant dense<1> : vector<3xi32>
%0 = spirv.UMod %arg0, %c1: i32
%1 = spirv.UMod %arg1, %cv1: vector<3xi32>
// CHECK: return %[[C0]], %[[CVEC0]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @umod_div_0
func.func @umod_div_0() -> i32 {
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[CN1:.*]] = spirv.Constant -1
%c0 = spirv.Constant 0 : i32
%cn1 = spirv.Constant -1 : i32
// CHECK: %0 = spirv.UMod %[[CN1]], %[[C0]]
%0 = spirv.UMod %cn1, %c0 : i32
return %0 : i32
}
// CHECK-LABEL: @const_fold_scalar_umod
func.func @const_fold_scalar_umod() -> (i32, i32, i32) {
%c56 = spirv.Constant 56 : i32
%c7 = spirv.Constant 7 : i32
%cn8 = spirv.Constant -8 : i32
%c3 = spirv.Constant 3 : i32
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[C2:.*]] = spirv.Constant 2
// CHECK-DAG: %[[C56:.*]] = spirv.Constant 56
%0 = spirv.UMod %c56, %c7 : i32
%1 = spirv.UMod %cn8, %c3 : i32
%2 = spirv.UMod %c56, %cn8 : i32
// CHECK: return %[[C0]], %[[C2]], %[[C56]]
return %0, %1, %2 : i32, i32, i32
}
// CHECK-LABEL: @const_fold_vector_umod
func.func @const_fold_vector_umod() -> vector<3xi32> {
// CHECK: %[[CVEC:.*]] = spirv.Constant dense<[42, 24, 0]>
%cv = spirv.Constant dense<[42, 24, -16]> : vector<3xi32>
%cv_mod = spirv.Constant dense<[76, -7, 5]> : vector<3xi32>
%0 = spirv.UMod %cv, %cv_mod : vector<3xi32>
// CHECK: return %[[CVEC]]
return %0 : vector<3xi32>
}
// CHECK-LABEL: @umod_fold
// CHECK-SAME: (%[[ARG:.*]]: i32)
func.func @umod_fold(%arg0: i32) -> (i32, i32) {
// CHECK: %[[CONST4:.*]] = spirv.Constant 4
// CHECK: %[[CONST32:.*]] = spirv.Constant 32
%const1 = spirv.Constant 32 : i32
%0 = spirv.UMod %arg0, %const1 : i32
%const2 = spirv.Constant 4 : i32
%1 = spirv.UMod %0, %const2 : i32
// CHECK: %[[UMOD0:.*]] = spirv.UMod %[[ARG]], %[[CONST32]]
// CHECK: %[[UMOD1:.*]] = spirv.UMod %[[ARG]], %[[CONST4]]
// CHECK: return %[[UMOD0]], %[[UMOD1]]
return %0, %1: i32, i32
}
// CHECK-LABEL: @umod_fail_vector_fold
// CHECK-SAME: (%[[ARG:.*]]: vector<4xi32>)
func.func @umod_fail_vector_fold(%arg0: vector<4xi32>) -> (vector<4xi32>, vector<4xi32>) {
// CHECK: %[[CONST4:.*]] = spirv.Constant dense<4> : vector<4xi32>
// CHECK: %[[CONST32:.*]] = spirv.Constant dense<32> : vector<4xi32>
%const1 = spirv.Constant dense<32> : vector<4xi32>
%0 = spirv.UMod %arg0, %const1 : vector<4xi32>
// CHECK: %[[UMOD0:.*]] = spirv.UMod %[[ARG]], %[[CONST32]]
%const2 = spirv.Constant dense<4> : vector<4xi32>
%1 = spirv.UMod %0, %const2 : vector<4xi32>
// CHECK: %[[UMOD1:.*]] = spirv.UMod %[[UMOD0]], %[[CONST4]]
// CHECK: return %[[UMOD0]], %[[UMOD1]]
return %0, %1: vector<4xi32>, vector<4xi32>
}
// CHECK-LABEL: @umod_fold_same_divisor
// CHECK-SAME: (%[[ARG:.*]]: i32)
func.func @umod_fold_same_divisor(%arg0: i32) -> (i32, i32) {
// CHECK: %[[CONST1:.*]] = spirv.Constant 32
%const1 = spirv.Constant 32 : i32
%0 = spirv.UMod %arg0, %const1 : i32
%const2 = spirv.Constant 32 : i32
%1 = spirv.UMod %0, %const2 : i32
// CHECK: %[[UMOD0:.*]] = spirv.UMod %[[ARG]], %[[CONST1]]
// CHECK: %[[UMOD1:.*]] = spirv.UMod %[[ARG]], %[[CONST1]]
// CHECK: return %[[UMOD0]], %[[UMOD1]]
return %0, %1: i32, i32
}
// CHECK-LABEL: @umod_fail_fold
// CHECK-SAME: (%[[ARG:.*]]: i32)
func.func @umod_fail_fold(%arg0: i32) -> (i32, i32) {
// CHECK: %[[CONST5:.*]] = spirv.Constant 5
// CHECK: %[[CONST32:.*]] = spirv.Constant 32
%const1 = spirv.Constant 32 : i32
%0 = spirv.UMod %arg0, %const1 : i32
// CHECK: %[[UMOD0:.*]] = spirv.UMod %[[ARG]], %[[CONST32]]
%const2 = spirv.Constant 5 : i32
%1 = spirv.UMod %0, %const2 : i32
// CHECK: %[[UMOD1:.*]] = spirv.UMod %[[UMOD0]], %[[CONST5]]
// CHECK: return %[[UMOD0]], %[[UMOD1]]
return %0, %1: i32, i32
}
// -----
//===----------------------------------------------------------------------===//
// spirv.SNegate
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @snegate_twice
// CHECK-SAME: (%[[ARG:.*]]: i32)
func.func @snegate_twice(%arg0 : i32) -> i32 {
%0 = spirv.SNegate %arg0 : i32
%1 = spirv.SNegate %0 : i32
// CHECK: return %[[ARG]] : i32
return %1 : i32
}
// CHECK-LABEL: @snegate_min
func.func @snegate_min() -> (i8, i8) {
// CHECK: %[[MIN:.*]] = spirv.Constant -128 : i8
%cmin = spirv.Constant -128 : i8
%0 = spirv.SNegate %cmin : i8
%1 = spirv.SNegate %0 : i8
// CHECK: return %[[MIN]], %[[MIN]]
return %0, %1 : i8, i8
}
// CHECK-LABEL: @const_fold_scalar_snegate
func.func @const_fold_scalar_snegate() -> (i32, i32, i32) {
%c0 = spirv.Constant 0 : i32
%c3 = spirv.Constant 3 : i32
%cn3 = spirv.Constant -3 : i32
// CHECK-DAG: %[[THREE:.*]] = spirv.Constant 3 : i32
// CHECK-DAG: %[[NTHREE:.*]] = spirv.Constant -3 : i32
// CHECK-DAG: %[[ZERO:.*]] = spirv.Constant 0 : i32
%0 = spirv.SNegate %c0 : i32
%1 = spirv.SNegate %c3 : i32
%2 = spirv.SNegate %cn3 : i32
// CHECK: return %[[ZERO]], %[[NTHREE]], %[[THREE]]
return %0, %1, %2 : i32, i32, i32
}
// CHECK-LABEL: @const_fold_vector_snegate
func.func @const_fold_vector_snegate() -> vector<3xi32> {
// CHECK: spirv.Constant dense<[0, 3, -3]>
%cv = spirv.Constant dense<[0, -3, 3]> : vector<3xi32>
%0 = spirv.SNegate %cv : vector<3xi32>
return %0 : vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.Not
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @not_twice
// CHECK-SAME: (%[[ARG:.*]]: i32)
func.func @not_twice(%arg0 : i32) -> i32 {
%0 = spirv.Not %arg0 : i32
%1 = spirv.Not %0 : i32
// CHECK: return %[[ARG]] : i32
return %1 : i32
}
// CHECK-LABEL: @const_fold_scalar_not
func.func @const_fold_scalar_not() -> (i32, i32, i32) {
%c0 = spirv.Constant 0 : i32
%c3 = spirv.Constant 3 : i32
%cn3 = spirv.Constant -3 : i32
// CHECK-DAG: %[[TWO:.*]] = spirv.Constant 2 : i32
// CHECK-DAG: %[[NFOUR:.*]] = spirv.Constant -4 : i32
// CHECK-DAG: %[[NONE:.*]] = spirv.Constant -1 : i32
%0 = spirv.Not %c0 : i32
%1 = spirv.Not %c3 : i32
%2 = spirv.Not %cn3 : i32
// CHECK: return %[[NONE]], %[[NFOUR]], %[[TWO]]
return %0, %1, %2 : i32, i32, i32
}
// CHECK-LABEL: @const_fold_vector_not
func.func @const_fold_vector_not() -> vector<3xi32> {
%cv = spirv.Constant dense<[-1, -4, 2]> : vector<3xi32>
// CHECK: spirv.Constant dense<[0, 3, -3]>
%0 = spirv.Not %cv : vector<3xi32>
return %0 : vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.LogicalAnd
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @convert_logical_and_true_false_scalar
// CHECK-SAME: %[[ARG:.+]]: i1
func.func @convert_logical_and_true_false_scalar(%arg: i1) -> (i1, i1) {
%true = spirv.Constant true
// CHECK: %[[FALSE:.+]] = spirv.Constant false
%false = spirv.Constant false
%0 = spirv.LogicalAnd %true, %arg: i1
%1 = spirv.LogicalAnd %arg, %false: i1
// CHECK: return %[[ARG]], %[[FALSE]]
return %0, %1: i1, i1
}
// CHECK-LABEL: @convert_logical_and_true_false_vector
// CHECK-SAME: %[[ARG:.+]]: vector<3xi1>
func.func @convert_logical_and_true_false_vector(%arg: vector<3xi1>) -> (vector<3xi1>, vector<3xi1>) {
%true = spirv.Constant dense<true> : vector<3xi1>
// CHECK: %[[FALSE:.+]] = spirv.Constant dense<false>
%false = spirv.Constant dense<false> : vector<3xi1>
%0 = spirv.LogicalAnd %true, %arg: vector<3xi1>
%1 = spirv.LogicalAnd %arg, %false: vector<3xi1>
// CHECK: return %[[ARG]], %[[FALSE]]
return %0, %1: vector<3xi1>, vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.LogicalNot
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @logical_not_twice
// CHECK-SAME: (%[[ARG:.*]]: i1)
func.func @logical_not_twice(%arg0 : i1) -> i1 {
%0 = spirv.LogicalNot %arg0 : i1
%1 = spirv.LogicalNot %0 : i1
// CHECK: return %[[ARG]] : i1
return %1 : i1
}
// CHECK-LABEL: @const_fold_scalar_logical_not
func.func @const_fold_scalar_logical_not() -> i1 {
%true = spirv.Constant true
// CHECK: spirv.Constant false
%0 = spirv.LogicalNot %true : i1
return %0 : i1
}
// CHECK-LABEL: @const_fold_vector_logical_not
func.func @const_fold_vector_logical_not() -> vector<2xi1> {
%cv = spirv.Constant dense<[true, false]> : vector<2xi1>
// CHECK: spirv.Constant dense<[false, true]>
%0 = spirv.LogicalNot %cv : vector<2xi1>
return %0 : vector<2xi1>
}
// -----
func.func @convert_logical_not_to_not_equal(%arg0: vector<3xi64>, %arg1: vector<3xi64>) -> vector<3xi1> {
// CHECK: %[[RESULT:.*]] = spirv.INotEqual {{%.*}}, {{%.*}} : vector<3xi64>
// CHECK-NEXT: spirv.ReturnValue %[[RESULT]] : vector<3xi1>
%2 = spirv.IEqual %arg0, %arg1 : vector<3xi64>
%3 = spirv.LogicalNot %2 : vector<3xi1>
spirv.ReturnValue %3 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.LogicalEqual
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @logical_equal_same
func.func @logical_equal_same(%arg0 : i1, %arg1 : vector<3xi1>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CVTRUE:.*]] = spirv.Constant dense<true>
%0 = spirv.LogicalEqual %arg0, %arg0 : i1
%1 = spirv.LogicalEqual %arg1, %arg1 : vector<3xi1>
// CHECK: return %[[CTRUE]], %[[CVTRUE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_logical_equal
func.func @const_fold_scalar_logical_equal() -> (i1, i1) {
%true = spirv.Constant true
%false = spirv.Constant false
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.LogicalEqual %true, %false : i1
%1 = spirv.LogicalEqual %false, %false : i1
// CHECK: return %[[CFALSE]], %[[CTRUE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_logical_equal
func.func @const_fold_vector_logical_equal() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[true, false, true]> : vector<3xi1>
%cv1 = spirv.Constant dense<[true, false, false]> : vector<3xi1>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[true, true, false]>
%0 = spirv.LogicalEqual %cv0, %cv1 : vector<3xi1>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.LogicalNotEqual
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @convert_logical_not_equal_false
// CHECK-SAME: %[[ARG:.+]]: vector<4xi1>
func.func @convert_logical_not_equal_false(%arg: vector<4xi1>) -> vector<4xi1> {
%cst = spirv.Constant dense<false> : vector<4xi1>
// CHECK: spirv.ReturnValue %[[ARG]] : vector<4xi1>
%0 = spirv.LogicalNotEqual %arg, %cst : vector<4xi1>
spirv.ReturnValue %0 : vector<4xi1>
}
// CHECK-LABEL: @logical_not_equal_same
func.func @logical_not_equal_same(%arg0 : i1, %arg1 : vector<3xi1>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
// CHECK-DAG: %[[CVFALSE:.*]] = spirv.Constant dense<false>
%0 = spirv.LogicalNotEqual %arg0, %arg0 : i1
%1 = spirv.LogicalNotEqual %arg1, %arg1 : vector<3xi1>
// CHECK: return %[[CFALSE]], %[[CVFALSE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_logical_not_equal
func.func @const_fold_scalar_logical_not_equal() -> (i1, i1) {
%true = spirv.Constant true
%false = spirv.Constant false
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.LogicalNotEqual %true, %false : i1
%1 = spirv.LogicalNotEqual %false, %false : i1
// CHECK: return %[[CTRUE]], %[[CFALSE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_logical_not_equal
func.func @const_fold_vector_logical_not_equal() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[true, false, true]> : vector<3xi1>
%cv1 = spirv.Constant dense<[true, false, false]> : vector<3xi1>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[false, false, true]>
%0 = spirv.LogicalNotEqual %cv0, %cv1 : vector<3xi1>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
func.func @convert_logical_not_to_equal(%arg0: vector<3xi64>, %arg1: vector<3xi64>) -> vector<3xi1> {
// CHECK: %[[RESULT:.*]] = spirv.IEqual {{%.*}}, {{%.*}} : vector<3xi64>
// CHECK-NEXT: spirv.ReturnValue %[[RESULT]] : vector<3xi1>
%2 = spirv.INotEqual %arg0, %arg1 : vector<3xi64>
%3 = spirv.LogicalNot %2 : vector<3xi1>
spirv.ReturnValue %3 : vector<3xi1>
}
// -----
func.func @convert_logical_not_parent_multi_use(%arg0: vector<3xi64>, %arg1: vector<3xi64>, %arg2: !spirv.ptr<vector<3xi1>, Uniform>) -> vector<3xi1> {
// CHECK: %[[RESULT_0:.*]] = spirv.INotEqual {{%.*}}, {{%.*}} : vector<3xi64>
// CHECK-NEXT: %[[RESULT_1:.*]] = spirv.IEqual {{%.*}}, {{%.*}} : vector<3xi64>
// CHECK-NEXT: spirv.Store "Uniform" {{%.*}}, %[[RESULT_0]]
// CHECK-NEXT: spirv.ReturnValue %[[RESULT_1]]
%0 = spirv.INotEqual %arg0, %arg1 : vector<3xi64>
%1 = spirv.LogicalNot %0 : vector<3xi1>
spirv.Store "Uniform" %arg2, %0 : vector<3xi1>
spirv.ReturnValue %1 : vector<3xi1>
}
// -----
func.func @convert_logical_not_to_logical_not_equal(%arg0: vector<3xi1>, %arg1: vector<3xi1>) -> vector<3xi1> {
// CHECK: %[[RESULT:.*]] = spirv.LogicalNotEqual {{%.*}}, {{%.*}} : vector<3xi1>
// CHECK-NEXT: spirv.ReturnValue %[[RESULT]] : vector<3xi1>
%2 = spirv.LogicalEqual %arg0, %arg1 : vector<3xi1>
%3 = spirv.LogicalNot %2 : vector<3xi1>
spirv.ReturnValue %3 : vector<3xi1>
}
// -----
func.func @convert_logical_not_to_logical_equal(%arg0: vector<3xi1>, %arg1: vector<3xi1>) -> vector<3xi1> {
// CHECK: %[[RESULT:.*]] = spirv.LogicalEqual {{%.*}}, {{%.*}} : vector<3xi1>
// CHECK-NEXT: spirv.ReturnValue %[[RESULT]] : vector<3xi1>
%2 = spirv.LogicalNotEqual %arg0, %arg1 : vector<3xi1>
%3 = spirv.LogicalNot %2 : vector<3xi1>
spirv.ReturnValue %3 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.LogicalOr
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @convert_logical_or_true_false_scalar
// CHECK-SAME: %[[ARG:.+]]: i1
func.func @convert_logical_or_true_false_scalar(%arg: i1) -> (i1, i1) {
// CHECK: %[[TRUE:.+]] = spirv.Constant true
%true = spirv.Constant true
%false = spirv.Constant false
%0 = spirv.LogicalOr %true, %arg: i1
%1 = spirv.LogicalOr %arg, %false: i1
// CHECK: return %[[TRUE]], %[[ARG]]
return %0, %1: i1, i1
}
// CHECK-LABEL: @convert_logical_or_true_false_vector
// CHECK-SAME: %[[ARG:.+]]: vector<3xi1>
func.func @convert_logical_or_true_false_vector(%arg: vector<3xi1>) -> (vector<3xi1>, vector<3xi1>) {
// CHECK: %[[TRUE:.+]] = spirv.Constant dense<true>
%true = spirv.Constant dense<true> : vector<3xi1>
%false = spirv.Constant dense<false> : vector<3xi1>
%0 = spirv.LogicalOr %true, %arg: vector<3xi1>
%1 = spirv.LogicalOr %arg, %false: vector<3xi1>
// CHECK: return %[[TRUE]], %[[ARG]]
return %0, %1: vector<3xi1>, vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.Select
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @convert_select_scalar
// CHECK-SAME: %[[ARG1:.+]]: i32, %[[ARG2:.+]]: i32
func.func @convert_select_scalar(%arg1: i32, %arg2: i32) -> (i32, i32) {
%true = spirv.Constant true
%false = spirv.Constant false
%0 = spirv.Select %true, %arg1, %arg2 : i1, i32
%1 = spirv.Select %false, %arg1, %arg2 : i1, i32
// CHECK: return %[[ARG1]], %[[ARG2]]
return %0, %1 : i32, i32
}
// CHECK-LABEL: @convert_select_vector
// CHECK-SAME: %[[ARG1:.+]]: vector<3xi32>, %[[ARG2:.+]]: vector<3xi32>
func.func @convert_select_vector(%arg1: vector<3xi32>, %arg2: vector<3xi32>) -> (vector<3xi32>, vector<3xi32>) {
%true = spirv.Constant dense<true> : vector<3xi1>
%false = spirv.Constant dense<false> : vector<3xi1>
%0 = spirv.Select %true, %arg1, %arg2 : vector<3xi1>, vector<3xi32>
%1 = spirv.Select %false, %arg1, %arg2 : vector<3xi1>, vector<3xi32>
// CHECK: return %[[ARG1]], %[[ARG2]]
return %0, %1: vector<3xi32>, vector<3xi32>
}
// CHECK-LABEL: @convert_select_vector_extra
// CHECK-SAME: %[[CONDITIONS:.+]]: vector<2xi1>, %[[ARG1:.+]]: vector<2xi32>
func.func @convert_select_vector_extra(%conditions: vector<2xi1>, %arg1: vector<2xi32>) -> (vector<2xi32>, vector<2xi32>) {
%true_false = spirv.Constant dense<[true, false]> : vector<2xi1>
%cvec_1 = spirv.Constant dense<[42, -132]> : vector<2xi32>
%cvec_2 = spirv.Constant dense<[0, 42]> : vector<2xi32>
// CHECK: %[[RES:.+]] = spirv.Constant dense<42>
%0 = spirv.Select %true_false, %cvec_1, %cvec_2: vector<2xi1>, vector<2xi32>
%1 = spirv.Select %conditions, %arg1, %arg1 : vector<2xi1>, vector<2xi32>
// CHECK: return %[[RES]], %[[ARG1]]
return %0, %1: vector<2xi32>, vector<2xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.IEqual
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @iequal_same
func.func @iequal_same(%arg0 : i32, %arg1 : vector<3xi32>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CVTRUE:.*]] = spirv.Constant dense<true>
%0 = spirv.IEqual %arg0, %arg0 : i32
%1 = spirv.IEqual %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[CTRUE]], %[[CVTRUE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_iequal
func.func @const_fold_scalar_iequal() -> (i1, i1) {
%c5 = spirv.Constant 5 : i32
%c6 = spirv.Constant 6 : i32
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.IEqual %c5, %c6 : i32
%1 = spirv.IEqual %c5, %c5 : i32
// CHECK: return %[[CFALSE]], %[[CTRUE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_iequal
func.func @const_fold_vector_iequal() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[-1, -4, 2]> : vector<3xi32>
%cv1 = spirv.Constant dense<[-1, -3, 2]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[true, false, true]>
%0 = spirv.IEqual %cv0, %cv1 : vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.INotEqual
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @inotequal_same
func.func @inotequal_same(%arg0 : i32, %arg1 : vector<3xi32>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
// CHECK-DAG: %[[CVFALSE:.*]] = spirv.Constant dense<false>
%0 = spirv.INotEqual %arg0, %arg0 : i32
%1 = spirv.INotEqual %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[CFALSE]], %[[CVFALSE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_inotequal
func.func @const_fold_scalar_inotequal() -> (i1, i1) {
%c5 = spirv.Constant 5 : i32
%c6 = spirv.Constant 6 : i32
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.INotEqual %c5, %c6 : i32
%1 = spirv.INotEqual %c5, %c5 : i32
// CHECK: return %[[CTRUE]], %[[CFALSE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_inotequal
func.func @const_fold_vector_inotequal() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[-1, -4, 2]> : vector<3xi32>
%cv1 = spirv.Constant dense<[-1, -3, 2]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[false, true, false]>
%0 = spirv.INotEqual %cv0, %cv1 : vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.SGreaterThan
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @sgt_same
func.func @sgt_same(%arg0 : i32, %arg1 : vector<3xi32>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
// CHECK-DAG: %[[CVFALSE:.*]] = spirv.Constant dense<false>
%0 = spirv.SGreaterThan %arg0, %arg0 : i32
%1 = spirv.SGreaterThan %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[CFALSE]], %[[CVFALSE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_sgt
func.func @const_fold_scalar_sgt() -> (i1, i1) {
%c4 = spirv.Constant 4 : i32
%c5 = spirv.Constant 5 : i32
%c6 = spirv.Constant 6 : i32
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.SGreaterThan %c5, %c6 : i32
%1 = spirv.SGreaterThan %c5, %c4 : i32
// CHECK: return %[[CFALSE]], %[[CTRUE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_sgt
func.func @const_fold_vector_sgt() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[-1, -4, 3]> : vector<3xi32>
%cv1 = spirv.Constant dense<[-1, -3, 2]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[false, false, true]>
%0 = spirv.SGreaterThan %cv0, %cv1 : vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.SGreaterThanEqual
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @sge_same
func.func @sge_same(%arg0 : i32, %arg1 : vector<3xi32>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CVTRUE:.*]] = spirv.Constant dense<true>
%0 = spirv.SGreaterThanEqual %arg0, %arg0 : i32
%1 = spirv.SGreaterThanEqual %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[CTRUE]], %[[CVTRUE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_sge
func.func @const_fold_scalar_sge() -> (i1, i1) {
%c4 = spirv.Constant 4 : i32
%c5 = spirv.Constant 5 : i32
%c6 = spirv.Constant 6 : i32
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.SGreaterThanEqual %c5, %c6 : i32
%1 = spirv.SGreaterThanEqual %c5, %c4 : i32
// CHECK: return %[[CFALSE]], %[[CTRUE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_sge
func.func @const_fold_vector_sge() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[-1, -4, 3]> : vector<3xi32>
%cv1 = spirv.Constant dense<[-1, -3, 2]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[true, false, true]>
%0 = spirv.SGreaterThanEqual %cv0, %cv1 : vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.UGreaterThan
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @ugt_same
func.func @ugt_same(%arg0 : i32, %arg1 : vector<3xi32>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
// CHECK-DAG: %[[CVFALSE:.*]] = spirv.Constant dense<false>
%0 = spirv.UGreaterThan %arg0, %arg0 : i32
%1 = spirv.UGreaterThan %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[CFALSE]], %[[CVFALSE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_ugt
func.func @const_fold_scalar_ugt() -> (i1, i1) {
%c4 = spirv.Constant 4 : i32
%c5 = spirv.Constant 5 : i32
%cn6 = spirv.Constant -6 : i32
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.UGreaterThan %c5, %cn6 : i32
%1 = spirv.UGreaterThan %c5, %c4 : i32
// CHECK: return %[[CFALSE]], %[[CTRUE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_ugt
func.func @const_fold_vector_ugt() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[-1, -4, 3]> : vector<3xi32>
%cv1 = spirv.Constant dense<[-1, -3, 2]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[false, false, true]>
%0 = spirv.UGreaterThan %cv0, %cv1 : vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.UGreaterThanEqual
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @uge_same
func.func @uge_same(%arg0 : i32, %arg1 : vector<3xi32>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CVTRUE:.*]] = spirv.Constant dense<true>
%0 = spirv.UGreaterThanEqual %arg0, %arg0 : i32
%1 = spirv.UGreaterThanEqual %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[CTRUE]], %[[CVTRUE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_uge
func.func @const_fold_scalar_uge() -> (i1, i1) {
%c4 = spirv.Constant 4 : i32
%c5 = spirv.Constant 5 : i32
%cn6 = spirv.Constant -6 : i32
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.UGreaterThanEqual %c5, %cn6 : i32
%1 = spirv.UGreaterThanEqual %c5, %c4 : i32
// CHECK: return %[[CFALSE]], %[[CTRUE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_uge
func.func @const_fold_vector_uge() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[-1, -4, 3]> : vector<3xi32>
%cv1 = spirv.Constant dense<[-1, -3, 2]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[true, false, true]>
%0 = spirv.UGreaterThanEqual %cv0, %cv1 : vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.SLessThan
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @slt_same
func.func @slt_same(%arg0 : i32, %arg1 : vector<3xi32>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
// CHECK-DAG: %[[CVFALSE:.*]] = spirv.Constant dense<false>
%0 = spirv.SLessThan %arg0, %arg0 : i32
%1 = spirv.SLessThan %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[CFALSE]], %[[CVFALSE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_slt
func.func @const_fold_scalar_slt() -> (i1, i1) {
%c4 = spirv.Constant 4 : i32
%c5 = spirv.Constant 5 : i32
%c6 = spirv.Constant 6 : i32
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.SLessThan %c5, %c6 : i32
%1 = spirv.SLessThan %c5, %c4 : i32
// CHECK: return %[[CTRUE]], %[[CFALSE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_slt
func.func @const_fold_vector_slt() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[-1, -4, 3]> : vector<3xi32>
%cv1 = spirv.Constant dense<[-1, -3, 2]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[false, true, false]>
%0 = spirv.SLessThan %cv0, %cv1 : vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.SLessThanEqual
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @sle_same
func.func @sle_same(%arg0 : i32, %arg1 : vector<3xi32>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CVTRUE:.*]] = spirv.Constant dense<true>
%0 = spirv.SLessThanEqual %arg0, %arg0 : i32
%1 = spirv.SLessThanEqual %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[CTRUE]], %[[CVTRUE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_sle
func.func @const_fold_scalar_sle() -> (i1, i1) {
%c4 = spirv.Constant 4 : i32
%c5 = spirv.Constant 5 : i32
%c6 = spirv.Constant 6 : i32
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.SLessThanEqual %c5, %c6 : i32
%1 = spirv.SLessThanEqual %c5, %c4 : i32
// CHECK: return %[[CTRUE]], %[[CFALSE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_sle
func.func @const_fold_vector_sle() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[-1, -4, 3]> : vector<3xi32>
%cv1 = spirv.Constant dense<[-1, -3, 2]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[true, true, false]>
%0 = spirv.SLessThanEqual %cv0, %cv1 : vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.ULessThan
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @ult_same
func.func @ult_same(%arg0 : i32, %arg1 : vector<3xi32>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
// CHECK-DAG: %[[CVFALSE:.*]] = spirv.Constant dense<false>
%0 = spirv.ULessThan %arg0, %arg0 : i32
%1 = spirv.ULessThan %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[CFALSE]], %[[CVFALSE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_ult
func.func @const_fold_scalar_ult() -> (i1, i1) {
%c4 = spirv.Constant 4 : i32
%c5 = spirv.Constant 5 : i32
%cn6 = spirv.Constant -6 : i32
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.ULessThan %c5, %cn6 : i32
%1 = spirv.ULessThan %c5, %c4 : i32
// CHECK: return %[[CTRUE]], %[[CFALSE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_ult
func.func @const_fold_vector_ult() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[-1, -4, 3]> : vector<3xi32>
%cv1 = spirv.Constant dense<[-1, -3, 2]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[false, true, false]>
%0 = spirv.ULessThan %cv0, %cv1 : vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.ULessThanEqual
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @ule_same
func.func @ule_same(%arg0 : i32, %arg1 : vector<3xi32>) -> (i1, vector<3xi1>) {
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CVTRUE:.*]] = spirv.Constant dense<true>
%0 = spirv.ULessThanEqual %arg0, %arg0 : i32
%1 = spirv.ULessThanEqual %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[CTRUE]], %[[CVTRUE]]
return %0, %1 : i1, vector<3xi1>
}
// CHECK-LABEL: @const_fold_scalar_ule
func.func @const_fold_scalar_ule() -> (i1, i1) {
%c4 = spirv.Constant 4 : i32
%c5 = spirv.Constant 5 : i32
%cn6 = spirv.Constant -6 : i32
// CHECK-DAG: %[[CTRUE:.*]] = spirv.Constant true
// CHECK-DAG: %[[CFALSE:.*]] = spirv.Constant false
%0 = spirv.ULessThanEqual %c5, %cn6 : i32
%1 = spirv.ULessThanEqual %c5, %c4 : i32
// CHECK: return %[[CTRUE]], %[[CFALSE]]
return %0, %1 : i1, i1
}
// CHECK-LABEL: @const_fold_vector_ule
func.func @const_fold_vector_ule() -> vector<3xi1> {
%cv0 = spirv.Constant dense<[-1, -4, 3]> : vector<3xi32>
%cv1 = spirv.Constant dense<[-1, -3, 2]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[true, true, false]>
%0 = spirv.ULessThanEqual %cv0, %cv1 : vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi1>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.LeftShiftLogical
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @lsl_x_0
// CHECK-SAME: (%[[ARG0:.*]]: i32, %[[ARG1:.*]]: vector<3xi32>)
func.func @lsl_x_0(%arg0 : i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
%c0 = spirv.Constant 0 : i32
%cv0 = spirv.Constant dense<0> : vector<3xi32>
%0 = spirv.ShiftLeftLogical %arg0, %c0 : i32, i32
%1 = spirv.ShiftLeftLogical %arg1, %cv0 : vector<3xi32>, vector<3xi32>
// CHECK: return %[[ARG0]], %[[ARG1]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @lsl_shift_overflow
// CHECK-SAME: (%[[ARG0:.*]]: i32, %[[ARG1:.*]]: vector<3xi32>)
func.func @lsl_shift_overflow(%arg0: i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
// CHECK-DAG: %[[C32:.*]] = spirv.Constant 32
// CHECK-DAG: %[[CV:.*]] = spirv.Constant dense<[6, 18, 128]>
%c32 = spirv.Constant 32 : i32
%cv = spirv.Constant dense<[6, 18, 128]> : vector<3xi32>
// CHECK: %0 = spirv.ShiftLeftLogical %[[ARG0]], %[[C32]]
// CHECK: %1 = spirv.ShiftLeftLogical %[[ARG1]], %[[CV]]
%0 = spirv.ShiftLeftLogical %arg0, %c32 : i32, i32
%1 = spirv.ShiftLeftLogical %arg1, %cv : vector<3xi32>, vector<3xi32>
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @const_fold_scalar_lsl
func.func @const_fold_scalar_lsl() -> i32 {
%c1 = spirv.Constant 65535 : i32 // 0x0000 ffff
%c2 = spirv.Constant 17 : i32
// CHECK: %[[RET:.*]] = spirv.Constant -131072
// 0x0000 ffff << 17 -> 0xfffe 0000
%0 = spirv.ShiftLeftLogical %c1, %c2 : i32, i32
// CHECK: return %[[RET]]
return %0 : i32
}
// CHECK-LABEL: @const_fold_vector_lsl
func.func @const_fold_vector_lsl() -> vector<3xi32> {
%c1 = spirv.Constant dense<[1, -1, 127]> : vector<3xi32>
%c2 = spirv.Constant dense<[31, 16, 13]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[-2147483648, -65536, 1040384]>
%0 = spirv.ShiftLeftLogical %c1, %c2 : vector<3xi32>, vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.RightShiftArithmetic
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @asr_x_0
// CHECK-SAME: (%[[ARG0:.*]]: i32, %[[ARG1:.*]]: vector<3xi32>)
func.func @asr_x_0(%arg0 : i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
%c0 = spirv.Constant 0 : i32
%cv0 = spirv.Constant dense<0> : vector<3xi32>
%0 = spirv.ShiftRightArithmetic %arg0, %c0 : i32, i32
%1 = spirv.ShiftRightArithmetic %arg1, %cv0 : vector<3xi32>, vector<3xi32>
// CHECK: return %[[ARG0]], %[[ARG1]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @asr_shift_overflow
// CHECK-SAME: (%[[ARG0:.*]]: i32, %[[ARG1:.*]]: vector<3xi32>)
func.func @asr_shift_overflow(%arg0: i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
// CHECK-DAG: %[[C32:.*]] = spirv.Constant 32
// CHECK-DAG: %[[CV:.*]] = spirv.Constant dense<[6, 18, 128]>
%c32 = spirv.Constant 32 : i32
%cv = spirv.Constant dense<[6, 18, 128]> : vector<3xi32>
// CHECK: %0 = spirv.ShiftRightArithmetic %[[ARG0]], %[[C32]]
// CHECK: %1 = spirv.ShiftRightArithmetic %[[ARG1]], %[[CV]]
%0 = spirv.ShiftRightArithmetic %arg0, %c32 : i32, i32
%1 = spirv.ShiftRightArithmetic %arg1, %cv : vector<3xi32>, vector<3xi32>
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @const_fold_scalar_asr
func.func @const_fold_scalar_asr() -> i32 {
%c1 = spirv.Constant -131072 : i32 // 0xfffe 0000
%c2 = spirv.Constant 17 : i32
// 0x0000 ffff ashr 17 -> 0xffff ffff
// CHECK: %[[RET:.*]] = spirv.Constant -1
%0 = spirv.ShiftRightArithmetic %c1, %c2 : i32, i32
// CHECK: return %[[RET]]
return %0 : i32
}
// CHECK-LABEL: @const_fold_vector_asr
func.func @const_fold_vector_asr() -> vector<3xi32> {
%c1 = spirv.Constant dense<[-2147483648, 239847, 127]> : vector<3xi32>
%c2 = spirv.Constant dense<[31, 16, 13]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[-1, 3, 0]>
%0 = spirv.ShiftRightArithmetic %c1, %c2 : vector<3xi32>, vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.RightShiftLogical
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @lsr_x_0
// CHECK-SAME: (%[[ARG0:.*]]: i32, %[[ARG1:.*]]: vector<3xi32>)
func.func @lsr_x_0(%arg0 : i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
%c0 = spirv.Constant 0 : i32
%cv0 = spirv.Constant dense<0> : vector<3xi32>
%0 = spirv.ShiftRightLogical %arg0, %c0 : i32, i32
%1 = spirv.ShiftRightLogical %arg1, %cv0 : vector<3xi32>, vector<3xi32>
// CHECK: return %[[ARG0]], %[[ARG1]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @lsr_shift_overflow
// CHECK-SAME: (%[[ARG0:.*]]: i32, %[[ARG1:.*]]: vector<3xi32>)
func.func @lsr_shift_overflow(%arg0: i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
// CHECK-DAG: %[[C32:.*]] = spirv.Constant 32
// CHECK-DAG: %[[CV:.*]] = spirv.Constant dense<[6, 18, 128]>
%c32 = spirv.Constant 32 : i32
%cv = spirv.Constant dense<[6, 18, 128]> : vector<3xi32>
// CHECK: %0 = spirv.ShiftRightLogical %[[ARG0]], %[[C32]]
// CHECK: %1 = spirv.ShiftRightLogical %[[ARG1]], %[[CV]]
%0 = spirv.ShiftRightLogical %arg0, %c32 : i32, i32
%1 = spirv.ShiftRightLogical %arg1, %cv : vector<3xi32>, vector<3xi32>
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @const_fold_scalar_lsr
func.func @const_fold_scalar_lsr() -> i32 {
%c1 = spirv.Constant -131072 : i32 // 0xfffe 0000
%c2 = spirv.Constant 17 : i32
// 0x0000 ffff << 17 -> 0x0000 7fff
// CHECK: %[[RET:.*]] = spirv.Constant 32767
%0 = spirv.ShiftRightLogical %c1, %c2 : i32, i32
// CHECK: return %[[RET]]
return %0 : i32
}
// CHECK-LABEL: @const_fold_vector_lsr
func.func @const_fold_vector_lsr() -> vector<3xi32> {
%c1 = spirv.Constant dense<[-2147483648, -1, -127]> : vector<3xi32>
%c2 = spirv.Constant dense<[31, 16, 13]> : vector<3xi32>
// CHECK: %[[RET:.*]] = spirv.Constant dense<[1, 65535, 524287]>
%0 = spirv.ShiftRightLogical %c1, %c2 : vector<3xi32>, vector<3xi32>
// CHECK: return %[[RET]]
return %0 : vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.BitwiseAnd
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @bitwise_and_x_x
// CHECK-SAME: (%[[ARG0:.*]]: i32, %[[ARG1:.*]]: vector<3xi32>)
func.func @bitwise_and_x_x(%arg0: i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
%0 = spirv.BitwiseAnd %arg0, %arg0 : i32
%1 = spirv.BitwiseAnd %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[ARG0]], %[[ARG1]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @bitwise_and_x_0
func.func @bitwise_and_x_0(%arg0 : i32, %arg1 : vector<3xi32>) -> (i32, vector<3xi32>) {
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0 : i32
// CHECK-DAG: %[[CV0:.*]] = spirv.Constant dense<0> : vector<3xi32>
%c0 = spirv.Constant 0 : i32
%cv0 = spirv.Constant dense<0> : vector<3xi32>
%0 = spirv.BitwiseAnd %arg0, %c0 : i32
%1 = spirv.BitwiseAnd %arg1, %cv0 : vector<3xi32>
// CHECK: return %[[C0]], %[[CV0]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @bitwise_and_x_n1
// CHECK-SAME: (%[[ARG0:.*]]: i32, %[[ARG1:.*]]: vector<3xi32>)
func.func @bitwise_and_x_n1(%arg0: i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
%cn1 = spirv.Constant -1 : i32
%cvn1 = spirv.Constant dense<-1> : vector<3xi32>
%0 = spirv.BitwiseAnd %arg0, %cn1 : i32
%1 = spirv.BitwiseAnd %arg1, %cvn1 : vector<3xi32>
// CHECK: return %[[ARG0]], %[[ARG1]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @const_fold_scalar_band
func.func @const_fold_scalar_band() -> i32 {
%c1 = spirv.Constant -268464129 : i32 // 0xefff 8fff
%c2 = spirv.Constant 268464128: i32 // 0x1000 7000
// 0xefff 8fff | 0x1000 7000 = 0xffff ffff = -1
// CHECK: %[[C0:.*]] = spirv.Constant 0
%0 = spirv.BitwiseAnd %c1, %c2 : i32
// CHECK: return %[[C0]]
return %0 : i32
}
// CHECK-LABEL: @const_fold_vector_band
func.func @const_fold_vector_band() -> vector<3xi32> {
%c1 = spirv.Constant dense<[42, -55, 127]> : vector<3xi32>
%c2 = spirv.Constant dense<[-3, -15, 28]> : vector<3xi32>
// CHECK: %[[CV:.*]] = spirv.Constant dense<[40, -63, 28]>
%0 = spirv.BitwiseAnd %c1, %c2 : vector<3xi32>
// CHECK: return %[[CV]]
return %0 : vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.BitwiseOr
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @bitwise_or_x_x
// CHECK-SAME: (%[[ARG0:.*]]: i32, %[[ARG1:.*]]: vector<3xi32>)
func.func @bitwise_or_x_x(%arg0: i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
%0 = spirv.BitwiseOr %arg0, %arg0 : i32
%1 = spirv.BitwiseOr %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[ARG0]], %[[ARG1]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @bitwise_or_x_0
// CHECK-SAME: (%[[ARG0:.*]]: i32, %[[ARG1:.*]]: vector<3xi32>)
func.func @bitwise_or_x_0(%arg0: i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
%c1 = spirv.Constant 0 : i32
%cv1 = spirv.Constant dense<0> : vector<3xi32>
%0 = spirv.BitwiseOr %arg0, %c1 : i32
%1 = spirv.BitwiseOr %arg1, %cv1 : vector<3xi32>
// CHECK: return %[[ARG0]], %[[ARG1]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @bitwise_or_x_n1
func.func @bitwise_or_x_n1(%arg0 : i32, %arg1 : vector<3xi32>) -> (i32, vector<3xi32>) {
// CHECK-DAG: %[[CN1:.*]] = spirv.Constant -1 : i32
// CHECK-DAG: %[[CVN1:.*]] = spirv.Constant dense<-1> : vector<3xi32>
%cn1 = spirv.Constant -1 : i32
%cvn1 = spirv.Constant dense<-1> : vector<3xi32>
%0 = spirv.BitwiseOr %arg0, %cn1 : i32
%1 = spirv.BitwiseOr %arg1, %cvn1 : vector<3xi32>
// CHECK: return %[[CN1]], %[[CVN1]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @const_fold_scalar_bor
func.func @const_fold_scalar_bor() -> i32 {
%c1 = spirv.Constant -268464129 : i32 // 0xefff 8fff
%c2 = spirv.Constant 268464128: i32 // 0x1000 7000
// 0xefff 8fff | 0x1000 7000 = 0xffff ffff = -1
// CHECK: %[[CN1:.*]] = spirv.Constant -1
%0 = spirv.BitwiseOr %c1, %c2 : i32
// CHECK: return %[[CN1]]
return %0 : i32
}
// CHECK-LABEL: @const_fold_vector_bor
func.func @const_fold_vector_bor() -> vector<3xi32> {
%c1 = spirv.Constant dense<[42, -55, 127]> : vector<3xi32>
%c2 = spirv.Constant dense<[-3, -15, 28]> : vector<3xi32>
// CHECK: %[[CV:.*]] = spirv.Constant dense<[-1, -7, 127]>
%0 = spirv.BitwiseOr %c1, %c2 : vector<3xi32>
// CHECK: return %[[CV]]
return %0 : vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.BitwiseXor
//===----------------------------------------------------------------------===//
// CHECK-LABEL: @bitwise_xor_x_0
// CHECK-SAME: (%[[ARG0:.*]]: i32, %[[ARG1:.*]]: vector<3xi32>)
func.func @bitwise_xor_x_0(%arg0: i32, %arg1: vector<3xi32>) -> (i32, vector<3xi32>) {
%c0 = spirv.Constant 0 : i32
%cv0 = spirv.Constant dense<0> : vector<3xi32>
%0 = spirv.BitwiseXor %arg0, %c0 : i32
%1 = spirv.BitwiseXor %arg1, %cv0 : vector<3xi32>
// CHECK: return %[[ARG0]], %[[ARG1]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @bitwise_xor_x_x
func.func @bitwise_xor_x_x(%arg0 : i32, %arg1 : vector<3xi32>) -> (i32, vector<3xi32>) {
// CHECK-DAG: %[[C0:.*]] = spirv.Constant 0
// CHECK-DAG: %[[CV0:.*]] = spirv.Constant dense<0>
%0 = spirv.BitwiseXor %arg0, %arg0 : i32
%1 = spirv.BitwiseXor %arg1, %arg1 : vector<3xi32>
// CHECK: return %[[C0]], %[[CV0]]
return %0, %1 : i32, vector<3xi32>
}
// CHECK-LABEL: @const_fold_scalar_bxor
func.func @const_fold_scalar_bxor() -> i32 {
%c1 = spirv.Constant 4294967295 : i32 // 2^32 - 1: 0xffff ffff
%c2 = spirv.Constant -2147483648 : i32 // -2^31 : 0x8000 0000
// 0x8000 0000 ^ 0xffff fffe = 0xefff ffff
// CHECK: %[[CBIG:.*]] = spirv.Constant 2147483647
%0 = spirv.BitwiseXor %c1, %c2 : i32
// CHECK: return %[[CBIG]]
return %0 : i32
}
// CHECK-LABEL: @const_fold_vector_bxor
func.func @const_fold_vector_bxor() -> vector<3xi32> {
%c1 = spirv.Constant dense<[42, -55, 127]> : vector<3xi32>
%c2 = spirv.Constant dense<[-3, -15, 28]> : vector<3xi32>
// CHECK: %[[CV:.*]] = spirv.Constant dense<[-41, 56, 99]>
%0 = spirv.BitwiseXor %c1, %c2 : vector<3xi32>
// CHECK: return %[[CV]]
return %0 : vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.mlir.selection
//===----------------------------------------------------------------------===//
func.func @canonicalize_selection_op_scalar_type(%cond: i1) -> () {
%0 = spirv.Constant 0: i32
// CHECK-DAG: %[[TRUE_VALUE:.*]] = spirv.Constant 1 : i32
%1 = spirv.Constant 1: i32
// CHECK-DAG: %[[FALSE_VALUE:.*]] = spirv.Constant 2 : i32
%2 = spirv.Constant 2: i32
// CHECK: %[[DST_VAR:.*]] = spirv.Variable init({{%.*}}) : !spirv.ptr<i32, Function>
%3 = spirv.Variable init(%0) : !spirv.ptr<i32, Function>
// CHECK: %[[SRC_VALUE:.*]] = spirv.Select {{%.*}}, %[[TRUE_VALUE]], %[[FALSE_VALUE]] : i1, i32
// CHECK-NEXT: spirv.Store "Function" %[[DST_VAR]], %[[SRC_VALUE]] ["Aligned", 4] : i32
// CHECK-NEXT: spirv.Return
spirv.mlir.selection {
spirv.BranchConditional %cond, ^then, ^else
^else:
spirv.Store "Function" %3, %2 ["Aligned", 4]: i32
spirv.Branch ^merge
^then:
spirv.Store "Function" %3, %1 ["Aligned", 4]: i32
spirv.Branch ^merge
^merge:
spirv.mlir.merge
}
spirv.Return
}
// -----
func.func @canonicalize_selection_op_vector_type(%cond: i1) -> () {
%0 = spirv.Constant dense<[0, 1, 2]> : vector<3xi32>
// CHECK-DAG: %[[TRUE_VALUE:.*]] = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
%1 = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
// CHECK-DAG: %[[FALSE_VALUE:.*]] = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
%2 = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
// CHECK: %[[DST_VAR:.*]] = spirv.Variable init({{%.*}}) : !spirv.ptr<vector<3xi32>, Function>
%3 = spirv.Variable init(%0) : !spirv.ptr<vector<3xi32>, Function>
// CHECK: %[[SRC_VALUE:.*]] = spirv.Select {{%.*}}, %[[TRUE_VALUE]], %[[FALSE_VALUE]] : i1, vector<3xi32>
// CHECK-NEXT: spirv.Store "Function" %[[DST_VAR]], %[[SRC_VALUE]] ["Aligned", 8] : vector<3xi32>
// CHECK-NEXT: spirv.Return
spirv.mlir.selection {
spirv.BranchConditional %cond, ^then, ^else
^then:
spirv.Store "Function" %3, %1 ["Aligned", 8]: vector<3xi32>
spirv.Branch ^merge
^else:
spirv.Store "Function" %3, %2 ["Aligned", 8] : vector<3xi32>
spirv.Branch ^merge
^merge:
spirv.mlir.merge
}
spirv.Return
}
// -----
// CHECK-LABEL: cannot_canonicalize_selection_op_0
// Store to a different variables.
func.func @cannot_canonicalize_selection_op_0(%cond: i1) -> () {
%0 = spirv.Constant dense<[0, 1, 2]> : vector<3xi32>
// CHECK-DAG: %[[SRC_VALUE_1:.*]] = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
// CHECK-DAG: %[[SRC_VALUE_0:.*]] = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
%1 = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
%2 = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
// CHECK: %[[DST_VAR_0:.*]] = spirv.Variable init({{%.*}}) : !spirv.ptr<vector<3xi32>, Function>
%3 = spirv.Variable init(%0) : !spirv.ptr<vector<3xi32>, Function>
// CHECK: %[[DST_VAR_1:.*]] = spirv.Variable init({{%.*}}) : !spirv.ptr<vector<3xi32>, Function>
%4 = spirv.Variable init(%0) : !spirv.ptr<vector<3xi32>, Function>
// CHECK: spirv.mlir.selection {
spirv.mlir.selection {
// CHECK: spirv.BranchConditional
// CHECK-SAME: ^bb1(%[[DST_VAR_0]], %[[SRC_VALUE_0]]
// CHECK-SAME: ^bb1(%[[DST_VAR_1]], %[[SRC_VALUE_1]]
spirv.BranchConditional %cond, ^then, ^else
^then:
// CHECK: ^bb1(%[[ARG0:.*]]: !spirv.ptr<vector<3xi32>, Function>, %[[ARG1:.*]]: vector<3xi32>):
// CHECK: spirv.Store "Function" %[[ARG0]], %[[ARG1]] ["Aligned", 8] : vector<3xi32>
spirv.Store "Function" %3, %1 ["Aligned", 8]: vector<3xi32>
spirv.Branch ^merge
^else:
spirv.Store "Function" %4, %2 ["Aligned", 8] : vector<3xi32>
spirv.Branch ^merge
^merge:
spirv.mlir.merge
}
spirv.Return
}
// -----
// CHECK-LABEL: cannot_canonicalize_selection_op_1
// A conditional block consists of more than 2 operations.
func.func @cannot_canonicalize_selection_op_1(%cond: i1) -> () {
%0 = spirv.Constant dense<[0, 1, 2]> : vector<3xi32>
// CHECK-DAG: %[[SRC_VALUE_0:.*]] = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
%1 = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
// CHECK-DAG: %[[SRC_VALUE_1:.*]] = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
%2 = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
// CHECK: %[[DST_VAR_0:.*]] = spirv.Variable init({{%.*}}) : !spirv.ptr<vector<3xi32>, Function>
%3 = spirv.Variable init(%0) : !spirv.ptr<vector<3xi32>, Function>
// CHECK: %[[DST_VAR_1:.*]] = spirv.Variable init({{%.*}}) : !spirv.ptr<vector<3xi32>, Function>
%4 = spirv.Variable init(%0) : !spirv.ptr<vector<3xi32>, Function>
// CHECK: spirv.mlir.selection {
spirv.mlir.selection {
spirv.BranchConditional %cond, ^then, ^else
^then:
// CHECK: spirv.Store "Function" %[[DST_VAR_0]], %[[SRC_VALUE_0]] ["Aligned", 8] : vector<3xi32>
spirv.Store "Function" %3, %1 ["Aligned", 8] : vector<3xi32>
// CHECK: spirv.Store "Function" %[[DST_VAR_1]], %[[SRC_VALUE_0]] ["Aligned", 8] : vector<3xi32>
spirv.Store "Function" %4, %1 ["Aligned", 8]: vector<3xi32>
spirv.Branch ^merge
^else:
// CHECK: spirv.Store "Function" %[[DST_VAR_1]], %[[SRC_VALUE_1]] ["Aligned", 8] : vector<3xi32>
spirv.Store "Function" %4, %2 ["Aligned", 8] : vector<3xi32>
spirv.Branch ^merge
^merge:
spirv.mlir.merge
}
spirv.Return
}
// -----
// CHECK-LABEL: cannot_canonicalize_selection_op_2
// A control-flow goes into `^then` block from `^else` block.
func.func @cannot_canonicalize_selection_op_2(%cond: i1) -> () {
%0 = spirv.Constant dense<[0, 1, 2]> : vector<3xi32>
// CHECK-DAG: %[[SRC_VALUE_0:.*]] = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
%1 = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
// CHECK-DAG: %[[SRC_VALUE_1:.*]] = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
%2 = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
// CHECK: %[[DST_VAR:.*]] = spirv.Variable init({{%.*}}) : !spirv.ptr<vector<3xi32>, Function>
%3 = spirv.Variable init(%0) : !spirv.ptr<vector<3xi32>, Function>
// CHECK: spirv.mlir.selection {
spirv.mlir.selection {
spirv.BranchConditional %cond, ^then, ^else
^then:
// CHECK: spirv.Store "Function" %[[DST_VAR]], %[[SRC_VALUE_0]] ["Aligned", 8] : vector<3xi32>
spirv.Store "Function" %3, %1 ["Aligned", 8]: vector<3xi32>
spirv.Branch ^merge
^else:
// CHECK: spirv.Store "Function" %[[DST_VAR]], %[[SRC_VALUE_1]] ["Aligned", 8] : vector<3xi32>
spirv.Store "Function" %3, %2 ["Aligned", 8] : vector<3xi32>
spirv.Branch ^then
^merge:
spirv.mlir.merge
}
spirv.Return
}
// -----
// CHECK-LABEL: cannot_canonicalize_selection_op_3
// `spirv.Return` as a block terminator.
func.func @cannot_canonicalize_selection_op_3(%cond: i1) -> () {
%0 = spirv.Constant dense<[0, 1, 2]> : vector<3xi32>
%1 = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
// CHECK-DAG: %[[SRC_VALUE_0:.*]] = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
// CHECK-DAG: %[[SRC_VALUE_1:.*]] = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
%2 = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
// CHECK: %[[DST_VAR:.*]] = spirv.Variable init({{%.*}}) : !spirv.ptr<vector<3xi32>, Function>
%3 = spirv.Variable init(%0) : !spirv.ptr<vector<3xi32>, Function>
// CHECK: spirv.mlir.selection {
spirv.mlir.selection {
spirv.BranchConditional %cond, ^then, ^else
^then:
// CHECK: spirv.Store "Function" %[[DST_VAR]], %[[SRC_VALUE_0]] ["Aligned", 8] : vector<3xi32>
spirv.Store "Function" %3, %1 ["Aligned", 8]: vector<3xi32>
spirv.Return
^else:
// CHECK: spirv.Store "Function" %[[DST_VAR]], %[[SRC_VALUE_1]] ["Aligned", 8] : vector<3xi32>
spirv.Store "Function" %3, %2 ["Aligned", 8] : vector<3xi32>
spirv.Branch ^merge
^merge:
spirv.mlir.merge
}
spirv.Return
}
// -----
// CHECK-LABEL: cannot_canonicalize_selection_op_4
// Different memory access attributes.
func.func @cannot_canonicalize_selection_op_4(%cond: i1) -> () {
%0 = spirv.Constant dense<[0, 1, 2]> : vector<3xi32>
// CHECK-DAG: %[[SRC_VALUE_0:.*]] = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
%1 = spirv.Constant dense<[1, 2, 3]> : vector<3xi32>
// CHECK-DAG: %[[SRC_VALUE_1:.*]] = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
%2 = spirv.Constant dense<[2, 3, 4]> : vector<3xi32>
// CHECK: %[[DST_VAR:.*]] = spirv.Variable init({{%.*}}) : !spirv.ptr<vector<3xi32>, Function>
%3 = spirv.Variable init(%0) : !spirv.ptr<vector<3xi32>, Function>
// CHECK: spirv.mlir.selection {
spirv.mlir.selection {
spirv.BranchConditional %cond, ^then, ^else
^then:
// CHECK: spirv.Store "Function" %[[DST_VAR]], %[[SRC_VALUE_0]] ["Aligned", 4] : vector<3xi32>
spirv.Store "Function" %3, %1 ["Aligned", 4]: vector<3xi32>
spirv.Branch ^merge
^else:
// CHECK: spirv.Store "Function" %[[DST_VAR]], %[[SRC_VALUE_1]] ["Aligned", 8] : vector<3xi32>
spirv.Store "Function" %3, %2 ["Aligned", 8] : vector<3xi32>
spirv.Branch ^merge
^merge:
spirv.mlir.merge
}
spirv.Return
}