myou-engine/src/graphics/framebuffer.nim

454 lines
No EOL
19 KiB
Nim

# The contents of this file are subject to the Common Public Attribution License
# Version 1.0 (the “License”); you may not use this file except in compliance
# with the License. You may obtain a copy of the License at
# https://myou.dev/licenses/LICENSE-CPAL. The License is based on the Mozilla
# Public License Version 1.1 but Sections 14 and 15 have been added to cover use
# of software over a computer network and provide for limited attribution for
# the Original Developer. In addition, Exhibit A has been modified to be
# consistent with Exhibit B.
#
# Software distributed under the License is distributed on an “AS IS” basis,
# WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for
# the specific language governing rights and limitations under the License.
#
# The Original Code is Myou Engine.
#
# the Original Developer is the Initial Developer.
#
# The Initial Developer of the Original Code is the Myou Engine developers.
# All portions of the code written by the Myou Engine developers are Copyright
# (c) 2024. All Rights Reserved.
#
# Alternatively, the contents of this file may be used under the terms of the
# GNU Affero General Public License version 3 (the [AGPL-3] License), in which
# case the provisions of [AGPL-3] License are applicable instead of those above.
#
# If you wish to allow use of your version of this file only under the terms of
# the [AGPL-3] License and not to allow others to use your version of this file
# under the CPAL, indicate your decision by deleting the provisions above and
# replace them with the notice and other provisions required by the [AGPL-3]
# License. If you do not delete the provisions above, a recipient may use your
# version of this file under either the CPAL or the [AGPL-3] License.
import ../types
import std/options
import std/tables
import vmath except Quat, quat
# Forward declarations
proc newFramebuffer*(engine: MyouEngine,
width, height: int,
format: TextureFormat,
depth_type: FramebufferDepthType = DepthNone,
depth_format = Depth_u16,
depth_only = false,
filter: TextureFilter = Linear,
tex_type: TextureType = Tex2D,
depth_filter: TextureFilter = Linear,
depth_tex_type: TextureType = Tex2D,
layer_count = 1,
texture: Texture = nil,
samples = 1,
): Framebuffer
proc set_attachments(self: Framebuffer; layer, mipmap_level: int)
proc set_texture*(self: Framebuffer, texture: Texture)
proc enable*(self: Framebuffer, rect = none((int32,int32,int32,int32)),
layer = -1, mipmap_level = 0, mark_textures = true): Framebuffer {.discardable.}
proc clear*(self: Framebuffer, color = vec4(0,0,0,0), clear_color = true, clear_depth = true, layer = -1) {.inline.}
proc disable*(self: Framebuffer)
proc draw*(dest: Framebuffer, shader_mat: Material, inputs: seq[Texture],
rect = none((int32,int32,int32,int32)), layer = -1, mipmap_level = 0)
proc blit_to*(self: Framebuffer, dest: Framebuffer,
src_rect = none((int32,int32,int32,int32)),
dst_rect = none((int32,int32,int32,int32)))
proc get_framebuffer_status_string*(self: Framebuffer): string
proc generate_mipmap*(self: Framebuffer, custom_shader: Material = nil)
proc destroy*(self: Framebuffer, remove_from_context: bool = true)
proc newMainFramebuffer*(engine: MyouEngine): Framebuffer
proc resize*(self: Framebuffer, width, height: int, samples = 0)
# End forward declarations
import std/strformat
import ../platform/gl
import ../objects/cubemap_probe
import ../postprocessing/effect_shaders
import ./render
import ./texture
import ./ubo
var active_buffer: Framebuffer
var active_rect: (int32,int32,int32,int32)
var active_layer: int
proc newFramebuffer*(engine: MyouEngine,
width, height: int,
format: TextureFormat,
depth_type: FramebufferDepthType = DepthNone,
depth_format = Depth_u16,
depth_only = false,
filter: TextureFilter = Linear,
tex_type: TextureType = Tex2D,
depth_filter: TextureFilter = Linear,
depth_tex_type: TextureType = Tex2D,
layer_count = 1,
texture: Texture = nil,
samples = 1,
): Framebuffer =
assert Tex3D notin [tex_type, depth_tex_type],
"Tex3D framebuffer not supported"
assert TexExternal notin [tex_type, depth_tex_type],
"TexExternal framebuffer not supported"
let self = new Framebuffer
# TODO: detect support for float textures and framebuffers in renderer
if width == 0 or height == 0:
raise newException(ValueError, "Invalid framebuffer size")
self.engine = engine
self.width = width
self.height = height
self.layer_count = layer_count
self.format = format
self.depth_format = depth_format
self.depth_type = depth_type
self.samples = samples
var tex_type = tex_type
# self.filters_should_blend = false
if samples > 1:
assert depth_type != DepthTexture,
"You can't use DepthTexture with a multisample framebuffer"
if not depth_only:
if texture == nil and samples <= 1:
if self.texture != nil:
# TODO: rely on destructor instead
self.texture.destroy()
self.texture = engine.newTexture("fb_tex", width, height,
layer_count, format, tex_type=tex_type, filter=filter)
# TODO: set as loaded only after having rendered something?
self.texture.loaded = true
self.texture.setExtrapolation Clamp
if self.texture.needsMipmap:
self.texture.generateMipmap()
else:
# Use an existing texture
# (to E.G. render to a mipmap of itself without depth buffer)
self.texture = texture
tex_type = texture.tex_type
# TODO: fall back to other texture types when not supported
if self.depth_texture != nil:
self.depth_texture.destroy()
self.depth_texture = nil
if depth_type == DepthTexture:
self.depth_texture = engine.newTexture("fb_depth", width, height,
layer_count, depth_format, tex_type=depth_tex_type,
filter=depth_filter)
self.depth_texture.loaded = true
var fb, rb: GLuint
glGenFramebuffers(1, fb.addr)
self.framebuffer = fb
glBindFramebuffer(GL_FRAMEBUFFER, fb)
if not depth_only and texture == nil and samples > 1:
glGenRenderbuffers(1, self.color_render_buffer.addr)
glBindRenderbuffer(GL_RENDERBUFFER, self.color_render_buffer)
glRenderbufferStorageMultisampleEXT(GL_RENDERBUFFER, samples.GLsizei, format.toInternalFormat, width.GLsizei, height.GLsizei)
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, self.color_render_buffer)
if depth_type == DepthRenderBuffer:
glGenRenderbuffers(1, rb.addr)
self.depth_render_buffer = rb
glBindRenderbuffer(GL_RENDERBUFFER, rb)
if samples > 1:
glRenderbufferStorageMultisampleEXT(GL_RENDERBUFFER, samples.GLsizei, depth_format.toInternalFormat, width.GLsizei, height.GLsizei)
else:
glRenderbufferStorage(GL_RENDERBUFFER, depth_format.toInternalFormat, width.GLsizei, height.GLsizei)
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, GL_RENDERBUFFER, rb)
self.current_mipmap_level = -1 # force attachment with the call below
self.set_attachments(0, 0)
self.is_complete = glCheckFramebufferStatus(GL_FRAMEBUFFER) == GL_FRAMEBUFFER_COMPLETE
doAssert self.is_complete, "Error creating framebuffer: " & self.get_framebuffer_status_string
self.has_mipmap = false
# self.last_viewport = nil
if self.texture != nil:
self.texture.unbind()
if self.depth_texture != nil:
self.depth_texture.unbind()
if rb != 0:
glBindRenderbuffer(GL_RENDERBUFFER, 0)
glBindFramebuffer(GL_FRAMEBUFFER, 0)
active_buffer = nil
engine.all_framebuffers.add(self)
return self
proc set_attachments(self: Framebuffer; layer, mipmap_level: int) =
let attachments = [GL_COLOR_ATTACHMENT0, GL_DEPTH_ATTACHMENT]
for i,tex in [self.texture, self.depth_texture]:
if tex != nil:
if tex.tex_type == Tex2DArray:
assert self.samples == 1, "unsupported"
assert layer >= 0, "Texture array layer must be specified"
glFramebufferTextureLayer(GL_FRAMEBUFFER, attachments[i],
tex.storage.tex.GLuint, mipmap_level.GLint, layer.GLint)
elif tex.tex_type == TexCube:
assert layer >= 0, "Texture cube side must be specified"
assert self.samples == 1, "unsupported"
glFramebufferTexture2D(GL_FRAMEBUFFER, attachments[i],
(GL_TEXTURE_CUBE_MAP_POSITIVE_X.int + layer).GLenum,
tex.storage.tex.GLuint, mipmap_level.GLint)
elif self.current_mipmap_level != mipmap_level:
if self.samples > 1:
glFramebufferTexture2DMultisampleEXT(GL_FRAMEBUFFER, attachments[i],
tex.storage.target, tex.storage.tex.GLuint, mipmap_level.GLint,
self.samples.GLsizei)
else:
glFramebufferTexture2D(GL_FRAMEBUFFER, attachments[i],
tex.storage.target, tex.storage.tex.GLuint, mipmap_level.GLint)
proc set_texture*(self: Framebuffer, texture: Texture) =
if active_buffer == self:
self.disable()
self.texture = texture
self.current_mipmap_level = -1 # this forces attachment
proc enable*(self: Framebuffer, rect = none((int32,int32,int32,int32)),
layer = -1, mipmap_level = 0, mark_textures = true): Framebuffer {.discardable.} =
self.has_mipmap = false
var left, bottom, width, height = 0'i32
if not rect.isSome:
width = self.width.int32
height = self.height.int32
else:
(left, bottom, width, height) = rect.get
width = width shr mipmap_level
height = height shr mipmap_level
if active_buffer == self and active_rect == (left,bottom,width,height) and
active_layer == layer and self.current_mipmap_level == mipmap_level:
return
self.current_width = width
self.current_height = height
if self.texture != nil:
self.texture.unbind()
if self.depth_texture != nil:
self.depth_texture.unbind()
glBindFramebuffer(GL_DRAW_FRAMEBUFFER, self.framebuffer)
when not defined(opengl_es):
if self.use_sRGB:
glEnable(GL_FRAMEBUFFER_SRGB)
self.set_attachments(layer, mipmap_level)
self.current_mipmap_level = mipmap_level
active_layer = layer
glViewport(left, bottom, width, height)
active_rect = (left,bottom,width,height)
if active_buffer != nil and active_buffer != self:
if active_buffer.texture != nil:
active_buffer.texture.is_framebuffer_active = false
if active_buffer.depth_texture != nil:
active_buffer.depth_texture.is_framebuffer_active = false
active_buffer = self
if mark_textures and self.texture != nil:
self.texture.is_framebuffer_active = true
if mark_textures and self.depth_texture != nil:
self.depth_texture.is_framebuffer_active = true
# filters_should_blend = self.filters_should_blend
return self
proc clear*(self: Framebuffer, color = vec4(0,0,0,0), clear_color = true, clear_depth = true, layer = -1) {.inline.} =
self.enable(layer=layer)
var bits: uint32
if clear_color:
glClearColor(color.r, color.g, color.b, color.a)
bits = GL_COLOR_BUFFER_BIT.uint32
if clear_depth:
bits = bits or GL_DEPTH_BUFFER_BIT.uint32
glClear(bits.GLbitfield)
proc disable*(self: Framebuffer) =
glBindFramebuffer(GL_FRAMEBUFFER, 0)
when not defined(opengl_es):
if self.use_sRGB:
glDisable(GL_FRAMEBUFFER_SRGB)
active_buffer = nil
active_layer = -1
if self.texture != nil:
self.texture.is_framebuffer_active = false
if self.depth_texture != nil:
self.depth_texture.is_framebuffer_active = false
proc draw*(dest: Framebuffer, shader_mat: Material, inputs: seq[Texture],
rect = none((int32,int32,int32,int32)), layer = -1, mipmap_level = 0) =
glUseProgram(0)
dest.enable(rect, layer, mipmap_level, mark_textures=false)
var i = 0
for tex in shader_mat.textures.mvalues:
if i >= inputs.len: break
tex = inputs[i]
inc(i)
if shader_mat.ubos.len != 0:
let ubo = shader_mat.ubos[0]
if ubo.name == "EffectShaderInputs":
var sto = ubo.storage(EffectShaderInput)
var i = 0
for tname,tex in shader_mat.textures:
sto[i] = EffectShaderInput(
size: tex.vec3_size,
px_size: vec3(1)/sto[i].size,
px_lod: tex.mipmapHigh.float32,
lod: mipmap_level.float32,
)
inc(i)
ubo.update()
# TODO: use framebuffers as inputs and pass the last camera data used?
# and the last depth buffer
let renderer = dest.engine.renderer
glDepthMask(false)
glDepthFunc(GL_ALWAYS)
let use_frustum_culling = renderer.use_frustum_culling
renderer.use_frustum_culling = false
var cd = RenderCameraData(world2cam: mat4(), cam2world: mat4(), projection_matrix: mat4())
if layer != -1 and inputs.len != 0 and inputs[0].tex_type == TexCube:
cd.world2cam = getCubemapSideMatrix(layer)
cd.cam2world = transpose(cd.world2cam)
renderer.draw_mesh(renderer.bg_mesh, mat4(), cd, -1, shader_mat)
renderer.use_frustum_culling = use_frustum_culling
glDepthFunc(GL_LEQUAL)
glDepthMask(true)
if shader_mat.ubos.len != 0:
let ubo = shader_mat.ubos[0]
if ubo.name == "EffectShaderInputs":
ubo.unbind()
proc blit_to*(self: Framebuffer, dest: Framebuffer,
src_rect = none((int32,int32,int32,int32)),
dst_rect = none((int32,int32,int32,int32))) =
dest.enable()
let (x1, y1, w1, h1) = src_rect.get((0'i32, 0'i32, self.width.int32, self.height.int32))
let (x2, y2, w2, h2) = dst_rect.get((0'i32, 0'i32, dest.width.int32, dest.height.int32))
glBindFramebuffer(GL_READ_FRAMEBUFFER, self.framebuffer);
glBlitFramebuffer(x1, y1, x1+w1, y1+h1, x2, y2, x2+w2, y2+h2,
GL_COLOR_BUFFER_BIT, GL_NEAREST);
glBindFramebuffer(GL_READ_FRAMEBUFFER, 0);
proc get_framebuffer_status_string*(self: Framebuffer): string =
glBindFramebuffer(GL_FRAMEBUFFER, self.framebuffer)
let status = glCheckFramebufferStatus(GL_FRAMEBUFFER)
result = case status:
of GL_FRAMEBUFFER_COMPLETE: "COMPLETE"
of GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT: "INCOMPLETE ATTACHMENT"
of GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS: "INCOMPLETE DIMENSIONS"
of GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT: "INCOMPLETE MISSING ATTACHMENT"
of GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE: "INCOMPLETE MULTISAMPLE"
of GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS: "INCOMPLETE LAYER TARGETS"
else:
&"Unknown status {status.GLint}"
self.disable()
# TODO: move to renderer or to engine
var cube_mipmap_shader: Material
proc generate_mipmap*(self: Framebuffer, custom_shader: Material = nil) =
if self.texture.needsMipmap and not self.has_mipmap:
if active_buffer == self:
self.disable()
var custom_shader = custom_shader
if custom_shader == nil and self.texture.tex_type == TexCube:
# cube mipmaps seem to be broken, use a custom shader for it
if cube_mipmap_shader == nil:
cube_mipmap_shader = self.engine.newEffectShader(@["cube"],
texture_types = @[TexCube],
code = "outColor = vec4(textureLod(cube, coord3D, 0.0).rgb, 1.0);",
add_input_size_ubo = false)
custom_shader = cube_mipmap_shader
if custom_shader == nil:
# use OpenGL built-in method
self.texture.bind_it()
glGenerateMipmap(self.texture.storage.target)
self.has_mipmap = true
return
let filter = self.texture.filter
let mipmap_range = self.texture.mipmap_range
let layer_range = if self.texture.tex_type == TexCube:
0 ..< 6
else:
-1 .. -1
self.texture.setFilter Linear
for m in mipmap_range[0] .. mipmap_range[1] - 1:
self.texture.setMipmapRange(m, m)
for i in layer_range:
self.draw(custom_shader, @[self.texture],
layer = i, mipmap_level = m+1)
self.texture.setFilter filter
self.texture.setMipmapRange(mipmap_range[0], mipmap_range[1])
self.has_mipmap = true
proc destroy*(self: Framebuffer, remove_from_context: bool = true) =
if active_buffer == self:
self.disable()
if self.texture != nil:
self.texture.destroy()
if self.depth_texture != nil:
self.depth_texture.destroy()
if self.color_render_buffer != 0:
glDeleteRenderbuffers(1, self.color_render_buffer.addr)
if self.depth_render_buffer != 0:
glDeleteRenderbuffers(1, self.depth_render_buffer.addr)
glDeleteFramebuffers(1, self.framebuffer.addr)
if remove_from_context:
var index = self.engine.all_framebuffers.find(self)
if index != -1:
self.engine.all_framebuffers.del(index)
proc newMainFramebuffer*(engine: MyouEngine): Framebuffer =
result = new Framebuffer
result.engine = engine
result.framebuffer = 0
result.is_complete = true
result.use_sRGB = not engine.use_glsl_tone_mapping
result.samples = 1
proc resize*(self: Framebuffer, width, height: int, samples = 0) =
if self.framebuffer == 0:
# It's the main framebuffer, just update width and height
self.width = width
self.height = height
return
let samples = if samples != 0: samples else: self.samples
if width == self.width and height == self.height and samples == self.samples:
# No change needed
return
var depth_only = false
var filter = Linear
var tex_type = Tex2D
var depth_filter = Linear
var depth_tex_type = Tex2D
if self.texture != nil:
filter = self.texture.filter
tex_type = self.texture.tex_type
elif self.color_render_buffer == 0:
depth_only = true
if self.depth_texture != nil:
depth_filter = self.depth_texture.filter
depth_tex_type = self.depth_texture.tex_type
let new_FB = newFramebuffer(
self.engine,
width, height, self.format,
depth_type = self.depth_type,
depth_format = self.depth_format,
depth_only = depth_only,
filter = filter,
tex_type = tex_type,
depth_filter = depth_filter,
depth_tex_type = depth_tex_type,
layer_count = self.layer_count,
samples = samples,
)
self[] = move new_FB[]