aboutsummaryrefslogtreecommitdiff
path: root/src/codegen/spirv.zig
diff options
context:
space:
mode:
authorVeikka Tuominen <git@vexu.eu>2022-01-29 15:59:42 +0200
committerGitHub <noreply@github.com>2022-01-29 15:59:42 +0200
commit9f16d9ed07275209946b9e733c30be1bb0a1ae33 (patch)
treeb4a226f51cb417231dd8e8835173210abc4e7b94 /src/codegen/spirv.zig
parente288148f60770a2cfa4c64f832b599172c383d36 (diff)
parent98ee39d1b0ed516428c611d8dc1e52d21c786f97 (diff)
downloadzig-9f16d9ed07275209946b9e733c30be1bb0a1ae33.tar.gz
zig-9f16d9ed07275209946b9e733c30be1bb0a1ae33.zip
Merge pull request #10665 from Snektron/spirv-improvements
spir-v improvements
Diffstat (limited to 'src/codegen/spirv.zig')
-rw-r--r--src/codegen/spirv.zig741
1 files changed, 346 insertions, 395 deletions
diff --git a/src/codegen/spirv.zig b/src/codegen/spirv.zig
index b4f02a14a7..0d9d1ae223 100644
--- a/src/codegen/spirv.zig
+++ b/src/codegen/spirv.zig
@@ -4,9 +4,6 @@ const Target = std.Target;
const log = std.log.scoped(.codegen);
const assert = std.debug.assert;
-const spec = @import("spirv/spec.zig");
-const Opcode = spec.Opcode;
-
const Module = @import("../Module.zig");
const Decl = Module.Decl;
const Type = @import("../type.zig").Type;
@@ -15,187 +12,78 @@ const LazySrcLoc = Module.LazySrcLoc;
const Air = @import("../Air.zig");
const Liveness = @import("../Liveness.zig");
-pub const Word = u32;
-pub const ResultId = u32;
+const spec = @import("spirv/spec.zig");
+const Opcode = spec.Opcode;
+const Word = spec.Word;
+const IdRef = spec.IdRef;
+const IdResult = spec.IdResult;
+const IdResultType = spec.IdResultType;
+
+const SpvModule = @import("spirv/Module.zig");
+const SpvSection = @import("spirv/Section.zig");
+const SpvType = @import("spirv/type.zig").Type;
-pub const TypeMap = std.HashMap(Type, u32, Type.HashContext64, std.hash_map.default_max_load_percentage);
-pub const InstMap = std.AutoHashMap(Air.Inst.Index, ResultId);
+const InstMap = std.AutoHashMapUnmanaged(Air.Inst.Index, IdRef);
const IncomingBlock = struct {
- src_label_id: ResultId,
- break_value_id: ResultId,
+ src_label_id: IdRef,
+ break_value_id: IdRef,
};
-pub const BlockMap = std.AutoHashMap(Air.Inst.Index, struct {
- label_id: ResultId,
+pub const BlockMap = std.AutoHashMapUnmanaged(Air.Inst.Index, struct {
+ label_id: IdRef,
incoming_blocks: *std.ArrayListUnmanaged(IncomingBlock),
});
-pub fn writeOpcode(code: *std.ArrayList(Word), opcode: Opcode, arg_count: u16) !void {
- const word_count: Word = arg_count + 1;
- try code.append((word_count << 16) | @enumToInt(opcode));
-}
-
-pub fn writeInstruction(code: *std.ArrayList(Word), opcode: Opcode, args: []const Word) !void {
- try writeOpcode(code, opcode, @intCast(u16, args.len));
- try code.appendSlice(args);
-}
-
-pub fn writeInstructionWithString(code: *std.ArrayList(Word), opcode: Opcode, args: []const Word, str: []const u8) !void {
- // Str needs to be written zero-terminated, so we need to add one to the length.
- const zero_terminated_len = str.len + 1;
- const str_words = (zero_terminated_len + @sizeOf(Word) - 1) / @sizeOf(Word);
-
- try writeOpcode(code, opcode, @intCast(u16, args.len + str_words));
- try code.ensureUnusedCapacity(args.len + str_words);
- code.appendSliceAssumeCapacity(args);
-
- // TODO: Not actually sure whether this is correct for big-endian.
- // See https://www.khronos.org/registry/spir-v/specs/unified1/SPIRV.html#Literal
- var i: usize = 0;
- while (i < zero_terminated_len) : (i += @sizeOf(Word)) {
- var word: Word = 0;
-
- var j: usize = 0;
- while (j < @sizeOf(Word) and i + j < str.len) : (j += 1) {
- word |= @as(Word, str[i + j]) << @intCast(std.math.Log2Int(Word), j * std.meta.bitCount(u8));
- }
-
- code.appendAssumeCapacity(word);
- }
-}
-
-/// This structure represents a SPIR-V (binary) module being compiled, and keeps track of all relevant information.
-/// That includes the actual instructions, the current result-id bound, and data structures for querying result-id's
-/// of data which needs to be persistent over different calls to Decl code generation.
-pub const SPIRVModule = struct {
- /// A general-purpose allocator which may be used to allocate temporary resources required for compilation.
- gpa: Allocator,
-
- /// The parent module.
+/// This structure is used to compile a declaration, and contains all relevant meta-information to deal with that.
+pub const DeclGen = struct {
+ /// The Zig module that we are generating decls for.
module: *Module,
- /// SPIR-V instructions return result-ids. This variable holds the module-wide counter for these.
- next_result_id: ResultId,
-
- /// Code of the actual SPIR-V binary, divided into the relevant logical sections.
- /// Note: To save some bytes, these could also be unmanaged, but since there is only one instance of SPIRVModule
- /// and this removes some clutter in the rest of the backend, it's fine like this.
- binary: struct {
- /// OpCapability and OpExtension instructions (in that order).
- capabilities_and_extensions: std.ArrayList(Word),
-
- /// OpString, OpSourceExtension, OpSource, OpSourceContinued.
- debug_strings: std.ArrayList(Word),
-
- /// Type declaration instructions, constant instructions, global variable declarations, OpUndef instructions.
- types_globals_constants: std.ArrayList(Word),
-
- /// Regular functions.
- fn_decls: std.ArrayList(Word),
- },
-
- /// Global type cache to reduce the amount of generated types.
- types: TypeMap,
-
- /// Cache for results of OpString instructions for module file names fed to OpSource.
- /// Since OpString is pretty much only used for those, we don't need to keep track of all strings,
- /// just the ones for OpLine. Note that OpLine needs the result of OpString, and not that of OpSource.
- file_names: std.StringHashMap(ResultId),
-
- pub fn init(gpa: Allocator, module: *Module) SPIRVModule {
- return .{
- .gpa = gpa,
- .module = module,
- .next_result_id = 1, // 0 is an invalid SPIR-V result ID.
- .binary = .{
- .capabilities_and_extensions = std.ArrayList(Word).init(gpa),
- .debug_strings = std.ArrayList(Word).init(gpa),
- .types_globals_constants = std.ArrayList(Word).init(gpa),
- .fn_decls = std.ArrayList(Word).init(gpa),
- },
- .types = TypeMap.init(gpa),
- .file_names = std.StringHashMap(ResultId).init(gpa),
- };
- }
-
- pub fn deinit(self: *SPIRVModule) void {
- self.file_names.deinit();
- self.types.deinit();
-
- self.binary.fn_decls.deinit();
- self.binary.types_globals_constants.deinit();
- self.binary.debug_strings.deinit();
- self.binary.capabilities_and_extensions.deinit();
- }
-
- pub fn allocResultId(self: *SPIRVModule) Word {
- defer self.next_result_id += 1;
- return self.next_result_id;
- }
-
- pub fn resultIdBound(self: *SPIRVModule) Word {
- return self.next_result_id;
- }
-
- fn resolveSourceFileName(self: *SPIRVModule, decl: *Decl) !ResultId {
- const path = decl.getFileScope().sub_file_path;
- const result = try self.file_names.getOrPut(path);
- if (!result.found_existing) {
- result.value_ptr.* = self.allocResultId();
- try writeInstructionWithString(&self.binary.debug_strings, .OpString, &[_]Word{result.value_ptr.*}, path);
- try writeInstruction(&self.binary.debug_strings, .OpSource, &[_]Word{
- @enumToInt(spec.SourceLanguage.Unknown), // TODO: Register Zig source language.
- 0, // TODO: Zig version as u32?
- result.value_ptr.*,
- });
- }
+ /// The SPIR-V module code should be put in.
+ spv: *SpvModule,
- return result.value_ptr.*;
- }
-};
-
-/// This structure is used to compile a declaration, and contains all relevant meta-information to deal with that.
-pub const DeclGen = struct {
- /// The SPIR-V module code should be put in.
- spv: *SPIRVModule,
+ /// The decl we are currently generating code for.
+ decl: *Decl,
+ /// The intermediate code of the declaration we are currently generating. Note: If
+ /// the declaration is not a function, this value will be undefined!
air: Air,
+
+ /// The liveness analysis of the intermediate code for the declaration we are currently generating.
+ /// Note: If the declaration is not a function, this value will be undefined!
liveness: Liveness,
/// An array of function argument result-ids. Each index corresponds with the
/// function argument of the same index.
- args: std.ArrayList(ResultId),
+ args: std.ArrayListUnmanaged(IdRef) = .{},
/// A counter to keep track of how many `arg` instructions we've seen yet.
next_arg_index: u32,
/// A map keeping track of which instruction generated which result-id.
- inst_results: InstMap,
+ inst_results: InstMap = .{},
/// We need to keep track of result ids for block labels, as well as the 'incoming'
/// blocks for a block.
- blocks: BlockMap,
+ blocks: BlockMap = .{},
/// The label of the SPIR-V block we are currently generating.
- current_block_label_id: ResultId,
+ current_block_label_id: IdRef,
/// The actual instructions for this function. We need to declare all locals in
/// the first block, and because we don't know which locals there are going to be,
/// we're just going to generate everything after the locals-section in this array.
/// Note: It will not contain OpFunction, OpFunctionParameter, OpVariable and the
- /// initial OpLabel. These will be generated into spv.binary.fn_decls directly.
- code: std.ArrayList(Word),
+ /// initial OpLabel. These will be generated into spv.sections.functions directly.
+ code: SpvSection = .{},
- /// The decl we are currently generating code for.
- decl: *Decl,
-
- /// If `gen` returned `Error.AnalysisFail`, this contains an explanatory message.
+ /// If `gen` returned `Error.CodegenFail`, this contains an explanatory message.
/// Memory is owned by `module.gpa`.
error_msg: ?*Module.ErrorMsg,
/// Possible errors the `gen` function may return.
- const Error = error{ AnalysisFail, OutOfMemory };
+ const Error = error{ CodegenFail, OutOfMemory };
/// This structure is used to return information about a type typically used for
/// arithmetic operations. These types may either be integers, floats, or a vector
@@ -244,18 +132,15 @@ pub const DeclGen = struct {
/// Initialize the common resources of a DeclGen. Some fields are left uninitialized,
/// only set when `gen` is called.
- pub fn init(spv: *SPIRVModule) DeclGen {
+ pub fn init(module: *Module, spv: *SpvModule) DeclGen {
return .{
+ .module = module,
.spv = spv,
+ .decl = undefined,
.air = undefined,
.liveness = undefined,
- .args = std.ArrayList(ResultId).init(spv.gpa),
.next_arg_index = undefined,
- .inst_results = InstMap.init(spv.gpa),
- .blocks = BlockMap.init(spv.gpa),
.current_block_label_id = undefined,
- .code = std.ArrayList(Word).init(spv.gpa),
- .decl = undefined,
.error_msg = undefined,
};
}
@@ -265,6 +150,7 @@ pub const DeclGen = struct {
/// returns such a reportable error, it is valid to be called again for a different decl.
pub fn gen(self: *DeclGen, decl: *Decl, air: Air, liveness: Liveness) !?*Module.ErrorMsg {
// Reset internal resources, we don't want to re-allocate these.
+ self.decl = decl;
self.air = air;
self.liveness = liveness;
self.args.items.len = 0;
@@ -272,35 +158,50 @@ pub const DeclGen = struct {
self.inst_results.clearRetainingCapacity();
self.blocks.clearRetainingCapacity();
self.current_block_label_id = undefined;
- self.code.items.len = 0;
- self.decl = decl;
+ self.code.reset();
self.error_msg = null;
- try self.genDecl();
- return self.error_msg;
+ self.genDecl() catch |err| switch (err) {
+ error.CodegenFail => return self.error_msg,
+ else => |others| return others,
+ };
+
+ return null;
}
/// Free resources owned by the DeclGen.
pub fn deinit(self: *DeclGen) void {
- self.args.deinit();
- self.inst_results.deinit();
- self.blocks.deinit();
- self.code.deinit();
+ self.args.deinit(self.spv.gpa);
+ self.inst_results.deinit(self.spv.gpa);
+ self.blocks.deinit(self.spv.gpa);
+ self.code.deinit(self.spv.gpa);
}
+ /// Return the target which we are currently compiling for.
fn getTarget(self: *DeclGen) std.Target {
- return self.spv.module.getTarget();
+ return self.module.getTarget();
}
fn fail(self: *DeclGen, comptime format: []const u8, args: anytype) Error {
@setCold(true);
const src: LazySrcLoc = .{ .node_offset = 0 };
const src_loc = src.toSrcLoc(self.decl);
- self.error_msg = try Module.ErrorMsg.create(self.spv.module.gpa, src_loc, format, args);
- return error.AnalysisFail;
+ assert(self.error_msg == null);
+ self.error_msg = try Module.ErrorMsg.create(self.module.gpa, src_loc, format, args);
+ return error.CodegenFail;
}
- fn resolve(self: *DeclGen, inst: Air.Inst.Ref) !ResultId {
+ fn todo(self: *DeclGen, comptime format: []const u8, args: anytype) Error {
+ @setCold(true);
+ const src: LazySrcLoc = .{ .node_offset = 0 };
+ const src_loc = src.toSrcLoc(self.decl);
+ assert(self.error_msg == null);
+ self.error_msg = try Module.ErrorMsg.create(self.module.gpa, src_loc, "TODO (SPIR-V): " ++ format, args);
+ return error.CodegenFail;
+ }
+
+ /// Fetch the result-id for a previously generated instruction or constant.
+ fn resolve(self: *DeclGen, inst: Air.Inst.Ref) !IdRef {
if (self.air.value(inst)) |val| {
return self.genConstant(self.air.typeOf(inst), val);
}
@@ -308,9 +209,13 @@ pub const DeclGen = struct {
return self.inst_results.get(index).?; // Assertion means instruction does not dominate usage.
}
- fn beginSPIRVBlock(self: *DeclGen, label_id: ResultId) !void {
- try writeInstruction(&self.code, .OpLabel, &[_]Word{label_id});
- self.current_block_label_id = label_id;
+ /// Start a new SPIR-V block, Emits the label of the new block, and stores which
+ /// block we are currently generating.
+ /// Note that there is no such thing as nested blocks like in ZIR or AIR, so we don't need to
+ /// keep track of the previous block.
+ fn beginSpvBlock(self: *DeclGen, label_id: IdResult) !void {
+ try self.code.emit(self.spv.gpa, .OpLabel, .{ .id_result = label_id });
+ self.current_block_label_id = label_id.toRef();
}
/// SPIR-V requires enabling specific integer sizes through capabilities, and so if they are not enabled, we need
@@ -392,32 +297,37 @@ pub const DeclGen = struct {
const int_info = ty.intInfo(target);
// TODO: Maybe it's useful to also return this value.
const maybe_backing_bits = self.backingIntBits(int_info.bits);
- break :blk ArithmeticTypeInfo{ .bits = int_info.bits, .is_vector = false, .signedness = int_info.signedness, .class = if (maybe_backing_bits) |backing_bits|
- if (backing_bits == int_info.bits)
- ArithmeticTypeInfo.Class.integer
+ break :blk ArithmeticTypeInfo{
+ .bits = int_info.bits,
+ .is_vector = false,
+ .signedness = int_info.signedness,
+ .class = if (maybe_backing_bits) |backing_bits|
+ if (backing_bits == int_info.bits)
+ ArithmeticTypeInfo.Class.integer
+ else
+ ArithmeticTypeInfo.Class.strange_integer
else
- ArithmeticTypeInfo.Class.strange_integer
- else
- .composite_integer };
+ .composite_integer,
+ };
},
// As of yet, there is no vector support in the self-hosted compiler.
- .Vector => self.fail("TODO: SPIR-V backend: implement arithmeticTypeInfo for Vector", .{}),
+ .Vector => self.todo("implement arithmeticTypeInfo for Vector", .{}),
// TODO: For which types is this the case?
- else => self.fail("TODO: SPIR-V backend: implement arithmeticTypeInfo for {}", .{ty}),
+ else => self.todo("implement arithmeticTypeInfo for {}", .{ty}),
};
}
/// Generate a constant representing `val`.
/// TODO: Deduplication?
- fn genConstant(self: *DeclGen, ty: Type, val: Value) Error!ResultId {
+ fn genConstant(self: *DeclGen, ty: Type, val: Value) Error!IdRef {
const target = self.getTarget();
- const code = &self.spv.binary.types_globals_constants;
- const result_id = self.spv.allocResultId();
- const result_type_id = try self.genType(ty);
+ const section = &self.spv.sections.types_globals_constants;
+ const result_id = self.spv.allocId();
+ const result_type_id = try self.resolveTypeId(ty);
if (val.isUndef()) {
- try writeInstruction(code, .OpUndef, &[_]Word{ result_type_id, result_id });
- return result_id;
+ try section.emit(self.spv.gpa, .OpUndef, .{ .id_result_type = result_type_id, .id_result = result_id });
+ return result_id.toRef();
}
switch (ty.zigTypeTag()) {
@@ -425,101 +335,96 @@ pub const DeclGen = struct {
const int_info = ty.intInfo(target);
const backing_bits = self.backingIntBits(int_info.bits) orelse {
// Integers too big for any native type are represented as "composite integers": An array of largestSupportedIntBits.
- return self.fail("TODO: SPIR-V backend: implement composite int constants for {}", .{ty});
+ return self.todo("implement composite int constants for {}", .{ty});
};
// We can just use toSignedInt/toUnsignedInt here as it returns u64 - a type large enough to hold any
// SPIR-V native type (up to i/u64 with Int64). If SPIR-V ever supports native ints of a larger size, this
// might need to be updated.
assert(self.largestSupportedIntBits() <= std.meta.bitCount(u64));
+
+ // Note, value is required to be sign-extended, so we don't need to mask off the upper bits.
+ // See https://www.khronos.org/registry/SPIR-V/specs/unified1/SPIRV.html#Literal
var int_bits = if (ty.isSignedInt()) @bitCast(u64, val.toSignedInt()) else val.toUnsignedInt();
- // Mask the low bits which make up the actual integer. This is to make sure that negative values
- // only use the actual bits of the type.
- // TODO: Should this be the backing type bits or the actual type bits?
- int_bits &= (@as(u64, 1) << @intCast(u6, backing_bits)) - 1;
-
- switch (backing_bits) {
- 0 => unreachable,
- 1...32 => try writeInstruction(code, .OpConstant, &[_]Word{
- result_type_id,
- result_id,
- @truncate(u32, int_bits),
- }),
- 33...64 => try writeInstruction(code, .OpConstant, &[_]Word{
- result_type_id,
- result_id,
- @truncate(u32, int_bits),
- @truncate(u32, int_bits >> @bitSizeOf(u32)),
- }),
- else => unreachable, // backing_bits is bounded by largestSupportedIntBits.
- }
+ const value: spec.LiteralContextDependentNumber = switch (backing_bits) {
+ 1...32 => .{ .uint32 = @truncate(u32, int_bits) },
+ 33...64 => .{ .uint64 = int_bits },
+ else => unreachable,
+ };
+
+ try section.emit(self.spv.gpa, .OpConstant, .{
+ .id_result_type = result_type_id,
+ .id_result = result_id,
+ .value = value,
+ });
},
.Bool => {
- const opcode: Opcode = if (val.toBool()) .OpConstantTrue else .OpConstantFalse;
- try writeInstruction(code, opcode, &[_]Word{ result_type_id, result_id });
+ const operands = .{ .id_result_type = result_type_id, .id_result = result_id };
+ if (val.toBool()) {
+ try section.emit(self.spv.gpa, .OpConstantTrue, operands);
+ } else {
+ try section.emit(self.spv.gpa, .OpConstantFalse, operands);
+ }
},
.Float => {
// At this point we are guaranteed that the target floating point type is supported, otherwise the function
- // would have exited at genType(ty).
-
- // f16 and f32 require one word of storage. f64 requires 2, low-order first.
-
- switch (ty.floatBits(target)) {
- 16 => try writeInstruction(code, .OpConstant, &[_]Word{ result_type_id, result_id, @bitCast(u16, val.toFloat(f16)) }),
- 32 => try writeInstruction(code, .OpConstant, &[_]Word{ result_type_id, result_id, @bitCast(u32, val.toFloat(f32)) }),
- 64 => {
- const float_bits = @bitCast(u64, val.toFloat(f64));
- try writeInstruction(code, .OpConstant, &[_]Word{
- result_type_id,
- result_id,
- @truncate(u32, float_bits),
- @truncate(u32, float_bits >> @bitSizeOf(u32)),
- });
- },
- 128 => unreachable, // Filtered out in the call to genType.
- // TODO: Insert case for long double when the layout for that is determined.
+ // would have exited at resolveTypeId(ty).
+
+ const value: spec.LiteralContextDependentNumber = switch (ty.floatBits(target)) {
+ // Prevent upcasting to f32 by bitcasting and writing as a uint32.
+ 16 => .{ .uint32 = @bitCast(u16, val.toFloat(f16)) },
+ 32 => .{ .float32 = val.toFloat(f32) },
+ 64 => .{ .float64 = val.toFloat(f64) },
+ 128 => unreachable, // Filtered out in the call to resolveTypeId.
+ // TODO: Insert case for long double when the layout for that is determined?
else => unreachable,
- }
+ };
+
+ try section.emit(self.spv.gpa, .OpConstant, .{
+ .id_result_type = result_type_id,
+ .id_result = result_id,
+ .value = value,
+ });
},
.Void => unreachable,
- else => return self.fail("TODO: SPIR-V backend: constant generation of type {}", .{ty}),
+ else => return self.todo("constant generation of type {}", .{ty}),
}
- return result_id;
+ return result_id.toRef();
}
- fn genType(self: *DeclGen, ty: Type) Error!ResultId {
- // We can't use getOrPut here so we can recursively generate types.
- if (self.spv.types.get(ty)) |already_generated| {
- return already_generated;
- }
+ /// Turn a Zig type into a SPIR-V Type, and return its type result-id.
+ fn resolveTypeId(self: *DeclGen, ty: Type) !IdResultType {
+ return self.spv.typeResultId(try self.resolveType(ty));
+ }
+ /// Turn a Zig type into a SPIR-V Type, and return a reference to it.
+ fn resolveType(self: *DeclGen, ty: Type) Error!SpvType.Ref {
const target = self.getTarget();
- const code = &self.spv.binary.types_globals_constants;
- const result_id = self.spv.allocResultId();
-
- switch (ty.zigTypeTag()) {
- .Void => try writeInstruction(code, .OpTypeVoid, &[_]Word{result_id}),
- .Bool => try writeInstruction(code, .OpTypeBool, &[_]Word{result_id}),
- .Int => {
+ return switch (ty.zigTypeTag()) {
+ .Void => try self.spv.resolveType(SpvType.initTag(.void)),
+ .Bool => blk: {
+ // TODO: SPIR-V booleans are opaque. For local variables this is fine, but for structs
+ // members we want to use integer types instead.
+ break :blk try self.spv.resolveType(SpvType.initTag(.bool));
+ },
+ .Int => blk: {
const int_info = ty.intInfo(target);
const backing_bits = self.backingIntBits(int_info.bits) orelse {
- // Integers too big for any native type are represented as "composite integers": An array of largestSupportedIntBits.
- return self.fail("TODO: SPIR-V backend: implement composite int {}", .{ty});
+ // TODO: Integers too big for any native type are represented as "composite integers":
+ // An array of largestSupportedIntBits.
+ return self.todo("Implement composite int type {}", .{ty});
};
- // TODO: If backing_bits != int_info.bits, a duplicate type might be generated here.
- try writeInstruction(code, .OpTypeInt, &[_]Word{
- result_id,
- backing_bits,
- switch (int_info.signedness) {
- .unsigned => 0,
- .signed => 1,
- },
- });
+ const payload = try self.spv.arena.create(SpvType.Payload.Int);
+ payload.* = .{
+ .width = backing_bits,
+ .signedness = int_info.signedness,
+ };
+ break :blk try self.spv.resolveType(SpvType.initPayload(&payload.base));
},
- .Float => {
+ .Float => blk: {
// We can (and want) not really emulate floating points with other floating point types like with the integer types,
// so if the float is not supported, just return an error.
const bits = ty.floatBits(target);
@@ -535,37 +440,34 @@ pub const DeclGen = struct {
return self.fail("Floating point width of {} bits is not supported for the current SPIR-V feature set", .{bits});
}
- try writeInstruction(code, .OpTypeFloat, &[_]Word{ result_id, bits });
+ const payload = try self.spv.arena.create(SpvType.Payload.Float);
+ payload.* = .{
+ .width = bits,
+ };
+ break :blk try self.spv.resolveType(SpvType.initPayload(&payload.base));
},
- .Fn => {
+ .Fn => blk: {
// We only support zig-calling-convention functions, no varargs.
if (ty.fnCallingConvention() != .Unspecified)
return self.fail("Unsupported calling convention for SPIR-V", .{});
if (ty.fnIsVarArgs())
- return self.fail("VarArgs unsupported for SPIR-V", .{});
-
- // In order to avoid a temporary here, first generate all the required types and then simply look them up
- // when generating the function type.
- const params = ty.fnParamLen();
- var i: usize = 0;
- while (i < params) : (i += 1) {
- _ = try self.genType(ty.fnParamType(i));
- }
+ return self.fail("VarArgs functions are unsupported for SPIR-V", .{});
- const return_type_id = try self.genType(ty.fnReturnType());
+ const param_types = try self.spv.arena.alloc(SpvType.Ref, ty.fnParamLen());
+ for (param_types) |*param, i| {
+ param.* = try self.resolveType(ty.fnParamType(i));
+ }
- // result id + result type id + parameter type ids.
- try writeOpcode(code, .OpTypeFunction, 2 + @intCast(u16, ty.fnParamLen()));
- try code.appendSlice(&.{ result_id, return_type_id });
+ const return_type = try self.resolveType(ty.fnReturnType());
- i = 0;
- while (i < params) : (i += 1) {
- const param_type_id = self.spv.types.get(ty.fnParamType(i)).?;
- try code.append(param_type_id);
- }
+ const payload = try self.spv.arena.create(SpvType.Payload.Function);
+ payload.* = .{ .return_type = return_type, .parameters = param_types };
+ break :blk try self.spv.resolveType(SpvType.initPayload(&payload.base));
+ },
+ .Pointer => {
+ // This type can now be properly implemented, but we still need to implement the storage classes as proper address spaces.
+ return self.todo("Implement type Pointer properly", .{});
},
- // When recursively generating a type, we cannot infer the pointer's storage class. See genPointerType.
- .Pointer => return self.fail("Cannot create pointer with unknown storage class", .{}),
.Vector => {
// Although not 100% the same, Zig vectors map quite neatly to SPIR-V vectors (including many integer and float operations
// which work on them), so simply use those.
@@ -575,41 +477,42 @@ pub const DeclGen = struct {
// is adequate at all for this.
// TODO: Vectors are not yet supported by the self-hosted compiler itself it seems.
- return self.fail("TODO: SPIR-V backend: implement type Vector", .{});
+ return self.todo("Implement type Vector", .{});
},
+
.Null,
.Undefined,
.EnumLiteral,
.ComptimeFloat,
.ComptimeInt,
.Type,
- => unreachable, // Must be const or comptime.
+ => unreachable, // Must be comptime.
.BoundFn => unreachable, // this type will be deleted from the language.
- else => |tag| return self.fail("TODO: SPIR-V backend: implement type {}s", .{tag}),
- }
-
- try self.spv.types.putNoClobber(ty, result_id);
- return result_id;
+ else => |tag| return self.todo("Implement zig type '{}'", .{tag}),
+ };
}
/// SPIR-V requires pointers to have a storage class (address space), and so we have a special function for that.
/// TODO: The result of this needs to be cached.
- fn genPointerType(self: *DeclGen, ty: Type, storage_class: spec.StorageClass) !ResultId {
+ fn genPointerType(self: *DeclGen, ty: Type, storage_class: spec.StorageClass) !IdResultType {
assert(ty.zigTypeTag() == .Pointer);
- const code = &self.spv.binary.types_globals_constants;
- const result_id = self.spv.allocResultId();
+ const result_id = self.spv.allocId();
// TODO: There are many constraints which are ignored for now: We may only create pointers to certain types, and to other types
// if more capabilities are enabled. For example, we may only create pointers to f16 if Float16Buffer is enabled.
// These also relates to the pointer's address space.
- const child_id = try self.genType(ty.elemType());
+ const child_id = try self.resolveTypeId(ty.elemType());
- try writeInstruction(code, .OpTypePointer, &[_]Word{ result_id, @enumToInt(storage_class), child_id });
+ try self.spv.sections.types_globals_constants.emit(self.spv.gpa, .OpTypePointer, .{
+ .id_result = result_id,
+ .storage_class = storage_class,
+ .type = child_id.toRef(),
+ });
- return result_id;
+ return result_id.toResultType();
}
fn genDecl(self: *DeclGen) !void {
@@ -618,41 +521,47 @@ pub const DeclGen = struct {
if (decl.val.castTag(.function)) |_| {
assert(decl.ty.zigTypeTag() == .Fn);
- const prototype_id = try self.genType(decl.ty);
- try writeInstruction(&self.spv.binary.fn_decls, .OpFunction, &[_]Word{
- self.spv.types.get(decl.ty.fnReturnType()).?, // This type should be generated along with the prototype.
- result_id,
- @bitCast(Word, spec.FunctionControl{}), // TODO: We can set inline here if the type requires it.
- prototype_id,
+ const prototype_id = try self.resolveTypeId(decl.ty);
+ try self.spv.sections.functions.emit(self.spv.gpa, .OpFunction, .{
+ .id_result_type = try self.resolveTypeId(decl.ty.fnReturnType()),
+ .id_result = result_id,
+ .function_control = .{}, // TODO: We can set inline here if the type requires it.
+ .function_type = prototype_id.toRef(),
});
const params = decl.ty.fnParamLen();
var i: usize = 0;
- try self.args.ensureUnusedCapacity(params);
+ try self.args.ensureUnusedCapacity(self.spv.gpa, params);
while (i < params) : (i += 1) {
- const param_type_id = self.spv.types.get(decl.ty.fnParamType(i)).?;
- const arg_result_id = self.spv.allocResultId();
- try writeInstruction(&self.spv.binary.fn_decls, .OpFunctionParameter, &[_]Word{ param_type_id, arg_result_id });
- self.args.appendAssumeCapacity(arg_result_id);
+ const param_type_id = try self.resolveTypeId(decl.ty.fnParamType(i));
+ const arg_result_id = self.spv.allocId();
+ try self.spv.sections.functions.emit(self.spv.gpa, .OpFunctionParameter, .{
+ .id_result_type = param_type_id,
+ .id_result = arg_result_id,
+ });
+ self.args.appendAssumeCapacity(arg_result_id.toRef());
}
// TODO: This could probably be done in a better way...
- const root_block_id = self.spv.allocResultId();
+ const root_block_id = self.spv.allocId();
- // We need to generate the label directly in the fn_decls here because we're going to write the local variables after
- // here. Since we're not generating in self.code, we're just going to bypass self.beginSPIRVBlock here.
- try writeInstruction(&self.spv.binary.fn_decls, .OpLabel, &[_]Word{root_block_id});
- self.current_block_label_id = root_block_id;
+ // We need to generate the label directly in the functions section here because we're going to write the local variables after
+ // here. Since we're not generating in self.code, we're just going to bypass self.beginSpvBlock here.
+ try self.spv.sections.functions.emit(self.spv.gpa, .OpLabel, .{
+ .id_result = root_block_id,
+ });
+ self.current_block_label_id = root_block_id.toRef();
const main_body = self.air.getMainBody();
try self.genBody(main_body);
- // Append the actual code into the fn_decls section.
- try self.spv.binary.fn_decls.appendSlice(self.code.items);
- try writeInstruction(&self.spv.binary.fn_decls, .OpFunctionEnd, &[_]Word{});
+ // Append the actual code into the functions section.
+ try self.spv.sections.functions.append(self.spv.gpa, self.code);
+ try self.spv.sections.functions.emit(self.spv.gpa, .OpFunctionEnd, {});
} else {
- return self.fail("TODO: SPIR-V backend: generate decl type {}", .{decl.ty.zigTypeTag()});
+ // TODO
+ // return self.todo("generate decl type {}", .{decl.ty.zigTypeTag()});
}
}
@@ -666,9 +575,9 @@ pub const DeclGen = struct {
const air_tags = self.air.instructions.items(.tag);
const result_id = switch (air_tags[inst]) {
// zig fmt: off
- .add, .addwrap => try self.airArithOp(inst, .{.OpFAdd, .OpIAdd, .OpIAdd}),
- .sub, .subwrap => try self.airArithOp(inst, .{.OpFSub, .OpISub, .OpISub}),
- .mul, .mulwrap => try self.airArithOp(inst, .{.OpFMul, .OpIMul, .OpIMul}),
+ .add, .addwrap => try self.airArithOp(inst, .OpFAdd, .OpIAdd, .OpIAdd),
+ .sub, .subwrap => try self.airArithOp(inst, .OpFSub, .OpISub, .OpISub),
+ .mul, .mulwrap => try self.airArithOp(inst, .OpFMul, .OpIMul, .OpIMul),
.bit_and => try self.airBinOpSimple(inst, .OpBitwiseAnd),
.bit_or => try self.airBinOpSimple(inst, .OpBitwiseOr),
@@ -678,12 +587,12 @@ pub const DeclGen = struct {
.not => try self.airNot(inst),
- .cmp_eq => try self.airCmp(inst, .{.OpFOrdEqual, .OpLogicalEqual, .OpIEqual}),
- .cmp_neq => try self.airCmp(inst, .{.OpFOrdNotEqual, .OpLogicalNotEqual, .OpINotEqual}),
- .cmp_gt => try self.airCmp(inst, .{.OpFOrdGreaterThan, .OpSGreaterThan, .OpUGreaterThan}),
- .cmp_gte => try self.airCmp(inst, .{.OpFOrdGreaterThanEqual, .OpSGreaterThanEqual, .OpUGreaterThanEqual}),
- .cmp_lt => try self.airCmp(inst, .{.OpFOrdLessThan, .OpSLessThan, .OpULessThan}),
- .cmp_lte => try self.airCmp(inst, .{.OpFOrdLessThanEqual, .OpSLessThanEqual, .OpULessThanEqual}),
+ .cmp_eq => try self.airCmp(inst, .OpFOrdEqual, .OpLogicalEqual, .OpIEqual),
+ .cmp_neq => try self.airCmp(inst, .OpFOrdNotEqual, .OpLogicalNotEqual, .OpINotEqual),
+ .cmp_gt => try self.airCmp(inst, .OpFOrdGreaterThan, .OpSGreaterThan, .OpUGreaterThan),
+ .cmp_gte => try self.airCmp(inst, .OpFOrdGreaterThanEqual, .OpSGreaterThanEqual, .OpUGreaterThanEqual),
+ .cmp_lt => try self.airCmp(inst, .OpFOrdLessThan, .OpSLessThan, .OpULessThan),
+ .cmp_lte => try self.airCmp(inst, .OpFOrdLessThanEqual, .OpSLessThanEqual, .OpULessThanEqual),
.arg => self.airArg(),
.alloc => try self.airAlloc(inst),
@@ -701,27 +610,30 @@ pub const DeclGen = struct {
.unreach => return self.airUnreach(),
// zig fmt: on
- else => |tag| return self.fail("TODO: SPIR-V backend: implement AIR tag {s}", .{
+ else => |tag| return self.todo("implement AIR tag {s}", .{
@tagName(tag),
}),
};
- try self.inst_results.putNoClobber(inst, result_id);
+ try self.inst_results.putNoClobber(self.spv.gpa, inst, result_id);
}
- fn airBinOpSimple(self: *DeclGen, inst: Air.Inst.Index, opcode: Opcode) !ResultId {
+ fn airBinOpSimple(self: *DeclGen, inst: Air.Inst.Index, comptime opcode: Opcode) !IdRef {
const bin_op = self.air.instructions.items(.data)[inst].bin_op;
const lhs_id = try self.resolve(bin_op.lhs);
const rhs_id = try self.resolve(bin_op.rhs);
- const result_id = self.spv.allocResultId();
- const result_type_id = try self.genType(self.air.typeOfIndex(inst));
- try writeInstruction(&self.code, opcode, &[_]Word{
- result_type_id, result_id, lhs_id, rhs_id,
+ const result_id = self.spv.allocId();
+ const result_type_id = try self.resolveTypeId(self.air.typeOfIndex(inst));
+ try self.code.emit(self.spv.gpa, opcode, .{
+ .id_result_type = result_type_id,
+ .id_result = result_id,
+ .operand_1 = lhs_id,
+ .operand_2 = rhs_id,
});
- return result_id;
+ return result_id.toRef();
}
- fn airArithOp(self: *DeclGen, inst: Air.Inst.Index, ops: [3]Opcode) !ResultId {
+ fn airArithOp(self: *DeclGen, inst: Air.Inst.Index, comptime fop: Opcode, comptime sop: Opcode, comptime uop: Opcode) !IdRef {
// LHS and RHS are guaranteed to have the same type, and AIR guarantees
// the result to be the same as the LHS and RHS, which matches SPIR-V.
const ty = self.air.typeOfIndex(inst);
@@ -729,8 +641,8 @@ pub const DeclGen = struct {
const lhs_id = try self.resolve(bin_op.lhs);
const rhs_id = try self.resolve(bin_op.rhs);
- const result_id = self.spv.allocResultId();
- const result_type_id = try self.genType(ty);
+ const result_id = self.spv.allocId();
+ const result_type_id = try self.resolveTypeId(ty);
assert(self.air.typeOf(bin_op.lhs).eql(ty));
assert(self.air.typeOf(bin_op.rhs).eql(ty));
@@ -741,10 +653,10 @@ pub const DeclGen = struct {
const opcode_index: usize = switch (info.class) {
.composite_integer => {
- return self.fail("TODO: SPIR-V backend: binary operations for composite integers", .{});
+ return self.todo("binary operations for composite integers", .{});
},
.strange_integer => {
- return self.fail("TODO: SPIR-V backend: binary operations for strange integers", .{});
+ return self.todo("binary operations for strange integers", .{});
},
.integer => switch (info.signedness) {
.signed => @as(usize, 1),
@@ -753,21 +665,32 @@ pub const DeclGen = struct {
.float => 0,
else => unreachable,
};
- const opcode = ops[opcode_index];
- try writeInstruction(&self.code, opcode, &[_]Word{ result_type_id, result_id, lhs_id, rhs_id });
+ const operands = .{
+ .id_result_type = result_type_id,
+ .id_result = result_id,
+ .operand_1 = lhs_id,
+ .operand_2 = rhs_id,
+ };
+
+ switch (opcode_index) {
+ 0 => try self.code.emit(self.spv.gpa, fop, operands),
+ 1 => try self.code.emit(self.spv.gpa, sop, operands),
+ 2 => try self.code.emit(self.spv.gpa, uop, operands),
+ else => unreachable,
+ }
// TODO: Trap on overflow? Probably going to be annoying.
// TODO: Look into SPV_KHR_no_integer_wrap_decoration which provides NoSignedWrap/NoUnsignedWrap.
- return result_id;
+ return result_id.toRef();
}
- fn airCmp(self: *DeclGen, inst: Air.Inst.Index, ops: [3]Opcode) !ResultId {
+ fn airCmp(self: *DeclGen, inst: Air.Inst.Index, comptime fop: Opcode, comptime sop: Opcode, comptime uop: Opcode) !IdRef {
const bin_op = self.air.instructions.items(.data)[inst].bin_op;
const lhs_id = try self.resolve(bin_op.lhs);
const rhs_id = try self.resolve(bin_op.rhs);
- const result_id = self.spv.allocResultId();
- const result_type_id = try self.genType(Type.initTag(.bool));
+ const result_id = self.spv.allocId();
+ const result_type_id = try self.resolveTypeId(Type.initTag(.bool));
const op_ty = self.air.typeOf(bin_op.lhs);
assert(op_ty.eql(self.air.typeOf(bin_op.rhs)));
@@ -777,10 +700,10 @@ pub const DeclGen = struct {
const opcode_index: usize = switch (info.class) {
.composite_integer => {
- return self.fail("TODO: SPIR-V backend: binary operations for composite integers", .{});
+ return self.todo("binary operations for composite integers", .{});
},
.strange_integer => {
- return self.fail("TODO: SPIR-V backend: comparison for strange integers", .{});
+ return self.todo("comparison for strange integers", .{});
},
.float => 0,
.bool => 1,
@@ -789,53 +712,71 @@ pub const DeclGen = struct {
.unsigned => @as(usize, 2),
},
};
- const opcode = ops[opcode_index];
- try writeInstruction(&self.code, opcode, &[_]Word{ result_type_id, result_id, lhs_id, rhs_id });
- return result_id;
+ const operands = .{
+ .id_result_type = result_type_id,
+ .id_result = result_id,
+ .operand_1 = lhs_id,
+ .operand_2 = rhs_id,
+ };
+
+ switch (opcode_index) {
+ 0 => try self.code.emit(self.spv.gpa, fop, operands),
+ 1 => try self.code.emit(self.spv.gpa, sop, operands),
+ 2 => try self.code.emit(self.spv.gpa, uop, operands),
+ else => unreachable,
+ }
+
+ return result_id.toRef();
}
- fn airNot(self: *DeclGen, inst: Air.Inst.Index) !ResultId {
+ fn airNot(self: *DeclGen, inst: Air.Inst.Index) !IdRef {
const ty_op = self.air.instructions.items(.data)[inst].ty_op;
const operand_id = try self.resolve(ty_op.operand);
- const result_id = self.spv.allocResultId();
- const result_type_id = try self.genType(Type.initTag(.bool));
- const opcode: Opcode = .OpLogicalNot;
- try writeInstruction(&self.code, opcode, &[_]Word{ result_type_id, result_id, operand_id });
- return result_id;
+ const result_id = self.spv.allocId();
+ const result_type_id = try self.resolveTypeId(Type.initTag(.bool));
+ try self.code.emit(self.spv.gpa, .OpLogicalNot, .{
+ .id_result_type = result_type_id,
+ .id_result = result_id,
+ .operand = operand_id,
+ });
+ return result_id.toRef();
}
- fn airAlloc(self: *DeclGen, inst: Air.Inst.Index) !ResultId {
+ fn airAlloc(self: *DeclGen, inst: Air.Inst.Index) !IdRef {
const ty = self.air.typeOfIndex(inst);
const storage_class = spec.StorageClass.Function;
const result_type_id = try self.genPointerType(ty, storage_class);
- const result_id = self.spv.allocResultId();
+ const result_id = self.spv.allocId();
- // Rather than generating into code here, we're just going to generate directly into the fn_decls section so that
+ // Rather than generating into code here, we're just going to generate directly into the functions section so that
// variable declarations appear in the first block of the function.
- try writeInstruction(&self.spv.binary.fn_decls, .OpVariable, &[_]Word{ result_type_id, result_id, @enumToInt(storage_class) });
-
- return result_id;
+ try self.spv.sections.functions.emit(self.spv.gpa, .OpVariable, .{
+ .id_result_type = result_type_id,
+ .id_result = result_id,
+ .storage_class = storage_class,
+ });
+ return result_id.toRef();
}
- fn airArg(self: *DeclGen) ResultId {
+ fn airArg(self: *DeclGen) IdRef {
defer self.next_arg_index += 1;
return self.args.items[self.next_arg_index];
}
- fn airBlock(self: *DeclGen, inst: Air.Inst.Index) !?ResultId {
- // In IR, a block doesn't really define an entry point like a block, but more like a scope that breaks can jump out of and
+ fn airBlock(self: *DeclGen, inst: Air.Inst.Index) !?IdRef {
+ // In AIR, a block doesn't really define an entry point like a block, but more like a scope that breaks can jump out of and
// "return" a value from. This cannot be directly modelled in SPIR-V, so in a block instruction, we're going to split up
// the current block by first generating the code of the block, then a label, and then generate the rest of the current
// ir.Block in a different SPIR-V block.
- const label_id = self.spv.allocResultId();
+ const label_id = self.spv.allocId();
// 4 chosen as arbitrary initial capacity.
var incoming_blocks = try std.ArrayListUnmanaged(IncomingBlock).initCapacity(self.spv.gpa, 4);
- try self.blocks.putNoClobber(inst, .{
- .label_id = label_id,
+ try self.blocks.putNoClobber(self.spv.gpa, inst, .{
+ .label_id = label_id.toRef(),
.incoming_blocks = &incoming_blocks,
});
defer {
@@ -849,7 +790,7 @@ pub const DeclGen = struct {
const body = self.air.extra[extra.end..][0..extra.data.body_len];
try self.genBody(body);
- try self.beginSPIRVBlock(label_id);
+ try self.beginSpvBlock(label_id);
// If this block didn't produce a value, simply return here.
if (!ty.hasRuntimeBits())
@@ -857,21 +798,21 @@ pub const DeclGen = struct {
// Combine the result from the blocks using the Phi instruction.
- const result_id = self.spv.allocResultId();
+ const result_id = self.spv.allocId();
// TODO: OpPhi is limited in the types that it may produce, such as pointers. Figure out which other types
- // are not allowed to be created from a phi node, and throw an error for those. For now, genType already throws
+ // are not allowed to be created from a phi node, and throw an error for those. For now, resolveTypeId already throws
// an error for pointers.
- const result_type_id = try self.genType(ty);
+ const result_type_id = try self.resolveTypeId(ty);
_ = result_type_id;
- try writeOpcode(&self.code, .OpPhi, 2 + @intCast(u16, incoming_blocks.items.len * 2)); // result type + result + variable/parent...
+ try self.code.emitRaw(self.spv.gpa, .OpPhi, 2 + @intCast(u16, incoming_blocks.items.len * 2)); // result type + result + variable/parent...
for (incoming_blocks.items) |incoming| {
- try self.code.appendSlice(&[_]Word{ incoming.break_value_id, incoming.src_label_id });
+ self.code.writeOperand(spec.PairIdRefIdRef, .{ incoming.break_value_id, incoming.src_label_id });
}
- return result_id;
+ return result_id.toRef();
}
fn airBr(self: *DeclGen, inst: Air.Inst.Index) !void {
@@ -885,7 +826,7 @@ pub const DeclGen = struct {
try block.incoming_blocks.append(self.spv.gpa, .{ .src_label_id = self.current_block_label_id, .break_value_id = operand_id });
}
- try writeInstruction(&self.code, .OpBranch, &[_]Word{block.label_id});
+ try self.code.emit(self.spv.gpa, .OpBranch, .{ .target_label = block.label_id });
}
fn airCondBr(self: *DeclGen, inst: Air.Inst.Index) !void {
@@ -896,63 +837,70 @@ pub const DeclGen = struct {
const condition_id = try self.resolve(pl_op.operand);
// These will always generate a new SPIR-V block, since they are ir.Body and not ir.Block.
- const then_label_id = self.spv.allocResultId();
- const else_label_id = self.spv.allocResultId();
+ const then_label_id = self.spv.allocId();
+ const else_label_id = self.spv.allocId();
// TODO: We can generate OpSelectionMerge here if we know the target block that both of these will resolve to,
// but i don't know if those will always resolve to the same block.
- try writeInstruction(&self.code, .OpBranchConditional, &[_]Word{
- condition_id,
- then_label_id,
- else_label_id,
+ try self.code.emit(self.spv.gpa, .OpBranchConditional, .{
+ .condition = condition_id,
+ .true_label = then_label_id.toRef(),
+ .false_label = else_label_id.toRef(),
});
- try self.beginSPIRVBlock(then_label_id);
+ try self.beginSpvBlock(then_label_id);
try self.genBody(then_body);
- try self.beginSPIRVBlock(else_label_id);
+ try self.beginSpvBlock(else_label_id);
try self.genBody(else_body);
}
fn airDbgStmt(self: *DeclGen, inst: Air.Inst.Index) !void {
const dbg_stmt = self.air.instructions.items(.data)[inst].dbg_stmt;
const src_fname_id = try self.spv.resolveSourceFileName(self.decl);
- try writeInstruction(&self.code, .OpLine, &[_]Word{ src_fname_id, dbg_stmt.line, dbg_stmt.column });
+ try self.code.emit(self.spv.gpa, .OpLine, .{
+ .file = src_fname_id,
+ .line = dbg_stmt.line,
+ .column = dbg_stmt.column,
+ });
}
- fn airLoad(self: *DeclGen, inst: Air.Inst.Index) !ResultId {
+ fn airLoad(self: *DeclGen, inst: Air.Inst.Index) !IdRef {
const ty_op = self.air.instructions.items(.data)[inst].ty_op;
const operand_id = try self.resolve(ty_op.operand);
const ty = self.air.typeOfIndex(inst);
- const result_type_id = try self.genType(ty);
- const result_id = self.spv.allocResultId();
+ const result_type_id = try self.resolveTypeId(ty);
+ const result_id = self.spv.allocId();
- const operands = if (ty.isVolatilePtr())
- &[_]Word{ result_type_id, result_id, operand_id, @bitCast(u32, spec.MemoryAccess{ .Volatile = true }) }
- else
- &[_]Word{ result_type_id, result_id, operand_id };
+ const access = spec.MemoryAccess.Extended{
+ .Volatile = ty.isVolatilePtr(),
+ };
- try writeInstruction(&self.code, .OpLoad, operands);
+ try self.code.emit(self.spv.gpa, .OpLoad, .{
+ .id_result_type = result_type_id,
+ .id_result = result_id,
+ .pointer = operand_id,
+ .memory_access = access,
+ });
- return result_id;
+ return result_id.toRef();
}
fn airLoop(self: *DeclGen, inst: Air.Inst.Index) !void {
const ty_pl = self.air.instructions.items(.data)[inst].ty_pl;
const loop = self.air.extraData(Air.Block, ty_pl.payload);
const body = self.air.extra[loop.end..][0..loop.data.body_len];
- const loop_label_id = self.spv.allocResultId();
+ const loop_label_id = self.spv.allocId();
// Jump to the loop entry point
- try writeInstruction(&self.code, .OpBranch, &[_]Word{loop_label_id});
+ try self.code.emit(self.spv.gpa, .OpBranch, .{ .target_label = loop_label_id.toRef() });
// TODO: Look into OpLoopMerge.
-
- try self.beginSPIRVBlock(loop_label_id);
+ try self.beginSpvBlock(loop_label_id);
try self.genBody(body);
- try writeInstruction(&self.code, .OpBranch, &[_]Word{loop_label_id});
+ try self.code.emit(self.spv.gpa, .OpBranch, .{ .target_label = loop_label_id.toRef() });
}
fn airRet(self: *DeclGen, inst: Air.Inst.Index) !void {
@@ -960,9 +908,9 @@ pub const DeclGen = struct {
const operand_ty = self.air.typeOf(operand);
if (operand_ty.hasRuntimeBits()) {
const operand_id = try self.resolve(operand);
- try writeInstruction(&self.code, .OpReturnValue, &[_]Word{operand_id});
+ try self.code.emit(self.spv.gpa, .OpReturnValue, .{ .value = operand_id });
} else {
- try writeInstruction(&self.code, .OpReturn, &[_]Word{});
+ try self.code.emit(self.spv.gpa, .OpReturn, {});
}
}
@@ -972,15 +920,18 @@ pub const DeclGen = struct {
const src_val_id = try self.resolve(bin_op.rhs);
const lhs_ty = self.air.typeOf(bin_op.lhs);
- const operands = if (lhs_ty.isVolatilePtr())
- &[_]Word{ dst_ptr_id, src_val_id, @bitCast(u32, spec.MemoryAccess{ .Volatile = true }) }
- else
- &[_]Word{ dst_ptr_id, src_val_id };
+ const access = spec.MemoryAccess.Extended{
+ .Volatile = lhs_ty.isVolatilePtr(),
+ };
- try writeInstruction(&self.code, .OpStore, operands);
+ try self.code.emit(self.spv.gpa, .OpStore, .{
+ .pointer = dst_ptr_id,
+ .object = src_val_id,
+ .memory_access = access,
+ });
}
fn airUnreach(self: *DeclGen) !void {
- try writeInstruction(&self.code, .OpUnreachable, &[_]Word{});
+ try self.code.emit(self.spv.gpa, .OpUnreachable, {});
}
};