Browse Source
Merge pull request #1217 from degasus/vbo_cache2
Merge pull request #1217 from degasus/vbo_cache2
renderer_opengl: Implement a buffer cache.nce_cpp
committed by
GitHub
5 changed files with 182 additions and 86 deletions
-
1src/video_core/CMakeLists.txt
-
90src/video_core/renderer_opengl/gl_buffer_cache.cpp
-
57src/video_core/renderer_opengl/gl_buffer_cache.h
-
101src/video_core/renderer_opengl/gl_rasterizer.cpp
-
19src/video_core/renderer_opengl/gl_rasterizer.h
@ -0,0 +1,90 @@ |
|||
// Copyright 2018 yuzu Emulator Project
|
|||
// Licensed under GPLv2 or any later version
|
|||
// Refer to the license.txt file included.
|
|||
|
|||
#include "common/alignment.h"
|
|||
#include "common/assert.h"
|
|||
#include "core/core.h"
|
|||
#include "core/memory.h"
|
|||
#include "video_core/renderer_opengl/gl_buffer_cache.h"
|
|||
|
|||
namespace OpenGL { |
|||
|
|||
OGLBufferCache::OGLBufferCache(size_t size) : stream_buffer(GL_ARRAY_BUFFER, size) {} |
|||
|
|||
GLintptr OGLBufferCache::UploadMemory(Tegra::GPUVAddr gpu_addr, size_t size, size_t alignment, |
|||
bool cache) { |
|||
auto& memory_manager = Core::System::GetInstance().GPU().MemoryManager(); |
|||
const boost::optional<VAddr> cpu_addr{memory_manager.GpuToCpuAddress(gpu_addr)}; |
|||
|
|||
// Cache management is a big overhead, so only cache entries with a given size.
|
|||
// TODO: Figure out which size is the best for given games.
|
|||
cache &= size >= 2048; |
|||
|
|||
if (cache) { |
|||
auto entry = TryGet(*cpu_addr); |
|||
if (entry) { |
|||
if (entry->size >= size && entry->alignment == alignment) { |
|||
return entry->offset; |
|||
} |
|||
Unregister(entry); |
|||
} |
|||
} |
|||
|
|||
AlignBuffer(alignment); |
|||
GLintptr uploaded_offset = buffer_offset; |
|||
|
|||
Memory::ReadBlock(*cpu_addr, buffer_ptr, size); |
|||
|
|||
buffer_ptr += size; |
|||
buffer_offset += size; |
|||
|
|||
if (cache) { |
|||
auto entry = std::make_shared<CachedBufferEntry>(); |
|||
entry->offset = uploaded_offset; |
|||
entry->size = size; |
|||
entry->alignment = alignment; |
|||
entry->addr = *cpu_addr; |
|||
Register(entry); |
|||
} |
|||
|
|||
return uploaded_offset; |
|||
} |
|||
|
|||
GLintptr OGLBufferCache::UploadHostMemory(const void* raw_pointer, size_t size, size_t alignment) { |
|||
AlignBuffer(alignment); |
|||
std::memcpy(buffer_ptr, raw_pointer, size); |
|||
GLintptr uploaded_offset = buffer_offset; |
|||
|
|||
buffer_ptr += size; |
|||
buffer_offset += size; |
|||
return uploaded_offset; |
|||
} |
|||
|
|||
void OGLBufferCache::Map(size_t max_size) { |
|||
bool invalidate; |
|||
std::tie(buffer_ptr, buffer_offset_base, invalidate) = |
|||
stream_buffer.Map(static_cast<GLsizeiptr>(max_size), 4); |
|||
buffer_offset = buffer_offset_base; |
|||
|
|||
if (invalidate) { |
|||
InvalidateAll(); |
|||
} |
|||
} |
|||
void OGLBufferCache::Unmap() { |
|||
stream_buffer.Unmap(buffer_offset - buffer_offset_base); |
|||
} |
|||
|
|||
GLuint OGLBufferCache::GetHandle() { |
|||
return stream_buffer.GetHandle(); |
|||
} |
|||
|
|||
void OGLBufferCache::AlignBuffer(size_t alignment) { |
|||
// Align the offset, not the mapped pointer
|
|||
GLintptr offset_aligned = |
|||
static_cast<GLintptr>(Common::AlignUp(static_cast<size_t>(buffer_offset), alignment)); |
|||
buffer_ptr += offset_aligned - buffer_offset; |
|||
buffer_offset = offset_aligned; |
|||
} |
|||
|
|||
} // namespace OpenGL
|
|||
@ -0,0 +1,57 @@ |
|||
// Copyright 2018 yuzu Emulator Project |
|||
// Licensed under GPLv2 or any later version |
|||
// Refer to the license.txt file included. |
|||
|
|||
#pragma once |
|||
|
|||
#include <memory> |
|||
#include <unordered_map> |
|||
|
|||
#include "common/common_types.h" |
|||
#include "video_core/rasterizer_cache.h" |
|||
#include "video_core/renderer_opengl/gl_resource_manager.h" |
|||
#include "video_core/renderer_opengl/gl_stream_buffer.h" |
|||
|
|||
namespace OpenGL { |
|||
|
|||
struct CachedBufferEntry final { |
|||
VAddr GetAddr() const { |
|||
return addr; |
|||
} |
|||
|
|||
size_t GetSizeInBytes() const { |
|||
return size; |
|||
} |
|||
|
|||
VAddr addr; |
|||
size_t size; |
|||
GLintptr offset; |
|||
size_t alignment; |
|||
}; |
|||
|
|||
class OGLBufferCache final : public RasterizerCache<std::shared_ptr<CachedBufferEntry>> { |
|||
public: |
|||
OGLBufferCache(size_t size); |
|||
|
|||
GLintptr UploadMemory(Tegra::GPUVAddr gpu_addr, size_t size, size_t alignment = 4, |
|||
bool cache = true); |
|||
|
|||
GLintptr UploadHostMemory(const void* raw_pointer, size_t size, size_t alignment = 4); |
|||
|
|||
void Map(size_t max_size); |
|||
void Unmap(); |
|||
|
|||
GLuint GetHandle(); |
|||
|
|||
protected: |
|||
void AlignBuffer(size_t alignment); |
|||
|
|||
private: |
|||
OGLStreamBuffer stream_buffer; |
|||
|
|||
u8* buffer_ptr; |
|||
GLintptr buffer_offset; |
|||
GLintptr buffer_offset_base; |
|||
}; |
|||
|
|||
} // namespace OpenGL |
|||
Write
Preview
Loading…
Cancel
Save
Reference in new issue