// RUN: mlir-opt --split-input-file --verify-diagnostics %s | FileCheck %s
// RUN: mlir-opt --split-input-file --verify-diagnostics --canonicalize %s \
// RUN: | FileCheck %s --check-prefix=CANON
//===----------------------------------------------------------------------===//
// spirv.BitCount
//===----------------------------------------------------------------------===//
func.func @bitcount(%arg: i32) -> i32 {
// CHECK: spirv.BitCount {{%.*}} : i32
%0 = spirv.BitCount %arg : i32
spirv.ReturnValue %0 : i32
}
// -----
//===----------------------------------------------------------------------===//
// spirv.BitFieldInsert
//===----------------------------------------------------------------------===//
func.func @bit_field_insert_vec(%base: vector<3xi32>, %insert: vector<3xi32>, %offset: i32, %count: i16) -> vector<3xi32> {
// CHECK: {{%.*}} = spirv.BitFieldInsert {{%.*}}, {{%.*}}, {{%.*}}, {{%.*}} : vector<3xi32>, i32, i16
%0 = spirv.BitFieldInsert %base, %insert, %offset, %count : vector<3xi32>, i32, i16
spirv.ReturnValue %0 : vector<3xi32>
}
// -----
func.func @bit_field_insert_invalid_insert_type(%base: vector<3xi32>, %insert: vector<2xi32>, %offset: i32, %count: i16) -> vector<3xi32> {
// TODO: expand post change in verification order. This is currently only
// verifying that the type verification is failing but not the specific error
// message. In final state the error should refer to mismatch in base and
// insert.
// expected-error @+1 {{type}}
%0 = "spirv.BitFieldInsert" (%base, %insert, %offset, %count) : (vector<3xi32>, vector<2xi32>, i32, i16) -> vector<3xi32>
spirv.ReturnValue %0 : vector<3xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.BitFieldSExtract
//===----------------------------------------------------------------------===//
func.func @bit_field_s_extract_vec(%base: vector<3xi32>, %offset: i8, %count: i8) -> vector<3xi32> {
// CHECK: {{%.*}} = spirv.BitFieldSExtract {{%.*}}, {{%.*}}, {{%.*}} : vector<3xi32>, i8, i8
%0 = spirv.BitFieldSExtract %base, %offset, %count : vector<3xi32>, i8, i8
spirv.ReturnValue %0 : vector<3xi32>
}
//===----------------------------------------------------------------------===//
// spirv.BitFieldUExtract
//===----------------------------------------------------------------------===//
func.func @bit_field_u_extract_vec(%base: vector<3xi32>, %offset: i8, %count: i8) -> vector<3xi32> {
// CHECK: {{%.*}} = spirv.BitFieldUExtract {{%.*}}, {{%.*}}, {{%.*}} : vector<3xi32>, i8, i8
%0 = spirv.BitFieldUExtract %base, %offset, %count : vector<3xi32>, i8, i8
spirv.ReturnValue %0 : vector<3xi32>
}
// -----
func.func @bit_field_u_extract_invalid_result_type(%base: vector<3xi32>, %offset: i32, %count: i16) -> vector<4xi32> {
// expected-error @+1 {{failed to verify that all of {base, result} have same type}}
%0 = "spirv.BitFieldUExtract" (%base, %offset, %count) : (vector<3xi32>, i32, i16) -> vector<4xi32>
spirv.ReturnValue %0 : vector<4xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.BitReverse
//===----------------------------------------------------------------------===//
func.func @bitreverse(%arg: i32) -> i32 {
// CHECK: spirv.BitReverse {{%.*}} : i32
%0 = spirv.BitReverse %arg : i32
spirv.ReturnValue %0 : i32
}
// -----
//===----------------------------------------------------------------------===//
// spirv.BitwiseOr
//===----------------------------------------------------------------------===//
// CHECK-LABEL: func @bitwise_or_scalar
func.func @bitwise_or_scalar(%arg: i32) -> i32 {
// CHECK: spirv.BitwiseOr
%0 = spirv.BitwiseOr %arg, %arg : i32
return %0 : i32
}
// CHECK-LABEL: func @bitwise_or_vector
func.func @bitwise_or_vector(%arg: vector<4xi32>) -> vector<4xi32> {
// CHECK: spirv.BitwiseOr
%0 = spirv.BitwiseOr %arg, %arg : vector<4xi32>
return %0 : vector<4xi32>
}
// CANON-LABEL: func @bitwise_or_zero
// CANON-SAME: (%[[ARG:.+]]: i32)
func.func @bitwise_or_zero(%arg: i32) -> i32 {
// CANON: return %[[ARG]]
%zero = spirv.Constant 0 : i32
%0 = spirv.BitwiseOr %arg, %zero : i32
return %0 : i32
}
// CANON-LABEL: func @bitwise_or_zero_vector
// CANON-SAME: (%[[ARG:.+]]: vector<4xi32>)
func.func @bitwise_or_zero_vector(%arg: vector<4xi32>) -> vector<4xi32> {
// CANON: return %[[ARG]]
%zero = spirv.Constant dense<0> : vector<4xi32>
%0 = spirv.BitwiseOr %arg, %zero : vector<4xi32>
return %0 : vector<4xi32>
}
// CANON-LABEL: func @bitwise_or_all_ones
func.func @bitwise_or_all_ones(%arg: i8) -> i8 {
// CANON: %[[CST:.+]] = spirv.Constant -1
// CANON: return %[[CST]]
%ones = spirv.Constant 255 : i8
%0 = spirv.BitwiseOr %arg, %ones : i8
return %0 : i8
}
// CANON-LABEL: func @bitwise_or_all_ones_vector
func.func @bitwise_or_all_ones_vector(%arg: vector<3xi8>) -> vector<3xi8> {
// CANON: %[[CST:.+]] = spirv.Constant dense<-1>
// CANON: return %[[CST]]
%ones = spirv.Constant dense<255> : vector<3xi8>
%0 = spirv.BitwiseOr %arg, %ones : vector<3xi8>
return %0 : vector<3xi8>
}
// -----
func.func @bitwise_or_float(%arg0: f16, %arg1: f16) -> f16 {
// expected-error @+1 {{operand #0 must be 8/16/32/64-bit integer or vector of 8/16/32/64-bit integer values of length 2/3/4}}
%0 = spirv.BitwiseOr %arg0, %arg1 : f16
return %0 : f16
}
// -----
//===----------------------------------------------------------------------===//
// spirv.BitwiseXor
//===----------------------------------------------------------------------===//
func.func @bitwise_xor_scalar(%arg: i32) -> i32 {
%c1 = spirv.Constant 1 : i32 // using constant to avoid folding
// CHECK: spirv.BitwiseXor
%0 = spirv.BitwiseXor %c1, %arg : i32
return %0 : i32
}
func.func @bitwise_xor_vector(%arg: vector<4xi32>) -> vector<4xi32> {
%c1 = spirv.Constant dense<1> : vector<4xi32> // using constant to avoid folding
// CHECK: spirv.BitwiseXor
%0 = spirv.BitwiseXor %c1, %arg : vector<4xi32>
return %0 : vector<4xi32>
}
// -----
func.func @bitwise_xor_float(%arg0: f16, %arg1: f16) -> f16 {
// expected-error @+1 {{operand #0 must be 8/16/32/64-bit integer or vector of 8/16/32/64-bit integer values of length 2/3/4}}
%0 = spirv.BitwiseXor %arg0, %arg1 : f16
return %0 : f16
}
// -----
//===----------------------------------------------------------------------===//
// spirv.BitwiseAnd
//===----------------------------------------------------------------------===//
// CHECK-LABEL: func @bitwise_and_scalar
func.func @bitwise_and_scalar(%arg: i32) -> i32 {
// CHECK: spirv.BitwiseAnd
%0 = spirv.BitwiseAnd %arg, %arg : i32
return %0 : i32
}
// CHECK-LABEL: func @bitwise_and_vector
func.func @bitwise_and_vector(%arg: vector<4xi32>) -> vector<4xi32> {
// CHECK: spirv.BitwiseAnd
%0 = spirv.BitwiseAnd %arg, %arg : vector<4xi32>
return %0 : vector<4xi32>
}
// CANON-LABEL: func @bitwise_and_zero
func.func @bitwise_and_zero(%arg: i32) -> i32 {
// CANON: %[[CST:.+]] = spirv.Constant 0
// CANON: return %[[CST]]
%zero = spirv.Constant 0 : i32
%0 = spirv.BitwiseAnd %arg, %zero : i32
return %0 : i32
}
// CANON-LABEL: func @bitwise_and_zero_vector
func.func @bitwise_and_zero_vector(%arg: vector<4xi32>) -> vector<4xi32> {
// CANON: %[[CST:.+]] = spirv.Constant dense<0>
// CANON: return %[[CST]]
%zero = spirv.Constant dense<0> : vector<4xi32>
%0 = spirv.BitwiseAnd %arg, %zero : vector<4xi32>
return %0 : vector<4xi32>
}
// CANON-LABEL: func @bitwise_and_all_ones
// CANON-SAME: (%[[ARG:.+]]: i8)
func.func @bitwise_and_all_ones(%arg: i8) -> i8 {
// CANON: return %[[ARG]]
%ones = spirv.Constant 255 : i8
%0 = spirv.BitwiseAnd %arg, %ones : i8
return %0 : i8
}
// CANON-LABEL: func @bitwise_and_all_ones_vector
// CANON-SAME: (%[[ARG:.+]]: vector<3xi8>)
func.func @bitwise_and_all_ones_vector(%arg: vector<3xi8>) -> vector<3xi8> {
// CANON: return %[[ARG]]
%ones = spirv.Constant dense<255> : vector<3xi8>
%0 = spirv.BitwiseAnd %arg, %ones : vector<3xi8>
return %0 : vector<3xi8>
}
// CANON-LABEL: func @bitwise_and_zext_1
// CANON-SAME: (%[[ARG:.+]]: i8)
func.func @bitwise_and_zext_1(%arg: i8) -> i32 {
// CANON: %[[ZEXT:.+]] = spirv.UConvert %[[ARG]]
// CANON: return %[[ZEXT]]
%zext = spirv.UConvert %arg : i8 to i32
%ones = spirv.Constant 255 : i32
%0 = spirv.BitwiseAnd %zext, %ones : i32
return %0 : i32
}
// CANON-LABEL: func @bitwise_and_zext_2
// CANON-SAME: (%[[ARG:.+]]: i8)
func.func @bitwise_and_zext_2(%arg: i8) -> i32 {
// CANON: %[[ZEXT:.+]] = spirv.UConvert %[[ARG]]
// CANON: return %[[ZEXT]]
%zext = spirv.UConvert %arg : i8 to i32
%ones = spirv.Constant 0x12345ff : i32
%0 = spirv.BitwiseAnd %zext, %ones : i32
return %0 : i32
}
// CANON-LABEL: func @bitwise_and_zext_3
// CANON-SAME: (%[[ARG:.+]]: i8)
func.func @bitwise_and_zext_3(%arg: i8) -> i32 {
// CANON: %[[ZEXT:.+]] = spirv.UConvert %[[ARG]]
// CANON: %[[AND:.+]] = spirv.BitwiseAnd %[[ZEXT]]
// CANON: return %[[AND]]
%zext = spirv.UConvert %arg : i8 to i32
%ones = spirv.Constant 254 : i32
%0 = spirv.BitwiseAnd %zext, %ones : i32
return %0 : i32
}
// CANON-LABEL: func @bitwise_and_zext_vector
// CANON-SAME: (%[[ARG:.+]]: vector<2xi8>)
func.func @bitwise_and_zext_vector(%arg: vector<2xi8>) -> vector<2xi32> {
// CANON: %[[ZEXT:.+]] = spirv.UConvert %[[ARG]]
// CANON: return %[[ZEXT]]
%zext = spirv.UConvert %arg : vector<2xi8> to vector<2xi32>
%ones = spirv.Constant dense<255> : vector<2xi32>
%0 = spirv.BitwiseAnd %zext, %ones : vector<2xi32>
return %0 : vector<2xi32>
}
// -----
func.func @bitwise_and_float(%arg0: f16, %arg1: f16) -> f16 {
// expected-error @+1 {{operand #0 must be 8/16/32/64-bit integer or vector of 8/16/32/64-bit integer values of length 2/3/4}}
%0 = spirv.BitwiseAnd %arg0, %arg1 : f16
return %0 : f16
}
// -----
//===----------------------------------------------------------------------===//
// spirv.Not
//===----------------------------------------------------------------------===//
func.func @not(%arg: i32) -> i32 {
// CHECK: spirv.Not {{%.*}} : i32
%0 = spirv.Not %arg : i32
spirv.ReturnValue %0 : i32
}
// -----
//===----------------------------------------------------------------------===//
// spirv.ShiftLeftLogical
//===----------------------------------------------------------------------===//
func.func @shift_left_logical(%arg0: i32, %arg1 : i16) -> i32 {
// CHECK: {{%.*}} = spirv.ShiftLeftLogical {{%.*}}, {{%.*}} : i32, i16
%0 = spirv.ShiftLeftLogical %arg0, %arg1: i32, i16
spirv.ReturnValue %0 : i32
}
// -----
func.func @shift_left_logical_invalid_result_type(%arg0: i32, %arg1 : i16) -> i16 {
// expected-error @+1 {{op failed to verify that all of {operand1, result} have same type}}
%0 = "spirv.ShiftLeftLogical" (%arg0, %arg1) : (i32, i16) -> (i16)
spirv.ReturnValue %0 : i16
}
// -----
//===----------------------------------------------------------------------===//
// spirv.ShiftRightArithmetic
//===----------------------------------------------------------------------===//
func.func @shift_right_arithmetic(%arg0: vector<4xi32>, %arg1 : vector<4xi8>) -> vector<4xi32> {
// CHECK: {{%.*}} = spirv.ShiftRightArithmetic {{%.*}}, {{%.*}} : vector<4xi32>, vector<4xi8>
%0 = spirv.ShiftRightArithmetic %arg0, %arg1: vector<4xi32>, vector<4xi8>
spirv.ReturnValue %0 : vector<4xi32>
}
// -----
//===----------------------------------------------------------------------===//
// spirv.ShiftRightLogical
//===----------------------------------------------------------------------===//
func.func @shift_right_logical(%arg0: vector<2xi32>, %arg1 : vector<2xi8>) -> vector<2xi32> {
// CHECK: {{%.*}} = spirv.ShiftRightLogical {{%.*}}, {{%.*}} : vector<2xi32>, vector<2xi8>
%0 = spirv.ShiftRightLogical %arg0, %arg1: vector<2xi32>, vector<2xi8>
spirv.ReturnValue %0 : vector<2xi32>
}