?? mb93493video.c
字號(hào):
/*** $Id: mb93493video.c,v 1.6 2005/04/22 03:33:36 weiym Exp $** ** Copyright (C) 2004 Feynman Software.*//*** 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-1307 USA*/#include <stdio.h>#include <stdlib.h>#include <string.h>#include <unistd.h>#include <sys/types.h>#include <sys/file.h>#include <sys/stat.h>#include <fcntl.h>#include <signal.h>#include <sched.h>#include "common.h"#include "minigui.h"#include "newgal.h"#include "sysvideo.h"#include "pixels_c.h"#include "vdc_init.h"#include "mb93493video.h"#define MB93493VID_DRIVER_NAME "mb93493"#define VRAM_W 720 /* width */#define VRAM_H 576 /* high */#define VRAM_PIXEL_LEN 2 /* color mode = 16bit/pix */#define LCD_BACKGROUND_COLOR 0x80 /* LCD background color */#define SHOWBUF_LEN 300*1024/* show buffur length */#define VRAM_LEN (VRAM_W*VRAM_H*VRAM_PIXEL_LEN)#define VRAM_PAGELEN (VRAM_LEN/2)#define RGB_RMASK 0x7c00#define RGB_GMASK 0x03e0#define RGB_BMASK 0x001f#define RGB_BPP 16#define RGB_PIXEL_LEN 2unsigned char *VRAM_addr; /* point to display memory of VDC */static unsigned char *y_table = NULL;static unsigned char *u_table = NULL;static unsigned char *v_table = NULL;typedef struct _YUV_Package{ unsigned char y0; unsigned char u; unsigned char y1; unsigned char v;} YUV_Package;static inline int _rgb2yuv_y(int r, int g, int b){ return (unsigned char)(( 299*r + 587*g + 114*b)/1000);}// [0, 0x100) => [-436, 436] => [0, 873) => [0, 0x100)static inline int _rgb2yuv_u(int r, int g, int b){ return (unsigned char)((-147*r - 289*g + 436*b + 436*0x100)/873);}// [0, 0x100) => [-615, 615] => [0, 1231) => [0, 0x100)static inline int _rgb2yuv_v(int r, int g, int b){ return (unsigned char)(( 615*r - 515*g - 100*b + 615*0x100)/1231);}static void init_yuv_table(void){ unsigned int i; int r, g, b; if(y_table != NULL) return; y_table = calloc(1, 2<<15); if(y_table == NULL) goto error_free; u_table = calloc(1, 2<<15); if(u_table == NULL) goto error_free; v_table = calloc(1, 2<<15); if(v_table == NULL) goto error_free; for(i=0; i<(2<<15); i++) { r = (i & RGB_RMASK) >> 7; g = (i & RGB_GMASK) >> 2; b = (i & RGB_BMASK) << 3; y_table[i] = _rgb2yuv_y(r, g, b); u_table[i] = _rgb2yuv_u(r, g, b); v_table[i] = _rgb2yuv_v(r, g, b); } return;error_free: if(y_table) free(y_table); y_table = NULL; if(u_table) free(u_table); u_table = NULL; if(v_table) free(v_table); v_table = NULL; return;}static inline int rgb2yuv_y(Uint16 rgb){ return y_table[rgb];}static inline int rgb2yuv_u(Uint16 rgb){ return u_table[rgb];}static inline int rgb2yuv_v(Uint16 rgb){ return v_table[rgb];}static BOOL is_suspend;#if 0static sigset_t oldmask;#define BLOCK_ALARM \do { \ sigset_t newmask; \ sigemptyset (&newmask); \ sigaddset (&newmask, SIGALRM); \ sigprocmask (SIG_BLOCK, &newmask, &oldmask); \} while (0);#define UNBLOCK_ALARM \ sigprocmask (SIG_SETMASK, &oldmask, NULL);#else#define BLOCK_ALARM#define UNBLOCK_ALARM#endif/* Initialization/Query functions */static int MB93493_VideoInit(_THIS, GAL_PixelFormat *vformat);static GAL_Rect **MB93493_ListModes(_THIS, GAL_PixelFormat *format, Uint32 flags);static GAL_Surface *MB93493_SetVideoMode(_THIS, GAL_Surface *current, int width, int height, int bpp, Uint32 flags);static int MB93493_SetColors(_THIS, int firstcolor, int ncolors, GAL_Color *colors);static void MB93493_UpdateRects (_THIS, int numrects, GAL_Rect *rects);static void MB93493_VideoQuit(_THIS);/* Hardware surface functions */static int MB93493_AllocHWSurface(_THIS, GAL_Surface *surface);static void MB93493_FreeHWSurface(_THIS, GAL_Surface *surface);static void* task_do_update (void* data);/* MB93493 driver bootstrap functions */static int MB93493_Available(void){ return (1);}static void MB93493_DeleteDevice(GAL_VideoDevice *device){ pthread_mutex_destroy (&device->hidden->lock); free(device->hidden); free(device);}static GAL_VideoDevice *MB93493_CreateDevice(int devindex){ GAL_VideoDevice *device; if(y_table == NULL) init_yuv_table(); if(y_table == NULL) { GAL_OutOfMemory(); return 0; } /* Initialize all variables that we clean on shutdown */ device = (GAL_VideoDevice *)calloc(1, sizeof(GAL_VideoDevice)); if (device == NULL) { GAL_OutOfMemory(); return 0; } device->hidden = (struct GAL_PrivateVideoData *)calloc(1, (sizeof *device->hidden)); if(device->hidden == NULL) { GAL_OutOfMemory(); free(device); return 0; } pthread_mutex_init (&device->hidden->lock, NULL); /* Set the function pointers */ device->VideoInit = MB93493_VideoInit; device->ListModes = MB93493_ListModes; device->SetVideoMode = MB93493_SetVideoMode; device->CreateYUVOverlay = NULL; device->SetColors = MB93493_SetColors; device->VideoQuit = MB93493_VideoQuit;#ifdef _LITE_VERSION device->RequestHWSurface = NULL;#endif device->AllocHWSurface = MB93493_AllocHWSurface; device->CheckHWBlit = NULL; device->FillHWRect = NULL; device->SetHWColorKey = NULL; device->SetHWAlpha = NULL; device->FreeHWSurface = MB93493_FreeHWSurface; device->UpdateRects = MB93493_UpdateRects; device->free = MB93493_DeleteDevice; return device;}VideoBootStrap MB93493_bootstrap = { MB93493VID_DRIVER_NAME, "MB93493 YUV framebuffer driver", MB93493_Available, MB93493_CreateDevice};static int MB93493_VideoInit(_THIS, GAL_PixelFormat *vformat){ fprintf(stderr, "WARNING: You are using the mb93493 video driver!\n"); VRAM_addr = NULL; if (open_vdc_device(MODE_PAL) != 0) { fprintf(stderr, "MB93493 NEWGAL Engine: can not open vdc device!\n"); this->hidden->buffer = NULL; this->hidden->is_opened = FALSE; return -1; } this->hidden->is_opened = TRUE; is_suspend = FALSE; this->hidden->dirty = FALSE; SetRect (&this->hidden->update, 0, 0, 0, 0); /* Determine the screen depth (use default 8-bit depth) */ /* we change this during the GAL_SetVideoMode implementation... */ vformat->BitsPerPixel = RGB_BPP; vformat->BytesPerPixel = RGB_PIXEL_LEN; /* We're done! */ return(0);}static GAL_Rect standard_mode = {0, 0, VRAM_W, VRAM_H};static GAL_Rect mini_mode = {0, 0, VRAM_W/2, VRAM_H/2};static GAL_Rect* modes [] = { &standard_mode, &mini_mode, NULL};static GAL_Rect **MB93493_ListModes(_THIS, GAL_PixelFormat *format, Uint32 flags){ if (format->BitsPerPixel == 16) return modes; return NULL;}static GAL_Surface *MB93493_SetVideoMode(_THIS, GAL_Surface *current, int width, int height, int bpp, Uint32 flags){ int ret; if ( this->hidden->buffer ) free( this->hidden->buffer ); this->hidden->buffer = NULL; this->hidden->buffer = calloc (1, width * height * (bpp / 8)); if ( this->hidden->buffer == NULL ) goto error_free; /* Allocate the new pixel format for the screen */ if ( ! GAL_ReallocFormat (current, bpp, RGB_RMASK, RGB_GMASK, RGB_BMASK, 0) ) goto error_free; /* Set up the new mode framebuffer */ current->flags = flags & GAL_FULLSCREEN; this->hidden->w = current->w = width; this->hidden->h = current->h = height; current->pitch = current->w * (bpp / 8); current->pixels = this->hidden->buffer; if(width == VRAM_W/2) this->hidden->mode = 1; ret = pthread_create (&this->hidden->th, NULL, task_do_update, this); if(ret!=0) goto error_free; /* We're done */ return (current);error_free: fprintf (stderr, "MB93493 NEWGAL Engine: Couldn't init.\n"); if(this->hidden->buffer) free(this->hidden->buffer); this->hidden->buffer = 0; return NULL;}/* We don't actually allow hardware surfaces other than the main one */static int MB93493_AllocHWSurface(_THIS, GAL_Surface *surface){ return (-1);}static void MB93493_FreeHWSurface(_THIS, GAL_Surface *surface){ surface->pixels = NULL;}static int MB93493_SetColors(_THIS, int firstcolor, int ncolors, GAL_Color *colors){ return (1);}static inline void round_rect_to_even (RECT* rc){ if (rc->left % 2) rc->left --; if (rc->right % 2) rc->right ++;}static inline void update_rgb_to_yuv_packet(Uint16* rgbbuf, YUV_Package* yuv){ yuv->y0 = (unsigned char) rgb2yuv_y(*rgbbuf); yuv->y1 = (unsigned char) rgb2yuv_y(*(rgbbuf+1)); yuv->u = (unsigned char) ((rgb2yuv_u(*rgbbuf) + rgb2yuv_u(*(rgbbuf+1)))>>1); yuv->v = (unsigned char) ((rgb2yuv_v(*rgbbuf) + rgb2yuv_v(*(rgbbuf+1)))>>1);}static inline void update_rgb_to_yuv_packet_minimode(Uint16* rgbbuf, YUV_Package* yuv){ yuv->y0 = (unsigned char) rgb2yuv_y(*rgbbuf); yuv->u = (unsigned char) rgb2yuv_u(*rgbbuf); yuv->v = (unsigned char) rgb2yuv_v(*rgbbuf); yuv->y1 = yuv->y0;}static inline void update_rect_to_yuv_buffer_page(_THIS, RECT* rect, int idx, int mode_alter){ YUV_Package *yuv; unsigned char* yuvrow; unsigned char* rgbrow; int row, col; int rgb_row_len, yuv_row_len; int row_alter = (mode_alter == 2)?1:2; idx %= 2; rgb_row_len = this->hidden->w*RGB_PIXEL_LEN; yuv_row_len = VRAM_W*VRAM_PIXEL_LEN; row = rect->top; if(mode_alter == 2) { if(idx && !(row%2)) row++; if(!idx && row%2) row++; } rgbrow = this->hidden->buffer + rgb_row_len*row; yuvrow = VRAM_addr + idx*VRAM_PAGELEN + yuv_row_len*row; if(VRAM_addr == NULL || this->hidden->buffer == NULL) return; for(; row<rect->bottom; row+=mode_alter) { yuv = (YUV_Package*)(yuvrow + rect->left*VRAM_PIXEL_LEN*row_alter); for(col=rect->left; col < rect->right; col+=mode_alter) { if(mode_alter == 2) update_rgb_to_yuv_packet((Uint16*)(rgbrow+RGB_PIXEL_LEN*col), yuv); else update_rgb_to_yuv_packet_minimode((Uint16*)(rgbrow+RGB_PIXEL_LEN*col), yuv); //memset(yuv, 0x80, sizeof(YUV_Package)); yuv++; } rgbrow += rgb_row_len*mode_alter; yuvrow += yuv_row_len; } vdc_set_data(idx);}static inline void update_rect_to_yuv_buffer(_THIS, RECT* rect){ int mode_alter = 1; if(this->hidden->mode == 0) mode_alter = 2; update_rect_to_yuv_buffer_page(this, rect, 0, mode_alter); update_rect_to_yuv_buffer_page(this, rect, 1, mode_alter);}static void* task_do_update (void* data){ _THIS; this = data; while (this->hidden->buffer && VRAM_addr) { usleep (10000); // 10 ms if (is_suspend) continue; if (this->hidden->dirty) { RECT bound; pthread_mutex_lock (&this->hidden->lock); bound = this->hidden->update; SetRect (&this->hidden->update, 0, 0, 0, 0); this->hidden->dirty = FALSE; round_rect_to_even (&bound); BLOCK_ALARM update_rect_to_yuv_buffer(this, &bound); UNBLOCK_ALARM pthread_mutex_unlock (&this->hidden->lock); } else vdc_set_data(0); } return 0;}static void MB93493_UpdateRects (_THIS, int numrects, GAL_Rect *rects){ int i; RECT bound; pthread_mutex_lock (&this->hidden->lock); bound = this->hidden->update; for (i = 0; i < numrects; i++) { RECT rc; SetRect (&rc, rects[i].x, rects[i].y, rects[i].x + rects[i].w, rects[i].y + rects[i].h); if (IsRectEmpty (&bound)) bound = rc; else GetBoundRect (&bound, &bound, &rc); } if (!IsRectEmpty (&bound)) { if (IntersectRect (&bound, &bound, &g_rcScr)) { this->hidden->update = bound; this->hidden->dirty = TRUE; } } pthread_mutex_unlock (&this->hidden->lock);}static void MB93493_VideoQuit(_THIS){ if (this->hidden->is_opened) { //close_vdc_device(); VRAM_addr = NULL; if(this->hidden->buffer) free(this->hidden->buffer); this->hidden->buffer = NULL; this->hidden->is_opened = FALSE; //pthread_join(this->hidden->th, NULL); } if (this->screen->pixels != NULL) { free(this->screen->pixels); this->screen->pixels = NULL; }}int mb93493_suspend_spi (void){ if (is_suspend) return -1; BLOCK_ALARM is_suspend = TRUE; return 0;}void mb93493_resume_spi (void){ if (!is_suspend) return; is_suspend = FALSE; UNBLOCK_ALARM MB93493_UpdateRects (current_video, 1, &standard_mode);}
?? 快捷鍵說明
復(fù)制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號(hào)
Ctrl + =
減小字號(hào)
Ctrl + -