2018-08-29 00:27:03 +02:00
|
|
|
// Copyright 2018 yuzu Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-09-06 20:57:40 +02:00
|
|
|
#include <cstring>
|
|
|
|
#include <memory>
|
|
|
|
|
2018-08-29 00:27:03 +02:00
|
|
|
#include "common/alignment.h"
|
|
|
|
#include "core/core.h"
|
2019-04-06 05:59:54 +02:00
|
|
|
#include "video_core/memory_manager.h"
|
2018-08-29 00:27:03 +02:00
|
|
|
#include "video_core/renderer_opengl/gl_buffer_cache.h"
|
2018-11-08 12:08:00 +01:00
|
|
|
#include "video_core/renderer_opengl/gl_rasterizer.h"
|
2018-08-29 00:27:03 +02:00
|
|
|
|
|
|
|
namespace OpenGL {
|
|
|
|
|
2019-02-19 02:58:32 +01:00
|
|
|
CachedBufferEntry::CachedBufferEntry(VAddr cpu_addr, std::size_t size, GLintptr offset,
|
|
|
|
std::size_t alignment, u8* host_ptr)
|
2019-03-27 17:35:31 +01:00
|
|
|
: RasterizerCacheObject{host_ptr}, cpu_addr{cpu_addr}, size{size}, offset{offset},
|
|
|
|
alignment{alignment} {}
|
2019-02-19 02:58:32 +01:00
|
|
|
|
2018-11-08 12:08:00 +01:00
|
|
|
OGLBufferCache::OGLBufferCache(RasterizerOpenGL& rasterizer, std::size_t size)
|
2019-01-06 06:07:20 +01:00
|
|
|
: RasterizerCache{rasterizer}, stream_buffer(size, true) {}
|
2018-08-29 00:27:03 +02:00
|
|
|
|
2019-03-04 05:17:35 +01:00
|
|
|
GLintptr OGLBufferCache::UploadMemory(GPUVAddr gpu_addr, std::size_t size, std::size_t alignment,
|
|
|
|
bool cache) {
|
2018-08-29 00:27:03 +02:00
|
|
|
auto& memory_manager = Core::System::GetInstance().GPU().MemoryManager();
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
|
2019-02-24 06:15:35 +01:00
|
|
|
const auto& host_ptr{memory_manager.GetPointer(gpu_addr)};
|
2018-08-29 00:27:03 +02:00
|
|
|
if (cache) {
|
2019-02-19 02:58:32 +01:00
|
|
|
auto entry = TryGet(host_ptr);
|
2018-08-29 00:27:03 +02:00
|
|
|
if (entry) {
|
2019-02-19 02:58:32 +01:00
|
|
|
if (entry->GetSize() >= size && entry->GetAlignment() == alignment) {
|
|
|
|
return entry->GetOffset();
|
2018-08-29 00:27:03 +02:00
|
|
|
}
|
|
|
|
Unregister(entry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
AlignBuffer(alignment);
|
2018-10-02 19:47:26 +02:00
|
|
|
const GLintptr uploaded_offset = buffer_offset;
|
2018-08-29 00:27:03 +02:00
|
|
|
|
2019-02-19 02:58:32 +01:00
|
|
|
if (!host_ptr) {
|
|
|
|
return uploaded_offset;
|
|
|
|
}
|
2018-08-29 00:27:03 +02:00
|
|
|
|
2019-02-19 02:58:32 +01:00
|
|
|
std::memcpy(buffer_ptr, host_ptr, size);
|
2018-08-29 00:27:03 +02:00
|
|
|
buffer_ptr += size;
|
|
|
|
buffer_offset += size;
|
|
|
|
|
|
|
|
if (cache) {
|
2019-02-24 06:15:35 +01:00
|
|
|
auto entry = std::make_shared<CachedBufferEntry>(
|
|
|
|
*memory_manager.GpuToCpuAddress(gpu_addr), size, uploaded_offset, alignment, host_ptr);
|
2018-08-29 00:27:03 +02:00
|
|
|
Register(entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
return uploaded_offset;
|
|
|
|
}
|
|
|
|
|
2018-09-15 15:21:06 +02:00
|
|
|
GLintptr OGLBufferCache::UploadHostMemory(const void* raw_pointer, std::size_t size,
|
|
|
|
std::size_t alignment) {
|
2018-08-29 00:27:03 +02:00
|
|
|
AlignBuffer(alignment);
|
|
|
|
std::memcpy(buffer_ptr, raw_pointer, size);
|
2018-10-02 19:47:26 +02:00
|
|
|
const GLintptr uploaded_offset = buffer_offset;
|
2018-08-29 00:27:03 +02:00
|
|
|
|
|
|
|
buffer_ptr += size;
|
|
|
|
buffer_offset += size;
|
|
|
|
return uploaded_offset;
|
|
|
|
}
|
|
|
|
|
2018-11-06 21:26:27 +01:00
|
|
|
bool OGLBufferCache::Map(std::size_t max_size) {
|
2018-08-29 00:27:03 +02:00
|
|
|
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();
|
|
|
|
}
|
2018-11-06 21:26:27 +01:00
|
|
|
return invalidate;
|
2018-08-29 00:27:03 +02:00
|
|
|
}
|
2018-10-02 19:47:26 +02:00
|
|
|
|
2018-08-29 00:27:03 +02:00
|
|
|
void OGLBufferCache::Unmap() {
|
|
|
|
stream_buffer.Unmap(buffer_offset - buffer_offset_base);
|
|
|
|
}
|
|
|
|
|
2018-09-06 20:58:15 +02:00
|
|
|
GLuint OGLBufferCache::GetHandle() const {
|
2018-08-29 00:27:03 +02:00
|
|
|
return stream_buffer.GetHandle();
|
|
|
|
}
|
|
|
|
|
2018-09-15 15:21:06 +02:00
|
|
|
void OGLBufferCache::AlignBuffer(std::size_t alignment) {
|
2018-08-29 00:27:03 +02:00
|
|
|
// Align the offset, not the mapped pointer
|
2018-10-02 19:47:26 +02:00
|
|
|
const GLintptr offset_aligned =
|
2018-09-15 15:21:06 +02:00
|
|
|
static_cast<GLintptr>(Common::AlignUp(static_cast<std::size_t>(buffer_offset), alignment));
|
2018-08-29 00:27:03 +02:00
|
|
|
buffer_ptr += offset_aligned - buffer_offset;
|
|
|
|
buffer_offset = offset_aligned;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace OpenGL
|