early-access version 1458
This commit is contained in:
parent
68b5ea0637
commit
6da9491544
10 changed files with 155 additions and 68 deletions
|
@ -1,7 +1,7 @@
|
||||||
yuzu emulator early access
|
yuzu emulator early access
|
||||||
=============
|
=============
|
||||||
|
|
||||||
This is the source code for early-access 1457.
|
This is the source code for early-access 1458.
|
||||||
|
|
||||||
## Legal Notice
|
## Legal Notice
|
||||||
|
|
||||||
|
|
|
@ -104,7 +104,7 @@ public:
|
||||||
: ServiceFramework{system_, "IUserLocalCommunicationService"} {
|
: ServiceFramework{system_, "IUserLocalCommunicationService"} {
|
||||||
// clang-format off
|
// clang-format off
|
||||||
static const FunctionInfo functions[] = {
|
static const FunctionInfo functions[] = {
|
||||||
{0, nullptr, "GetState"},
|
{0, &IUserLocalCommunicationService::GetState, "GetState"},
|
||||||
{1, nullptr, "GetNetworkInfo"},
|
{1, nullptr, "GetNetworkInfo"},
|
||||||
{2, nullptr, "GetIpv4Address"},
|
{2, nullptr, "GetIpv4Address"},
|
||||||
{3, nullptr, "GetDisconnectReason"},
|
{3, nullptr, "GetDisconnectReason"},
|
||||||
|
@ -139,14 +139,38 @@ public:
|
||||||
RegisterHandlers(functions);
|
RegisterHandlers(functions);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Initialize2(Kernel::HLERequestContext& ctx) {
|
void GetState(Kernel::HLERequestContext& ctx) {
|
||||||
LOG_WARNING(Service_LDN, "(STUBBED) called");
|
LOG_WARNING(Service_LDN, "(STUBBED) called");
|
||||||
|
|
||||||
// Return the disabled error to indicate that LDN is currently unavailable, otherwise games
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
// will continue to try to make a connection.
|
|
||||||
IPC::ResponseBuilder rb{ctx, 2};
|
// Indicate a network error, as we do not actually emulate LDN
|
||||||
rb.Push(ERROR_DISABLED);
|
rb.Push(static_cast<u32>(State::Error));
|
||||||
|
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Initialize2(Kernel::HLERequestContext& ctx) {
|
||||||
|
LOG_DEBUG(Service_LDN, "called");
|
||||||
|
|
||||||
|
is_initialized = true;
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
enum class State {
|
||||||
|
None,
|
||||||
|
Initialized,
|
||||||
|
AccessPointOpened,
|
||||||
|
AccessPointCreated,
|
||||||
|
StationOpened,
|
||||||
|
StationConnected,
|
||||||
|
Error,
|
||||||
|
};
|
||||||
|
|
||||||
|
bool is_initialized{};
|
||||||
};
|
};
|
||||||
|
|
||||||
class LDNS final : public ServiceFramework<LDNS> {
|
class LDNS final : public ServiceFramework<LDNS> {
|
||||||
|
|
|
@ -33,11 +33,16 @@ void Mouse::UpdateThread() {
|
||||||
info.motion.UpdateOrientation(update_time * 1000);
|
info.motion.UpdateOrientation(update_time * 1000);
|
||||||
info.tilt_speed = 0;
|
info.tilt_speed = 0;
|
||||||
info.data.motion = info.motion.GetMotion();
|
info.data.motion = info.motion.GetMotion();
|
||||||
|
if (Settings::values.mouse_panning) {
|
||||||
|
info.last_mouse_change *= 0.96f;
|
||||||
|
info.data.axis = {static_cast<int>(16 * info.last_mouse_change.x),
|
||||||
|
static_cast<int>(16 * -info.last_mouse_change.y)};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (configuring) {
|
if (configuring) {
|
||||||
UpdateYuzuSettings();
|
UpdateYuzuSettings();
|
||||||
}
|
}
|
||||||
if (mouse_panning_timout++ > 8) {
|
if (mouse_panning_timout++ > 20) {
|
||||||
StopPanning();
|
StopPanning();
|
||||||
}
|
}
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(update_time));
|
std::this_thread::sleep_for(std::chrono::milliseconds(update_time));
|
||||||
|
@ -82,16 +87,27 @@ void Mouse::StopPanning() {
|
||||||
void Mouse::MouseMove(int x, int y, int center_x, int center_y) {
|
void Mouse::MouseMove(int x, int y, int center_x, int center_y) {
|
||||||
for (MouseInfo& info : mouse_info) {
|
for (MouseInfo& info : mouse_info) {
|
||||||
if (Settings::values.mouse_panning) {
|
if (Settings::values.mouse_panning) {
|
||||||
const auto mouse_change = Common::MakeVec(x, y) - Common::MakeVec(center_x, center_y);
|
auto mouse_change =
|
||||||
|
(Common::MakeVec(x, y) - Common::MakeVec(center_x, center_y)).Cast<float>();
|
||||||
mouse_panning_timout = 0;
|
mouse_panning_timout = 0;
|
||||||
|
|
||||||
if (mouse_change.y == 0 && mouse_change.x == 0) {
|
if (mouse_change.y == 0 && mouse_change.x == 0) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
const auto mouse_change_length = mouse_change.Length();
|
||||||
|
if (mouse_change_length < 3.0f) {
|
||||||
|
mouse_change /= mouse_change_length / 3.0f;
|
||||||
|
}
|
||||||
|
|
||||||
|
info.last_mouse_change = (info.last_mouse_change * 0.91f) + (mouse_change * 0.09f);
|
||||||
|
|
||||||
|
const auto last_mouse_change_length = info.last_mouse_change.Length();
|
||||||
|
if (last_mouse_change_length > 8.0f) {
|
||||||
|
info.last_mouse_change /= last_mouse_change_length / 8.0f;
|
||||||
|
} else if (last_mouse_change_length < 1.0f) {
|
||||||
|
info.last_mouse_change = mouse_change / mouse_change.Length();
|
||||||
|
}
|
||||||
|
|
||||||
info.last_mouse_change = (info.last_mouse_change * 0.8f) + (mouse_change * 0.2f);
|
|
||||||
info.data.axis = {static_cast<int>(16 * info.last_mouse_change.x),
|
|
||||||
static_cast<int>(16 * -info.last_mouse_change.y)};
|
|
||||||
info.tilt_direction = info.last_mouse_change;
|
info.tilt_direction = info.last_mouse_change;
|
||||||
info.tilt_speed = info.tilt_direction.Normalize() * info.sensitivity;
|
info.tilt_speed = info.tilt_direction.Normalize() * info.sensitivity;
|
||||||
continue;
|
continue;
|
||||||
|
|
|
@ -12,14 +12,15 @@
|
||||||
#include "common/cityhash.h"
|
#include "common/cityhash.h"
|
||||||
#include "common/common_types.h"
|
#include "common/common_types.h"
|
||||||
#include "video_core/renderer_vulkan/fixed_pipeline_state.h"
|
#include "video_core/renderer_vulkan/fixed_pipeline_state.h"
|
||||||
|
#include "video_core/renderer_vulkan/vk_state_tracker.h"
|
||||||
|
|
||||||
namespace Vulkan {
|
namespace Vulkan {
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
constexpr std::size_t POINT = 0;
|
constexpr size_t POINT = 0;
|
||||||
constexpr std::size_t LINE = 1;
|
constexpr size_t LINE = 1;
|
||||||
constexpr std::size_t POLYGON = 2;
|
constexpr size_t POLYGON = 2;
|
||||||
constexpr std::array POLYGON_OFFSET_ENABLE_LUT = {
|
constexpr std::array POLYGON_OFFSET_ENABLE_LUT = {
|
||||||
POINT, // Points
|
POINT, // Points
|
||||||
LINE, // Lines
|
LINE, // Lines
|
||||||
|
@ -40,10 +41,14 @@ constexpr std::array POLYGON_OFFSET_ENABLE_LUT = {
|
||||||
|
|
||||||
} // Anonymous namespace
|
} // Anonymous namespace
|
||||||
|
|
||||||
void FixedPipelineState::Fill(const Maxwell& regs, bool has_extended_dynamic_state) {
|
void FixedPipelineState::Refresh(Tegra::Engines::Maxwell3D& maxwell3d,
|
||||||
const std::array enabled_lut = {regs.polygon_offset_point_enable,
|
bool has_extended_dynamic_state) {
|
||||||
regs.polygon_offset_line_enable,
|
const Maxwell& regs = maxwell3d.regs;
|
||||||
regs.polygon_offset_fill_enable};
|
const std::array enabled_lut{
|
||||||
|
regs.polygon_offset_point_enable,
|
||||||
|
regs.polygon_offset_line_enable,
|
||||||
|
regs.polygon_offset_fill_enable,
|
||||||
|
};
|
||||||
const u32 topology_index = static_cast<u32>(regs.draw.topology.Value());
|
const u32 topology_index = static_cast<u32>(regs.draw.topology.Value());
|
||||||
|
|
||||||
raw1 = 0;
|
raw1 = 0;
|
||||||
|
@ -64,45 +69,53 @@ void FixedPipelineState::Fill(const Maxwell& regs, bool has_extended_dynamic_sta
|
||||||
|
|
||||||
raw2 = 0;
|
raw2 = 0;
|
||||||
const auto test_func =
|
const auto test_func =
|
||||||
regs.alpha_test_enabled == 1 ? regs.alpha_test_func : Maxwell::ComparisonOp::Always;
|
regs.alpha_test_enabled != 0 ? regs.alpha_test_func : Maxwell::ComparisonOp::Always;
|
||||||
alpha_test_func.Assign(PackComparisonOp(test_func));
|
alpha_test_func.Assign(PackComparisonOp(test_func));
|
||||||
early_z.Assign(regs.force_early_fragment_tests != 0 ? 1 : 0);
|
early_z.Assign(regs.force_early_fragment_tests != 0 ? 1 : 0);
|
||||||
|
|
||||||
alpha_test_ref = Common::BitCast<u32>(regs.alpha_test_ref);
|
alpha_test_ref = Common::BitCast<u32>(regs.alpha_test_ref);
|
||||||
point_size = Common::BitCast<u32>(regs.point_size);
|
point_size = Common::BitCast<u32>(regs.point_size);
|
||||||
|
|
||||||
for (std::size_t index = 0; index < Maxwell::NumVertexArrays; ++index) {
|
if (maxwell3d.dirty.flags[Dirty::InstanceDivisors]) {
|
||||||
binding_divisors[index] =
|
maxwell3d.dirty.flags[Dirty::InstanceDivisors] = false;
|
||||||
regs.instanced_arrays.IsInstancingEnabled(index) ? regs.vertex_array[index].divisor : 0;
|
for (size_t index = 0; index < Maxwell::NumVertexArrays; ++index) {
|
||||||
|
const bool is_enabled = regs.instanced_arrays.IsInstancingEnabled(index);
|
||||||
|
binding_divisors[index] = is_enabled ? regs.vertex_array[index].divisor : 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
if (maxwell3d.dirty.flags[Dirty::VertexAttributes]) {
|
||||||
for (size_t index = 0; index < Maxwell::NumVertexAttributes; ++index) {
|
maxwell3d.dirty.flags[Dirty::VertexAttributes] = false;
|
||||||
const auto& input = regs.vertex_attrib_format[index];
|
for (size_t index = 0; index < Maxwell::NumVertexAttributes; ++index) {
|
||||||
auto& attribute = attributes[index];
|
const auto& input = regs.vertex_attrib_format[index];
|
||||||
attribute.raw = 0;
|
auto& attribute = attributes[index];
|
||||||
attribute.enabled.Assign(input.IsConstant() ? 0 : 1);
|
attribute.raw = 0;
|
||||||
attribute.buffer.Assign(input.buffer);
|
attribute.enabled.Assign(input.IsConstant() ? 0 : 1);
|
||||||
attribute.offset.Assign(input.offset);
|
attribute.buffer.Assign(input.buffer);
|
||||||
attribute.type.Assign(static_cast<u32>(input.type.Value()));
|
attribute.offset.Assign(input.offset);
|
||||||
attribute.size.Assign(static_cast<u32>(input.size.Value()));
|
attribute.type.Assign(static_cast<u32>(input.type.Value()));
|
||||||
attribute.binding_index_enabled.Assign(regs.vertex_array[index].IsEnabled() ? 1 : 0);
|
attribute.size.Assign(static_cast<u32>(input.size.Value()));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
if (maxwell3d.dirty.flags[Dirty::Blending]) {
|
||||||
for (std::size_t index = 0; index < std::size(attachments); ++index) {
|
maxwell3d.dirty.flags[Dirty::Blending] = false;
|
||||||
attachments[index].Fill(regs, index);
|
for (size_t index = 0; index < attachments.size(); ++index) {
|
||||||
|
attachments[index].Refresh(regs, index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (maxwell3d.dirty.flags[Dirty::ViewportSwizzles]) {
|
||||||
|
maxwell3d.dirty.flags[Dirty::ViewportSwizzles] = false;
|
||||||
|
const auto& transform = regs.viewport_transform;
|
||||||
|
std::ranges::transform(transform, viewport_swizzles.begin(), [](const auto& viewport) {
|
||||||
|
return static_cast<u16>(viewport.swizzle.raw);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto& transform = regs.viewport_transform;
|
|
||||||
std::transform(transform.begin(), transform.end(), viewport_swizzles.begin(),
|
|
||||||
[](const auto& viewport) { return static_cast<u16>(viewport.swizzle.raw); });
|
|
||||||
|
|
||||||
if (!has_extended_dynamic_state) {
|
if (!has_extended_dynamic_state) {
|
||||||
no_extended_dynamic_state.Assign(1);
|
no_extended_dynamic_state.Assign(1);
|
||||||
dynamic_state.Fill(regs);
|
dynamic_state.Refresh(regs);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void FixedPipelineState::BlendingAttachment::Fill(const Maxwell& regs, std::size_t index) {
|
void FixedPipelineState::BlendingAttachment::Refresh(const Maxwell& regs, size_t index) {
|
||||||
const auto& mask = regs.color_mask[regs.color_mask_common ? 0 : index];
|
const auto& mask = regs.color_mask[regs.color_mask_common ? 0 : index];
|
||||||
|
|
||||||
raw = 0;
|
raw = 0;
|
||||||
|
@ -141,7 +154,7 @@ void FixedPipelineState::BlendingAttachment::Fill(const Maxwell& regs, std::size
|
||||||
enable.Assign(1);
|
enable.Assign(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
void FixedPipelineState::DynamicState::Fill(const Maxwell& regs) {
|
void FixedPipelineState::DynamicState::Refresh(const Maxwell& regs) {
|
||||||
u32 packed_front_face = PackFrontFace(regs.front_face);
|
u32 packed_front_face = PackFrontFace(regs.front_face);
|
||||||
if (regs.screen_y_control.triangle_rast_flip != 0) {
|
if (regs.screen_y_control.triangle_rast_flip != 0) {
|
||||||
// Flip front face
|
// Flip front face
|
||||||
|
@ -178,9 +191,9 @@ void FixedPipelineState::DynamicState::Fill(const Maxwell& regs) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
std::size_t FixedPipelineState::Hash() const noexcept {
|
size_t FixedPipelineState::Hash() const noexcept {
|
||||||
const u64 hash = Common::CityHash64(reinterpret_cast<const char*>(this), Size());
|
const u64 hash = Common::CityHash64(reinterpret_cast<const char*>(this), Size());
|
||||||
return static_cast<std::size_t>(hash);
|
return static_cast<size_t>(hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FixedPipelineState::operator==(const FixedPipelineState& rhs) const noexcept {
|
bool FixedPipelineState::operator==(const FixedPipelineState& rhs) const noexcept {
|
||||||
|
|
|
@ -58,7 +58,7 @@ struct FixedPipelineState {
|
||||||
BitField<30, 1, u32> enable;
|
BitField<30, 1, u32> enable;
|
||||||
};
|
};
|
||||||
|
|
||||||
void Fill(const Maxwell& regs, std::size_t index);
|
void Refresh(const Maxwell& regs, size_t index);
|
||||||
|
|
||||||
constexpr std::array<bool, 4> Mask() const noexcept {
|
constexpr std::array<bool, 4> Mask() const noexcept {
|
||||||
return {mask_r != 0, mask_g != 0, mask_b != 0, mask_a != 0};
|
return {mask_r != 0, mask_g != 0, mask_b != 0, mask_a != 0};
|
||||||
|
@ -96,8 +96,6 @@ struct FixedPipelineState {
|
||||||
BitField<6, 14, u32> offset;
|
BitField<6, 14, u32> offset;
|
||||||
BitField<20, 3, u32> type;
|
BitField<20, 3, u32> type;
|
||||||
BitField<23, 6, u32> size;
|
BitField<23, 6, u32> size;
|
||||||
// Not really an element of a vertex attribute, but it can be packed here
|
|
||||||
BitField<29, 1, u32> binding_index_enabled;
|
|
||||||
|
|
||||||
constexpr Maxwell::VertexAttribute::Type Type() const noexcept {
|
constexpr Maxwell::VertexAttribute::Type Type() const noexcept {
|
||||||
return static_cast<Maxwell::VertexAttribute::Type>(type.Value());
|
return static_cast<Maxwell::VertexAttribute::Type>(type.Value());
|
||||||
|
@ -108,7 +106,7 @@ struct FixedPipelineState {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <std::size_t Position>
|
template <size_t Position>
|
||||||
union StencilFace {
|
union StencilFace {
|
||||||
BitField<Position + 0, 3, u32> action_stencil_fail;
|
BitField<Position + 0, 3, u32> action_stencil_fail;
|
||||||
BitField<Position + 3, 3, u32> action_depth_fail;
|
BitField<Position + 3, 3, u32> action_depth_fail;
|
||||||
|
@ -152,7 +150,7 @@ struct FixedPipelineState {
|
||||||
// Vertex stride is a 12 bits value, we have 4 bits to spare per element
|
// Vertex stride is a 12 bits value, we have 4 bits to spare per element
|
||||||
std::array<u16, Maxwell::NumVertexArrays> vertex_strides;
|
std::array<u16, Maxwell::NumVertexArrays> vertex_strides;
|
||||||
|
|
||||||
void Fill(const Maxwell& regs);
|
void Refresh(const Maxwell& regs);
|
||||||
|
|
||||||
Maxwell::ComparisonOp DepthTestFunc() const noexcept {
|
Maxwell::ComparisonOp DepthTestFunc() const noexcept {
|
||||||
return UnpackComparisonOp(depth_test_func);
|
return UnpackComparisonOp(depth_test_func);
|
||||||
|
@ -199,9 +197,9 @@ struct FixedPipelineState {
|
||||||
std::array<u16, Maxwell::NumViewports> viewport_swizzles;
|
std::array<u16, Maxwell::NumViewports> viewport_swizzles;
|
||||||
DynamicState dynamic_state;
|
DynamicState dynamic_state;
|
||||||
|
|
||||||
void Fill(const Maxwell& regs, bool has_extended_dynamic_state);
|
void Refresh(Tegra::Engines::Maxwell3D& maxwell3d, bool has_extended_dynamic_state);
|
||||||
|
|
||||||
std::size_t Hash() const noexcept;
|
size_t Hash() const noexcept;
|
||||||
|
|
||||||
bool operator==(const FixedPipelineState& rhs) const noexcept;
|
bool operator==(const FixedPipelineState& rhs) const noexcept;
|
||||||
|
|
||||||
|
@ -209,8 +207,8 @@ struct FixedPipelineState {
|
||||||
return !operator==(rhs);
|
return !operator==(rhs);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::size_t Size() const noexcept {
|
size_t Size() const noexcept {
|
||||||
const std::size_t total_size = sizeof *this;
|
const size_t total_size = sizeof *this;
|
||||||
return total_size - (no_extended_dynamic_state != 0 ? 0 : sizeof(DynamicState));
|
return total_size - (no_extended_dynamic_state != 0 ? 0 : sizeof(DynamicState));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -224,7 +222,7 @@ namespace std {
|
||||||
|
|
||||||
template <>
|
template <>
|
||||||
struct hash<Vulkan::FixedPipelineState> {
|
struct hash<Vulkan::FixedPipelineState> {
|
||||||
std::size_t operator()(const Vulkan::FixedPipelineState& k) const noexcept {
|
size_t operator()(const Vulkan::FixedPipelineState& k) const noexcept {
|
||||||
return k.Hash();
|
return k.Hash();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -221,9 +221,6 @@ vk::Pipeline VKGraphicsPipeline::CreatePipeline(const SPIRVProgram& program,
|
||||||
std::vector<VkVertexInputBindingDescription> vertex_bindings;
|
std::vector<VkVertexInputBindingDescription> vertex_bindings;
|
||||||
std::vector<VkVertexInputBindingDivisorDescriptionEXT> vertex_binding_divisors;
|
std::vector<VkVertexInputBindingDivisorDescriptionEXT> vertex_binding_divisors;
|
||||||
for (std::size_t index = 0; index < Maxwell::NumVertexArrays; ++index) {
|
for (std::size_t index = 0; index < Maxwell::NumVertexArrays; ++index) {
|
||||||
if (state.attributes[index].binding_index_enabled == 0) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
const bool instanced = state.binding_divisors[index] != 0;
|
const bool instanced = state.binding_divisors[index] != 0;
|
||||||
const auto rate = instanced ? VK_VERTEX_INPUT_RATE_INSTANCE : VK_VERTEX_INPUT_RATE_VERTEX;
|
const auto rate = instanced ? VK_VERTEX_INPUT_RATE_INSTANCE : VK_VERTEX_INPUT_RATE_VERTEX;
|
||||||
vertex_bindings.push_back({
|
vertex_bindings.push_back({
|
||||||
|
|
|
@ -270,8 +270,7 @@ void RasterizerVulkan::Draw(bool is_indexed, bool is_instanced) {
|
||||||
|
|
||||||
query_cache.UpdateCounters();
|
query_cache.UpdateCounters();
|
||||||
|
|
||||||
GraphicsPipelineCacheKey key;
|
graphics_key.fixed_state.Refresh(maxwell3d, device.IsExtExtendedDynamicStateSupported());
|
||||||
key.fixed_state.Fill(maxwell3d.regs, device.IsExtExtendedDynamicStateSupported());
|
|
||||||
|
|
||||||
std::scoped_lock lock{buffer_cache.mutex, texture_cache.mutex};
|
std::scoped_lock lock{buffer_cache.mutex, texture_cache.mutex};
|
||||||
|
|
||||||
|
@ -279,14 +278,16 @@ void RasterizerVulkan::Draw(bool is_indexed, bool is_instanced) {
|
||||||
texture_cache.UpdateRenderTargets(false);
|
texture_cache.UpdateRenderTargets(false);
|
||||||
|
|
||||||
const auto shaders = pipeline_cache.GetShaders();
|
const auto shaders = pipeline_cache.GetShaders();
|
||||||
key.shaders = GetShaderAddresses(shaders);
|
graphics_key.shaders = GetShaderAddresses(shaders);
|
||||||
|
|
||||||
|
graphics_key.shaders = GetShaderAddresses(shaders);
|
||||||
SetupShaderDescriptors(shaders, is_indexed);
|
SetupShaderDescriptors(shaders, is_indexed);
|
||||||
|
|
||||||
const Framebuffer* const framebuffer = texture_cache.GetFramebuffer();
|
const Framebuffer* const framebuffer = texture_cache.GetFramebuffer();
|
||||||
key.renderpass = framebuffer->RenderPass();
|
graphics_key.renderpass = framebuffer->RenderPass();
|
||||||
|
|
||||||
auto* const pipeline =
|
VKGraphicsPipeline* const pipeline = pipeline_cache.GetGraphicsPipeline(
|
||||||
pipeline_cache.GetGraphicsPipeline(key, framebuffer->NumColorBuffers(), async_shaders);
|
graphics_key, framebuffer->NumColorBuffers(), async_shaders);
|
||||||
if (pipeline == nullptr || pipeline->GetHandle() == VK_NULL_HANDLE) {
|
if (pipeline == nullptr || pipeline->GetHandle() == VK_NULL_HANDLE) {
|
||||||
// Async graphics pipeline was not ready.
|
// Async graphics pipeline was not ready.
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -20,6 +20,7 @@
|
||||||
#include "video_core/renderer_vulkan/vk_buffer_cache.h"
|
#include "video_core/renderer_vulkan/vk_buffer_cache.h"
|
||||||
#include "video_core/renderer_vulkan/vk_descriptor_pool.h"
|
#include "video_core/renderer_vulkan/vk_descriptor_pool.h"
|
||||||
#include "video_core/renderer_vulkan/vk_fence_manager.h"
|
#include "video_core/renderer_vulkan/vk_fence_manager.h"
|
||||||
|
#include "video_core/renderer_vulkan/vk_graphics_pipeline.h"
|
||||||
#include "video_core/renderer_vulkan/vk_pipeline_cache.h"
|
#include "video_core/renderer_vulkan/vk_pipeline_cache.h"
|
||||||
#include "video_core/renderer_vulkan/vk_query_cache.h"
|
#include "video_core/renderer_vulkan/vk_query_cache.h"
|
||||||
#include "video_core/renderer_vulkan/vk_scheduler.h"
|
#include "video_core/renderer_vulkan/vk_scheduler.h"
|
||||||
|
@ -176,6 +177,8 @@ private:
|
||||||
BlitImageHelper blit_image;
|
BlitImageHelper blit_image;
|
||||||
ASTCDecoderPass astc_decoder_pass;
|
ASTCDecoderPass astc_decoder_pass;
|
||||||
|
|
||||||
|
GraphicsPipelineCacheKey graphics_key;
|
||||||
|
|
||||||
TextureCacheRuntime texture_cache_runtime;
|
TextureCacheRuntime texture_cache_runtime;
|
||||||
TextureCache texture_cache;
|
TextureCache texture_cache;
|
||||||
BufferCacheRuntime buffer_cache_runtime;
|
BufferCacheRuntime buffer_cache_runtime;
|
||||||
|
|
|
@ -18,9 +18,7 @@
|
||||||
#define NUM(field_name) (sizeof(Maxwell3D::Regs::field_name) / (sizeof(u32)))
|
#define NUM(field_name) (sizeof(Maxwell3D::Regs::field_name) / (sizeof(u32)))
|
||||||
|
|
||||||
namespace Vulkan {
|
namespace Vulkan {
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
using namespace Dirty;
|
using namespace Dirty;
|
||||||
using namespace VideoCommon::Dirty;
|
using namespace VideoCommon::Dirty;
|
||||||
using Tegra::Engines::Maxwell3D;
|
using Tegra::Engines::Maxwell3D;
|
||||||
|
@ -128,6 +126,34 @@ void SetupDirtyStencilTestEnable(Tables& tables) {
|
||||||
tables[0][OFF(stencil_enable)] = StencilTestEnable;
|
tables[0][OFF(stencil_enable)] = StencilTestEnable;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SetupDirtyBlending(Tables& tables) {
|
||||||
|
tables[0][OFF(color_mask_common)] = Blending;
|
||||||
|
tables[0][OFF(independent_blend_enable)] = Blending;
|
||||||
|
FillBlock(tables[0], OFF(color_mask), NUM(color_mask), Blending);
|
||||||
|
FillBlock(tables[0], OFF(blend), NUM(blend), Blending);
|
||||||
|
FillBlock(tables[0], OFF(independent_blend), NUM(independent_blend), Blending);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetupDirtyInstanceDivisors(Tables& tables) {
|
||||||
|
static constexpr size_t divisor_offset = 3;
|
||||||
|
for (size_t index = 0; index < Regs::NumVertexArrays; ++index) {
|
||||||
|
tables[0][OFF(instanced_arrays) + index] = InstanceDivisors;
|
||||||
|
tables[0][OFF(vertex_array) + index * NUM(vertex_array[0]) + divisor_offset] =
|
||||||
|
InstanceDivisors;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetupDirtyVertexAttributes(Tables& tables) {
|
||||||
|
FillBlock(tables[0], OFF(vertex_attrib_format), NUM(vertex_attrib_format), VertexAttributes);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetupDirtyViewportSwizzles(Tables& tables) {
|
||||||
|
static constexpr size_t swizzle_offset = 6;
|
||||||
|
for (size_t index = 0; index < Regs::NumViewports; ++index) {
|
||||||
|
tables[0][OFF(viewport_transform) + index * NUM(viewport_transform[0]) + swizzle_offset] =
|
||||||
|
ViewportSwizzles;
|
||||||
|
}
|
||||||
|
}
|
||||||
} // Anonymous namespace
|
} // Anonymous namespace
|
||||||
|
|
||||||
StateTracker::StateTracker(Tegra::GPU& gpu)
|
StateTracker::StateTracker(Tegra::GPU& gpu)
|
||||||
|
@ -148,6 +174,10 @@ StateTracker::StateTracker(Tegra::GPU& gpu)
|
||||||
SetupDirtyFrontFace(tables);
|
SetupDirtyFrontFace(tables);
|
||||||
SetupDirtyStencilOp(tables);
|
SetupDirtyStencilOp(tables);
|
||||||
SetupDirtyStencilTestEnable(tables);
|
SetupDirtyStencilTestEnable(tables);
|
||||||
|
SetupDirtyBlending(tables);
|
||||||
|
SetupDirtyInstanceDivisors(tables);
|
||||||
|
SetupDirtyVertexAttributes(tables);
|
||||||
|
SetupDirtyViewportSwizzles(tables);
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Vulkan
|
} // namespace Vulkan
|
||||||
|
|
|
@ -35,6 +35,11 @@ enum : u8 {
|
||||||
StencilOp,
|
StencilOp,
|
||||||
StencilTestEnable,
|
StencilTestEnable,
|
||||||
|
|
||||||
|
Blending,
|
||||||
|
InstanceDivisors,
|
||||||
|
VertexAttributes,
|
||||||
|
ViewportSwizzles,
|
||||||
|
|
||||||
Last
|
Last
|
||||||
};
|
};
|
||||||
static_assert(Last <= std::numeric_limits<u8>::max());
|
static_assert(Last <= std::numeric_limits<u8>::max());
|
||||||
|
|
Loading…
Reference in a new issue