⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 voutgl.m

📁 uclinux 下的vlc播放器源代码
💻 M
📖 第 1 页 / 共 2 页
字号:
/***************************************************************************** * voutgl.m: MacOS X OpenGL provider ***************************************************************************** * Copyright (C) 2001-2004 the VideoLAN team * $Id: vout.m 8351 2004-08-02 13:06:38Z hartman $ * * Authors: Colin Delacroix <colin@zoy.org> *          Florian G. Pflug <fgp@phlo.org> *          Jon Lech Johansen <jon-vl@nanocrew.net> *          Derk-Jan Hartman <hartman at videolan dot org> *          Eric Petit <titer@m0k.org> *          Benjamin Pracht <bigben at videolan dot org> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 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 General Public License for more details. * * You should have received a copy of the GNU 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. *****************************************************************************//***************************************************************************** * Preamble *****************************************************************************/#include <errno.h>                                                 /* ENOMEM */#include <stdlib.h>                                                /* free() */#include <string.h>                                            /* strerror() */#include <vlc_keys.h>#include "intf.h"#include "vout.h"#include <OpenGL/OpenGL.h>#include <OpenGL/gl.h>#include <AGL/agl.h>/***************************************************************************** * VLCView interface *****************************************************************************/@interface VLCGLView : NSOpenGLView{    vout_thread_t * p_vout;}- (id) initWithVout: (vout_thread_t *) p_vout;@endstruct vout_sys_t{    NSAutoreleasePool * o_pool;    VLCGLView         * o_glview;    VLCVoutView       * o_vout_view;    vlc_bool_t          b_saved_frame;    NSRect              s_frame;    vlc_bool_t          b_got_frame;    vlc_mutex_t         lock;    /* Mozilla plugin-related variables */    vlc_bool_t          b_embedded;    AGLContext          agl_ctx;    AGLDrawable         agl_drawable;    int                 i_offx, i_offy;    int                 i_width, i_height;    WindowRef           theWindow;    WindowGroupRef      winGroup;};/***************************************************************************** * Local prototypes *****************************************************************************/static int  Init   ( vout_thread_t * p_vout );static void End    ( vout_thread_t * p_vout );static int  Manage ( vout_thread_t * p_vout );static int  Control( vout_thread_t *, int, va_list );static void Swap   ( vout_thread_t * p_vout );static int  Lock   ( vout_thread_t * p_vout );static void Unlock ( vout_thread_t * p_vout );static int  aglInit   ( vout_thread_t * p_vout );static void aglEnd    ( vout_thread_t * p_vout );static int  aglManage ( vout_thread_t * p_vout );static int  aglControl( vout_thread_t *, int, va_list );static void aglSwap   ( vout_thread_t * p_vout );int E_(OpenVideoGL)  ( vlc_object_t * p_this ){    vout_thread_t * p_vout = (vout_thread_t *) p_this;    vlc_value_t value_drawable;    if( !CGDisplayUsesOpenGLAcceleration( kCGDirectMainDisplay ) )    {        msg_Warn( p_vout, "no OpenGL hardware acceleration found. "                          "Video display will be slow" );        return( 1 );    }    msg_Dbg( p_vout, "display is Quartz Extreme accelerated" );    p_vout->p_sys = malloc( sizeof( vout_sys_t ) );    if( p_vout->p_sys == NULL )    {        msg_Err( p_vout, "out of memory" );        return( 1 );    }    memset( p_vout->p_sys, 0, sizeof( vout_sys_t ) );    vlc_mutex_init( p_vout, &p_vout->p_sys->lock );    var_Get( p_vout->p_vlc, "drawable", &value_drawable );    if( value_drawable.i_int != 0 )    {        static const GLint ATTRIBUTES[] = {             AGL_WINDOW,            AGL_RGBA,            AGL_NO_RECOVERY,            AGL_ACCELERATED,            AGL_DOUBLEBUFFER,            AGL_RED_SIZE,   8,            AGL_GREEN_SIZE, 8,            AGL_BLUE_SIZE,  8,            AGL_ALPHA_SIZE, 8,            AGL_DEPTH_SIZE, 24,            AGL_NONE };        AGLPixelFormat pixFormat;        p_vout->p_sys->b_embedded = VLC_TRUE;        pixFormat = aglChoosePixelFormat(NULL, 0, ATTRIBUTES);        if( NULL == pixFormat )        {            msg_Err( p_vout, "no screen renderer available for required attributes." );            return VLC_EGENERIC;        }                p_vout->p_sys->agl_ctx = aglCreateContext(pixFormat, NULL);        aglDestroyPixelFormat(pixFormat);        if( NULL == p_vout->p_sys->agl_ctx )        {            msg_Err( p_vout, "cannot create AGL context." );            return VLC_EGENERIC;        }        else {            // tell opengl not to sync buffer swap with vertical retrace (too inefficient)            GLint param = 0;            aglSetInteger(p_vout->p_sys->agl_ctx, AGL_SWAP_INTERVAL, &param);            aglEnable(p_vout->p_sys->agl_ctx, AGL_SWAP_INTERVAL);        }        p_vout->pf_init             = aglInit;        p_vout->pf_end              = aglEnd;        p_vout->pf_manage           = aglManage;        p_vout->pf_control          = aglControl;        p_vout->pf_swap             = aglSwap;        p_vout->pf_lock             = Lock;        p_vout->pf_unlock           = Unlock;    }    else    {        p_vout->p_sys->b_embedded = VLC_FALSE;        p_vout->p_sys->o_pool = [[NSAutoreleasePool alloc] init];        /* Create the GL view */        p_vout->p_sys->o_glview = [[VLCGLView alloc] initWithVout: p_vout];        [p_vout->p_sys->o_glview autorelease];        /* Spawn the window */        if( !(p_vout->p_sys->o_vout_view = [VLCVoutView getVoutView: p_vout                        subView: p_vout->p_sys->o_glview frame: nil]) )        {            return VLC_EGENERIC;        }        p_vout->pf_init   = Init;        p_vout->pf_end    = End;        p_vout->pf_manage = Manage;        p_vout->pf_control= Control;        p_vout->pf_swap   = Swap;        p_vout->pf_lock   = Lock;        p_vout->pf_unlock = Unlock;    }    p_vout->p_sys->b_got_frame = VLC_FALSE;    return VLC_SUCCESS;}void E_(CloseVideoGL) ( vlc_object_t * p_this ){    vout_thread_t * p_vout = (vout_thread_t *) p_this;    if( p_vout->p_sys->b_embedded )    {        aglDestroyContext(p_vout->p_sys->agl_ctx);    }    else    {        NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];        /* Close the window */        [p_vout->p_sys->o_vout_view closeVout];        [o_pool release];    }    /* Clean up */    vlc_mutex_destroy( &p_vout->p_sys->lock );    free( p_vout->p_sys );}static int Init( vout_thread_t * p_vout ){    [[p_vout->p_sys->o_glview openGLContext] makeCurrentContext];    return VLC_SUCCESS;}static void End( vout_thread_t * p_vout ){    [[p_vout->p_sys->o_glview openGLContext] makeCurrentContext];}static int Manage( vout_thread_t * p_vout ){    if( p_vout->i_changes & VOUT_ASPECT_CHANGE )    {        [p_vout->p_sys->o_glview reshape];        p_vout->i_changes &= ~VOUT_ASPECT_CHANGE;    }    if( p_vout->i_changes & VOUT_CROP_CHANGE )    {        [p_vout->p_sys->o_glview reshape];        p_vout->i_changes &= ~VOUT_CROP_CHANGE;    }    if( p_vout->i_changes & VOUT_FULLSCREEN_CHANGE )    {        NSAutoreleasePool *o_pool = [[NSAutoreleasePool alloc] init];        if( !p_vout->b_fullscreen )        {            /* Save window size and position */            p_vout->p_sys->s_frame.size =                [p_vout->p_sys->o_vout_view frame].size;            p_vout->p_sys->s_frame.origin =                [[p_vout->p_sys->o_vout_view getWindow ]frame].origin;            p_vout->p_sys->b_saved_frame = VLC_TRUE;        }        [p_vout->p_sys->o_vout_view closeVout];        p_vout->b_fullscreen = !p_vout->b_fullscreen;#define o_glview p_vout->p_sys->o_glview        o_glview = [[VLCGLView alloc] initWithVout: p_vout];        [o_glview autorelease];        if( p_vout->p_sys->b_saved_frame )        {            p_vout->p_sys->o_vout_view = [VLCVoutView getVoutView: p_vout                        subView: o_glview                        frame: &p_vout->p_sys->s_frame];        }        else        {            p_vout->p_sys->o_vout_view = [VLCVoutView getVoutView: p_vout                        subView: o_glview frame: nil];        }        [[o_glview openGLContext] makeCurrentContext];#undef o_glview        [o_pool release];        p_vout->i_changes &= ~VOUT_FULLSCREEN_CHANGE;    }    [p_vout->p_sys->o_vout_view manage];    return VLC_SUCCESS;}/***************************************************************************** * Control: control facility for the vout *****************************************************************************/static int Control( vout_thread_t *p_vout, int i_query, va_list args ){    vlc_bool_t b_arg;    switch( i_query )    {        case VOUT_SET_STAY_ON_TOP:            b_arg = va_arg( args, vlc_bool_t );            [p_vout->p_sys->o_vout_view setOnTop: b_arg];            return VLC_SUCCESS;        case VOUT_CLOSE:        case VOUT_REPARENT:        default:            return vout_vaControlDefault( p_vout, i_query, args );    }}static void Swap( vout_thread_t * p_vout ){    p_vout->p_sys->b_got_frame = VLC_TRUE;    [[p_vout->p_sys->o_glview openGLContext] makeCurrentContext];    glFlush();}static int Lock( vout_thread_t * p_vout ){    vlc_mutex_lock( &p_vout->p_sys->lock );#if __INTEL__    CGLLockContext( (CGLContextObj)p_vout->p_sys->agl_ctx ); #endif    return 0;}static void Unlock( vout_thread_t * p_vout ){    vlc_mutex_unlock( &p_vout->p_sys->lock );#if __INTEL__    CGLUnlockContext( (CGLContextObj)p_vout->p_sys->agl_ctx ); #endif}/***************************************************************************** * VLCGLView implementation *****************************************************************************/@implementation VLCGLView- (id) initWithVout: (vout_thread_t *) vout{    p_vout = vout;    NSOpenGLPixelFormatAttribute attribs[] =    {        NSOpenGLPFAAccelerated,        NSOpenGLPFANoRecovery,        NSOpenGLPFAColorSize, 24,        NSOpenGLPFAAlphaSize, 8,        NSOpenGLPFADepthSize, 24,        NSOpenGLPFAWindow,        0    };    NSOpenGLPixelFormat * fmt = [[NSOpenGLPixelFormat alloc]        initWithAttributes: attribs];    if( !fmt )    {        msg_Warn( p_vout, "could not create OpenGL video output" );        return nil;    }    self = [super initWithFrame: NSMakeRect(0,0,10,10) pixelFormat: fmt];    [fmt release];    [[self openGLContext] makeCurrentContext];    [[self openGLContext] update];    /* Swap buffers only during the vertical retrace of the monitor.       http://developer.apple.com/documentation/GraphicsImaging/       Conceptual/OpenGL/chap5/chapter_5_section_44.html */    long params[] = { 1 };    CGLSetParameter( CGLGetCurrentContext(), kCGLCPSwapInterval,                     params );    return self;}- (void) reshape{    int x, y;    vlc_value_t val;    Lock( p_vout );    NSRect bounds = [self bounds];    [[self openGLContext] makeCurrentContext];    var_Get( p_vout, "macosx-stretch", &val );    if( val.b_bool )    {        x = bounds.size.width;        y = bounds.size.height;    }    else if( bounds.size.height * p_vout->fmt_in.i_visible_width *             p_vout->fmt_in.i_sar_num <             bounds.size.width * p_vout->fmt_in.i_visible_height *             p_vout->fmt_in.i_sar_den )    {        x = ( bounds.size.height * p_vout->fmt_in.i_visible_width *              p_vout->fmt_in.i_sar_num ) /            ( p_vout->fmt_in.i_visible_height * p_vout->fmt_in.i_sar_den);        y = bounds.size.height;    }    else    {        x = bounds.size.width;        y = ( bounds.size.width * p_vout->fmt_in.i_visible_height *              p_vout->fmt_in.i_sar_den) /            ( p_vout->fmt_in.i_visible_width * p_vout->fmt_in.i_sar_num  );    }    glViewport( ( bounds.size.width - x ) / 2,                ( bounds.size.height - y ) / 2, x, y );    if( p_vout->p_sys->b_got_frame )    {        /* Ask the opengl module to redraw */        vout_thread_t * p_parent;        p_parent = (vout_thread_t *) p_vout->p_parent;        Unlock( p_vout );        if( p_parent && p_parent->pf_display )        {            p_parent->pf_display( p_parent, NULL );        }    }    else    {        glClear( GL_COLOR_BUFFER_BIT );        Unlock( p_vout );    }    [super reshape];}- (void) update{    Lock( p_vout );    [super update];    Unlock( p_vout );}- (void) drawRect: (NSRect) rect{    Lock( p_vout );

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -