Implement OpenGL 3 renderer for pixelpresenter

This commit is contained in:
Elias Batek 2023-12-21 02:34:19 +01:00
parent 7266c4883a
commit 1d2e57f61a
4 changed files with 660 additions and 1 deletions

View File

@ -22,6 +22,8 @@ Future release, likely May 2024 or later.
Nothing is planned for it at this time.
arsd.pixelpresenter was added.
## 11.0
Released: Planned for May 2023, actually out August 2023.

View File

@ -86,6 +86,8 @@
$(H4 $(ID desktop-game) Games)
See [arsd.simpledisplay] and [arsd.gamehelpers].
Check out [arsd.pixelpresenter] for old-skool games that blit fully-rendered frames to the screen.
$(H4 $(ID desktop-gui) GUIs)
See [arsd.minigui], [arsd.nanovega], and also: https://github.com/drug007/nanogui

View File

@ -4,9 +4,663 @@
Distributed under the Boost Software License, Version 1.0.
+/
/++
# Pixel Presenter
$(B Pixel Presenter) is a high-level display library for one specific scenario:
Blitting fully-rendered frames to the screen.
This is useful for software-rendered applications.
Think of old-skool games, emulators etc.
This library builds upon [arsd.simpledisplay] and [arsd.color].
It wraps a [arsd.simpledisplay.SimpleWindow|SimpleWindow]) and displays the provided frame data.
Each frame is automatically centered on, and optionally scaled to, the carrier window.
This processing is done with hardware acceleration (OpenGL).
Later versions might add a software-mode.
Several $(B scaling) modes are supported.
Most notably `keepAspectRatio` that scales frames to the while preserving the original aspect ratio.
See [Scaling] for details.
$(PITFALL
This module is $(B work in progress).
API is subject to changes until further notice.
)
+/
module arsd.pixelpresenter;
import arsd.color;
import arsd.simpledisplay;
/*
## TODO
- Complete documentation
- Usage example(s)
- Additional renderer implementations:
- a `ScreenPainter`-based renderer
- a legacy OpenGL renderer (maybe)
- Is there something in arsd that serves a similar purpose to `PixelBuffer`?
- Minimum window size
- or something similar
- to ensure `Scaling.integer` doesnt break unexpectedly
- Hybrid scaling mode: integer up, FP down
*/
///
alias Pixel = Color;
///
alias ColorF = arsd.color.ColorF;
///
alias Size = arsd.color.Size;
///
alias Point = arsd.color.Point;
// verify assumption(s)
static assert(Pixel.sizeof == uint.sizeof);
/// casts value `v` to type `T`
auto ref T typeCast(T, S)(auto ref S v) {
return cast(T) v;
}
@safe pure nothrow @nogc {
///
Pixel rgba(ubyte r, ubyte g, ubyte b, ubyte a = 0xFF) {
return Pixel(r, g, b, a);
}
///
Pixel rgb(ubyte r, ubyte g, ubyte b) {
return rgba(r, g, b, 0xFF);
}
}
/++
Pixel data container
+/
struct PixelBuffer {
/// Pixel data
Pixel[] data;
/// Pixel per row
int width;
@safe pure nothrow:
this(Size size) {
this.size = size;
}
// undocumented: really shouldnt be used.
// carries the risks of `length` and `width` getting out of sync accidentally.
deprecated("Use `size` instead.")
void length(int value) {
data.length = value;
}
/++
Changes the size of the buffer
Reallocates the underlying pixel array.
+/
void size(Size value) {
data.length = value.area;
width = value.width;
}
/// ditto
void size(int totalPixels, int width)
in (length % width == 0) {
data.length = totalPixels;
this.width = width;
}
@safe pure nothrow @nogc:
/// Height of the buffer, i.e. the number of lines
int height() inout {
if (data.length == 0)
return 0;
return (cast(int) data.length / width);
}
/// Rectangular size of the buffer
Size size() inout {
return Size(width, height);
}
/// Length of the buffer, i.e. the number of pixels
int length() inout {
return cast(int) data.length;
}
/++
Number of bytes per line
Returns:
width × Pixel.sizeof
+/
int pitch() inout {
return (width * int(Pixel.sizeof));
}
/// Clears the buffers contents (by setting each pixel to the same color)
void clear(Pixel value) {
data[] = value;
}
}
@safe pure nothrow @nogc {
// keep aspect ratio (contain)
int karContainScalingFactorInt(const Size drawing, const Size canvas) {
const int w = canvas.width / drawing.width;
const int h = canvas.height / drawing.height;
return (w < h) ? w : h;
}
// keep aspect ratio (contain; FP variant)
float karContainScalingFactorF(const Size drawing, const Size canvas) {
const w = float(canvas.width) / float(drawing.width);
const h = float(canvas.height) / float(drawing.height);
return (w < h) ? w : h;
}
// keep aspect ratio (cover)
float karCoverScalingFactorF(const Size drawing, const Size canvas) {
const w = float(canvas.width) / float(drawing.width);
const h = float(canvas.height) / float(drawing.height);
return (w > h) ? w : h;
}
Size deltaPerimeter(const Size a, const Size b) {
return Size(
a.width - b.width,
a.height - b.height,
);
}
Point offsetCenter(const Size drawing, const Size canvas) {
auto delta = canvas.deltaPerimeter(drawing);
return (cast(Point) delta) >> 1;
}
}
/++
Scaling/Fit Modes
Each scaling modes has unique behavior for different window-size to frame-size ratios.
Unfortunately, there are no universally applicable naming conventions for these modes.
In fact, different implementations tend to contradict each other.
$(SMALL_TABLE
Mode feature matrix
Mode | Aspect Ratio | Pixel Ratio | Cropping | Border | Comment(s)
`none` | preserved | preserved | yes | 4 |
`stretch` | no | no | no | none |
`contain` | preserved | no | no | 4 | letterboxing/pillarboxing
`integer` | preserved | preserved | no | 2 | works only if `window.size >= frame.size`
`cover` | preserved | no | yes | none |
)
$(SMALL_TABLE
Feature | Definition
Aspect Ratio | Whether the original aspect ratio (width ÷ height) of the input frame is preserved
Pixel Ratio | Whether the orignal pixel ratio (= square) is preserved
Cropping | Whether the outer areas of the input frame might get cut off
Border | The number of padding-areas/borders that can potentially appear around the frame
)
For your convience, aliases matching the [`object-fit`](https://developer.mozilla.org/en-US/docs/Web/CSS/object-fit)
CSS property are provided, too. These are prefixed with `css`.
Currently there is no equivalent for `scale-down` as it does not appear to be particularly useful here.
+/
enum Scaling {
none = 0, ///
stretch, ///
contain, ///
integer, ///
cover, ///
// aliases
center = none, ///
keepAspectRatio = contain, ///
// CSS `object-fit` style aliases
cssNone = none, ///
cssContain = contain, ///
cssFill = stretch, ///
cssCover = cover, ///
}
///
enum ScalingFilter {
nearest, /// nearest neighbor → blocky/pixelish
linear, /// (bi-)linear interpolation → smooth/blurry
}
///
struct PresenterConfig {
Window window; ///
Renderer renderer; ///
///
static struct Renderer {
/++
Internal resolution
+/
Size resolution;
/++
Scaling method
to apply when `window.size` != `resolution`
+/
Scaling scaling = Scaling.keepAspectRatio;
/++
Filter
+/
ScalingFilter filter = ScalingFilter.nearest;
/++
Background color
+/
ColorF background = ColorF(0.0f, 0.0f, 0.0f, 1.0f);
///
void setPixelPerfect() {
scaling = Scaling.integer;
filter = ScalingFilter.nearest;
}
}
///
static struct Window {
string title = "ARSD Pixel Presenter";
Size size;
}
}
// undocumented
struct PresenterObjects {
PixelBuffer framebuffer;
SimpleWindow window;
PresenterConfig config;
}
///
struct WantsOpenGl {
bool wanted; /// Is OpenGL wanted?
ubyte vMaj; /// major version
ubyte vMin; /// minor version
}
///
interface PixelRenderer {
/++
Does this renderer use OpenGL?
Returns:
Whether the renderer requires an OpenGL-enabled window
and which version is expected.
+/
public WantsOpenGl wantsOpenGl() @safe pure nothrow @nogc;
/++
Setup function
Called once during setup.
Perform initialization tasks in here.
Params:
pro = Pointer to the [PresenterObjects] of the presenter. To be stored for later use.
+/
public void setup(PresenterObjects* pro);
/++
Reconfigure renderer
Called upon configuration changes.
The new config can be found in the [PresenterObjects] received during `setup()`.
+/
public void reconfigure();
}
///
final class OpenGL3PixelRenderer : PixelRenderer {
private {
PresenterObjects* _pro;
bool _clear = true;
GLfloat[16] _vertices;
OpenGlShader _shader;
GLuint _vao;
GLuint _vbo;
GLuint _ebo;
GLuint _texture = 0;
}
///
public this() {
}
public WantsOpenGl wantsOpenGl() @safe pure nothrow @nogc {
return WantsOpenGl(true, 3, 0);
}
// TODO: make this ctor?
public void setup(PresenterObjects* pro) {
_pro = pro;
_pro.window.visibleForTheFirstTime = &this.visibleForTheFirstTime;
_pro.window.redrawOpenGlScene = &this.redrawOpenGlScene;
}
private {
void visibleForTheFirstTime() {
_pro.window.setAsCurrentOpenGlContext();
gl3.loadDynamicLibrary();
this.compileLinkShader();
this.setupVertexObjects();
this.reconfigure();
}
void redrawOpenGlScene() {
if (_clear) {
glClearColor(
_pro.config.renderer.background.r,
_pro.config.renderer.background.g,
_pro.config.renderer.background.b,
_pro.config.renderer.background.a
);
glClear(GL_COLOR_BUFFER_BIT);
_clear = false;
}
glActiveTexture(GL_TEXTURE0);
glBindTexture(GL_TEXTURE_2D, _texture);
glTexSubImage2D(
GL_TEXTURE_2D,
0,
0, 0,
_pro.config.renderer.resolution.width, _pro.config.renderer.resolution.height,
GL_RGBA, GL_UNSIGNED_BYTE,
cast(void*) _pro.framebuffer.data.ptr
);
glUseProgram(_shader.shaderProgram);
glBindVertexArray(_vao);
glDrawElements(GL_TRIANGLES, 6, GL_UNSIGNED_INT, null);
}
}
private {
void compileLinkShader() {
_shader = new OpenGlShader(
OpenGlShader.Source(GL_VERTEX_SHADER, `
#version 330 core
layout (location = 0) in vec2 aPos;
layout (location = 1) in vec2 aTexCoord;
out vec2 TexCoord;
void main() {
gl_Position = vec4(aPos.x, aPos.y, 0.0, 1.0);
TexCoord = aTexCoord;
}
`),
OpenGlShader.Source(GL_FRAGMENT_SHADER, `
#version 330 core
out vec4 FragColor;
in vec2 TexCoord;
uniform sampler2D sampler;
void main() {
FragColor = texture(sampler, TexCoord);
}
`),
);
}
void setupVertexObjects() {
glGenVertexArrays(1, &_vao);
glBindVertexArray(_vao);
glGenBuffers(1, &_vbo);
glGenBuffers(1, &_ebo);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, _ebo);
glBufferDataSlice(GL_ELEMENT_ARRAY_BUFFER, indices, GL_STATIC_DRAW);
glBindBuffer(GL_ARRAY_BUFFER, _vbo);
glBufferDataSlice(GL_ARRAY_BUFFER, vertices, GL_STATIC_DRAW);
glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 4 * GLfloat.sizeof, null);
glEnableVertexAttribArray(0);
glVertexAttribPointer(1, 2, GL_FLOAT, GL_FALSE, 4 * GLfloat.sizeof, cast(void*)(2 * GLfloat.sizeof));
glEnableVertexAttribArray(1);
}
void setupTexture() {
if (_texture == 0) {
glGenTextures(1, &_texture);
}
glBindTexture(GL_TEXTURE_2D, _texture);
final switch (_pro.config.renderer.filter) with (ScalingFilter) {
case nearest:
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
break;
case linear:
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
break;
}
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
glTexImage2D(
GL_TEXTURE_2D,
0,
GL_RGBA8,
_pro.config.renderer.resolution.width, _pro.config.renderer.resolution.height,
0,
GL_RGBA, GL_UNSIGNED_BYTE,
null
);
glBindTexture(GL_TEXTURE_2D, 0);
}
}
public void reconfigure() {
Size viewport;
final switch (_pro.config.renderer.scaling) {
case Scaling.none:
viewport = _pro.config.renderer.resolution;
break;
case Scaling.stretch:
viewport = _pro.config.window.size;
break;
case Scaling.contain:
const float scaleF = karContainScalingFactorF(_pro.config.renderer.resolution, _pro.config.window.size);
viewport = Size(
typeCast!int(scaleF * _pro.config.renderer.resolution.width),
typeCast!int(scaleF * _pro.config.renderer.resolution.height),
);
break;
case Scaling.integer:
const int scaleI = karContainScalingFactorInt(_pro.config.renderer.resolution, _pro.config.window.size);
viewport = (_pro.config.renderer.resolution * scaleI);
break;
case Scaling.cover:
const float fillF = karCoverScalingFactorF(_pro.config.renderer.resolution, _pro.config.window.size);
viewport = Size(
typeCast!int(fillF * _pro.config.renderer.resolution.width),
typeCast!int(fillF * _pro.config.renderer.resolution.height),
);
break;
}
const Point viewportPos = offsetCenter(viewport, _pro.config.window.size);
glViewport(viewportPos.x, viewportPos.y, viewport.width, viewport.height);
this.setupTexture();
_clear = true;
}
private {
static immutable GLfloat[] vertices = [
//dfmt off
// positions // texture coordinates
1.0f, 1.0f, 1.0f, 0.0f,
1.0f, -1.0f, 1.0f, 1.0f,
-1.0f, -1.0f, 0.0f, 1.0f,
-1.0f, 1.0f, 0.0f, 0.0f,
//dfmt on
];
static immutable GLuint[] indices = [
//dfmt off
0, 1, 3,
1, 2, 3,
//dfmt on
];
}
}
/++
+/
final class PixelPresenter {
private {
PresenterObjects* _pro;
PixelRenderer _renderer;
}
// ctors
public {
///
this(const PresenterConfig config, bool useOpenGl = true) {
if (useOpenGl) {
this(config, new OpenGL3PixelRenderer());
} else {
assert(false, "Not implemented");
}
}
///
this(const PresenterConfig config, PixelRenderer renderer) {
_renderer = renderer;
// create software framebuffer
auto framebuffer = PixelBuffer(config.renderer.resolution);
// OpenGL?
auto openGlOptions = OpenGlOptions.no;
const openGl = _renderer.wantsOpenGl;
if (openGl.wanted) {
setOpenGLContextVersion(openGl.vMaj, openGl.vMin);
openGLContextCompatible = false;
openGlOptions = OpenGlOptions.yes;
}
// spawn window
auto window = new SimpleWindow(
config.window.size,
config.window.title,
openGlOptions,
Resizability.allowResizing,
);
window.windowResized = &this.windowResized;
// alloc objects
_pro = new PresenterObjects(
framebuffer,
window,
config,
);
_renderer.setup(_pro);
}
}
// public functions
public {
///
int eventLoop(T...)(T eventHandlers) if (T.length == 0 || is(T[0] == delegate)) {
return _pro.window.eventLoop(
16,
delegate() { eventHandlers[0](); _pro.window.redrawOpenGlSceneSoon(); },
);
}
///
PixelBuffer framebuffer() @safe pure nothrow @nogc {
return _pro.framebuffer;
}
///
void reconfigure(const PresenterConfig config) {
assert(false, "Not implemented");
//_framebuffer.size = config.internalResolution;
//_renderer.reconfigure(config);
}
///
bool isFullscreen() {
return _pro.window.fullscreen;
}
/// ditto
void isFullscreen(bool enabled) {
return _pro.window.fullscreen = enabled;
}
/++
Returns the underlying `SimpleWindow`
$(WARNING
This is unsupported; use at your own risk.
Tinkering with the window directly can break all sort of things
that a presenter or renderer could possibly have set up.
)
+/
SimpleWindow tinker() @safe pure nothrow @nogc {
return _pro.window;
}
}
// event handlers
private {
void windowResized(int width, int height) {
_pro.config.window.size = Size(width, height);
_renderer.reconfigure();
}
}
}

View File

@ -18767,6 +18767,7 @@ extern(System) nothrow @nogc {
enum uint GL_RGB = 0x1907;
enum uint GL_BGRA = 0x80e1;
enum uint GL_RGBA = 0x1908;
enum uint GL_RGBA8 = 0x8058;
enum uint GL_TEXTURE_2D = 0x0DE1;
enum uint GL_TEXTURE_MIN_FILTER = 0x2801;
enum uint GL_NEAREST = 0x2600;