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

/*****************************************************************************
* 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);
}
}