adding buffer-image copies
All checks were successful
Build / build (push) Successful in 2m11s
Test / build_and_test (push) Successful in 48m53s

This commit is contained in:
2026-04-12 22:34:58 +02:00
parent 3717fb3790
commit d54c28dcaa
11 changed files with 351 additions and 110 deletions

View File

@@ -18,6 +18,7 @@ extent: vk.Extent3D,
mip_levels: u32,
array_layers: u32,
samples: vk.SampleCountFlags,
flags: vk.ImageCreateFlags,
tiling: vk.ImageTiling,
usage: vk.ImageUsageFlags,
memory: ?*DeviceMemory,
@@ -28,7 +29,8 @@ vtable: *const VTable,
pub const VTable = struct {
destroy: *const fn (*Self, std.mem.Allocator) void,
getMemoryRequirements: *const fn (*Self, *vk.MemoryRequirements) void,
getMemoryRequirements: *const fn (*Self, *vk.MemoryRequirements) VkError!void,
getTotalSizeForAspect: *const fn (*const Self, vk.ImageAspectFlags) VkError!usize,
};
pub fn init(device: *Device, allocator: std.mem.Allocator, info: *const vk.ImageCreateInfo) VkError!Self {
@@ -41,6 +43,7 @@ pub fn init(device: *Device, allocator: std.mem.Allocator, info: *const vk.Image
.mip_levels = info.mip_levels,
.array_layers = info.array_layers,
.samples = info.samples,
.flags = info.flags,
.tiling = info.tiling,
.usage = info.usage,
.memory = null,
@@ -55,7 +58,7 @@ pub inline fn destroy(self: *Self, allocator: std.mem.Allocator) void {
}
pub inline fn bindMemory(self: *Self, memory: *DeviceMemory, offset: vk.DeviceSize) VkError!void {
const image_size = self.getTotalSize();
const image_size = try self.getTotalSize();
if (offset >= image_size or !self.allowed_memory_types.isSet(memory.memory_type_index)) {
return VkError.ValidationFailed;
}
@@ -63,11 +66,10 @@ pub inline fn bindMemory(self: *Self, memory: *DeviceMemory, offset: vk.DeviceSi
self.memory_offset = offset;
}
pub inline fn getMemoryRequirements(self: *Self, requirements: *vk.MemoryRequirements) void {
const image_size = self.getTotalSize();
requirements.size = image_size;
pub inline fn getMemoryRequirements(self: *Self, requirements: *vk.MemoryRequirements) VkError!void {
requirements.size = try self.getTotalSize();
requirements.memory_type_bits = self.allowed_memory_types.mask;
self.vtable.getMemoryRequirements(self, requirements);
try self.vtable.getMemoryRequirements(self, requirements);
}
pub inline fn getClearFormat(self: *Self) vk.Format {
@@ -79,78 +81,22 @@ pub inline fn getClearFormat(self: *Self) vk.Format {
.r32g32b32a32_sfloat;
}
pub inline fn getPixelSize(self: *const Self) usize {
return lib.vku.vkuFormatTexelBlockSize(@intCast(@intFromEnum(self.format)));
pub inline fn getTexelSize(self: *const Self) usize {
return lib.format.texelSize(self.format);
}
pub inline fn getTotalSize(self: *const Self) usize {
const pixel_size = self.getPixelSize();
return self.extent.width * self.extent.height * self.extent.depth * pixel_size;
pub inline fn getTotalSizeForAspect(self: *const Self, aspect: vk.ImageAspectFlags) VkError!usize {
return self.vtable.getTotalSizeForAspect(self, aspect);
}
pub inline fn getFormatPixelSize(format: vk.Format) usize {
return lib.vku.vkuFormatTexelBlockSize(@intCast(@intFromEnum(format)));
pub inline fn getTotalSize(self: *const Self) VkError!usize {
return self.vtable.getTotalSizeForAspect(self, lib.format.toAspect(self.format));
}
pub inline fn getFormatTotalSize(self: *const Self, format: vk.Format) usize {
const pixel_size = self.getFormatPixelSize(format);
return self.extent.width * self.extent.height * self.extent.depth * pixel_size;
pub inline fn formatFromAspect(self: *const Self, aspect_mask: vk.ImageAspectFlags) vk.Format {
return lib.format.fromAspect(self.format, aspect_mask);
}
pub fn formatSupportsColorAttachemendBlend(format: vk.Format) bool {
return switch (format) {
// Vulkan 1.1 mandatory
.r5g6b5_unorm_pack16,
.a1r5g5b5_unorm_pack16,
.r8_unorm,
.r8g8_unorm,
.r8g8b8a8_unorm,
.r8g8b8a8_srgb,
.b8g8r8a8_unorm,
.b8g8r8a8_srgb,
.a8b8g8r8_unorm_pack32,
.a8b8g8r8_srgb_pack32,
.a2b10g10r10_unorm_pack32,
.r16_sfloat,
.r16g16_sfloat,
.r16g16b16a16_sfloat,
// optional
.r4g4b4a4_unorm_pack16,
.b4g4r4a4_unorm_pack16,
.b5g6r5_unorm_pack16,
.r5g5b5a1_unorm_pack16,
.b5g5r5a1_unorm_pack16,
.a2r10g10b10_unorm_pack32,
.r16_unorm,
.r16g16_unorm,
.r16g16b16a16_unorm,
.r32_sfloat,
.r32g32_sfloat,
.r32g32b32a32_sfloat,
.b10g11r11_ufloat_pack32,
.a4r4g4b4_unorm_pack16,
.a4b4g4r4_unorm_pack16,
=> true,
else => false,
};
}
pub fn formatFromAspect(base_format: vk.Format, aspect: vk.ImageAspectFlags) vk.Format {
if (aspect.color_bit or (aspect.color_bit and aspect.stencil_bit)) {
return base_format;
} else if (aspect.depth_bit) {
if (base_format == .d16_unorm or base_format == .d16_unorm_s8_uint) {
return .d16_unorm;
} else if (base_format == .d24_unorm_s8_uint) {
return .x8_d24_unorm_pack32;
} else if (base_format == .d32_sfloat or base_format == .d32_sfloat_s8_uint) {
return .d32_sfloat;
}
} else if (aspect.stencil_bit) {
if (base_format == .s8_uint or base_format == .d16_unorm_s8_uint or base_format == .d24_unorm_s8_uint or base_format == .d32_sfloat_s8_uint) {
return .s8_uint;
}
}
lib.unsupported("format {d}", .{@intFromEnum(base_format)});
return base_format;
pub inline fn formatToAspect(self: *const Self, aspect_mask: vk.ImageAspectFlags) vk.ImageAspectFlags {
return lib.format.toAspect(self.format, aspect_mask);
}

88
src/vulkan/format.zig git.filemode.normal_file
View File

@@ -0,0 +1,88 @@
const std = @import("std");
const vk = @import("vulkan");
const lib = @import("lib.zig");
pub fn fromAspect(format: vk.Format, aspect: vk.ImageAspectFlags) vk.Format {
if (aspect.color_bit or (aspect.color_bit and aspect.stencil_bit)) {
return format;
} else if (aspect.depth_bit) {
if (format == .d16_unorm or format == .d16_unorm_s8_uint) {
return .d16_unorm;
} else if (format == .d24_unorm_s8_uint) {
return .x8_d24_unorm_pack32;
} else if (format == .d32_sfloat or format == .d32_sfloat_s8_uint) {
return .d32_sfloat;
}
} else if (aspect.stencil_bit) {
if (format == .s8_uint or format == .d16_unorm_s8_uint or format == .d24_unorm_s8_uint or format == .d32_sfloat_s8_uint) {
return .s8_uint;
}
}
lib.unsupported("format {s}", .{@tagName(format)});
return format;
}
pub fn toAspect(format: vk.Format) vk.ImageAspectFlags {
var aspect: vk.ImageAspectFlags = .{};
if (lib.vku.vkuFormatHasDepth(@intCast(@intFromEnum(format))))
aspect.depth_bit = true;
if (lib.vku.vkuFormatHasStencil(@intCast(@intFromEnum(format))))
aspect.stencil_bit = true;
if (aspect.toInt() == 0)
aspect.color_bit = true;
return aspect;
}
pub inline fn texelSize(format: vk.Format) usize {
return lib.vku.vkuFormatTexelBlockSize(@intCast(@intFromEnum(format)));
}
pub inline fn supportsColorAttachemendBlend(format: vk.Format) bool {
return switch (format) {
// Vulkan 1.1 mandatory
.r5g6b5_unorm_pack16,
.a1r5g5b5_unorm_pack16,
.r8_unorm,
.r8g8_unorm,
.r8g8b8a8_unorm,
.r8g8b8a8_srgb,
.b8g8r8a8_unorm,
.b8g8r8a8_srgb,
.a8b8g8r8_unorm_pack32,
.a8b8g8r8_srgb_pack32,
.a2b10g10r10_unorm_pack32,
.r16_sfloat,
.r16g16_sfloat,
.r16g16b16a16_sfloat,
// optional
.r4g4b4a4_unorm_pack16,
.b4g4r4a4_unorm_pack16,
.b5g6r5_unorm_pack16,
.r5g5b5a1_unorm_pack16,
.b5g5r5a1_unorm_pack16,
.a2r10g10b10_unorm_pack32,
.r16_unorm,
.r16g16_unorm,
.r16g16b16a16_unorm,
.r32_sfloat,
.r32g32_sfloat,
.r32g32b32a32_sfloat,
.b10g11r11_ufloat_pack32,
.a4r4g4b4_unorm_pack16,
.a4b4g4r4_unorm_pack16,
=> true,
else => false,
};
}
pub inline fn pitchMemSize(format: vk.Format, width: usize) usize {
// To be updated for compressed formats handling
return texelSize(format) * width;
}
pub inline fn sliceMemSize(format: vk.Format, width: usize, height: usize) usize {
// To be updated for compressed formats handling
return pitchMemSize(format, width) * height;
}

View File

@@ -10,6 +10,7 @@ pub const vku = @cImport({
pub const errors = @import("error_set.zig");
pub const lib_vulkan = @import("lib_vulkan.zig");
pub const logger = @import("logger/logger.zig");
pub const format = @import("format.zig");
pub const Dispatchable = @import("Dispatchable.zig").Dispatchable;
pub const NonDispatchable = @import("NonDispatchable.zig").NonDispatchable;

View File

@@ -624,7 +624,7 @@ pub export fn strollAllocateCommandBuffers(p_device: vk.Device, info: *const vk.
}
pub export fn strollAllocateDescriptorSets(p_device: vk.Device, info: *const vk.DescriptorSetAllocateInfo, p_sets: [*]vk.DescriptorSet) callconv(vk.vulkan_call_conv) vk.Result {
entryPointBeginLogTrace(.vkAllocateCommandBuffers);
entryPointBeginLogTrace(.vkAllocateDescriptorSets);
defer entryPointEndLogTrace();
Dispatchable(Device).checkHandleValidity(p_device) catch |err| return toVkResult(err);
@@ -1385,7 +1385,7 @@ pub export fn strollGetImageMemoryRequirements(p_device: vk.Device, p_image: vk.
Dispatchable(Device).checkHandleValidity(p_device) catch |err| return errorLogger(err);
const image = NonDispatchable(Image).fromHandleObject(p_image) catch |err| return errorLogger(err);
image.getMemoryRequirements(requirements);
image.getMemoryRequirements(requirements) catch |err| return errorLogger(err);
}
pub export fn strollGetImageSparseMemoryRequirements(p_device: vk.Device, p_image: vk.Image, requirements: *vk.SparseImageMemoryRequirements) callconv(vk.vulkan_call_conv) void {