mirror of https://github.com/yuzu-mirror/yuzu
Merge pull request #9556 from vonchenplus/draw_texture
video_core: Implement maxwell3d draw texture methodpull/8/head
commit
475370c8f8
@ -0,0 +1,59 @@
|
|||||||
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
|
#include "video_core/host_shaders/blit_color_float_frag.h"
|
||||||
|
#include "video_core/host_shaders/full_screen_triangle_vert.h"
|
||||||
|
#include "video_core/renderer_opengl/blit_image.h"
|
||||||
|
#include "video_core/renderer_opengl/gl_shader_manager.h"
|
||||||
|
#include "video_core/renderer_opengl/gl_shader_util.h"
|
||||||
|
|
||||||
|
namespace OpenGL {
|
||||||
|
|
||||||
|
BlitImageHelper::BlitImageHelper(ProgramManager& program_manager_)
|
||||||
|
: program_manager(program_manager_),
|
||||||
|
full_screen_vert(CreateProgram(HostShaders::FULL_SCREEN_TRIANGLE_VERT, GL_VERTEX_SHADER)),
|
||||||
|
blit_color_to_color_frag(
|
||||||
|
CreateProgram(HostShaders::BLIT_COLOR_FLOAT_FRAG, GL_FRAGMENT_SHADER)) {}
|
||||||
|
|
||||||
|
BlitImageHelper::~BlitImageHelper() = default;
|
||||||
|
|
||||||
|
void BlitImageHelper::BlitColor(GLuint dst_framebuffer, GLuint src_image_view, GLuint src_sampler,
|
||||||
|
const Region2D& dst_region, const Region2D& src_region,
|
||||||
|
const Extent3D& src_size) {
|
||||||
|
glEnable(GL_CULL_FACE);
|
||||||
|
glDisable(GL_COLOR_LOGIC_OP);
|
||||||
|
glDisable(GL_DEPTH_TEST);
|
||||||
|
glDisable(GL_STENCIL_TEST);
|
||||||
|
glDisable(GL_POLYGON_OFFSET_FILL);
|
||||||
|
glDisable(GL_RASTERIZER_DISCARD);
|
||||||
|
glDisable(GL_ALPHA_TEST);
|
||||||
|
glDisablei(GL_BLEND, 0);
|
||||||
|
glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
|
||||||
|
glCullFace(GL_BACK);
|
||||||
|
glFrontFace(GL_CW);
|
||||||
|
glColorMaski(0, GL_TRUE, GL_TRUE, GL_TRUE, GL_TRUE);
|
||||||
|
glDepthRangeIndexed(0, 0.0, 0.0);
|
||||||
|
|
||||||
|
program_manager.BindPresentPrograms(full_screen_vert.handle, blit_color_to_color_frag.handle);
|
||||||
|
glProgramUniform2f(full_screen_vert.handle, 0,
|
||||||
|
static_cast<float>(src_region.end.x - src_region.start.x) /
|
||||||
|
static_cast<float>(src_size.width),
|
||||||
|
static_cast<float>(src_region.end.y - src_region.start.y) /
|
||||||
|
static_cast<float>(src_size.height));
|
||||||
|
glProgramUniform2f(full_screen_vert.handle, 1,
|
||||||
|
static_cast<float>(src_region.start.x) / static_cast<float>(src_size.width),
|
||||||
|
static_cast<float>(src_region.start.y) /
|
||||||
|
static_cast<float>(src_size.height));
|
||||||
|
glViewport(std::min(dst_region.start.x, dst_region.end.x),
|
||||||
|
std::min(dst_region.start.y, dst_region.end.y),
|
||||||
|
std::abs(dst_region.end.x - dst_region.start.x),
|
||||||
|
std::abs(dst_region.end.y - dst_region.start.y));
|
||||||
|
glBindFramebuffer(GL_DRAW_FRAMEBUFFER, dst_framebuffer);
|
||||||
|
glBindSampler(0, src_sampler);
|
||||||
|
glBindTextureUnit(0, src_image_view);
|
||||||
|
glClear(GL_COLOR_BUFFER_BIT);
|
||||||
|
glDrawArrays(GL_TRIANGLES, 0, 3);
|
||||||
|
}
|
||||||
|
} // namespace OpenGL
|
@ -0,0 +1,38 @@
|
|||||||
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <glad/glad.h>
|
||||||
|
|
||||||
|
#include "video_core/engines/fermi_2d.h"
|
||||||
|
#include "video_core/renderer_opengl/gl_resource_manager.h"
|
||||||
|
#include "video_core/texture_cache/types.h"
|
||||||
|
|
||||||
|
namespace OpenGL {
|
||||||
|
|
||||||
|
using VideoCommon::Extent3D;
|
||||||
|
using VideoCommon::Offset2D;
|
||||||
|
using VideoCommon::Region2D;
|
||||||
|
|
||||||
|
class ProgramManager;
|
||||||
|
class Framebuffer;
|
||||||
|
class ImageView;
|
||||||
|
|
||||||
|
class BlitImageHelper {
|
||||||
|
public:
|
||||||
|
explicit BlitImageHelper(ProgramManager& program_manager);
|
||||||
|
~BlitImageHelper();
|
||||||
|
|
||||||
|
void BlitColor(GLuint dst_framebuffer, GLuint src_image_view, GLuint src_sampler,
|
||||||
|
const Region2D& dst_region, const Region2D& src_region,
|
||||||
|
const Extent3D& src_size);
|
||||||
|
|
||||||
|
private:
|
||||||
|
ProgramManager& program_manager;
|
||||||
|
|
||||||
|
OGLProgram full_screen_vert;
|
||||||
|
OGLProgram blit_color_to_color_frag;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace OpenGL
|
@ -1,2 +1,123 @@
|
|||||||
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
|
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include <glad/glad.h>
|
||||||
|
|
||||||
|
#include "video_core/renderer_opengl/gl_shader_manager.h"
|
||||||
|
|
||||||
|
namespace OpenGL {
|
||||||
|
|
||||||
|
static constexpr std::array ASSEMBLY_PROGRAM_ENUMS{
|
||||||
|
GL_VERTEX_PROGRAM_NV, GL_TESS_CONTROL_PROGRAM_NV, GL_TESS_EVALUATION_PROGRAM_NV,
|
||||||
|
GL_GEOMETRY_PROGRAM_NV, GL_FRAGMENT_PROGRAM_NV,
|
||||||
|
};
|
||||||
|
|
||||||
|
ProgramManager::ProgramManager(const Device& device) {
|
||||||
|
glCreateProgramPipelines(1, &pipeline.handle);
|
||||||
|
if (device.UseAssemblyShaders()) {
|
||||||
|
glEnable(GL_COMPUTE_PROGRAM_NV);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProgramManager::BindComputeProgram(GLuint program) {
|
||||||
|
glUseProgram(program);
|
||||||
|
is_compute_bound = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProgramManager::BindComputeAssemblyProgram(GLuint program) {
|
||||||
|
if (current_assembly_compute_program != program) {
|
||||||
|
current_assembly_compute_program = program;
|
||||||
|
glBindProgramARB(GL_COMPUTE_PROGRAM_NV, program);
|
||||||
|
}
|
||||||
|
UnbindPipeline();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProgramManager::BindSourcePrograms(std::span<const OGLProgram, NUM_STAGES> programs) {
|
||||||
|
static constexpr std::array<GLenum, 5> stage_enums{
|
||||||
|
GL_VERTEX_SHADER_BIT, GL_TESS_CONTROL_SHADER_BIT, GL_TESS_EVALUATION_SHADER_BIT,
|
||||||
|
GL_GEOMETRY_SHADER_BIT, GL_FRAGMENT_SHADER_BIT,
|
||||||
|
};
|
||||||
|
for (size_t stage = 0; stage < NUM_STAGES; ++stage) {
|
||||||
|
if (current_programs[stage] != programs[stage].handle) {
|
||||||
|
current_programs[stage] = programs[stage].handle;
|
||||||
|
glUseProgramStages(pipeline.handle, stage_enums[stage], programs[stage].handle);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
BindPipeline();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProgramManager::BindPresentPrograms(GLuint vertex, GLuint fragment) {
|
||||||
|
if (current_programs[0] != vertex) {
|
||||||
|
current_programs[0] = vertex;
|
||||||
|
glUseProgramStages(pipeline.handle, GL_VERTEX_SHADER_BIT, vertex);
|
||||||
|
}
|
||||||
|
if (current_programs[4] != fragment) {
|
||||||
|
current_programs[4] = fragment;
|
||||||
|
glUseProgramStages(pipeline.handle, GL_FRAGMENT_SHADER_BIT, fragment);
|
||||||
|
}
|
||||||
|
glUseProgramStages(
|
||||||
|
pipeline.handle,
|
||||||
|
GL_TESS_CONTROL_SHADER_BIT | GL_TESS_EVALUATION_SHADER_BIT | GL_GEOMETRY_SHADER_BIT, 0);
|
||||||
|
current_programs[1] = 0;
|
||||||
|
current_programs[2] = 0;
|
||||||
|
current_programs[3] = 0;
|
||||||
|
|
||||||
|
if (current_stage_mask != 0) {
|
||||||
|
current_stage_mask = 0;
|
||||||
|
for (const GLenum program_type : ASSEMBLY_PROGRAM_ENUMS) {
|
||||||
|
glDisable(program_type);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
BindPipeline();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProgramManager::BindAssemblyPrograms(std::span<const OGLAssemblyProgram, NUM_STAGES> programs,
|
||||||
|
u32 stage_mask) {
|
||||||
|
const u32 changed_mask = current_stage_mask ^ stage_mask;
|
||||||
|
current_stage_mask = stage_mask;
|
||||||
|
|
||||||
|
if (changed_mask != 0) {
|
||||||
|
for (size_t stage = 0; stage < NUM_STAGES; ++stage) {
|
||||||
|
if (((changed_mask >> stage) & 1) != 0) {
|
||||||
|
if (((stage_mask >> stage) & 1) != 0) {
|
||||||
|
glEnable(ASSEMBLY_PROGRAM_ENUMS[stage]);
|
||||||
|
} else {
|
||||||
|
glDisable(ASSEMBLY_PROGRAM_ENUMS[stage]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (size_t stage = 0; stage < NUM_STAGES; ++stage) {
|
||||||
|
if (current_programs[stage] != programs[stage].handle) {
|
||||||
|
current_programs[stage] = programs[stage].handle;
|
||||||
|
glBindProgramARB(ASSEMBLY_PROGRAM_ENUMS[stage], programs[stage].handle);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
UnbindPipeline();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProgramManager::RestoreGuestCompute() {}
|
||||||
|
|
||||||
|
void ProgramManager::BindPipeline() {
|
||||||
|
if (!is_pipeline_bound) {
|
||||||
|
is_pipeline_bound = true;
|
||||||
|
glBindProgramPipeline(pipeline.handle);
|
||||||
|
}
|
||||||
|
UnbindCompute();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProgramManager::UnbindPipeline() {
|
||||||
|
if (is_pipeline_bound) {
|
||||||
|
is_pipeline_bound = false;
|
||||||
|
glBindProgramPipeline(0);
|
||||||
|
}
|
||||||
|
UnbindCompute();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProgramManager::UnbindCompute() {
|
||||||
|
if (is_compute_bound) {
|
||||||
|
is_compute_bound = false;
|
||||||
|
glUseProgram(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} // namespace OpenGL
|
||||||
|
Loading…
Reference in New Issue