217 lines
7.5 KiB
C++
217 lines
7.5 KiB
C++
// Copyright (C) 2020 Jérôme Leclercq
|
|
// This file is part of the "Nazara Engine - OpenGL Renderer"
|
|
// For conditions of distribution and use, see copyright notice in Config.hpp
|
|
|
|
#include <Nazara/OpenGLRenderer/OpenGLDevice.hpp>
|
|
#include <Nazara/Renderer/CommandPool.hpp>
|
|
#include <Nazara/OpenGLRenderer/OpenGLBuffer.hpp>
|
|
#include <Nazara/OpenGLRenderer/OpenGLCommandPool.hpp>
|
|
#include <Nazara/OpenGLRenderer/OpenGLFboFramebuffer.hpp>
|
|
#include <Nazara/OpenGLRenderer/OpenGLRenderPass.hpp>
|
|
#include <Nazara/OpenGLRenderer/OpenGLRenderPipeline.hpp>
|
|
#include <Nazara/OpenGLRenderer/OpenGLRenderPipelineLayout.hpp>
|
|
#include <Nazara/OpenGLRenderer/OpenGLShaderModule.hpp>
|
|
#include <Nazara/OpenGLRenderer/OpenGLTexture.hpp>
|
|
#include <Nazara/OpenGLRenderer/OpenGLTextureSampler.hpp>
|
|
#include <Nazara/OpenGLRenderer/Wrapper/Loader.hpp>
|
|
#include <stdexcept>
|
|
#include <Nazara/OpenGLRenderer/Debug.hpp>
|
|
|
|
namespace Nz
|
|
{
|
|
OpenGLDevice::OpenGLDevice(GL::Loader& loader) :
|
|
m_loader(loader)
|
|
{
|
|
m_referenceContext = loader.CreateContext(this, {});
|
|
if (!m_referenceContext)
|
|
throw std::runtime_error("failed to create reference context");
|
|
|
|
if (!GL::Context::SetCurrentContext(m_referenceContext.get()))
|
|
throw std::runtime_error("failed to activate reference context");
|
|
|
|
const GLubyte* vendorStr = m_referenceContext->glGetString(GL_VENDOR);
|
|
const GLubyte* rendererStr = m_referenceContext->glGetString(GL_RENDERER);
|
|
|
|
m_deviceInfo.name = "OpenGL Device (";
|
|
|
|
if (vendorStr)
|
|
m_deviceInfo.name.append(reinterpret_cast<const char*>(vendorStr));
|
|
|
|
if (rendererStr)
|
|
{
|
|
if (vendorStr)
|
|
m_deviceInfo.name += " - ";
|
|
|
|
m_deviceInfo.name.append(reinterpret_cast<const char*>(rendererStr));
|
|
}
|
|
|
|
m_deviceInfo.name += ')';
|
|
|
|
m_deviceInfo.type = RenderDeviceType::Unknown;
|
|
|
|
GLint minUboOffsetAlignment;
|
|
m_referenceContext->glGetIntegerv(GL_UNIFORM_BUFFER_OFFSET_ALIGNMENT, &minUboOffsetAlignment);
|
|
|
|
assert(minUboOffsetAlignment >= 1);
|
|
m_deviceInfo.limits.minUniformBufferOffsetAlignment = static_cast<UInt64>(minUboOffsetAlignment);
|
|
|
|
m_contexts.insert(m_referenceContext.get());
|
|
}
|
|
|
|
OpenGLDevice::~OpenGLDevice()
|
|
{
|
|
m_referenceContext.reset();
|
|
}
|
|
|
|
std::unique_ptr<GL::Context> OpenGLDevice::CreateContext(const GL::ContextParams& params) const
|
|
{
|
|
auto contextPtr = m_loader.CreateContext(this, params, m_referenceContext.get());
|
|
m_contexts.insert(contextPtr.get());
|
|
|
|
return contextPtr;
|
|
}
|
|
|
|
std::unique_ptr<GL::Context> OpenGLDevice::CreateContext(const GL::ContextParams& params, WindowHandle handle) const
|
|
{
|
|
auto contextPtr = m_loader.CreateContext(this, params, handle, m_referenceContext.get());
|
|
m_contexts.insert(contextPtr.get());
|
|
|
|
return contextPtr;
|
|
}
|
|
|
|
const RenderDeviceInfo& OpenGLDevice::GetDeviceInfo() const
|
|
{
|
|
return m_deviceInfo;
|
|
}
|
|
|
|
std::shared_ptr<AbstractBuffer> OpenGLDevice::InstantiateBuffer(BufferType type)
|
|
{
|
|
return std::make_shared<OpenGLBuffer>(*this, type);
|
|
}
|
|
|
|
std::shared_ptr<CommandPool> OpenGLDevice::InstantiateCommandPool(QueueType /*queueType*/)
|
|
{
|
|
return std::make_shared<OpenGLCommandPool>();
|
|
}
|
|
|
|
std::shared_ptr<Framebuffer> OpenGLDevice::InstantiateFramebuffer(unsigned int /*width*/, unsigned int /*height*/, const std::shared_ptr<RenderPass>& /*renderPass*/, const std::vector<std::shared_ptr<Texture>>& attachments)
|
|
{
|
|
return std::make_shared<OpenGLFboFramebuffer>(*this, attachments);
|
|
}
|
|
|
|
std::shared_ptr<RenderPass> OpenGLDevice::InstantiateRenderPass(std::vector<RenderPass::Attachment> attachments, std::vector<RenderPass::SubpassDescription> subpassDescriptions, std::vector<RenderPass::SubpassDependency> subpassDependencies)
|
|
{
|
|
return std::make_shared<OpenGLRenderPass>(std::move(attachments), std::move(subpassDescriptions), std::move(subpassDependencies));
|
|
}
|
|
|
|
std::shared_ptr<RenderPipeline> OpenGLDevice::InstantiateRenderPipeline(RenderPipelineInfo pipelineInfo)
|
|
{
|
|
return std::make_shared<OpenGLRenderPipeline>(*this, std::move(pipelineInfo));
|
|
}
|
|
|
|
std::shared_ptr<RenderPipelineLayout> OpenGLDevice::InstantiateRenderPipelineLayout(RenderPipelineLayoutInfo pipelineLayoutInfo)
|
|
{
|
|
return std::make_shared<OpenGLRenderPipelineLayout>(std::move(pipelineLayoutInfo));
|
|
}
|
|
|
|
std::shared_ptr<ShaderModule> OpenGLDevice::InstantiateShaderModule(ShaderStageTypeFlags shaderStages, ShaderAst::StatementPtr& shaderAst, const ShaderWriter::States& states)
|
|
{
|
|
return std::make_shared<OpenGLShaderModule>(*this, shaderStages, shaderAst, states);
|
|
}
|
|
|
|
std::shared_ptr<ShaderModule> OpenGLDevice::InstantiateShaderModule(ShaderStageTypeFlags shaderStages, ShaderLanguage lang, const void* source, std::size_t sourceSize, const ShaderWriter::States& states)
|
|
{
|
|
return std::make_shared<OpenGLShaderModule>(*this, shaderStages, lang, source, sourceSize, states);
|
|
}
|
|
|
|
std::shared_ptr<Texture> OpenGLDevice::InstantiateTexture(const TextureInfo& params)
|
|
{
|
|
return std::make_shared<OpenGLTexture>(*this, params);
|
|
}
|
|
|
|
std::shared_ptr<TextureSampler> OpenGLDevice::InstantiateTextureSampler(const TextureSamplerInfo& params)
|
|
{
|
|
return std::make_shared<OpenGLTextureSampler>(*this, params);
|
|
}
|
|
|
|
bool OpenGLDevice::IsTextureFormatSupported(PixelFormat format, TextureUsage usage) const
|
|
{
|
|
switch (format)
|
|
{
|
|
case PixelFormat::Undefined:
|
|
return false;
|
|
|
|
case PixelFormat::A8:
|
|
case PixelFormat::BGR8:
|
|
case PixelFormat::BGR8_SRGB:
|
|
case PixelFormat::BGRA8:
|
|
case PixelFormat::BGRA8_SRGB:
|
|
case PixelFormat::L8:
|
|
case PixelFormat::LA8:
|
|
case PixelFormat::R8:
|
|
case PixelFormat::R8I:
|
|
case PixelFormat::R8UI:
|
|
case PixelFormat::R16:
|
|
case PixelFormat::R16F:
|
|
case PixelFormat::R16I:
|
|
case PixelFormat::R16UI:
|
|
case PixelFormat::R32F:
|
|
case PixelFormat::R32I:
|
|
case PixelFormat::R32UI:
|
|
case PixelFormat::RG8:
|
|
case PixelFormat::RG8I:
|
|
case PixelFormat::RG8UI:
|
|
case PixelFormat::RG16:
|
|
case PixelFormat::RG16F:
|
|
case PixelFormat::RG16I:
|
|
case PixelFormat::RG16UI:
|
|
case PixelFormat::RG32F:
|
|
case PixelFormat::RG32I:
|
|
case PixelFormat::RG32UI:
|
|
case PixelFormat::RGB5A1:
|
|
case PixelFormat::RGB8:
|
|
case PixelFormat::RGB8_SRGB:
|
|
case PixelFormat::RGB16F:
|
|
case PixelFormat::RGB16I:
|
|
case PixelFormat::RGB16UI:
|
|
case PixelFormat::RGB32F:
|
|
case PixelFormat::RGB32I:
|
|
case PixelFormat::RGB32UI:
|
|
case PixelFormat::RGBA4:
|
|
case PixelFormat::RGBA8:
|
|
case PixelFormat::RGBA8_SRGB:
|
|
case PixelFormat::RGBA16F:
|
|
case PixelFormat::RGBA16I:
|
|
case PixelFormat::RGBA16UI:
|
|
case PixelFormat::RGBA32F:
|
|
case PixelFormat::RGBA32I:
|
|
case PixelFormat::RGBA32UI:
|
|
return usage == TextureUsage::ColorAttachment || usage == TextureUsage::InputAttachment || usage == TextureUsage::ShaderSampling || usage == TextureUsage::TransferDestination || usage == TextureUsage::TransferSource;
|
|
|
|
case PixelFormat::DXT1:
|
|
case PixelFormat::DXT3:
|
|
case PixelFormat::DXT5:
|
|
{
|
|
if (!m_referenceContext->IsExtensionSupported(GL::Extension::TextureCompressionS3tc))
|
|
return false;
|
|
|
|
return usage == TextureUsage::InputAttachment || usage == TextureUsage::ShaderSampling || usage == TextureUsage::TransferDestination || usage == TextureUsage::TransferSource;
|
|
}
|
|
|
|
case PixelFormat::Depth16:
|
|
case PixelFormat::Depth16Stencil8:
|
|
case PixelFormat::Depth24:
|
|
case PixelFormat::Depth24Stencil8:
|
|
case PixelFormat::Depth32F:
|
|
case PixelFormat::Depth32FStencil8:
|
|
case PixelFormat::Stencil1:
|
|
case PixelFormat::Stencil4:
|
|
case PixelFormat::Stencil8:
|
|
case PixelFormat::Stencil16:
|
|
return usage == TextureUsage::DepthStencilAttachment || usage == TextureUsage::ShaderSampling || usage == TextureUsage::TransferDestination || usage == TextureUsage::TransferSource;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|