mirror of
https://github.com/RPCS3/rpcs3.git
synced 2025-02-05 06:39:52 +00:00
GL: re-use common fp/vp decompiler (#2100)
This commit is contained in:
parent
7f3cb4d3c9
commit
a270ac7f02
@ -217,7 +217,7 @@ public:
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void fill_fragment_constans_buffer(gsl::span<f32, gsl::dynamic_range> dst_buffer, const RSXFragmentProgram &fragment_program) const
|
void fill_fragment_constants_buffer(gsl::span<f32, gsl::dynamic_range> dst_buffer, const RSXFragmentProgram &fragment_program) const
|
||||||
{
|
{
|
||||||
const auto I = m_fragment_shader_cache.find(fragment_program);
|
const auto I = m_fragment_shader_cache.find(fragment_program);
|
||||||
if (I == m_fragment_shader_cache.end())
|
if (I == m_fragment_shader_cache.end())
|
||||||
|
@ -301,7 +301,7 @@ D3D12_CONSTANT_BUFFER_VIEW_DESC D3D12GSRender::upload_fragment_shader_constants(
|
|||||||
|
|
||||||
size_t offset = 0;
|
size_t offset = 0;
|
||||||
float *mapped_buffer = m_buffer_data.map<float>(CD3DX12_RANGE(heap_offset, heap_offset + buffer_size));
|
float *mapped_buffer = m_buffer_data.map<float>(CD3DX12_RANGE(heap_offset, heap_offset + buffer_size));
|
||||||
m_pso_cache.fill_fragment_constans_buffer({ mapped_buffer, ::narrow<int>(buffer_size) }, m_fragment_program);
|
m_pso_cache.fill_fragment_constants_buffer({ mapped_buffer, ::narrow<int>(buffer_size) }, m_fragment_program);
|
||||||
m_buffer_data.unmap(CD3DX12_RANGE(heap_offset, heap_offset + buffer_size));
|
m_buffer_data.unmap(CD3DX12_RANGE(heap_offset, heap_offset + buffer_size));
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
@ -2,7 +2,6 @@
|
|||||||
#include "Utilities/Config.h"
|
#include "Utilities/Config.h"
|
||||||
#include "Emu/Memory/Memory.h"
|
#include "Emu/Memory/Memory.h"
|
||||||
#include "GLGSRender.h"
|
#include "GLGSRender.h"
|
||||||
#include "rsx_gl_cache.h"
|
|
||||||
#include "../rsx_methods.h"
|
#include "../rsx_methods.h"
|
||||||
#include "../Common/BufferUtils.h"
|
#include "../Common/BufferUtils.h"
|
||||||
#include "../rsx_utils.h"
|
#include "../rsx_utils.h"
|
||||||
@ -27,7 +26,7 @@ namespace
|
|||||||
|
|
||||||
GLGSRender::GLGSRender() : GSRender(frame_type::OpenGL)
|
GLGSRender::GLGSRender() : GSRender(frame_type::OpenGL)
|
||||||
{
|
{
|
||||||
init_glsl_cache_program_context(programs_cache.context);
|
shaders_cache.load(rsx::old_shaders_cache::shader_language::glsl);
|
||||||
}
|
}
|
||||||
|
|
||||||
u32 GLGSRender::enable(u32 condition, u32 cap)
|
u32 GLGSRender::enable(u32 condition, u32 cap)
|
||||||
@ -144,7 +143,7 @@ namespace
|
|||||||
case rsx::logic_op::logic_and_inverted: return GL_AND_INVERTED;
|
case rsx::logic_op::logic_and_inverted: return GL_AND_INVERTED;
|
||||||
case rsx::logic_op::logic_noop: return GL_NOOP;
|
case rsx::logic_op::logic_noop: return GL_NOOP;
|
||||||
case rsx::logic_op::logic_xor: return GL_XOR;
|
case rsx::logic_op::logic_xor: return GL_XOR;
|
||||||
case rsx::logic_op::logic_or : return GL_OR;
|
case rsx::logic_op::logic_or: return GL_OR;
|
||||||
case rsx::logic_op::logic_nor: return GL_NOR;
|
case rsx::logic_op::logic_nor: return GL_NOR;
|
||||||
case rsx::logic_op::logic_equiv: return GL_EQUIV;
|
case rsx::logic_op::logic_equiv: return GL_EQUIV;
|
||||||
case rsx::logic_op::logic_invert: return GL_INVERT;
|
case rsx::logic_op::logic_invert: return GL_INVERT;
|
||||||
@ -347,50 +346,6 @@ void GLGSRender::end()
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
rsx::user_clip_plane_op clip_plane_0 = rsx::method_registers.clip_plane_0_enabled();
|
|
||||||
rsx::user_clip_plane_op clip_plane_1 = rsx::method_registers.clip_plane_1_enabled();
|
|
||||||
rsx::user_clip_plane_op clip_plane_2 = rsx::method_registers.clip_plane_2_enabled();
|
|
||||||
rsx::user_clip_plane_op clip_plane_3 = rsx::method_registers.clip_plane_3_enabled();
|
|
||||||
rsx::user_clip_plane_op clip_plane_4 = rsx::method_registers.clip_plane_4_enabled();
|
|
||||||
rsx::user_clip_plane_op clip_plane_5 = rsx::method_registers.clip_plane_5_enabled();
|
|
||||||
|
|
||||||
auto set_clip_plane_control = [&](int index, rsx::user_clip_plane_op control)
|
|
||||||
{
|
|
||||||
int value = 0;
|
|
||||||
int location;
|
|
||||||
if (m_program->uniforms.has_location("uc_m" + std::to_string(index), &location))
|
|
||||||
{
|
|
||||||
switch (control)
|
|
||||||
{
|
|
||||||
default:
|
|
||||||
LOG_ERROR(RSX, "bad clip plane control (0x%x)", (u32)control);
|
|
||||||
|
|
||||||
case rsx::user_clip_plane_op::disable:
|
|
||||||
value = 0;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case rsx::user_clip_plane_op::greather_or_equal:
|
|
||||||
value = 1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case rsx::user_clip_plane_op::less_than:
|
|
||||||
value = -1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
__glcheck m_program->uniforms[location] = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
__glcheck enable(value, GL_CLIP_DISTANCE0 + index);
|
|
||||||
};
|
|
||||||
|
|
||||||
set_clip_plane_control(0, clip_plane_0);
|
|
||||||
set_clip_plane_control(1, clip_plane_1);
|
|
||||||
set_clip_plane_control(2, clip_plane_2);
|
|
||||||
set_clip_plane_control(3, clip_plane_3);
|
|
||||||
set_clip_plane_control(4, clip_plane_4);
|
|
||||||
set_clip_plane_control(5, clip_plane_5);
|
|
||||||
|
|
||||||
draw_fbo.bind();
|
draw_fbo.bind();
|
||||||
m_program->use();
|
m_program->use();
|
||||||
|
|
||||||
@ -408,82 +363,24 @@ void GLGSRender::end()
|
|||||||
ds->set_cleared();
|
ds->set_cleared();
|
||||||
}
|
}
|
||||||
|
|
||||||
//setup textures
|
//Setup textures
|
||||||
{
|
|
||||||
for (int i = 0; i < rsx::limits::fragment_textures_count; ++i)
|
for (int i = 0; i < rsx::limits::fragment_textures_count; ++i)
|
||||||
{
|
{
|
||||||
int location;
|
int location;
|
||||||
if (m_program->uniforms.has_location("ftexture" + std::to_string(i), &location))
|
if (m_program->uniforms.has_location("tex" + std::to_string(i), &location))
|
||||||
{
|
{
|
||||||
if (!rsx::method_registers.fragment_textures[i].enabled())
|
if (!rsx::method_registers.fragment_textures[i].enabled())
|
||||||
{
|
{
|
||||||
glActiveTexture(GL_TEXTURE0 + i);
|
glActiveTexture(GL_TEXTURE0 + i);
|
||||||
glBindTexture(GL_TEXTURE_2D, 0);
|
glBindTexture(GL_TEXTURE_2D, 0);
|
||||||
glProgramUniform1i(m_program->id(), location, i);
|
glProgramUniform1i(m_program->id(), location, i);
|
||||||
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
m_gl_textures[i].set_target(get_gl_target_for_texture(rsx::method_registers.fragment_textures[i]));
|
m_gl_textures[i].set_target(get_gl_target_for_texture(rsx::method_registers.fragment_textures[i]));
|
||||||
|
|
||||||
__glcheck m_gl_texture_cache.upload_texture(i, rsx::method_registers.fragment_textures[i], m_gl_textures[i], m_rtts);
|
__glcheck m_gl_texture_cache.upload_texture(i, rsx::method_registers.fragment_textures[i], m_gl_textures[i], m_rtts);
|
||||||
__glcheck glProgramUniform1i(m_program->id(), location, i);
|
|
||||||
|
|
||||||
if (m_program->uniforms.has_location("ftexture" + std::to_string(i) + "_cm", &location))
|
|
||||||
{
|
|
||||||
if (rsx::method_registers.fragment_textures[i].format() & CELL_GCM_TEXTURE_UN)
|
|
||||||
{
|
|
||||||
u32 width = std::max<u32>(rsx::method_registers.fragment_textures[i].width(), 1);
|
|
||||||
u32 height = std::max<u32>(rsx::method_registers.fragment_textures[i].height(), 1);
|
|
||||||
u32 depth = std::max<u32>(rsx::method_registers.fragment_textures[i].depth(), 1);
|
|
||||||
|
|
||||||
glProgramUniform4f(m_program->id(), location, 1.f / width, 1.f / height, 1.f / depth, 1.0f);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
//This shader may have been re-used with a different texture config. Have to reset this
|
|
||||||
glProgramUniform4f(m_program->id(), location, 1.f, 1.f, 1.f, 1.f);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (int i = 0; i < rsx::limits::vertex_textures_count; ++i)
|
|
||||||
{
|
|
||||||
int location;
|
|
||||||
if (m_program->uniforms.has_location("vtexture" + std::to_string(i), &location))
|
|
||||||
{
|
|
||||||
if (!rsx::method_registers.fragment_textures[i].enabled())
|
|
||||||
{
|
|
||||||
glActiveTexture(GL_TEXTURE0 + i);
|
|
||||||
glBindTexture(GL_TEXTURE_2D, 0);
|
|
||||||
glProgramUniform1i(m_program->id(), location, i);
|
glProgramUniform1i(m_program->id(), location, i);
|
||||||
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
m_gl_vertex_textures[i].set_target(get_gl_target_for_texture(rsx::method_registers.vertex_textures[i]));
|
|
||||||
|
|
||||||
__glcheck m_gl_texture_cache.upload_texture(i, rsx::method_registers.vertex_textures[i], m_gl_vertex_textures[i], m_rtts);
|
|
||||||
__glcheck glProgramUniform1i(m_program->id(), location, i);
|
|
||||||
|
|
||||||
if (m_program->uniforms.has_location("vtexture" + std::to_string(i) + "_cm", &location))
|
|
||||||
{
|
|
||||||
if (rsx::method_registers.fragment_textures[i].format() & CELL_GCM_TEXTURE_UN)
|
|
||||||
{
|
|
||||||
u32 width = std::max<u32>(rsx::method_registers.fragment_textures[i].width(), 1);
|
|
||||||
u32 height = std::max<u32>(rsx::method_registers.fragment_textures[i].height(), 1);
|
|
||||||
u32 depth = std::max<u32>(rsx::method_registers.fragment_textures[i].depth(), 1);
|
|
||||||
|
|
||||||
glProgramUniform4f(m_program->id(), location, 1.f / width, 1.f / height, 1.f / depth, 1.0f);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
//This shader may have been re-used with a different texture config. Have to reset this
|
|
||||||
glProgramUniform4f(m_program->id(), location, 1.f, 1.f, 1.f, 1.f);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -580,7 +477,7 @@ void GLGSRender::on_exit()
|
|||||||
{
|
{
|
||||||
glDisable(GL_VERTEX_PROGRAM_POINT_SIZE);
|
glDisable(GL_VERTEX_PROGRAM_POINT_SIZE);
|
||||||
|
|
||||||
programs_cache.clear();
|
m_prog_buffer.clear();
|
||||||
|
|
||||||
if (draw_fbo)
|
if (draw_fbo)
|
||||||
{
|
{
|
||||||
@ -616,7 +513,6 @@ void GLGSRender::on_exit()
|
|||||||
|
|
||||||
void nv4097_clear_surface(u32 arg, GLGSRender* renderer)
|
void nv4097_clear_surface(u32 arg, GLGSRender* renderer)
|
||||||
{
|
{
|
||||||
//LOG_NOTICE(Log::RSX, "nv4097_clear_surface(0x%x)", arg);
|
|
||||||
if (rsx::method_registers.surface_color_target() == rsx::surface_target::none) return;
|
if (rsx::method_registers.surface_color_target() == rsx::surface_target::none) return;
|
||||||
|
|
||||||
if ((arg & 0xf3) == 0)
|
if ((arg & 0xf3) == 0)
|
||||||
@ -625,14 +521,6 @@ void nv4097_clear_surface(u32 arg, GLGSRender* renderer)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
u16 clear_x = rsx::method_registers[NV4097_SET_CLEAR_RECT_HORIZONTAL];
|
|
||||||
u16 clear_y = rsx::method_registers[NV4097_SET_CLEAR_RECT_VERTICAL];
|
|
||||||
u16 clear_w = rsx::method_registers[NV4097_SET_CLEAR_RECT_HORIZONTAL] >> 16;
|
|
||||||
u16 clear_h = rsx::method_registers[NV4097_SET_CLEAR_RECT_VERTICAL] >> 16;
|
|
||||||
glScissor(clear_x, clear_y, clear_w, clear_h);
|
|
||||||
*/
|
|
||||||
|
|
||||||
renderer->init_buffers(true);
|
renderer->init_buffers(true);
|
||||||
renderer->draw_fbo.bind();
|
renderer->draw_fbo.bind();
|
||||||
|
|
||||||
@ -712,27 +600,23 @@ bool GLGSRender::do_method(u32 cmd, u32 arg)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
//binding 0
|
struct alignas(4) glsl_scale_buffer
|
||||||
struct alignas(4) glsl_matrix_buffer
|
|
||||||
{
|
{
|
||||||
float viewport_matrix[4][4];
|
float viewport_matrix[4][4];
|
||||||
float window_matrix[4][4];
|
float window_matrix[4][4];
|
||||||
float normalize_matrix[4][4];
|
float normalize_matrix[4][4];
|
||||||
};
|
};
|
||||||
|
|
||||||
//binding 1
|
|
||||||
struct alignas(4) glsl_vertex_constants_buffer
|
struct alignas(4) glsl_vertex_constants_buffer
|
||||||
{
|
{
|
||||||
float vc[468][4];
|
float vc[468][4];
|
||||||
};
|
};
|
||||||
|
|
||||||
//binding 2
|
|
||||||
struct alignas(4) glsl_fragment_constants_buffer
|
struct alignas(4) glsl_fragment_constants_buffer
|
||||||
{
|
{
|
||||||
float fc[2048][4];
|
float fc[2048][4];
|
||||||
};
|
};
|
||||||
|
|
||||||
//binding 3
|
|
||||||
struct alignas(4) glsl_fragment_state_buffer
|
struct alignas(4) glsl_fragment_state_buffer
|
||||||
{
|
{
|
||||||
float fog_param0;
|
float fog_param0;
|
||||||
@ -741,49 +625,6 @@ struct alignas(4) glsl_fragment_state_buffer
|
|||||||
float alpha_ref;
|
float alpha_ref;
|
||||||
};
|
};
|
||||||
|
|
||||||
static void fill_matrix_buffer(glsl_matrix_buffer *buffer)
|
|
||||||
{
|
|
||||||
rsx::fill_viewport_matrix(buffer->viewport_matrix, true);
|
|
||||||
rsx::fill_window_matrix(buffer->window_matrix, true);
|
|
||||||
|
|
||||||
f32 viewport_x = rsx::method_registers.viewport_origin_x();
|
|
||||||
f32 viewport_y = rsx::method_registers.viewport_origin_y();
|
|
||||||
f32 viewport_w = rsx::method_registers.viewport_width();
|
|
||||||
f32 viewport_h = rsx::method_registers.viewport_height();
|
|
||||||
|
|
||||||
rsx::window_origin shader_window_origin = rsx::method_registers.shader_window_origin();
|
|
||||||
u16 shader_window_height = rsx::method_registers.shader_window_height();
|
|
||||||
|
|
||||||
f32 left = viewport_x;
|
|
||||||
f32 right = viewport_x + viewport_w;
|
|
||||||
f32 top = viewport_y;
|
|
||||||
f32 bottom = viewport_y + viewport_h;
|
|
||||||
//f32 far_ = (f32&)rsx::method_registers[NV4097_SET_CLIP_MAX];
|
|
||||||
//f32 near_ = (f32&)rsx::method_registers[NV4097_SET_CLIP_MIN];
|
|
||||||
|
|
||||||
if (shader_window_origin == rsx::window_origin::bottom)
|
|
||||||
{
|
|
||||||
top = shader_window_height - (viewport_y + viewport_h) + 1;
|
|
||||||
bottom = shader_window_height - viewport_y + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
f32 scale_x = 2.0f / (right - left);
|
|
||||||
f32 scale_y = 2.0f / (top - bottom);
|
|
||||||
f32 scale_z = 2.0f;
|
|
||||||
|
|
||||||
f32 offset_x = -(right + left) / (right - left);
|
|
||||||
f32 offset_y = -(top + bottom) / (top - bottom);
|
|
||||||
f32 offset_z = -1.0;
|
|
||||||
|
|
||||||
if (shader_window_origin == rsx::window_origin::top)
|
|
||||||
{
|
|
||||||
scale_y = -scale_y;
|
|
||||||
offset_y = -offset_y;
|
|
||||||
}
|
|
||||||
|
|
||||||
rsx::fill_scale_offset_matrix(buffer->normalize_matrix, true, offset_x, offset_y, offset_z, scale_x, scale_y, scale_z);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void fill_fragment_state_buffer(glsl_fragment_state_buffer *buffer)
|
static void fill_fragment_state_buffer(glsl_fragment_state_buffer *buffer)
|
||||||
{
|
{
|
||||||
const float fog_params[2] = { rsx::method_registers.fog_params_0(), rsx::method_registers.fog_params_1() };
|
const float fog_params[2] = { rsx::method_registers.fog_params_0(), rsx::method_registers.fog_params_1() };
|
||||||
@ -795,94 +636,66 @@ static void fill_fragment_state_buffer(glsl_fragment_state_buffer *buffer)
|
|||||||
|
|
||||||
bool GLGSRender::load_program()
|
bool GLGSRender::load_program()
|
||||||
{
|
{
|
||||||
rsx::raw_program prog = get_raw_program();
|
RSXVertexProgram vertex_program = get_current_vertex_program();
|
||||||
rsx::program_info info = programs_cache.get(prog, rsx::decompile_language::glsl);
|
RSXFragmentProgram fragment_program = get_current_fragment_program();
|
||||||
m_program = (gl::glsl::program*)info.program;
|
|
||||||
|
m_program = &m_prog_buffer.getGraphicPipelineState(vertex_program, fragment_program, nullptr);
|
||||||
m_program->use();
|
m_program->use();
|
||||||
|
|
||||||
u32 fragment_constants_count = info.fragment_shader.decompiled->constants.size();
|
u32 fragment_constants_size = m_prog_buffer.get_fragment_constants_buffer_size(fragment_program);
|
||||||
u32 fragment_constants_size = fragment_constants_count * sizeof(rsx::fragment_program::ucode_instr);
|
fragment_constants_size = std::max(32U, fragment_constants_size);
|
||||||
|
|
||||||
u32 max_buffer_sz =
|
u32 max_buffer_sz =
|
||||||
align(sizeof(glsl_matrix_buffer), m_uniform_buffer_offset_align) +
|
align(sizeof(glsl_scale_buffer), m_uniform_buffer_offset_align) +
|
||||||
align(sizeof(glsl_vertex_constants_buffer), m_uniform_buffer_offset_align) +
|
align(sizeof(glsl_vertex_constants_buffer), m_uniform_buffer_offset_align) +
|
||||||
align(sizeof(glsl_fragment_state_buffer), m_uniform_buffer_offset_align) +
|
align(fragment_constants_size, m_uniform_buffer_offset_align) +
|
||||||
align(fragment_constants_size, m_uniform_buffer_offset_align);
|
align(sizeof(glsl_fragment_state_buffer), m_uniform_buffer_offset_align);
|
||||||
|
|
||||||
m_uniform_ring_buffer.reserve_and_map(max_buffer_sz);
|
|
||||||
|
|
||||||
u32 scale_offset_offset;
|
u32 scale_offset_offset;
|
||||||
u32 vertex_constants_offset;
|
u32 vertex_constants_offset;
|
||||||
u32 fragment_constants_offset;
|
u32 fragment_constants_offset;
|
||||||
u32 fragment_state_offset;
|
u32 fragment_state_offset;
|
||||||
|
|
||||||
|
m_uniform_ring_buffer.reserve_and_map(max_buffer_sz);
|
||||||
|
|
||||||
|
// Scale offset
|
||||||
{
|
{
|
||||||
auto mapping = m_uniform_ring_buffer.alloc_from_reserve(sizeof(glsl_matrix_buffer), m_uniform_buffer_offset_align);
|
auto mapping = m_uniform_ring_buffer.alloc_from_reserve(sizeof(glsl_scale_buffer), m_uniform_buffer_offset_align);
|
||||||
fill_matrix_buffer((glsl_matrix_buffer *)mapping.first);
|
fill_scale_offset_data((glsl_scale_buffer *)mapping.first, false);
|
||||||
scale_offset_offset = mapping.second;
|
scale_offset_offset = mapping.second;
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
// Fragment state
|
||||||
auto mapping = m_uniform_ring_buffer.alloc_from_reserve(sizeof(glsl_vertex_constants_buffer), m_uniform_buffer_offset_align);
|
|
||||||
fill_vertex_program_constants_data(mapping.first);
|
|
||||||
vertex_constants_offset = mapping.second;
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
{
|
||||||
auto mapping = m_uniform_ring_buffer.alloc_from_reserve(sizeof(glsl_fragment_state_buffer), m_uniform_buffer_offset_align);
|
auto mapping = m_uniform_ring_buffer.alloc_from_reserve(sizeof(glsl_fragment_state_buffer), m_uniform_buffer_offset_align);
|
||||||
fill_fragment_state_buffer((glsl_fragment_state_buffer *)mapping.first);
|
fill_fragment_state_buffer((glsl_fragment_state_buffer *)mapping.first);
|
||||||
fragment_state_offset = mapping.second;
|
fragment_state_offset = mapping.second;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Vertex constants
|
||||||
|
{
|
||||||
|
auto mapping = m_uniform_ring_buffer.alloc_from_reserve(sizeof(glsl_vertex_constants_buffer), m_uniform_buffer_offset_align);
|
||||||
|
fill_vertex_program_constants_data(mapping.first);
|
||||||
|
vertex_constants_offset = mapping.second;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fragment constants
|
||||||
if (fragment_constants_size)
|
if (fragment_constants_size)
|
||||||
{
|
{
|
||||||
auto mapping = m_uniform_ring_buffer.alloc_from_reserve(fragment_constants_size, m_uniform_buffer_offset_align);
|
auto mapping = m_uniform_ring_buffer.alloc_from_reserve(fragment_constants_size, m_uniform_buffer_offset_align);
|
||||||
|
u8 *buf = static_cast<u8*>(mapping.first);
|
||||||
|
m_prog_buffer.fill_fragment_constants_buffer({ reinterpret_cast<float*>(buf), gsl::narrow<int>(fragment_constants_size) }, fragment_program);
|
||||||
fragment_constants_offset = mapping.second;
|
fragment_constants_offset = mapping.second;
|
||||||
|
|
||||||
static const __m128i mask = _mm_set_epi8(
|
|
||||||
0xE, 0xF, 0xC, 0xD,
|
|
||||||
0xA, 0xB, 0x8, 0x9,
|
|
||||||
0x6, 0x7, 0x4, 0x5,
|
|
||||||
0x2, 0x3, 0x0, 0x1);
|
|
||||||
|
|
||||||
//The shader may be the same, but the value of the constants (and the shader location in memory) may have changed
|
|
||||||
//Point to the current shader location, not the cached version
|
|
||||||
auto ucode = (const rsx::fragment_program::ucode_instr *)prog.fragment_shader.ucode_ptr;
|
|
||||||
|
|
||||||
auto dst = (const rsx::fragment_program::ucode_instr *)mapping.first;
|
|
||||||
|
|
||||||
for (const auto& constant : info.fragment_shader.decompiled->constants)
|
|
||||||
{
|
|
||||||
const void *src = ucode + u32(constant.id / sizeof(*ucode));
|
|
||||||
|
|
||||||
const __m128i &vector = _mm_loadu_si128((const __m128i*)src);
|
|
||||||
const __m128i &shuffled_vector = _mm_shuffle_epi8(vector, mask);
|
|
||||||
_mm_stream_si128((__m128i*)dst, shuffled_vector);
|
|
||||||
|
|
||||||
if (0)
|
|
||||||
{
|
|
||||||
float x = ((float*)dst)[0];
|
|
||||||
float y = ((float*)dst)[1];
|
|
||||||
float z = ((float*)dst)[2];
|
|
||||||
float w = ((float*)dst)[3];
|
|
||||||
|
|
||||||
LOG_WARNING(RSX, "fc%u = {%g, %g, %g, %g}", constant.id, x, y, z, w);
|
|
||||||
}
|
|
||||||
|
|
||||||
++dst;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
m_uniform_ring_buffer.unmap();
|
m_uniform_ring_buffer.unmap();
|
||||||
|
|
||||||
m_uniform_ring_buffer.bind_range(0, scale_offset_offset, sizeof(glsl_matrix_buffer));
|
m_uniform_ring_buffer.bind_range(0, scale_offset_offset, sizeof(glsl_scale_buffer));
|
||||||
m_uniform_ring_buffer.bind_range(1, vertex_constants_offset, sizeof(glsl_vertex_constants_buffer));
|
m_uniform_ring_buffer.bind_range(1, vertex_constants_offset, sizeof(glsl_vertex_constants_buffer));
|
||||||
|
|
||||||
if (fragment_constants_size)
|
if (fragment_constants_size)
|
||||||
{
|
{
|
||||||
m_uniform_ring_buffer.bind_range(2, fragment_constants_offset, fragment_constants_size);
|
m_uniform_ring_buffer.bind_range(2, fragment_constants_offset, fragment_constants_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
m_uniform_ring_buffer.bind_range(3, fragment_state_offset, sizeof(glsl_fragment_state_buffer));
|
m_uniform_ring_buffer.bind_range(3, fragment_state_offset, sizeof(glsl_fragment_state_buffer));
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@ -5,9 +5,6 @@
|
|||||||
#include "gl_texture_cache.h"
|
#include "gl_texture_cache.h"
|
||||||
#include "gl_render_targets.h"
|
#include "gl_render_targets.h"
|
||||||
#include <Utilities/optional.hpp>
|
#include <Utilities/optional.hpp>
|
||||||
|
|
||||||
#define RSX_DEBUG 1
|
|
||||||
|
|
||||||
#include "GLProgramBuffer.h"
|
#include "GLProgramBuffer.h"
|
||||||
|
|
||||||
#pragma comment(lib, "opengl32.lib")
|
#pragma comment(lib, "opengl32.lib")
|
||||||
@ -15,6 +12,8 @@
|
|||||||
class GLGSRender : public GSRender
|
class GLGSRender : public GSRender
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
|
GLFragmentProgram m_fragment_prog;
|
||||||
|
GLVertexProgram m_vertex_prog;
|
||||||
|
|
||||||
rsx::gl::texture m_gl_textures[rsx::limits::fragment_textures_count];
|
rsx::gl::texture m_gl_textures[rsx::limits::fragment_textures_count];
|
||||||
rsx::gl::texture m_gl_vertex_textures[rsx::limits::vertex_textures_count];
|
rsx::gl::texture m_gl_vertex_textures[rsx::limits::vertex_textures_count];
|
||||||
@ -43,6 +42,7 @@ public:
|
|||||||
gl::fbo draw_fbo;
|
gl::fbo draw_fbo;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
GLProgramBuffer m_prog_buffer;
|
||||||
|
|
||||||
//buffer
|
//buffer
|
||||||
gl::fbo m_flip_fbo;
|
gl::fbo m_flip_fbo;
|
||||||
|
@ -4,6 +4,19 @@
|
|||||||
#include "../Common/BufferUtils.h"
|
#include "../Common/BufferUtils.h"
|
||||||
#include "gl_helpers.h"
|
#include "gl_helpers.h"
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
static constexpr std::array<const char*, 16> s_reg_table =
|
||||||
|
{
|
||||||
|
"in_pos_buffer", "in_weight_buffer", "in_normal_buffer",
|
||||||
|
"in_diff_color_buffer", "in_spec_color_buffer",
|
||||||
|
"in_fog_buffer",
|
||||||
|
"in_point_size_buffer", "in_7_buffer",
|
||||||
|
"in_tc0_buffer", "in_tc1_buffer", "in_tc2_buffer", "in_tc3_buffer",
|
||||||
|
"in_tc4_buffer", "in_tc5_buffer", "in_tc6_buffer", "in_tc7_buffer"
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
u32 to_gl_internal_type(rsx::vertex_base_type type, u8 size)
|
u32 to_gl_internal_type(rsx::vertex_base_type type, u8 size)
|
||||||
@ -289,7 +302,7 @@ namespace
|
|||||||
void operator()(const rsx::vertex_array_buffer& vertex_array)
|
void operator()(const rsx::vertex_array_buffer& vertex_array)
|
||||||
{
|
{
|
||||||
int location;
|
int location;
|
||||||
if (!m_program->uniforms.has_location(rsx::vertex_program::input_attrib_names[vertex_array.index] + "_buffer", &location))
|
if (!m_program->uniforms.has_location(s_reg_table[vertex_array.index], &location))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Fill vertex_array
|
// Fill vertex_array
|
||||||
@ -318,7 +331,7 @@ namespace
|
|||||||
void operator()(const rsx::vertex_array_register& vertex_register)
|
void operator()(const rsx::vertex_array_register& vertex_register)
|
||||||
{
|
{
|
||||||
int location;
|
int location;
|
||||||
if (!m_program->uniforms.has_location(rsx::vertex_program::input_attrib_names[vertex_register.index] + "_buffer", &location))
|
if (!m_program->uniforms.has_location(s_reg_table[vertex_register.index], &location))
|
||||||
return;
|
return;
|
||||||
switch (vertex_register.type)
|
switch (vertex_register.type)
|
||||||
{
|
{
|
||||||
@ -349,7 +362,7 @@ namespace
|
|||||||
void operator()(const rsx::empty_vertex_array& vbo)
|
void operator()(const rsx::empty_vertex_array& vbo)
|
||||||
{
|
{
|
||||||
int location;
|
int location;
|
||||||
if (!m_program->uniforms.has_location(rsx::vertex_program::input_attrib_names[vbo.index] + "_buffer", &location))
|
if (!m_program->uniforms.has_location(s_reg_table[vbo.index], &location))
|
||||||
return;
|
return;
|
||||||
glActiveTexture(GL_TEXTURE0 + vbo.index + texture_index_offset);
|
glActiveTexture(GL_TEXTURE0 + vbo.index + texture_index_offset);
|
||||||
glBindTexture(GL_TEXTURE_BUFFER, 0);
|
glBindTexture(GL_TEXTURE_BUFFER, 0);
|
||||||
@ -376,7 +389,7 @@ void GLGSRender::upload_vertex_buffers(u32 min_index, u32 max_index, const u32&
|
|||||||
for (int index = 0; index < rsx::limits::vertex_count; ++index)
|
for (int index = 0; index < rsx::limits::vertex_count; ++index)
|
||||||
{
|
{
|
||||||
int location;
|
int location;
|
||||||
if (!m_program->uniforms.has_location(rsx::vertex_program::input_attrib_names[index] + "_buffer", &location))
|
if (!m_program->uniforms.has_location(s_reg_table[index], &location))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
glActiveTexture(GL_TEXTURE0 + index + texture_index_offset);
|
glActiveTexture(GL_TEXTURE0 + index + texture_index_offset);
|
||||||
@ -413,7 +426,7 @@ u32 GLGSRender::upload_inline_array(const u32 &max_vertex_attrib_size, const u32
|
|||||||
auto &vertex_info = rsx::method_registers.vertex_arrays_info[index];
|
auto &vertex_info = rsx::method_registers.vertex_arrays_info[index];
|
||||||
|
|
||||||
int location;
|
int location;
|
||||||
if (!m_program->uniforms.has_location(rsx::vertex_program::input_attrib_names[index] + "_buffer", &location))
|
if (!m_program->uniforms.has_location(s_reg_table[index], &location))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (!vertex_info.size) // disabled, bind a null sampler
|
if (!vertex_info.size) // disabled, bind a null sampler
|
||||||
|
@ -1101,7 +1101,7 @@ bool VKGSRender::load_program()
|
|||||||
if (fragment_constants_sz)
|
if (fragment_constants_sz)
|
||||||
{
|
{
|
||||||
buf = (u8*)m_uniform_buffer_ring_info.map(fragment_constants_offset, fragment_constants_sz);
|
buf = (u8*)m_uniform_buffer_ring_info.map(fragment_constants_offset, fragment_constants_sz);
|
||||||
m_prog_buffer.fill_fragment_constans_buffer({ reinterpret_cast<float*>(buf), ::narrow<int>(fragment_constants_sz) }, fragment_program);
|
m_prog_buffer.fill_fragment_constants_buffer({ reinterpret_cast<float*>(buf), ::narrow<int>(fragment_constants_sz) }, fragment_program);
|
||||||
m_uniform_buffer_ring_info.unmap();
|
m_uniform_buffer_ring_info.unmap();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user