openMSX
GLHQScaler.cc
Go to the documentation of this file.
1#include "GLHQScaler.hh"
2
3#include "File.hh"
4#include "FileContext.hh"
5#include "FrameSource.hh"
6#include "GLUtil.hh"
7#include "HQCommon.hh"
8
9#include "narrow.hh"
10#include "ranges.hh"
11#include "vla.hh"
12
13#include <array>
14#include <cstring>
15#include <utility>
16
17namespace openmsx {
18
20 : GLScaler("hq")
21 , fallback(fallback_)
22{
23 for (const auto& p : program) {
24 p.activate();
25 glUniform1i(p.getUniformLocation("edgeTex"), 2);
26 glUniform1i(p.getUniformLocation("offsetTex"), 3);
27 glUniform1i(p.getUniformLocation("weightTex"), 4);
28 }
29
30 // GL_LUMINANCE_ALPHA is no longer supported in newer openGL versions
31 auto format = (OPENGL_VERSION >= OPENGL_3_3) ? GL_RG : GL_LUMINANCE_ALPHA;
32 edgeTexture.bind();
33 glTexImage2D(GL_TEXTURE_2D, // target
34 0, // level
35 format, // internal format
36 320, // width
37 240, // height
38 0, // border
39 format, // format
40 GL_UNSIGNED_BYTE, // type
41 nullptr); // data
42#if OPENGL_VERSION >= OPENGL_3_3
43 GLint swizzleMask[] = {GL_RED, GL_RED, GL_RED, GL_ONE};
44 glTexParameteriv(GL_TEXTURE_2D, GL_TEXTURE_SWIZZLE_RGBA, swizzleMask);
45#endif
46 edgeBuffer.setImage(320, 240);
47
48 const auto& context = systemFileContext();
49 glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
50 std::string offsetsName = "shaders/HQ_xOffsets.dat";
51 std::string weightsName = "shaders/HQ_xWeights.dat";
52 for (auto i : xrange(3)) {
53 int n = i + 2;
54 offsetsName[10] = narrow<char>('0' + n);
55 File offsetsFile(context.resolve(offsetsName));
56 offsetTexture[i].bind();
57 glTexImage2D(GL_TEXTURE_2D, // target
58 0, // level
59 GL_RGBA, // internal format
60 n * 64, // width
61 n * 64, // height
62 0, // border
63 GL_RGBA, // format
64 GL_UNSIGNED_BYTE, // type
65 offsetsFile.mmap().data());// data
66
67 weightsName[10] = narrow<char>('0' + n);
68 File weightsFile(context.resolve(weightsName));
69 weightTexture[i].bind();
70 glTexImage2D(GL_TEXTURE_2D, // target
71 0, // level
72 GL_RGB, // internal format
73 n * 64, // width
74 n * 64, // height
75 0, // border
76 GL_RGB, // format
77 GL_UNSIGNED_BYTE, // type
78 weightsFile.mmap().data());// data
79 }
80 glPixelStorei(GL_UNPACK_ALIGNMENT, 4); // restore to default
81}
82
84 gl::ColorTexture& src, gl::ColorTexture* superImpose,
85 unsigned srcStartY, unsigned srcEndY, unsigned srcWidth,
86 unsigned dstStartY, unsigned dstEndY, unsigned dstWidth,
87 unsigned logSrcHeight)
88{
89 unsigned factorX = dstWidth / srcWidth; // 1 - 4
90 unsigned factorY = (dstEndY - dstStartY) / (srcEndY - srcStartY);
91
92 if ((srcWidth == 320) && (factorX > 1) && (factorX == factorY)) {
93 assert(src.getHeight() == 2 * 240);
94 setup(superImpose != nullptr);
95 glActiveTexture(GL_TEXTURE4);
96 weightTexture[factorX - 2].bind();
97 glActiveTexture(GL_TEXTURE3);
98 offsetTexture[factorX - 2].bind();
99 glActiveTexture(GL_TEXTURE2);
100 edgeTexture.bind();
101 glActiveTexture(GL_TEXTURE0);
102 execute(src, superImpose,
103 srcStartY, srcEndY, srcWidth,
104 dstStartY, dstEndY, dstWidth,
105 logSrcHeight);
106 } else {
107 fallback.scaleImage(src, superImpose,
108 srcStartY, srcEndY, srcWidth,
109 dstStartY, dstEndY, dstWidth,
110 logSrcHeight);
111 }
112}
113
114using Pixel = uint32_t;
116 unsigned srcStartY, unsigned srcEndY, unsigned lineWidth,
117 FrameSource& paintFrame)
118{
119 if ((lineWidth != 320) || (srcEndY > 240)) return;
120
121 std::array<Endian::L32, 320 / 2> tmpBuf2; // 2 x uint16_t
122 #ifndef NDEBUG
123 // Avoid UMR. In optimized mode we don't care.
124 ranges::fill(tmpBuf2, 0);
125 #endif
126
127 VLA_SSE_ALIGNED(Pixel, buf1, lineWidth);
128 VLA_SSE_ALIGNED(Pixel, buf2, lineWidth);
129 auto curr = paintFrame.getLine(narrow<int>(srcStartY) - 1, buf1);
130 auto next = paintFrame.getLine(narrow<int>(srcStartY) + 0, buf2);
131 EdgeHQ edgeOp(0, 8, 16);
132 calcEdgesGL(curr, next, tmpBuf2, edgeOp);
133
134 edgeBuffer.bind();
135 if (auto* mapped = edgeBuffer.mapWrite()) {
136 for (auto y : xrange(srcStartY, srcEndY)) {
137 curr = next;
138 std::swap(buf1, buf2);
139 next = paintFrame.getLine(narrow<int>(y + 1), buf2);
140 calcEdgesGL(curr, next, tmpBuf2, edgeOp);
141 memcpy(mapped + 320 * size_t(y), tmpBuf2.data(), 320 * sizeof(uint16_t));
142 }
143 edgeBuffer.unmap();
144
145 auto format = (OPENGL_VERSION >= OPENGL_3_3) ? GL_RG : GL_LUMINANCE_ALPHA;
146 edgeTexture.bind();
147 glTexSubImage2D(GL_TEXTURE_2D, // target
148 0, // level
149 0, // offset x
150 narrow<GLint>(srcStartY), // offset y
151 narrow<GLint>(lineWidth), // width
152 narrow<GLint>(srcEndY - srcStartY), // height
153 format, // format
154 GL_UNSIGNED_BYTE, // type
155 edgeBuffer.getOffset(0, srcStartY));// data
156 }
157 edgeBuffer.unbind();
158}
159
160} // namespace openmsx
#define OPENGL_3_3
Definition GLUtil.hh:23
#define OPENGL_VERSION
Definition GLUtil.hh:24
GLsizei getHeight() const
Definition GLUtil.hh:117
void unmap() const
Unmaps the contents of this buffer.
Definition GLUtil.hh:318
void unbind() const
Unbind this buffer.
Definition GLUtil.hh:286
void setImage(GLuint width, GLuint height)
Sets the image for this buffer.
Definition GLUtil.hh:260
T * getOffset(GLuint x, GLuint y)
Gets a pointer relative to the start of this buffer.
Definition GLUtil.hh:294
void bind() const
Bind this PixelBuffer.
Definition GLUtil.hh:278
T * mapWrite()
Maps the contents of this buffer into memory.
Definition GLUtil.hh:307
void bind() const
Makes this texture the active GL texture.
Definition GLUtil.hh:87
std::span< const uint8_t > mmap()
Map file in memory.
Definition File.cc:102
Interface for getting lines from a video frame.
std::span< const Pixel > getLine(int line, std::span< Pixel > buf) const
Gets a pointer to the pixels of the given line number.
void uploadBlock(unsigned srcStartY, unsigned srcEndY, unsigned lineWidth, FrameSource &paintFrame) override
GLHQScaler(GLScaler &fallback)
Definition GLHQScaler.cc:19
void scaleImage(gl::ColorTexture &src, gl::ColorTexture *superImpose, unsigned srcStartY, unsigned srcEndY, unsigned srcWidth, unsigned dstStartY, unsigned dstEndY, unsigned dstWidth, unsigned logSrcHeight) override
Scales the image in the given area, which must consist of lines which are all equally wide.
Definition GLHQScaler.cc:83
Abstract base class for OpenGL scalers.
Definition GLScaler.hh:16
void setup(bool superImpose)
Setup scaler.
Definition GLScaler.cc:40
std::array< gl::ShaderProgram, 2 > program
Definition GLScaler.hh:85
void execute(const gl::ColorTexture &src, const gl::ColorTexture *superImpose, unsigned srcStartY, unsigned srcEndY, unsigned srcWidth, unsigned dstStartY, unsigned dstEndY, unsigned dstWidth, unsigned logSrcHeight, bool textureFromZero=false)
Helper method to draw a rectangle with multiple texture coordinates.
Definition GLScaler.cc:48
virtual void scaleImage(gl::ColorTexture &src, gl::ColorTexture *superImpose, unsigned srcStartY, unsigned srcEndY, unsigned srcWidth, unsigned dstStartY, unsigned dstEndY, unsigned dstWidth, unsigned logSrcHeight)=0
Scales the image in the given area, which must consist of lines which are all equally wide.
EndianT< uint32_t, ConvLittle< BIG > > L32
Definition endian.hh:122
void format(SectorAccessibleDisk &disk, MSXBootSectorType bootType)
Format the given disk (= a single partition).
This file implemented 3 utility functions:
Definition Autofire.cc:11
const FileContext & systemFileContext()
CharacterConverter::Pixel Pixel
void calcEdgesGL(std::span< const uint32_t > curr, std::span< const uint32_t > next, std::span< Endian::L32 > edges2, EdgeOp edgeOp)
Definition HQCommon.hh:63
constexpr void fill(ForwardRange &&range, const T &value)
Definition ranges.hh:315
#define VLA_SSE_ALIGNED(TYPE, NAME, LENGTH)
Definition vla.hh:50
constexpr auto xrange(T e)
Definition xrange.hh:132