You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
289 lines
8.2 KiB
289 lines
8.2 KiB
/*****************************************************************************
|
|
* glwin32.c: Windows OpenGL provider
|
|
*****************************************************************************
|
|
* Copyright (C) 2001-2009 VLC authors and VideoLAN
|
|
*
|
|
* Authors: Gildas Bazin <gbazin@videolan.org>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation; either version 2.1 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
|
|
*****************************************************************************/
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include <assert.h>
|
|
|
|
#include <vlc_common.h>
|
|
#include <vlc_plugin.h>
|
|
#include <vlc_vout_display.h>
|
|
#include <vlc_opengl.h>
|
|
|
|
#include <windows.h>
|
|
#include <versionhelpers.h>
|
|
|
|
#define GLEW_STATIC
|
|
#include "../opengl/renderer.h"
|
|
#include "../opengl/vout_helper.h"
|
|
|
|
#include "common.h"
|
|
|
|
/*****************************************************************************
|
|
* Module descriptor
|
|
*****************************************************************************/
|
|
static int Open (vout_display_t *,
|
|
video_format_t *, vlc_video_context *);
|
|
static void Close(vout_display_t *);
|
|
|
|
vlc_module_begin()
|
|
set_subcategory(SUBCAT_VIDEO_VOUT)
|
|
set_shortname("OpenGL")
|
|
set_description(N_("OpenGL video output for Windows"))
|
|
add_shortcut("glwin32", "opengl")
|
|
set_callback_display(Open, 275)
|
|
add_glopts()
|
|
|
|
add_opengl_submodule_renderer()
|
|
vlc_module_end()
|
|
|
|
/*****************************************************************************
|
|
* Local prototypes.
|
|
*****************************************************************************/
|
|
typedef struct vout_display_sys_t
|
|
{
|
|
struct event_thread_t *video_wnd;
|
|
bool place_changed;
|
|
|
|
vlc_gl_t *gl;
|
|
vout_display_opengl_t *vgl;
|
|
vlc_viewpoint_t viewpoint;
|
|
} vout_display_sys_t;
|
|
|
|
static void Prepare(vout_display_t *, picture_t *, const vlc_render_subpicture *, vlc_tick_t);
|
|
static void Display(vout_display_t *, picture_t *);
|
|
|
|
static int SetViewpoint(vout_display_t *vd, const vlc_viewpoint_t *vp)
|
|
{
|
|
vout_display_sys_t *sys = vd->sys;
|
|
int ret = vout_display_opengl_SetViewpoint(sys->vgl, vp);
|
|
if (ret != VLC_SUCCESS)
|
|
return ret;
|
|
sys->viewpoint = *vp;
|
|
return VLC_SUCCESS;
|
|
}
|
|
|
|
static int SetDisplaySize(vout_display_t *vd, unsigned width, unsigned height)
|
|
{
|
|
VLC_UNUSED(width); VLC_UNUSED(height);
|
|
vout_display_sys_t *sys = vd->sys;
|
|
CommonDisplaySizeChanged(sys->video_wnd);
|
|
return VLC_SUCCESS;
|
|
}
|
|
|
|
static int Control(vout_display_t *vd, int query)
|
|
{
|
|
vout_display_sys_t *sys = vd->sys;
|
|
switch (query) {
|
|
case VOUT_DISPLAY_CHANGE_SOURCE_PLACE:
|
|
sys->place_changed = true;
|
|
break;
|
|
case VOUT_DISPLAY_CHANGE_SOURCE_ASPECT:
|
|
case VOUT_DISPLAY_CHANGE_SOURCE_CROP:
|
|
break;
|
|
}
|
|
return VLC_SUCCESS;
|
|
}
|
|
|
|
static const struct vlc_window_operations embedVideoWindow_Ops =
|
|
{
|
|
};
|
|
|
|
static vlc_window_t *EmbedVideoWindow_Create(vout_display_t *vd)
|
|
{
|
|
vout_display_sys_t *sys = vd->sys;
|
|
|
|
vlc_window_t *wnd = vlc_object_create(vd, sizeof(vlc_window_t));
|
|
if (!wnd)
|
|
return NULL;
|
|
|
|
wnd->type = VLC_WINDOW_TYPE_HWND;
|
|
wnd->handle.hwnd = CommonVideoHWND(sys->video_wnd);
|
|
wnd->ops = &embedVideoWindow_Ops;
|
|
return wnd;
|
|
}
|
|
|
|
static int
|
|
UpdateFormat(vout_display_t *vd, const video_format_t *fmt,
|
|
vlc_video_context *vctx)
|
|
{
|
|
vout_display_sys_t *sys = vd->sys;
|
|
|
|
int ret = vlc_gl_MakeCurrent(sys->gl);
|
|
if (ret != VLC_SUCCESS)
|
|
return ret;
|
|
|
|
ret = vout_display_opengl_UpdateFormat(sys->vgl, fmt, vctx);
|
|
|
|
// /* Force to recompute the viewport on next picture */
|
|
sys->place_changed = true;
|
|
|
|
/* Restore viewpoint */
|
|
int vp_ret = vout_display_opengl_SetViewpoint(sys->vgl, &sys->viewpoint);
|
|
/* The viewpoint previously applied is necessarily valid */
|
|
assert(vp_ret == VLC_SUCCESS);
|
|
(void) vp_ret;
|
|
|
|
vlc_gl_ReleaseCurrent(sys->gl);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static const struct vlc_display_operations ops = {
|
|
.close = Close,
|
|
.prepare = Prepare,
|
|
.display = Display,
|
|
.set_display_size = SetDisplaySize,
|
|
.control = Control,
|
|
.set_viewpoint = SetViewpoint,
|
|
.update_format = UpdateFormat,
|
|
};
|
|
|
|
/**
|
|
* It creates an OpenGL vout display.
|
|
*/
|
|
static int Open(vout_display_t *vd,
|
|
video_format_t *fmtp, vlc_video_context *context)
|
|
{
|
|
vout_display_sys_t *sys;
|
|
|
|
/* do not use OpenGL on XP unless forced */
|
|
if(!vd->obj.force && !IsWindowsVistaOrGreater())
|
|
return VLC_EGENERIC;
|
|
|
|
/* Allocate structure */
|
|
vd->sys = sys = calloc(1, sizeof(*sys));
|
|
if (!sys)
|
|
return VLC_ENOMEM;
|
|
|
|
/* */
|
|
sys->place_changed = true;
|
|
if (CommonWindowInit(vd, &sys->video_wnd,
|
|
vd->source->projection_mode != PROJECTION_MODE_RECTANGULAR))
|
|
goto error;
|
|
|
|
vlc_window_SetTitle(vd->cfg->window, VOUT_TITLE " (OpenGL output)");
|
|
|
|
vout_display_cfg_t embed_cfg = *vd->cfg;
|
|
embed_cfg.window = EmbedVideoWindow_Create(vd);
|
|
if (!embed_cfg.window)
|
|
goto error;
|
|
|
|
char *modlist = var_InheritString(embed_cfg.window, "gl");
|
|
sys->gl = vlc_gl_Create(&embed_cfg, VLC_OPENGL, modlist, NULL);
|
|
free(modlist);
|
|
if (!sys->gl)
|
|
{
|
|
vlc_object_delete(embed_cfg.window);
|
|
goto error;
|
|
}
|
|
|
|
vlc_gl_Resize (sys->gl, vd->cfg->display.width, vd->cfg->display.height);
|
|
|
|
const vlc_fourcc_t *subpicture_chromas;
|
|
if (vlc_gl_MakeCurrent (sys->gl))
|
|
goto error;
|
|
sys->vgl = vout_display_opengl_New(fmtp, &subpicture_chromas, sys->gl,
|
|
&vd->cfg->viewpoint, context);
|
|
vlc_gl_ReleaseCurrent (sys->gl);
|
|
if (!sys->vgl)
|
|
goto error;
|
|
|
|
sys->viewpoint = vd->cfg->viewpoint;
|
|
|
|
/* Setup vout_display now that everything is fine */
|
|
vd->info.subpicture_chromas = subpicture_chromas;
|
|
|
|
vd->ops = &ops;
|
|
|
|
return VLC_SUCCESS;
|
|
|
|
error:
|
|
Close(vd);
|
|
return VLC_EGENERIC;
|
|
}
|
|
|
|
/**
|
|
* It destroys an OpenGL vout display.
|
|
*/
|
|
static void Close(vout_display_t *vd)
|
|
{
|
|
vout_display_sys_t *sys = vd->sys;
|
|
vlc_gl_t *gl = sys->gl;
|
|
|
|
if (gl)
|
|
{
|
|
vlc_window_t *surface = gl->surface;
|
|
if (sys->vgl)
|
|
{
|
|
vlc_gl_MakeCurrent (gl);
|
|
vout_display_opengl_Delete(sys->vgl);
|
|
vlc_gl_ReleaseCurrent (gl);
|
|
}
|
|
vlc_gl_Delete(gl);
|
|
vlc_object_delete(surface);
|
|
}
|
|
|
|
CommonWindowClean(sys->video_wnd);
|
|
|
|
free(sys);
|
|
}
|
|
|
|
/* */
|
|
static void Prepare(vout_display_t *vd, picture_t *picture,
|
|
const vlc_render_subpicture *subpicture,
|
|
vlc_tick_t date)
|
|
{
|
|
VLC_UNUSED(date);
|
|
vout_display_sys_t *sys = vd->sys;
|
|
|
|
if (vlc_gl_MakeCurrent (sys->gl) != VLC_SUCCESS)
|
|
return;
|
|
if (sys->place_changed)
|
|
{
|
|
vout_display_place_t place = *vd->place;
|
|
|
|
/* Reverse vertical alignment as the GL tex are Y inverted */
|
|
place.y = vd->cfg->display.height - (place.y + place.height);
|
|
|
|
vlc_gl_Resize (sys->gl, place.width, place.height);
|
|
vout_display_opengl_SetOutputSize(sys->vgl, vd->cfg->display.width, vd->cfg->display.height);
|
|
vout_display_opengl_Viewport(sys->vgl, place.x, place.y, place.width, place.height);
|
|
sys->place_changed = false;
|
|
}
|
|
vout_display_opengl_Prepare (sys->vgl, picture, subpicture);
|
|
vlc_gl_ReleaseCurrent (sys->gl);
|
|
}
|
|
|
|
static void Display(vout_display_t *vd, picture_t *picture)
|
|
{
|
|
vout_display_sys_t *sys = vd->sys;
|
|
VLC_UNUSED(picture);
|
|
|
|
if (vlc_gl_MakeCurrent (sys->gl) == VLC_SUCCESS)
|
|
{
|
|
vout_display_opengl_Display(sys->vgl);
|
|
vlc_gl_ReleaseCurrent (sys->gl);
|
|
vlc_gl_Swap(sys->gl);
|
|
}
|
|
}
|
|
|