/***************************************************************************** * vout_fb.c: Linux framebuffer video output display method ***************************************************************************** * Copyright (C) 1999, 2000 VideoLAN * * Authors: * * 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., 59 Temple Place - Suite 330, Boston, MA 02111, USA. *****************************************************************************/ /***************************************************************************** * Preamble *****************************************************************************/ #include "defs.h" #include <errno.h> /* ENOMEM */ #include <stdlib.h> /* free() */ #include <string.h> /* strerror() */ #include <fcntl.h> /* open() */ #include <unistd.h> /* close() */ #include <linux/fb.h> #include <sys/ioctl.h> #include <sys/mman.h> /* mmap() */ #include "config.h" #include "common.h" #include "threads.h" #include "mtime.h" #include "plugins.h" #include "video.h" #include "video_output.h" #include "intf_msg.h" #include "main.h" /***************************************************************************** * vout_sys_t: video output framebuffer method descriptor ***************************************************************************** * This structure is part of the video output thread descriptor. * It describes the FB specific properties of an output thread. *****************************************************************************/ typedef struct vout_sys_s { /* System informations */ int i_fb_dev; /* framebuffer device handle */ struct fb_var_screeninfo var_info; /* framebuffer mode informations */ /* Video memory */ byte_t * p_video; /* base adress */ size_t i_page_size; /* page size */ struct fb_cmap fb_cmap; /* original colormap */ unsigned short *fb_palette; /* original palette */ } vout_sys_t; /***************************************************************************** * Local prototypes *****************************************************************************/ static int FBOpenDisplay ( vout_thread_t *p_vout ); static void FBCloseDisplay ( vout_thread_t *p_vout ); /***************************************************************************** * vout_SysCreate: allocates FB video thread output method ***************************************************************************** * This function allocates and initializes a FB vout method. *****************************************************************************/ int vout_SysCreate( vout_thread_t *p_vout, char *psz_display, int i_root_window, void *p_data ) { /* Allocate structure */ p_vout->p_sys = malloc( sizeof( vout_sys_t ) ); if( p_vout->p_sys == NULL ) { intf_ErrMsg("error: %s\n", strerror(ENOMEM) ); return( 1 ); } /* Open and initialize device */ if( FBOpenDisplay( p_vout ) ) { intf_ErrMsg("vout error: can't open display\n"); free( p_vout->p_sys ); return( 1 ); } return( 0 ); } /***************************************************************************** * vout_SysInit: initialize framebuffer video thread output method *****************************************************************************/ int vout_SysInit( vout_thread_t *p_vout ) { return( 0 ); } /***************************************************************************** * vout_SysEnd: terminate FB video thread output method *****************************************************************************/ void vout_SysEnd( vout_thread_t *p_vout ) { ; } /***************************************************************************** * vout_SysDestroy: destroy FB video thread output method ***************************************************************************** * Terminate an output method created by vout_CreateOutputMethod *****************************************************************************/ void vout_SysDestroy( vout_thread_t *p_vout ) { FBCloseDisplay( p_vout ); free( p_vout->p_sys ); } /***************************************************************************** * vout_SysManage: handle FB events ***************************************************************************** * This function should be called regularly by video output thread. It manages * console events. It returns a non null value on error. *****************************************************************************/ int vout_SysManage( vout_thread_t *p_vout ) { /* * Size change */ if( p_vout->i_changes & VOUT_SIZE_CHANGE ) { intf_DbgMsg("resizing window\n"); p_vout->i_changes &= ~VOUT_SIZE_CHANGE; /* Destroy XImages to change their size */ vout_SysEnd( p_vout ); /* Recreate XImages. If SysInit failed, the thread can't go on. */ if( vout_SysInit( p_vout ) ) { intf_ErrMsg("error: can't resize display\n"); return( 1 ); } #if 1 /* Tell the video output thread that it will need to rebuild YUV * tables. This is needed since conversion buffer size may have changed */ p_vout->i_changes |= VOUT_YUV_CHANGE; intf_Msg("Video display resized (%dx%d)\n", p_vout->i_width, p_vout->i_height); #endif } return 0; } /***************************************************************************** * vout_SysDisplay: displays previously rendered output ***************************************************************************** * This function send the currently rendered image to FB image, waits until * it is displayed and switch the two rendering buffers, preparing next frame. *****************************************************************************/ void vout_SysDisplay( vout_thread_t *p_vout ) { /* swap the two Y offsets */ p_vout->p_sys->var_info.yoffset = p_vout->i_buffer_index ? p_vout->p_sys->var_info.yres : 0; /* the X offset should be 0, but who knows ... * some other app might have played with the framebuffer */ p_vout->p_sys->var_info.xoffset = 0; //ioctl( p_vout->p_sys->i_fb_dev, FBIOPUT_VSCREENINFO, &p_vout->p_sys->var_info ); ioctl( p_vout->p_sys->i_fb_dev, FBIOPAN_DISPLAY, &p_vout->p_sys->var_info ); } /***************************************************************************** * vout_SetPalette: sets an 8 bpp palette ***************************************************************************** * This function sets the palette given as an argument. It does not return * anything, but could later send information on which colors it was unable * to set. *****************************************************************************/ void vout_SetPalette( p_vout_thread_t p_vout, u16 *red, u16 *green, u16 *blue, u16 *transp ) { struct fb_cmap cmap = { 0, 256, red, green, blue, transp }; ioctl( p_vout->p_sys->i_fb_dev, FBIOPUTCMAP, &cmap ); } /* following functions are local */ /***************************************************************************** * FBOpenDisplay: open and initialize framebuffer device ***************************************************************************** * XXX?? The framebuffer mode is only provided as a fast and efficient way to * display video, providing the card is configured and the mode ok. It is * not portable, and is not supposed to work with many cards. Use at your * own risk ! *****************************************************************************/ static int FBOpenDisplay( vout_thread_t *p_vout ) { char *psz_device; /* framebuffer device path */ struct fb_fix_screeninfo fix_info; /* framebuffer fix information */ /* Open framebuffer device */ psz_device = main_GetPszVariable( VOUT_FB_DEV_VAR, VOUT_FB_DEV_DEFAULT ); p_vout->p_sys->i_fb_dev = open( psz_device, O_RDWR); if( p_vout->p_sys->i_fb_dev == -1 ) { intf_ErrMsg("vout error: can't open %s (%s)\n", psz_device, strerror(errno) ); return( 1 ); } /* Get framebuffer device informations */ if( ioctl( p_vout->p_sys->i_fb_dev, FBIOGET_VSCREENINFO, &p_vout->p_sys->var_info ) ) { intf_ErrMsg( "vout error: can't get framebuffer informations (%s)\n", strerror(errno) ); close( p_vout->p_sys->i_fb_dev ); return( 1 ); } /* Framebuffer must have some basic properties to be usable */ /* XXX?? */ /* Set some attributes */ p_vout->p_sys->var_info.activate = FB_ACTIVATE_NXTOPEN; p_vout->p_sys->var_info.xoffset = 0; p_vout->p_sys->var_info.yoffset = 0; intf_ErrMsg( "vout: ypanstep is %i\n", fix_info.ypanstep ); /* XXX?? ask sam p_vout->p_sys->mode_info.sync = FB_SYNC_VERT_HIGH_ACT; */ if( ioctl( p_vout->p_sys->i_fb_dev, FBIOPUT_VSCREENINFO, &p_vout->p_sys->var_info ) ) { intf_ErrMsg("vout error: can't set framebuffer informations (%s)\n", strerror(errno) ); close( p_vout->p_sys->i_fb_dev ); return( 1 ); } /* Get some informations again, in the definitive configuration */ if( ioctl( p_vout->p_sys->i_fb_dev, FBIOGET_FSCREENINFO, &fix_info ) || ioctl( p_vout->p_sys->i_fb_dev, FBIOGET_VSCREENINFO, &p_vout->p_sys->var_info ) ) { intf_ErrMsg("vout error: can't get framebuffer informations (%s)\n", strerror(errno) ); /* FIXME: restore fb config ?? */ close( p_vout->p_sys->i_fb_dev ); return( 1 ); } /* FIXME: if the image is full-size, it gets cropped on the left * because of the xres / xres_virtual slight difference */ intf_Msg( "%ix%i (virtual %ix%i)\n", p_vout->p_sys->var_info.xres, p_vout->p_sys->var_info.yres, p_vout->p_sys->var_info.xres_virtual, p_vout->p_sys->var_info.yres_virtual ); p_vout->i_width = p_vout->p_sys->var_info.xres_virtual ? p_vout->p_sys->var_info.xres_virtual : p_vout->p_sys->var_info.xres; p_vout->i_height = p_vout->p_sys->var_info.yres; p_vout->i_screen_depth = p_vout->p_sys->var_info.bits_per_pixel; switch( p_vout->i_screen_depth ) { case 8: /* 8 bpp */ p_vout->p_sys->fb_palette = malloc( 8 * 256 * sizeof(unsigned short) ); p_vout->p_sys->fb_cmap.start = 0; p_vout->p_sys->fb_cmap.len = 256; p_vout->p_sys->fb_cmap.red = p_vout->p_sys->fb_palette; p_vout->p_sys->fb_cmap.green = p_vout->p_sys->fb_palette + 256 * sizeof(unsigned short); p_vout->p_sys->fb_cmap.blue = p_vout->p_sys->fb_palette + 2 * 256 * sizeof(unsigned short); p_vout->p_sys->fb_cmap.transp = p_vout->p_sys->fb_palette + 3 * 256 * sizeof(unsigned short); /* saves the colormap */ ioctl( p_vout->p_sys->i_fb_dev, FBIOGETCMAP, &p_vout->p_sys->fb_cmap ); p_vout->i_bytes_per_pixel = 1; p_vout->i_bytes_per_line = p_vout->i_width; break; case 15: /* 15 bpp (16bpp with a missing green bit) */ case 16: /* 16 bpp (65536 colors) */ p_vout->i_bytes_per_pixel = 2; p_vout->i_bytes_per_line = p_vout->i_width * 2; break; case 24: /* 24 bpp (millions of colors) */ p_vout->i_bytes_per_pixel = 3; p_vout->i_bytes_per_line = p_vout->i_width * 3; break; case 32: /* 32 bpp (millions of colors) */ p_vout->i_bytes_per_pixel = 4; p_vout->i_bytes_per_line = p_vout->i_width * 4; break; default: /* unsupported screen depth */ intf_ErrMsg( "vout error: screen depth %d is not supported\n", p_vout->i_screen_depth); return( 1 ); break; } switch( p_vout->i_screen_depth ) { case 15: case 16: case 24: case 32: p_vout->i_red_mask = ( (1 << p_vout->p_sys->var_info.red.length) - 1 ) << p_vout->p_sys->var_info.red.offset; p_vout->i_green_mask = ( (1 << p_vout->p_sys->var_info.green.length) - 1 ) << p_vout->p_sys->var_info.green.offset; p_vout->i_blue_mask = ( (1 << p_vout->p_sys->var_info.blue.length) - 1 ) << p_vout->p_sys->var_info.blue.offset; } p_vout->p_sys->i_page_size = p_vout->i_width * p_vout->i_height * p_vout->i_bytes_per_pixel; /* Map two framebuffers a the very beginning of the fb */ p_vout->p_sys->p_video = mmap(0, p_vout->p_sys->i_page_size * 2, PROT_READ | PROT_WRITE, MAP_SHARED, p_vout->p_sys->i_fb_dev, 0 ); if( (int)p_vout->p_sys->p_video == -1 ) /* XXX?? according to man, it is -1. What about NULL ? */ { intf_ErrMsg("vout error: can't map video memory (%s)\n", strerror(errno) ); /* FIXME: restore fb config ?? */ close( p_vout->p_sys->i_fb_dev ); return( 1 ); } /* Set and initialize buffers */ vout_SetBuffers( p_vout, p_vout->p_sys->p_video, p_vout->p_sys->p_video + p_vout->p_sys->i_page_size ); intf_DbgMsg("framebuffer type=%d, visual=%d, ypanstep=%d, ywrap=%d, accel=%d\n", fix_info.type, fix_info.visual, fix_info.ypanstep, fix_info.ywrapstep, fix_info.accel ); return( 0 ); } /***************************************************************************** * FBCloseDisplay: close and reset framebuffer device ***************************************************************************** * Returns all resources allocated by FBOpenDisplay and restore the original * state of the device. *****************************************************************************/ static void FBCloseDisplay( vout_thread_t *p_vout ) { /* Restore palette */ if( p_vout->i_screen_depth == 8 ); { ioctl( p_vout->p_sys->i_fb_dev, FBIOPUTCMAP, &p_vout->p_sys->fb_cmap ); free( p_vout->p_sys->fb_palette ); } /* Destroy window and close display */ close( p_vout->p_sys->i_fb_dev ); }