Hyprland/src/render/OpenGL.hpp

259 lines
10 KiB
C++
Raw Normal View History

2022-04-04 19:44:25 +02:00
#pragma once
#include "../defines.hpp"
#include "../helpers/Monitor.hpp"
#include "../helpers/Color.hpp"
#include "../helpers/Timer.hpp"
2023-07-19 20:09:49 +02:00
#include "../helpers/Region.hpp"
#include "../helpers/Format.hpp"
2022-04-04 19:44:25 +02:00
#include <list>
2022-04-05 20:49:15 +02:00
#include <unordered_map>
#include <map>
2022-04-04 19:44:25 +02:00
2022-07-10 15:41:26 +02:00
#include <cairo/cairo.h>
2022-04-04 19:44:25 +02:00
#include "Shader.hpp"
2022-04-05 14:33:54 +02:00
#include "Texture.hpp"
2022-04-05 20:49:15 +02:00
#include "Framebuffer.hpp"
2023-10-21 15:15:48 +02:00
#include "Transformer.hpp"
#include "Renderbuffer.hpp"
#include <GLES2/gl2ext.h>
2022-04-04 19:44:25 +02:00
2023-07-20 17:47:49 +02:00
#include "../debug/TracyDefines.hpp"
class CHyprRenderer;
2022-04-04 21:45:35 +02:00
inline const float fullVerts[] = {
1, 0, // top right
0, 0, // top left
1, 1, // bottom right
0, 1, // bottom left
};
inline const float fanVertsFull[] = {-1.0f, -1.0f, 1.0f, -1.0f, 1.0f, 1.0f, -1.0f, 1.0f};
2022-04-04 21:45:35 +02:00
enum eDiscardMode {
DISCARD_OPAQUE = 1,
DISCARD_ALPHA = 1 << 1
2023-03-18 00:16:13 +01:00
};
struct SRenderModifData {
2024-01-07 18:35:44 +01:00
enum eRenderModifType {
RMOD_TYPE_SCALE, /* scale by a float */
RMOD_TYPE_SCALECENTER, /* scale by a float from the center */
RMOD_TYPE_TRANSLATE, /* translate by a Vector2D */
RMOD_TYPE_ROTATE, /* rotate by a float in rad from top left */
RMOD_TYPE_ROTATECENTER, /* rotate by a float in rad from center */
};
std::vector<std::pair<eRenderModifType, std::any>> modifs;
void applyToBox(CBox& box);
2024-04-03 15:09:58 +02:00
void applyToRegion(CRegion& rg);
float combinedScale();
bool enabled = true;
};
2022-06-29 12:54:53 +02:00
struct SMonitorRenderData {
CFramebuffer offloadFB;
2023-07-19 20:09:49 +02:00
CFramebuffer mirrorFB; // these are used for some effects,
CFramebuffer mirrorSwapFB; // etc
2023-10-21 15:15:48 +02:00
CFramebuffer offMainFB;
2022-09-13 15:25:42 +02:00
CFramebuffer monitorMirrorFB; // used for mirroring outputs, does not contain artifacts like offloadFB
2022-06-29 12:54:53 +02:00
SP<CTexture> stencilTex = makeShared<CTexture>();
2022-07-31 23:44:04 +02:00
CFramebuffer blurFB;
bool blurFBDirty = true;
bool blurFBShouldRender = false;
// Shaders
bool m_bShadersInitialized = false;
CShader m_shQUAD;
CShader m_shRGBA;
CShader m_shPASSTHRURGBA;
2023-11-04 20:32:50 +01:00
CShader m_shMATTE;
CShader m_shRGBX;
CShader m_shEXT;
CShader m_shBLUR1;
CShader m_shBLUR2;
CShader m_shBLURPREPARE;
CShader m_shBLURFINISH;
CShader m_shSHADOW;
CShader m_shBORDER1;
2023-04-04 15:49:58 +02:00
CShader m_shGLITCH;
//
2022-06-29 12:54:53 +02:00
};
2022-04-04 19:44:25 +02:00
struct SCurrentRenderData {
CMonitor* pMonitor = nullptr;
PHLWORKSPACE pWorkspace = nullptr;
float projection[9];
float savedProjection[9];
std::array<float, 9> monitorProjection;
SMonitorRenderData* pCurrentMonData = nullptr;
CFramebuffer* currentFB = nullptr; // current rendering to
CFramebuffer* mainFB = nullptr; // main to render to
CFramebuffer* outFB = nullptr; // out to render to (if offloaded, etc)
CRegion damage;
CRegion finalDamage; // damage used for funal off -> main
SRenderModifData renderModif;
float mouseZoomFactor = 1.f;
bool mouseZoomUseMouse = true; // true by default
bool useNearestNeighbor = false;
bool forceIntrospection = false; // cleaned in ::end()
bool blockScreenShader = false;
bool simplePass = false;
Vector2D primarySurfaceUVTopLeft = Vector2D(-1, -1);
Vector2D primarySurfaceUVBottomRight = Vector2D(-1, -1);
CBox clipBox = {}; // scaled coordinates
uint32_t discardMode = DISCARD_OPAQUE;
float discardOpacity = 0.f;
2022-04-04 19:44:25 +02:00
};
2022-11-26 18:56:43 +01:00
class CGradientValueData;
2022-04-04 19:44:25 +02:00
class CHyprOpenGLImpl {
public:
2022-04-04 19:44:25 +02:00
CHyprOpenGLImpl();
void begin(CMonitor*, const CRegion& damage, CFramebuffer* fb = nullptr, std::optional<CRegion> finalDamage = {});
void beginSimple(CMonitor*, const CRegion& damage, CRenderbuffer* rb = nullptr, CFramebuffer* fb = nullptr);
void end();
2022-04-04 19:44:25 +02:00
void renderRect(CBox*, const CColor&, int round = 0);
void renderRectWithBlur(CBox*, const CColor&, int round = 0, float blurA = 1.f, bool xray = false);
void renderRectWithDamage(CBox*, const CColor&, CRegion* damage, int round = 0);
void renderTexture(SP<CTexture>, CBox*, float a, int round = 0, bool discardActive = false, bool allowCustomUV = false);
void renderTextureWithDamage(SP<CTexture>, CBox*, CRegion* damage, float a, int round = 0, bool discardActive = false, bool allowCustomUV = false);
void renderTextureWithBlur(SP<CTexture>, CBox*, float a, SP<CWLSurfaceResource> pSurface, int round = 0, bool blockBlurOptimization = false, float blurA = 1.f);
void renderRoundedShadow(CBox*, int round, int range, const CColor& color, float a = 1.0);
void renderBorder(CBox*, const CGradientValueData&, int round, int borderSize, float a = 1.0, int outerRound = -1 /* use round */);
void renderTextureMatte(SP<CTexture> tex, CBox* pBox, CFramebuffer& matte);
void setMonitorTransformEnabled(bool enabled);
void setRenderModifEnabled(bool enabled);
2022-04-04 21:45:35 +02:00
void saveMatrix();
void setMatrixScaleTranslate(const Vector2D& translate, const float& scale);
void restoreMatrix();
void blend(bool enabled);
void makeWindowSnapshot(PHLWINDOW);
void makeRawWindowSnapshot(PHLWINDOW, CFramebuffer*);
void makeLayerSnapshot(PHLLS);
void renderSnapshot(PHLWINDOW);
void renderSnapshot(PHLLS);
bool shouldUseNewBlurOptimizations(PHLLS pLayer, PHLWINDOW pWindow);
2022-04-05 20:49:15 +02:00
void clear(const CColor&);
void clearWithTex();
void scissor(const CBox*, bool transform = true);
void scissor(const pixman_box32*, bool transform = true);
void scissor(const int x, const int y, const int w, const int h, bool transform = true);
2022-04-04 19:44:25 +02:00
void destroyMonitorResources(CMonitor*);
2022-04-19 19:01:23 +02:00
void markBlurDirtyForMonitor(CMonitor*);
void preWindowPass();
bool preBlurQueued();
void preRender(CMonitor*);
2022-08-01 12:23:09 +02:00
void saveBufferForMirror(CBox*);
void renderMirrored();
2022-09-13 15:25:42 +02:00
void applyScreenShader(const std::string& path);
2022-12-01 14:36:07 +01:00
void bindOffMain();
void renderOffToMain(CFramebuffer* off);
void bindBackOnMain();
2023-10-21 15:15:48 +02:00
void setDamage(const CRegion& damage, std::optional<CRegion> finalDamage = {});
uint32_t getPreferredReadFormat(CMonitor* pMonitor);
std::vector<SDRMFormat> getDRMFormats();
EGLImageKHR createEGLImage(const SDMABUFAttrs& attrs);
SCurrentRenderData m_RenderData;
2022-04-04 19:44:25 +02:00
GLint m_iCurrentOutputFb = 0;
2022-04-05 20:49:15 +02:00
bool m_bReloadScreenShader = true; // at launch it can be set
2022-12-01 14:36:07 +01:00
PHLWINDOWREF m_pCurrentWindow; // hack to get the current rendered window
PHLLS m_pCurrentLayer; // hack to get the current rendered layer
2022-05-17 13:16:37 +02:00
std::map<PHLWINDOWREF, CFramebuffer> m_mWindowFramebuffers;
std::map<PHLLSREF, CFramebuffer> m_mLayerFramebuffers;
std::unordered_map<CMonitor*, SMonitorRenderData> m_mMonitorRenderResources;
std::unordered_map<CMonitor*, CFramebuffer> m_mMonitorBGFBs;
2022-04-05 20:49:15 +02:00
struct {
PFNGLEGLIMAGETARGETRENDERBUFFERSTORAGEOESPROC glEGLImageTargetRenderbufferStorageOES = nullptr;
PFNGLEGLIMAGETARGETTEXTURE2DOESPROC glEGLImageTargetTexture2DOES = nullptr;
PFNEGLCREATEIMAGEKHRPROC eglCreateImageKHR = nullptr;
PFNEGLDESTROYIMAGEKHRPROC eglDestroyImageKHR = nullptr;
PFNEGLQUERYDMABUFFORMATSEXTPROC eglQueryDmaBufFormatsEXT = nullptr;
PFNEGLQUERYDMABUFMODIFIERSEXTPROC eglQueryDmaBufModifiersEXT = nullptr;
} m_sProc;
struct {
bool EXT_read_format_bgra = false;
bool EXT_image_dma_buf_import = false;
bool EXT_image_dma_buf_import_modifiers = false;
} m_sExts;
private:
std::list<GLuint> m_lBuffers;
std::list<GLuint> m_lTextures;
std::vector<SDRMFormat> drmFormats;
bool m_bHasModifiers = false;
2022-04-04 19:44:25 +02:00
int m_iDRMFD;
std::string m_szExtensions;
2022-04-04 19:44:25 +02:00
bool m_bFakeFrame = false;
bool m_bEndFrame = false;
bool m_bApplyFinalShader = false;
bool m_bBlend = false;
bool m_bOffloadedFramebuffer = false;
2022-12-01 14:36:07 +01:00
CShader m_sFinalScreenShader;
CTimer m_tGlobalTimer;
void logShaderError(const GLuint&, bool program = false);
GLuint createProgram(const std::string&, const std::string&, bool dynamic = false);
GLuint compileShader(const GLuint&, std::string, bool dynamic = false);
void createBGTextureForMonitor(CMonitor*);
void initShaders();
void initDRMFormats();
std::vector<uint64_t> getModsForFormat(EGLint format);
2022-04-14 16:43:29 +02:00
// returns the out FB, can be either Mirror or MirrorSwap
CFramebuffer* blurMainFramebufferWithDamage(float a, CRegion* damage);
void renderTextureInternalWithDamage(SP<CTexture>, CBox* pBox, float a, CRegion* damage, int round = 0, bool discardOpaque = false, bool noAA = false,
bool allowCustomUV = false, bool allowDim = false);
void renderTexturePrimitive(SP<CTexture> tex, CBox* pBox);
void renderSplash(cairo_t* const, cairo_surface_t* const, double offset, const Vector2D& size);
void preBlurForCurrentMonitor();
bool passRequiresIntrospection(CMonitor* pMonitor);
friend class CHyprRenderer;
2022-04-04 19:44:25 +02:00
};
inline std::unique_ptr<CHyprOpenGLImpl> g_pHyprOpenGL;