| /* 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 |