Browse Source
Add rasterizer_accelerated (#79)
Add rasterizer_accelerated (#79)
Credit: Antique - [Sudachi] Dev (https://sudachi.emuplace.app/) Reviewed-on: https://git.eden-emu.dev/eden-emu/eden/pulls/79 Co-authored-by: JPikachu <jpikachu@noreply.localhost> Co-committed-by: JPikachu <jpikachu@noreply.localhost>nce_cpp
committed by
edendev
2 changed files with 121 additions and 0 deletions
@ -0,0 +1,72 @@ |
|||
// SPDX-FileCopyrightText: Copyright 2019 yuzu Emulator Project
|
|||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|||
|
|||
#include <atomic>
|
|||
|
|||
#include "common/assert.h"
|
|||
#include "common/common_types.h"
|
|||
#include "common/div_ceil.h"
|
|||
#include "core/memory.h"
|
|||
#include "video_core/rasterizer_accelerated.h"
|
|||
|
|||
namespace VideoCore { |
|||
|
|||
using namespace Core::Memory; |
|||
|
|||
RasterizerAccelerated::RasterizerAccelerated(Memory& cpu_memory_) |
|||
: cached_pages(std::make_unique<CachedPages>()), cpu_memory{cpu_memory_} {} |
|||
|
|||
RasterizerAccelerated::~RasterizerAccelerated() = default; |
|||
|
|||
void RasterizerAccelerated::UpdatePagesCachedCount(VAddr addr, u64 size, int delta) { |
|||
u64 uncache_begin = 0; |
|||
u64 cache_begin = 0; |
|||
u64 uncache_bytes = 0; |
|||
u64 cache_bytes = 0; |
|||
|
|||
std::atomic_thread_fence(std::memory_order_acquire); |
|||
const u64 page_end = Common::DivCeil(addr + size, YUZU_PAGESIZE); |
|||
for (u64 page = addr >> YUZU_PAGEBITS; page != page_end; ++page) { |
|||
std::atomic_uint16_t& count = cached_pages->at(page >> 2).Count(page); |
|||
|
|||
if (delta > 0) { |
|||
ASSERT_MSG(count.load(std::memory_order::relaxed) < UINT16_MAX, "Count may overflow!"); |
|||
} else if (delta < 0) { |
|||
ASSERT_MSG(count.load(std::memory_order::relaxed) > 0, "Count may underflow!"); |
|||
} else { |
|||
ASSERT_MSG(false, "Delta must be non-zero!"); |
|||
} |
|||
|
|||
// Adds or subtracts 1, as count is a unsigned 8-bit value
|
|||
count.fetch_add(static_cast<u16>(delta), std::memory_order_release); |
|||
|
|||
// Assume delta is either -1 or 1
|
|||
if (count.load(std::memory_order::relaxed) == 0) { |
|||
if (uncache_bytes == 0) { |
|||
uncache_begin = page; |
|||
} |
|||
uncache_bytes += YUZU_PAGESIZE; |
|||
} else if (uncache_bytes > 0) { |
|||
cpu_memory.RasterizerMarkRegionCached(uncache_begin << YUZU_PAGEBITS, uncache_bytes, |
|||
false); |
|||
uncache_bytes = 0; |
|||
} |
|||
if (count.load(std::memory_order::relaxed) == 1 && delta > 0) { |
|||
if (cache_bytes == 0) { |
|||
cache_begin = page; |
|||
} |
|||
cache_bytes += YUZU_PAGESIZE; |
|||
} else if (cache_bytes > 0) { |
|||
cpu_memory.RasterizerMarkRegionCached(cache_begin << YUZU_PAGEBITS, cache_bytes, true); |
|||
cache_bytes = 0; |
|||
} |
|||
} |
|||
if (uncache_bytes > 0) { |
|||
cpu_memory.RasterizerMarkRegionCached(uncache_begin << YUZU_PAGEBITS, uncache_bytes, false); |
|||
} |
|||
if (cache_bytes > 0) { |
|||
cpu_memory.RasterizerMarkRegionCached(cache_begin << YUZU_PAGEBITS, cache_bytes, true); |
|||
} |
|||
} |
|||
|
|||
} // namespace VideoCore
|
|||
@ -0,0 +1,49 @@ |
|||
// SPDX-FileCopyrightText: Copyright 2019 yuzu Emulator Project |
|||
// SPDX-License-Identifier: GPL-2.0-or-later |
|||
|
|||
#pragma once |
|||
|
|||
#include <array> |
|||
#include <atomic> |
|||
|
|||
#include "common/common_types.h" |
|||
#include "video_core/rasterizer_interface.h" |
|||
|
|||
namespace Core::Memory { |
|||
class Memory; |
|||
} |
|||
|
|||
namespace VideoCore { |
|||
|
|||
/// Implements the shared part in GPU accelerated rasterizers in RasterizerInterface. |
|||
class RasterizerAccelerated : public RasterizerInterface { |
|||
public: |
|||
explicit RasterizerAccelerated(Core::Memory::Memory& cpu_memory_); |
|||
~RasterizerAccelerated() override; |
|||
|
|||
void UpdatePagesCachedCount(VAddr addr, u64 size, int delta) override; |
|||
|
|||
private: |
|||
class CacheEntry final { |
|||
public: |
|||
CacheEntry() = default; |
|||
|
|||
std::atomic_uint16_t& Count(std::size_t page) { |
|||
return values[page & 3]; |
|||
} |
|||
|
|||
const std::atomic_uint16_t& Count(std::size_t page) const { |
|||
return values[page & 3]; |
|||
} |
|||
|
|||
private: |
|||
std::array<std::atomic_uint16_t, 4> values{}; |
|||
}; |
|||
static_assert(sizeof(CacheEntry) == 8, "CacheEntry should be 8 bytes!"); |
|||
|
|||
using CachedPages = std::array<CacheEntry, 0x2000000>; |
|||
std::unique_ptr<CachedPages> cached_pages; |
|||
Core::Memory::Memory& cpu_memory; |
|||
}; |
|||
|
|||
} // namespace VideoCore |
|||
Write
Preview
Loading…
Cancel
Save
Reference in new issue