/* $Id: Framebuffer.h 81537 2019-10-25 11:46:30Z vboxsync $ */ /** @file * * VBox frontends: VBoxSDL (simple frontend based on SDL): * Declaration of VBoxSDLFB (SDL framebuffer) class */ /* * Copyright (C) 2006-2019 Oracle Corporation * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. */ #ifndef VBOX_INCLUDED_SRC_VBoxSDL_Framebuffer_h #define VBOX_INCLUDED_SRC_VBoxSDL_Framebuffer_h #ifndef RT_WITHOUT_PRAGMA_ONCE # pragma once #endif #include "VBoxSDL.h" #include #include #ifdef VBOX_SECURELABEL #include /* function pointers */ extern "C" { extern DECLSPEC int (SDLCALL *pTTF_Init)(void); extern DECLSPEC TTF_Font* (SDLCALL *pTTF_OpenFont)(const char *file, int ptsize); extern DECLSPEC SDL_Surface* (SDLCALL *pTTF_RenderUTF8_Solid)(TTF_Font *font, const char *text, SDL_Color fg); extern DECLSPEC SDL_Surface* (SDLCALL *pTTF_RenderUTF8_Shaded)(TTF_Font *font, const char *text, SDL_Color fg, SDL_Color bg); extern DECLSPEC SDL_Surface* (SDLCALL *pTTF_RenderUTF8_Blended)(TTF_Font *font, const char *text, SDL_Color fg); extern DECLSPEC void (SDLCALL *pTTF_CloseFont)(TTF_Font *font); extern DECLSPEC void (SDLCALL *pTTF_Quit)(void); } #endif /* VBOX_SECURELABEL && !VBOX_WITH_SDL2 */ class VBoxSDLFBOverlay; class ATL_NO_VTABLE VBoxSDLFB : public ATL::CComObjectRootEx, VBOX_SCRIPTABLE_IMPL(IFramebuffer) { public: VBoxSDLFB(); virtual ~VBoxSDLFB(); HRESULT init(uint32_t uScreenId, bool fFullscreen, bool fResizable, bool fShowSDLConfig, bool fKeepHostRes, uint32_t u32FixedWidth, uint32_t u32FixedHeight, uint32_t u32FixedBPP, bool fUpdateImage); static bool init(bool fShowSDLConfig); static void uninit(); DECLARE_NOT_AGGREGATABLE(VBoxSDLFB) DECLARE_PROTECT_FINAL_CONSTRUCT() BEGIN_COM_MAP(VBoxSDLFB) COM_INTERFACE_ENTRY(IFramebuffer) COM_INTERFACE_ENTRY2(IDispatch,IFramebuffer) COM_INTERFACE_ENTRY_AGGREGATE(IID_IMarshal, m_pUnkMarshaler.m_p) END_COM_MAP() HRESULT FinalConstruct(); void FinalRelease(); STDMETHOD(COMGETTER(Width))(ULONG *width); STDMETHOD(COMGETTER(Height))(ULONG *height); STDMETHOD(COMGETTER(BitsPerPixel))(ULONG *bitsPerPixel); STDMETHOD(COMGETTER(BytesPerLine))(ULONG *bytesPerLine); STDMETHOD(COMGETTER(PixelFormat))(BitmapFormat_T *pixelFormat); STDMETHOD(COMGETTER(HeightReduction))(ULONG *heightReduction); STDMETHOD(COMGETTER(Overlay))(IFramebufferOverlay **aOverlay); STDMETHOD(COMGETTER(WinId))(LONG64 *winId); STDMETHOD(COMGETTER(Capabilities))(ComSafeArrayOut(FramebufferCapabilities_T, aCapabilities)); STDMETHOD(NotifyUpdate)(ULONG x, ULONG y, ULONG w, ULONG h); STDMETHOD(NotifyUpdateImage)(ULONG x, ULONG y, ULONG w, ULONG h, ComSafeArrayIn(BYTE, aImage)); STDMETHOD(NotifyChange)(ULONG aScreenId, ULONG aXOrigin, ULONG aYOrigin, ULONG aWidth, ULONG aHeight); STDMETHOD(VideoModeSupported)(ULONG width, ULONG height, ULONG bpp, BOOL *supported); STDMETHOD(GetVisibleRegion)(BYTE *aRectangles, ULONG aCount, ULONG *aCountCopied); STDMETHOD(SetVisibleRegion)(BYTE *aRectangles, ULONG aCount); STDMETHOD(ProcessVHWACommand)(BYTE *pCommand, LONG enmCmd, BOOL fGuestCmd); STDMETHOD(Notify3DEvent)(ULONG uType, ComSafeArrayIn(BYTE, aData)); // internal public methods bool initialized() { return mfInitialized; } void notifyChange(ULONG aScreenId); void resizeGuest(); void resizeSDL(); void update(int x, int y, int w, int h, bool fGuestRelative); void repaint(); void setFullscreen(bool fFullscreen); void getFullscreenGeometry(uint32_t *width, uint32_t *height); uint32_t getScreenId() { return mScreenId; } uint32_t getGuestXRes() { return mGuestXRes; } uint32_t getGuestYRes() { return mGuestYRes; } int32_t getOriginX() { return mOriginX; } int32_t getOriginY() { return mOriginY; } int32_t getXOffset() { return mCenterXOffset; } int32_t getYOffset() { return mCenterYOffset; } #ifdef VBOX_WITH_SDL2 SDL_Window *getWindow() { return mpWindow; } bool hasWindow(uint32_t id) { return mScreen && SDL_GetWindowID(mpWindow) == id; } int setWindowTitle(const char *pcszTitle); #endif #ifdef VBOX_SECURELABEL int initSecureLabel(uint32_t height, char *font, uint32_t pointsize, uint32_t labeloffs); void setSecureLabelText(const char *text); void setSecureLabelColor(uint32_t colorFG, uint32_t colorBG); void paintSecureLabel(int x, int y, int w, int h, bool fForce); #endif void setWinId(int64_t winId) { mWinId = winId; } void setOrigin(int32_t axOrigin, int32_t ayOrigin) { mOriginX = axOrigin; mOriginY = ayOrigin; } bool getFullscreen() { return mfFullscreen; } private: /** current SDL framebuffer pointer (also includes screen width/height) */ SDL_Surface *mScreen; #ifdef VBOX_WITH_SDL2 /** the SDL window */ SDL_Window *mpWindow; /** the texture */ SDL_Texture *mpTexture; /** renderer */ SDL_Renderer *mpRenderer; /** render info */ SDL_RendererInfo mRenderInfo; #endif /** false if constructor failed */ bool mfInitialized; /** the screen number of this framebuffer */ uint32_t mScreenId; /** use NotifyUpdateImage */ bool mfUpdateImage; /** maximum possible screen width in pixels (~0 = no restriction) */ uint32_t mMaxScreenWidth; /** maximum possible screen height in pixels (~0 = no restriction) */ uint32_t mMaxScreenHeight; /** current guest screen width in pixels */ ULONG mGuestXRes; /** current guest screen height in pixels */ ULONG mGuestYRes; int32_t mOriginX; int32_t mOriginY; /** fixed SDL screen width (~0 = not set) */ uint32_t mFixedSDLWidth; /** fixed SDL screen height (~0 = not set) */ uint32_t mFixedSDLHeight; /** fixed SDL bits per pixel (~0 = not set) */ uint32_t mFixedSDLBPP; /** Y offset in pixels, i.e. guest-nondrawable area at the top */ uint32_t mTopOffset; /** X offset for guest screen centering */ uint32_t mCenterXOffset; /** Y offset for guest screen centering */ uint32_t mCenterYOffset; /** flag whether we're in fullscreen mode */ bool mfFullscreen; /** flag whether we keep the host screen resolution when switching to * fullscreen or not */ bool mfKeepHostRes; /** framebuffer update semaphore */ RTCRITSECT mUpdateLock; /** flag whether the SDL window should be resizable */ bool mfResizable; /** flag whether we print out SDL information */ bool mfShowSDLConfig; /** handle to window where framebuffer context is being drawn*/ int64_t mWinId; #ifdef VBOX_SECURELABEL /** current secure label text */ Utf8Str mSecureLabelText; /** current secure label foreground color (RGB) */ uint32_t mSecureLabelColorFG; /** current secure label background color (RGB) */ uint32_t mSecureLabelColorBG; /** secure label font handle */ TTF_Font *mLabelFont; /** secure label height in pixels */ uint32_t mLabelHeight; /** secure label offset from the top of the secure label */ uint32_t mLabelOffs; #endif SDL_Surface *mSurfVRAM; BYTE *mPtrVRAM; ULONG mBitsPerPixel; ULONG mBytesPerLine; BOOL mfSameSizeRequested; ComPtr mpSourceBitmap; ComPtr mpPendingSourceBitmap; bool mfUpdates; #ifdef RT_OS_WINDOWS ComPtr m_pUnkMarshaler; #endif }; class VBoxSDLFBOverlay : public IFramebufferOverlay { public: VBoxSDLFBOverlay(ULONG x, ULONG y, ULONG width, ULONG height, BOOL visible, VBoxSDLFB *aParent); virtual ~VBoxSDLFBOverlay(); #ifdef RT_OS_WINDOWS STDMETHOD_(ULONG, AddRef)() { return ::InterlockedIncrement(&refcnt); } STDMETHOD_(ULONG, Release)() { long cnt = ::InterlockedDecrement(&refcnt); if (cnt == 0) delete this; return cnt; } #endif VBOX_SCRIPTABLE_DISPATCH_IMPL(IFramebuffer) NS_DECL_ISUPPORTS STDMETHOD(COMGETTER(X))(ULONG *x); STDMETHOD(COMGETTER(Y))(ULONG *y); STDMETHOD(COMGETTER(Width))(ULONG *width); STDMETHOD(COMGETTER(Height))(ULONG *height); STDMETHOD(COMGETTER(Visible))(BOOL *visible); STDMETHOD(COMSETTER(Visible))(BOOL visible); STDMETHOD(COMGETTER(Alpha))(ULONG *alpha); STDMETHOD(COMSETTER(Alpha))(ULONG alpha); STDMETHOD(COMGETTER(BytesPerLine))(ULONG *bytesPerLine); /* These are not used, or return standard values. */ STDMETHOD(COMGETTER(BitsPerPixel))(ULONG *bitsPerPixel); STDMETHOD(COMGETTER(PixelFormat))(ULONG *pixelFormat); STDMETHOD(COMGETTER(UsesGuestVRAM))(BOOL *usesGuestVRAM); STDMETHOD(COMGETTER(HeightReduction))(ULONG *heightReduction); STDMETHOD(COMGETTER(Overlay))(IFramebufferOverlay **aOverlay); STDMETHOD(COMGETTER(WinId))(LONG64 *winId); STDMETHOD(Lock)(); STDMETHOD(Unlock)(); STDMETHOD(Move)(ULONG x, ULONG y); STDMETHOD(NotifyUpdate)(ULONG x, ULONG y, ULONG w, ULONG h); STDMETHOD(RequestResize)(ULONG aScreenId, ULONG pixelFormat, ULONG vram, ULONG bitsPerPixel, ULONG bytesPerLine, ULONG w, ULONG h, BOOL *finished); STDMETHOD(VideoModeSupported)(ULONG width, ULONG height, ULONG bpp, BOOL *supported); // internal public methods HRESULT init(); private: /** Overlay X offset */ ULONG mOverlayX; /** Overlay Y offset */ ULONG mOverlayY; /** Overlay width */ ULONG mOverlayWidth; /** Overlay height */ ULONG mOverlayHeight; /** Whether the overlay is currently active */ BOOL mOverlayVisible; /** The parent IFramebuffer */ VBoxSDLFB *mParent; /** SDL surface containing the actual framebuffer bits */ SDL_Surface *mOverlayBits; /** Additional SDL surface used for combining the framebuffer and the overlay */ SDL_Surface *mBlendedBits; #ifdef RT_OS_WINDOWS long refcnt; #endif }; #endif /* !VBOX_INCLUDED_SRC_VBoxSDL_Framebuffer_h */