|  | // Copyright 2014 The Chromium Authors | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "gpu/command_buffer/common/buffer.h" | 
|  |  | 
|  | #include <stddef.h> | 
|  | #include <stdint.h> | 
|  |  | 
|  | #include "base/atomic_sequence_num.h" | 
|  | #include "base/check_op.h" | 
|  | #include "base/format_macros.h" | 
|  | #include "base/no_destructor.h" | 
|  | #include "base/numerics/safe_math.h" | 
|  | #include "base/strings/stringprintf.h" | 
|  |  | 
|  | namespace gpu { | 
|  | namespace { | 
|  |  | 
|  | // Global atomic to generate unique buffer IDs. | 
|  | base::AtomicSequenceNumber g_next_buffer_id; | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | const base::UnsafeSharedMemoryRegion& BufferBacking::shared_memory_region() | 
|  | const { | 
|  | static const base::NoDestructor<base::UnsafeSharedMemoryRegion> | 
|  | kInvalidRegion; | 
|  | return *kInvalidRegion; | 
|  | } | 
|  |  | 
|  | base::UnguessableToken BufferBacking::GetGUID() const { | 
|  | return base::UnguessableToken(); | 
|  | } | 
|  |  | 
|  | MemoryBufferBacking::MemoryBufferBacking(uint32_t size) | 
|  | : memory_(new char[size]), size_(size) {} | 
|  |  | 
|  | MemoryBufferBacking::~MemoryBufferBacking() = default; | 
|  |  | 
|  | void* MemoryBufferBacking::GetMemory() const { | 
|  | return memory_.get(); | 
|  | } | 
|  |  | 
|  | uint32_t MemoryBufferBacking::GetSize() const { | 
|  | return size_; | 
|  | } | 
|  |  | 
|  | SharedMemoryBufferBacking::SharedMemoryBufferBacking( | 
|  | base::UnsafeSharedMemoryRegion shared_memory_region, | 
|  | base::WritableSharedMemoryMapping shared_memory_mapping) | 
|  | : shared_memory_region_(std::move(shared_memory_region)), | 
|  | shared_memory_mapping_(std::move(shared_memory_mapping)) { | 
|  | DCHECK_EQ(shared_memory_region_.GetGUID(), shared_memory_mapping_.guid()); | 
|  | DCHECK_LE(shared_memory_mapping_.size(), static_cast<size_t>(UINT32_MAX)); | 
|  | } | 
|  |  | 
|  | SharedMemoryBufferBacking::~SharedMemoryBufferBacking() = default; | 
|  |  | 
|  | const base::UnsafeSharedMemoryRegion& | 
|  | SharedMemoryBufferBacking::shared_memory_region() const { | 
|  | return shared_memory_region_; | 
|  | } | 
|  |  | 
|  | base::UnguessableToken SharedMemoryBufferBacking::GetGUID() const { | 
|  | return shared_memory_region_.GetGUID(); | 
|  | } | 
|  |  | 
|  | void* SharedMemoryBufferBacking::GetMemory() const { | 
|  | return shared_memory_mapping_.memory(); | 
|  | } | 
|  |  | 
|  | uint32_t SharedMemoryBufferBacking::GetSize() const { | 
|  | return static_cast<uint32_t>(shared_memory_mapping_.size()); | 
|  | } | 
|  |  | 
|  | Buffer::Buffer(std::unique_ptr<BufferBacking> backing) | 
|  | : backing_(std::move(backing)), | 
|  | memory_(backing_->GetMemory()), | 
|  | size_(backing_->GetSize()) { | 
|  | DCHECK(memory_) << "The memory must be mapped to create a Buffer"; | 
|  | } | 
|  |  | 
|  | Buffer::~Buffer() = default; | 
|  |  | 
|  | void* Buffer::GetDataAddress(uint32_t data_offset, uint32_t data_size) const { | 
|  | base::CheckedNumeric<uint32_t> end = data_offset; | 
|  | end += data_size; | 
|  | if (!end.IsValid() || end.ValueOrDie() > static_cast<uint32_t>(size_)) | 
|  | return nullptr; | 
|  | return static_cast<uint8_t*>(memory_) + data_offset; | 
|  | } | 
|  |  | 
|  | void* Buffer::GetDataAddressAndSize(uint32_t data_offset, | 
|  | uint32_t* data_size) const { | 
|  | if (data_offset > static_cast<uint32_t>(size_)) | 
|  | return nullptr; | 
|  | *data_size = GetRemainingSize(data_offset); | 
|  | return static_cast<uint8_t*>(memory_) + data_offset; | 
|  | } | 
|  |  | 
|  | uint32_t Buffer::GetRemainingSize(uint32_t data_offset) const { | 
|  | if (data_offset > static_cast<uint32_t>(size_)) | 
|  | return 0; | 
|  | return static_cast<uint32_t>(size_) - data_offset; | 
|  | } | 
|  |  | 
|  | int32_t GetNextBufferId() { | 
|  | // 0 is a reserved value. | 
|  | return g_next_buffer_id.GetNext() + 1; | 
|  | } | 
|  |  | 
|  | base::trace_event::MemoryAllocatorDumpGuid GetBufferGUIDForTracing( | 
|  | uint64_t tracing_process_id, | 
|  | int32_t buffer_id) { | 
|  | return base::trace_event::MemoryAllocatorDumpGuid(base::StringPrintf( | 
|  | "gpu-buffer-x-process/%" PRIx64 "/%d", tracing_process_id, buffer_id)); | 
|  | } | 
|  |  | 
|  | } // namespace gpu |