mirror of
https://github.com/ziglang/zig.git
synced 2026-02-20 16:24:51 +00:00
Merge branch 'merge-shawnl-simd5'
This is the commit from Shawn's SIMD patchset regarding `@splat`, plus my fixups.
This commit is contained in:
commit
1eb33966b1
@ -5864,7 +5864,7 @@ volatile (
|
||||
: [number] "{rax}" (number),
|
||||
[arg1] "{rdi}" (arg1)
|
||||
// Next is the list of clobbers. These declare a set of registers whose
|
||||
// values will not be preserved by the execution of this assembly code.
|
||||
// values will not be preserved by the execution of this assembly code.
|
||||
// These do not include output or input registers. The special clobber
|
||||
// value of "memory" means that the assembly writes to arbitrary undeclared
|
||||
// memory locations - not only the memory pointed to by a declared indirect
|
||||
@ -5885,7 +5885,7 @@ volatile (
|
||||
</p>
|
||||
{#header_open|Output Constraints#}
|
||||
<p>
|
||||
Output constraints are still considered to be unstable in Zig, and
|
||||
Output constraints are still considered to be unstable in Zig, and
|
||||
so
|
||||
<a href="http://releases.llvm.org/8.0.0/docs/LangRef.html#inline-asm-constraint-string">LLVM documentation</a>
|
||||
and
|
||||
@ -5900,7 +5900,7 @@ volatile (
|
||||
|
||||
{#header_open|Input Constraints#}
|
||||
<p>
|
||||
Input constraints are still considered to be unstable in Zig, and
|
||||
Input constraints are still considered to be unstable in Zig, and
|
||||
so
|
||||
<a href="http://releases.llvm.org/8.0.0/docs/LangRef.html#inline-asm-constraint-string">LLVM documentation</a>
|
||||
and
|
||||
@ -5919,7 +5919,7 @@ volatile (
|
||||
the assembly code. These do not include output or input registers. The special clobber
|
||||
value of {#syntax#}"memory"{#endsyntax#} means that the assembly causes writes to
|
||||
arbitrary undeclared memory locations - not only the memory pointed to by a declared
|
||||
indirect output.
|
||||
indirect output.
|
||||
</p>
|
||||
<p>
|
||||
Failure to declare the full set of clobbers for a given inline assembly
|
||||
@ -7746,6 +7746,30 @@ test "@setRuntimeSafety" {
|
||||
</p>
|
||||
{#header_close#}
|
||||
|
||||
{#header_open|@splat#}
|
||||
<pre>{#syntax#}@splat(comptime len: u32, scalar: var) @Vector(len, @typeOf(scalar)){#endsyntax#}</pre>
|
||||
<p>
|
||||
Produces a vector of length {#syntax#}len{#endsyntax#} where each element is the value
|
||||
{#syntax#}scalar{#endsyntax#}:
|
||||
</p>
|
||||
{#code_begin|test#}
|
||||
const std = @import("std");
|
||||
const assert = std.debug.assert;
|
||||
|
||||
test "vector @splat" {
|
||||
const scalar: u32 = 5;
|
||||
const result = @splat(4, scalar);
|
||||
comptime assert(@typeOf(result) == @Vector(4, u32));
|
||||
assert(std.mem.eql(u32, ([4]u32)(result), [_]u32{ 5, 5, 5, 5 }));
|
||||
}
|
||||
{#code_end#}
|
||||
<p>
|
||||
{#syntax#}scalar{#endsyntax#} must be an {#link|integer|Integers#}, {#link|bool|Primitive Types#},
|
||||
{#link|float|Floats#}, or {#link|pointer|Pointers#}.
|
||||
</p>
|
||||
{#see_also|Vectors|@shuffle#}
|
||||
{#header_close#}
|
||||
|
||||
{#header_open|@sqrt#}
|
||||
<pre>{#syntax#}@sqrt(comptime T: type, value: T) T{#endsyntax#}</pre>
|
||||
<p>
|
||||
@ -9456,8 +9480,8 @@ const c = @cImport({
|
||||
<li>Does not support Zig-only pointer attributes such as alignment. Use normal {#link|Pointers#}
|
||||
please!</li>
|
||||
</ul>
|
||||
<p>When a C pointer is pointing to a single struct (not an array), deference the C pointer to
|
||||
access to the struct's fields or member data. That syntax looks like
|
||||
<p>When a C pointer is pointing to a single struct (not an array), deference the C pointer to
|
||||
access to the struct's fields or member data. That syntax looks like
|
||||
this: </p>
|
||||
<p>{#syntax#}ptr_to_struct.*.struct_member{#endsyntax#}</p>
|
||||
<p>This is comparable to doing {#syntax#}->{#endsyntax#} in C.</p>
|
||||
|
||||
@ -1612,6 +1612,7 @@ enum BuiltinFnId {
|
||||
BuiltinFnIdIntType,
|
||||
BuiltinFnIdVectorType,
|
||||
BuiltinFnIdShuffle,
|
||||
BuiltinFnIdSplat,
|
||||
BuiltinFnIdSetCold,
|
||||
BuiltinFnIdSetRuntimeSafety,
|
||||
BuiltinFnIdSetFloatMode,
|
||||
@ -2431,6 +2432,8 @@ enum IrInstructionId {
|
||||
IrInstructionIdIntType,
|
||||
IrInstructionIdVectorType,
|
||||
IrInstructionIdShuffleVector,
|
||||
IrInstructionIdSplatSrc,
|
||||
IrInstructionIdSplatGen,
|
||||
IrInstructionIdBoolNot,
|
||||
IrInstructionIdMemset,
|
||||
IrInstructionIdMemcpy,
|
||||
@ -3681,6 +3684,19 @@ struct IrInstructionShuffleVector {
|
||||
IrInstruction *mask; // This is in zig-format, not llvm format
|
||||
};
|
||||
|
||||
struct IrInstructionSplatSrc {
|
||||
IrInstruction base;
|
||||
|
||||
IrInstruction *len;
|
||||
IrInstruction *scalar;
|
||||
};
|
||||
|
||||
struct IrInstructionSplatGen {
|
||||
IrInstruction base;
|
||||
|
||||
IrInstruction *scalar;
|
||||
};
|
||||
|
||||
struct IrInstructionAssertZero {
|
||||
IrInstruction base;
|
||||
|
||||
|
||||
@ -4619,6 +4619,18 @@ static LLVMValueRef ir_render_shuffle_vector(CodeGen *g, IrExecutable *executabl
|
||||
llvm_mask_value, "");
|
||||
}
|
||||
|
||||
static LLVMValueRef ir_render_splat(CodeGen *g, IrExecutable *executable, IrInstructionSplatGen *instruction) {
|
||||
ZigType *result_type = instruction->base.value.type;
|
||||
src_assert(result_type->id == ZigTypeIdVector, instruction->base.source_node);
|
||||
uint32_t len = result_type->data.vector.len;
|
||||
LLVMTypeRef op_llvm_type = LLVMVectorType(get_llvm_type(g, instruction->scalar->value.type), 1);
|
||||
LLVMTypeRef mask_llvm_type = LLVMVectorType(LLVMInt32Type(), len);
|
||||
LLVMValueRef undef_vector = LLVMGetUndef(op_llvm_type);
|
||||
LLVMValueRef op_vector = LLVMBuildInsertElement(g->builder, undef_vector,
|
||||
ir_llvm_value(g, instruction->scalar), LLVMConstInt(LLVMInt32Type(), 0, false), "");
|
||||
return LLVMBuildShuffleVector(g->builder, op_vector, undef_vector, LLVMConstNull(mask_llvm_type), "");
|
||||
}
|
||||
|
||||
static LLVMValueRef ir_render_pop_count(CodeGen *g, IrExecutable *executable, IrInstructionPopCount *instruction) {
|
||||
ZigType *int_type = instruction->op->value.type;
|
||||
LLVMValueRef fn_val = get_int_builtin_fn(g, int_type, BuiltinFnIdPopCount);
|
||||
@ -5986,6 +5998,7 @@ static LLVMValueRef ir_render_instruction(CodeGen *g, IrExecutable *executable,
|
||||
case IrInstructionIdFrameSizeSrc:
|
||||
case IrInstructionIdAllocaGen:
|
||||
case IrInstructionIdAwaitSrc:
|
||||
case IrInstructionIdSplatSrc:
|
||||
zig_unreachable();
|
||||
|
||||
case IrInstructionIdDeclVarGen:
|
||||
@ -6146,6 +6159,8 @@ static LLVMValueRef ir_render_instruction(CodeGen *g, IrExecutable *executable,
|
||||
return ir_render_spill_end(g, executable, (IrInstructionSpillEnd *)instruction);
|
||||
case IrInstructionIdShuffleVector:
|
||||
return ir_render_shuffle_vector(g, executable, (IrInstructionShuffleVector *) instruction);
|
||||
case IrInstructionIdSplatGen:
|
||||
return ir_render_splat(g, executable, (IrInstructionSplatGen *) instruction);
|
||||
}
|
||||
zig_unreachable();
|
||||
}
|
||||
@ -7837,6 +7852,7 @@ static void define_builtin_fns(CodeGen *g) {
|
||||
create_builtin_fn(g, BuiltinFnIdIntType, "IntType", 2); // TODO rename to Int
|
||||
create_builtin_fn(g, BuiltinFnIdVectorType, "Vector", 2);
|
||||
create_builtin_fn(g, BuiltinFnIdShuffle, "shuffle", 4);
|
||||
create_builtin_fn(g, BuiltinFnIdSplat, "splat", 2);
|
||||
create_builtin_fn(g, BuiltinFnIdSetCold, "setCold", 1);
|
||||
create_builtin_fn(g, BuiltinFnIdSetRuntimeSafety, "setRuntimeSafety", 1);
|
||||
create_builtin_fn(g, BuiltinFnIdSetFloatMode, "setFloatMode", 1);
|
||||
|
||||
111
src/ir.cpp
111
src/ir.cpp
@ -721,6 +721,14 @@ static constexpr IrInstructionId ir_instruction_id(IrInstructionShuffleVector *)
|
||||
return IrInstructionIdShuffleVector;
|
||||
}
|
||||
|
||||
static constexpr IrInstructionId ir_instruction_id(IrInstructionSplatSrc *) {
|
||||
return IrInstructionIdSplatSrc;
|
||||
}
|
||||
|
||||
static constexpr IrInstructionId ir_instruction_id(IrInstructionSplatGen *) {
|
||||
return IrInstructionIdSplatGen;
|
||||
}
|
||||
|
||||
static constexpr IrInstructionId ir_instruction_id(IrInstructionBoolNot *) {
|
||||
return IrInstructionIdBoolNot;
|
||||
}
|
||||
@ -2300,6 +2308,19 @@ static IrInstruction *ir_build_shuffle_vector(IrBuilder *irb, Scope *scope, AstN
|
||||
return &instruction->base;
|
||||
}
|
||||
|
||||
static IrInstruction *ir_build_splat_src(IrBuilder *irb, Scope *scope, AstNode *source_node,
|
||||
IrInstruction *len, IrInstruction *scalar)
|
||||
{
|
||||
IrInstructionSplatSrc *instruction = ir_build_instruction<IrInstructionSplatSrc>(irb, scope, source_node);
|
||||
instruction->len = len;
|
||||
instruction->scalar = scalar;
|
||||
|
||||
ir_ref_instruction(len, irb->current_basic_block);
|
||||
ir_ref_instruction(scalar, irb->current_basic_block);
|
||||
|
||||
return &instruction->base;
|
||||
}
|
||||
|
||||
static IrInstruction *ir_build_bool_not(IrBuilder *irb, Scope *scope, AstNode *source_node, IrInstruction *value) {
|
||||
IrInstructionBoolNot *instruction = ir_build_instruction<IrInstructionBoolNot>(irb, scope, source_node);
|
||||
instruction->value = value;
|
||||
@ -2356,6 +2377,19 @@ static IrInstruction *ir_build_slice_src(IrBuilder *irb, Scope *scope, AstNode *
|
||||
return &instruction->base;
|
||||
}
|
||||
|
||||
static IrInstruction *ir_build_splat_gen(IrAnalyze *ira, IrInstruction *source_instruction, ZigType *result_type,
|
||||
IrInstruction *scalar)
|
||||
{
|
||||
IrInstructionSplatGen *instruction = ir_build_instruction<IrInstructionSplatGen>(
|
||||
&ira->new_irb, source_instruction->scope, source_instruction->source_node);
|
||||
instruction->base.value.type = result_type;
|
||||
instruction->scalar = scalar;
|
||||
|
||||
ir_ref_instruction(scalar, ira->new_irb.current_basic_block);
|
||||
|
||||
return &instruction->base;
|
||||
}
|
||||
|
||||
static IrInstruction *ir_build_slice_gen(IrAnalyze *ira, IrInstruction *source_instruction, ZigType *slice_type,
|
||||
IrInstruction *ptr, IrInstruction *start, IrInstruction *end, bool safety_check_on, IrInstruction *result_loc)
|
||||
{
|
||||
@ -4985,6 +5019,22 @@ static IrInstruction *ir_gen_builtin_fn_call(IrBuilder *irb, Scope *scope, AstNo
|
||||
arg0_value, arg1_value, arg2_value, arg3_value);
|
||||
return ir_lval_wrap(irb, scope, shuffle_vector, lval, result_loc);
|
||||
}
|
||||
case BuiltinFnIdSplat:
|
||||
{
|
||||
AstNode *arg0_node = node->data.fn_call_expr.params.at(0);
|
||||
IrInstruction *arg0_value = ir_gen_node(irb, arg0_node, scope);
|
||||
if (arg0_value == irb->codegen->invalid_instruction)
|
||||
return arg0_value;
|
||||
|
||||
AstNode *arg1_node = node->data.fn_call_expr.params.at(1);
|
||||
IrInstruction *arg1_value = ir_gen_node(irb, arg1_node, scope);
|
||||
if (arg1_value == irb->codegen->invalid_instruction)
|
||||
return arg1_value;
|
||||
|
||||
IrInstruction *splat = ir_build_splat_src(irb, scope, node,
|
||||
arg0_value, arg1_value);
|
||||
return ir_lval_wrap(irb, scope, splat, lval, result_loc);
|
||||
}
|
||||
case BuiltinFnIdMemcpy:
|
||||
{
|
||||
AstNode *arg0_node = node->data.fn_call_expr.params.at(0);
|
||||
@ -11049,16 +11099,23 @@ static ZigType *ir_resolve_type(IrAnalyze *ira, IrInstruction *type_value) {
|
||||
return ir_resolve_const_type(ira->codegen, ira->new_irb.exec, type_value->source_node, val);
|
||||
}
|
||||
|
||||
static Error ir_validate_vector_elem_type(IrAnalyze *ira, IrInstruction *source_instr, ZigType *elem_type) {
|
||||
if (!is_valid_vector_elem_type(elem_type)) {
|
||||
ir_add_error(ira, source_instr,
|
||||
buf_sprintf("vector element type must be integer, float, bool, or pointer; '%s' is invalid",
|
||||
buf_ptr(&elem_type->name)));
|
||||
return ErrorSemanticAnalyzeFail;
|
||||
}
|
||||
return ErrorNone;
|
||||
}
|
||||
|
||||
static ZigType *ir_resolve_vector_elem_type(IrAnalyze *ira, IrInstruction *elem_type_value) {
|
||||
Error err;
|
||||
ZigType *elem_type = ir_resolve_type(ira, elem_type_value);
|
||||
if (type_is_invalid(elem_type))
|
||||
return ira->codegen->builtin_types.entry_invalid;
|
||||
if (!is_valid_vector_elem_type(elem_type)) {
|
||||
ir_add_error(ira, elem_type_value,
|
||||
buf_sprintf("vector element type must be integer, float, bool, or pointer; '%s' is invalid",
|
||||
buf_ptr(&elem_type->name)));
|
||||
if ((err = ir_validate_vector_elem_type(ira, elem_type_value, elem_type)))
|
||||
return ira->codegen->builtin_types.entry_invalid;
|
||||
}
|
||||
return elem_type;
|
||||
}
|
||||
|
||||
@ -22324,6 +22381,45 @@ static IrInstruction *ir_analyze_instruction_shuffle_vector(IrAnalyze *ira, IrIn
|
||||
return ir_analyze_shuffle_vector(ira, &instruction->base, scalar_type, a, b, mask);
|
||||
}
|
||||
|
||||
static IrInstruction *ir_analyze_instruction_splat(IrAnalyze *ira, IrInstructionSplatSrc *instruction) {
|
||||
Error err;
|
||||
|
||||
IrInstruction *len = instruction->len->child;
|
||||
if (type_is_invalid(len->value.type))
|
||||
return ira->codegen->invalid_instruction;
|
||||
|
||||
IrInstruction *scalar = instruction->scalar->child;
|
||||
if (type_is_invalid(scalar->value.type))
|
||||
return ira->codegen->invalid_instruction;
|
||||
|
||||
uint64_t len_u64;
|
||||
if (!ir_resolve_unsigned(ira, len, ira->codegen->builtin_types.entry_u32, &len_u64))
|
||||
return ira->codegen->invalid_instruction;
|
||||
uint32_t len_int = len_u64;
|
||||
|
||||
if ((err = ir_validate_vector_elem_type(ira, scalar, scalar->value.type)))
|
||||
return ira->codegen->invalid_instruction;
|
||||
|
||||
ZigType *return_type = get_vector_type(ira->codegen, len_int, scalar->value.type);
|
||||
|
||||
if (instr_is_comptime(scalar)) {
|
||||
ConstExprValue *scalar_val = ir_resolve_const(ira, scalar, UndefOk);
|
||||
if (scalar_val == nullptr)
|
||||
return ira->codegen->invalid_instruction;
|
||||
if (scalar_val->special == ConstValSpecialUndef)
|
||||
return ir_const_undef(ira, &instruction->base, return_type);
|
||||
|
||||
IrInstruction *result = ir_const(ira, &instruction->base, return_type);
|
||||
result->value.data.x_array.data.s_none.elements = create_const_vals(len_int);
|
||||
for (uint32_t i = 0; i < len_int; i += 1) {
|
||||
copy_const_val(&result->value.data.x_array.data.s_none.elements[i], scalar_val, false);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
return ir_build_splat_gen(ira, &instruction->base, return_type, scalar);
|
||||
}
|
||||
|
||||
static IrInstruction *ir_analyze_instruction_bool_not(IrAnalyze *ira, IrInstructionBoolNot *instruction) {
|
||||
IrInstruction *value = instruction->value->child;
|
||||
if (type_is_invalid(value->value.type))
|
||||
@ -25778,6 +25874,7 @@ static IrInstruction *ir_analyze_instruction_base(IrAnalyze *ira, IrInstruction
|
||||
case IrInstructionIdTestErrGen:
|
||||
case IrInstructionIdFrameSizeGen:
|
||||
case IrInstructionIdAwaitGen:
|
||||
case IrInstructionIdSplatGen:
|
||||
zig_unreachable();
|
||||
|
||||
case IrInstructionIdReturn:
|
||||
@ -25908,6 +26005,8 @@ static IrInstruction *ir_analyze_instruction_base(IrAnalyze *ira, IrInstruction
|
||||
return ir_analyze_instruction_vector_type(ira, (IrInstructionVectorType *)instruction);
|
||||
case IrInstructionIdShuffleVector:
|
||||
return ir_analyze_instruction_shuffle_vector(ira, (IrInstructionShuffleVector *)instruction);
|
||||
case IrInstructionIdSplatSrc:
|
||||
return ir_analyze_instruction_splat(ira, (IrInstructionSplatSrc *)instruction);
|
||||
case IrInstructionIdBoolNot:
|
||||
return ir_analyze_instruction_bool_not(ira, (IrInstructionBoolNot *)instruction);
|
||||
case IrInstructionIdMemset:
|
||||
@ -26244,6 +26343,8 @@ bool ir_has_side_effects(IrInstruction *instruction) {
|
||||
case IrInstructionIdIntType:
|
||||
case IrInstructionIdVectorType:
|
||||
case IrInstructionIdShuffleVector:
|
||||
case IrInstructionIdSplatSrc:
|
||||
case IrInstructionIdSplatGen:
|
||||
case IrInstructionIdBoolNot:
|
||||
case IrInstructionIdSliceSrc:
|
||||
case IrInstructionIdMemberCount:
|
||||
|
||||
@ -44,6 +44,10 @@ static const char* ir_instruction_type_str(IrInstruction* instruction) {
|
||||
return "Invalid";
|
||||
case IrInstructionIdShuffleVector:
|
||||
return "Shuffle";
|
||||
case IrInstructionIdSplatSrc:
|
||||
return "SplatSrc";
|
||||
case IrInstructionIdSplatGen:
|
||||
return "SplatGen";
|
||||
case IrInstructionIdDeclVarSrc:
|
||||
return "DeclVarSrc";
|
||||
case IrInstructionIdDeclVarGen:
|
||||
@ -1222,6 +1226,20 @@ static void ir_print_shuffle_vector(IrPrint *irp, IrInstructionShuffleVector *in
|
||||
fprintf(irp->f, ")");
|
||||
}
|
||||
|
||||
static void ir_print_splat_src(IrPrint *irp, IrInstructionSplatSrc *instruction) {
|
||||
fprintf(irp->f, "@splat(");
|
||||
ir_print_other_instruction(irp, instruction->len);
|
||||
fprintf(irp->f, ", ");
|
||||
ir_print_other_instruction(irp, instruction->scalar);
|
||||
fprintf(irp->f, ")");
|
||||
}
|
||||
|
||||
static void ir_print_splat_gen(IrPrint *irp, IrInstructionSplatGen *instruction) {
|
||||
fprintf(irp->f, "@splat(");
|
||||
ir_print_other_instruction(irp, instruction->scalar);
|
||||
fprintf(irp->f, ")");
|
||||
}
|
||||
|
||||
static void ir_print_bool_not(IrPrint *irp, IrInstructionBoolNot *instruction) {
|
||||
fprintf(irp->f, "! ");
|
||||
ir_print_other_instruction(irp, instruction->value);
|
||||
@ -2160,6 +2178,12 @@ static void ir_print_instruction(IrPrint *irp, IrInstruction *instruction, bool
|
||||
case IrInstructionIdShuffleVector:
|
||||
ir_print_shuffle_vector(irp, (IrInstructionShuffleVector *)instruction);
|
||||
break;
|
||||
case IrInstructionIdSplatSrc:
|
||||
ir_print_splat_src(irp, (IrInstructionSplatSrc *)instruction);
|
||||
break;
|
||||
case IrInstructionIdSplatGen:
|
||||
ir_print_splat_gen(irp, (IrInstructionSplatGen *)instruction);
|
||||
break;
|
||||
case IrInstructionIdBoolNot:
|
||||
ir_print_bool_not(irp, (IrInstructionBoolNot *)instruction);
|
||||
break;
|
||||
|
||||
@ -6507,6 +6507,16 @@ pub fn addCases(cases: *tests.CompileErrorContext) void {
|
||||
"tmp.zig:2:26: error: vector element type must be integer, float, bool, or pointer; '@Vector(4, u8)' is invalid",
|
||||
);
|
||||
|
||||
cases.addTest(
|
||||
"bad @splat type",
|
||||
\\export fn entry() void {
|
||||
\\ const c = 4;
|
||||
\\ var v = @splat(4, c);
|
||||
\\}
|
||||
,
|
||||
"tmp.zig:3:23: error: vector element type must be integer, float, bool, or pointer; 'comptime_int' is invalid",
|
||||
);
|
||||
|
||||
cases.add("compileLog of tagged enum doesn't crash the compiler",
|
||||
\\const Bar = union(enum(u32)) {
|
||||
\\ X: i32 = 1
|
||||
|
||||
@ -35,12 +35,12 @@ test "vector bin compares with mem.eql" {
|
||||
fn doTheTest() void {
|
||||
var v: @Vector(4, i32) = [4]i32{ 2147483647, -2, 30, 40 };
|
||||
var x: @Vector(4, i32) = [4]i32{ 1, 2147483647, 30, 4 };
|
||||
expect(mem.eql(bool, ([4]bool)(v == x), [4]bool{ false, false, true, false}));
|
||||
expect(mem.eql(bool, ([4]bool)(v != x), [4]bool{ true, true, false, true}));
|
||||
expect(mem.eql(bool, ([4]bool)(v < x), [4]bool{ false, true, false, false}));
|
||||
expect(mem.eql(bool, ([4]bool)(v > x), [4]bool{ true, false, false, true}));
|
||||
expect(mem.eql(bool, ([4]bool)(v <= x), [4]bool{ false, true, true, false}));
|
||||
expect(mem.eql(bool, ([4]bool)(v >= x), [4]bool{ true, false, true, true}));
|
||||
expect(mem.eql(bool, ([4]bool)(v == x), [4]bool{ false, false, true, false }));
|
||||
expect(mem.eql(bool, ([4]bool)(v != x), [4]bool{ true, true, false, true }));
|
||||
expect(mem.eql(bool, ([4]bool)(v < x), [4]bool{ false, true, false, false }));
|
||||
expect(mem.eql(bool, ([4]bool)(v > x), [4]bool{ true, false, false, true }));
|
||||
expect(mem.eql(bool, ([4]bool)(v <= x), [4]bool{ false, true, true, false }));
|
||||
expect(mem.eql(bool, ([4]bool)(v >= x), [4]bool{ true, false, true, true }));
|
||||
}
|
||||
};
|
||||
S.doTheTest();
|
||||
@ -114,22 +114,22 @@ test "vector casts of sizes not divisable by 8" {
|
||||
const S = struct {
|
||||
fn doTheTest() void {
|
||||
{
|
||||
var v: @Vector(4, u3) = [4]u3{ 5, 2, 3, 0};
|
||||
var v: @Vector(4, u3) = [4]u3{ 5, 2, 3, 0 };
|
||||
var x: [4]u3 = v;
|
||||
expect(mem.eql(u3, x, ([4]u3)(v)));
|
||||
}
|
||||
{
|
||||
var v: @Vector(4, u2) = [4]u2{ 1, 2, 3, 0};
|
||||
var v: @Vector(4, u2) = [4]u2{ 1, 2, 3, 0 };
|
||||
var x: [4]u2 = v;
|
||||
expect(mem.eql(u2, x, ([4]u2)(v)));
|
||||
}
|
||||
{
|
||||
var v: @Vector(4, u1) = [4]u1{ 1, 0, 1, 0};
|
||||
var v: @Vector(4, u1) = [4]u1{ 1, 0, 1, 0 };
|
||||
var x: [4]u1 = v;
|
||||
expect(mem.eql(u1, x, ([4]u1)(v)));
|
||||
}
|
||||
{
|
||||
var v: @Vector(4, bool) = [4]bool{ false, false, true, false};
|
||||
var v: @Vector(4, bool) = [4]bool{ false, false, true, false };
|
||||
var x: [4]bool = v;
|
||||
expect(mem.eql(bool, x, ([4]bool)(v)));
|
||||
}
|
||||
@ -138,3 +138,20 @@ test "vector casts of sizes not divisable by 8" {
|
||||
S.doTheTest();
|
||||
comptime S.doTheTest();
|
||||
}
|
||||
|
||||
test "vector @splat" {
|
||||
const S = struct {
|
||||
fn doTheTest() void {
|
||||
var v: u32 = 5;
|
||||
var x = @splat(4, v);
|
||||
expect(@typeOf(x) == @Vector(4, u32));
|
||||
var array_x: [4]u32 = x;
|
||||
expect(array_x[0] == 5);
|
||||
expect(array_x[1] == 5);
|
||||
expect(array_x[2] == 5);
|
||||
expect(array_x[3] == 5);
|
||||
}
|
||||
};
|
||||
S.doTheTest();
|
||||
comptime S.doTheTest();
|
||||
}
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user