blob: e7f213f55eb3ab329e68610c0da3c0eed4340d3a [file] [log] [blame]
/* Copyright (c) 2015-2026 The Khronos Group Inc.
* Copyright (c) 2015-2026 Valve Corporation
* Copyright (c) 2015-2026 LunarG, Inc.
* Copyright (C) 2015-2026 Google Inc.
* Modifications Copyright (C) 2020 Advanced Micro Devices, Inc. All rights reserved.
* Modifications Copyright (C) 2022 RasterGrid Kft.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#pragma once
#include <variant>
#include "state_tracker/device_memory_state.h"
#include "containers/range.h"
namespace vvl {
class BufferSubState;
class BufferViewSubState;
class DeviceState;
class VideoProfileDesc;
class Buffer : public Bindable, public SubStateManager<BufferSubState> {
public:
const vku::safe_VkBufferCreateInfo safe_create_info;
const VkBufferCreateInfo &create_info;
const VkMemoryRequirements requirements;
VkDeviceAddress deviceAddress = 0;
// VkBufferUsageFlags2CreateInfo can be used instead over the VkBufferCreateInfo::usage
const VkBufferUsageFlags2 usage;
unordered_set<std::shared_ptr<const VideoProfileDesc>> supported_video_profiles;
Buffer(DeviceState &dev_data, VkBuffer handle, const VkBufferCreateInfo *pCreateInfo);
Buffer(Buffer const &rh_obj) = delete;
// This destructor is needed because Bindable depends on the tracker_ variant defined in this
// class. So we need to do the Destroy() work before tracker_ is destroyed.
virtual ~Buffer() {
if (!Destroyed()) {
Destroy();
}
}
void Destroy() override;
void NotifyInvalidate(const StateObject::NodeList &invalid_nodes, bool unlink) override;
VkBuffer VkHandle() const { return handle_.Cast<VkBuffer>(); }
VkDeviceSize GetRegionSize(VkDeviceSize offset, VkDeviceSize size) const {
if (offset < create_info.size) {
if (size == VK_WHOLE_SIZE) {
return create_info.size - offset;
} else if ((offset + size) <= create_info.size) {
return size;
}
}
return 0;
}
static VkDeviceSize GetRegionSize(const std::shared_ptr<const Buffer> &buffer_state, VkDeviceSize offset, VkDeviceSize size) {
return buffer_state ? buffer_state->GetRegionSize(offset, size) : 0;
}
vvl::range<VkDeviceAddress> DeviceAddressRange() const { return {deviceAddress, deviceAddress + create_info.size}; }
// This function is only used for comparing Imported External Dedicated Memory
bool CompareCreateInfo(const Buffer &other) const;
// Used to help unify the way we print the BDA info
std::string Describe(const Logger& dev_data) const;
private:
std::variant<std::monostate, BindableLinearMemoryTracker, BindableSparseMemoryTracker> tracker_;
};
class BufferSubState {
public:
explicit BufferSubState(Buffer &buf) : base(buf) {}
BufferSubState(const BufferSubState &) = delete;
BufferSubState &operator=(const BufferSubState &) = delete;
virtual ~BufferSubState() {}
virtual void Destroy() {}
virtual void NotifyInvalidate(const StateObject::NodeList &, bool) {}
Buffer &base;
};
class BufferView : public StateObject, public SubStateManager<BufferViewSubState> {
public:
const vku::safe_VkBufferViewCreateInfo safe_create_info;
const VkBufferViewCreateInfo &create_info;
std::shared_ptr<Buffer> buffer_state;
#ifdef VK_USE_PLATFORM_METAL_EXT
const bool metal_bufferview_export;
#endif // VK_USE_PLATFORM_METAL_EXT
// Format features that matter when accessing the buffer
// both as a buffer (ex OpLoad) or image (ex OpImageWrite)
const VkFormatFeatureFlags2 buffer_format_features;
BufferView(const std::shared_ptr<Buffer> &bf, VkBufferView handle, const VkBufferViewCreateInfo *pCreateInfo,
VkFormatFeatureFlags2 format_features);
void LinkChildNodes() override {
// Connect child node(s), which cannot safely be done in the constructor.
buffer_state->AddParent(this);
}
virtual ~BufferView() {
if (!Destroyed()) {
Destroy();
}
}
BufferView(const BufferView &rh_obj) = delete;
VkBufferView VkHandle() const { return handle_.Cast<VkBufferView>(); }
void Destroy() override;
void NotifyInvalidate(const StateObject::NodeList &invalid_nodes, bool unlink) override;
bool Invalid() const override { return Destroyed() || !buffer_state || buffer_state->Invalid(); }
VkDeviceSize Size() const {
VkDeviceSize size = create_info.range;
if (size == VK_WHOLE_SIZE) {
size = buffer_state->create_info.size - create_info.offset;
}
return size;
}
};
class BufferViewSubState {
public:
explicit BufferViewSubState(BufferView &buf) : base(buf) {}
BufferViewSubState(const BufferViewSubState &) = delete;
BufferViewSubState &operator=(const BufferViewSubState &) = delete;
virtual ~BufferViewSubState() {}
virtual void Destroy() {}
virtual void NotifyInvalidate(const StateObject::NodeList &, bool) {}
BufferView &base;
};
} // namespace vvl