Use SetErrorMode so Windows will not show all kinds of error dialogs
[mplayer/glamo.git] / libvo / vo_svga.c
blobf745c41c290ef9c4edb5045d2907cb69285f1c17
1 /*
2 * video driver for SVGAlib
3 * by Zoltan Mark Vician <se7en@sch.bme.hu>
4 * Code started: Mon Apr 1 23:25:47 2001
5 * Some changes by Matan Ziv-Av <matan@svgalib.org>
6 * complete rewrite by Ivan Kalvachev 19 Mar 2003
8 * This file is part of MPlayer.
10 * MPlayer is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
15 * MPlayer is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
20 * You should have received a copy of the GNU General Public License along
21 * with MPlayer; if not, write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
26 Wrangings:
27 - 1bpp doesn't work right for me with '-double' and svgalib 1.4.3,
28 but works OK with svgalib 1.9.17
29 - The HW acceleration is not tested - svgalibs supports few chipsets,
30 and i don't have any of them. If it works for you then let me know.
31 I will remove this warning after confirm its status.
32 - retrace sync works only in doublebuffer mode.
33 - the retrace sync may slow down decoding a lot - mplayer is blocked while
34 waiting for retrace
35 - denoise3d fails to find common colorspace, use -vf denoise3d,scale
37 TODO:
38 - let choose_best_mode take aspect into account
39 - set palette from mpi->palette or mpi->plane[1]
40 - make faster OSD black bars clear - need some OSD changes
41 - Make nicer CONFIG parsing
42 - change video mode logical width to match img->stride[0] - for HW only
45 #include <stdio.h>
46 #include <stdlib.h>
47 #include <string.h>
49 #include <vga.h>
51 #include <limits.h>
53 #include "config.h"
54 #include "video_out.h"
55 #include "video_out_internal.h"
56 #include "fastmemcpy.h"
57 #include "osdep/getch2.h"
58 #ifdef CONFIG_VIDIX
59 #include "vosub_vidix.h"
60 #endif
62 #include "sub.h"
64 #include "mp_msg.h"
65 #include "help_mp.h"
66 //#include "mp_image.h"
68 #include <assert.h>
70 //silence warnings, probably it have to go in some global header
71 #define UNUSED(x) ((void)(x))
74 static void draw_alpha(int x0, int y0, int w, int h, unsigned char *src,
75 unsigned char *srca, int stride);
76 static uint32_t get_image(mp_image_t *mpi);
78 #define MAXPAGES 16
79 #define PAGE_EMPTY 0
80 #define PAGE_BUSY 1
82 #define CAP_ACCEL_CLEAR 8
83 #define CAP_ACCEL_PUTIMAGE 4
84 #define CAP_ACCEL_BACKGR 2
85 #define CAP_LINEAR 1
87 static uint8_t zerobuf[8192];//used when clear screen with vga_draw
89 static int squarepix;
90 static int force_vm=0;
91 static int force_native=0;
92 static int sync_flip=0;
93 static int blackbar_osd=0;
94 static int cpage,max_pages,old_page;
96 static vga_modeinfo * modeinfo;
97 static int mode_stride; //keep it in case of vga_setlogicalwidth
98 static int stride_granularity; //not yet used
99 static int mode_bpp;
100 static int mode_capabilities;
102 static int image_width,image_height; // used by OSD
103 static int x_pos, y_pos;
105 static struct {
106 int yoffset;//y position of the page
107 int doffset;//display start of the page
108 uint8_t * vbase;//memory start address of the page
109 int locks;
110 }PageStore[MAXPAGES];
112 static const vo_info_t info = {
113 "SVGAlib",
114 "svga",
115 "Ivan Kalvachev <iive@users.sf.net>",
119 #ifdef CONFIG_VIDIX
120 static char vidix_name[32] = "";
121 static vidix_grkey_t gr_key;
122 #endif
124 LIBVO_EXTERN(svga)
127 //return number of 1'st free page or -1 if no free one
128 static inline int page_find_free(void){
129 int i;
130 for(i=0;i<max_pages;i++)
131 if(PageStore[i].locks == PAGE_EMPTY) return i;
132 return -1;
135 static int preinit(const char *arg)
137 int i,rez;
138 char s[64];
140 getch2_disable();
141 memset(zerobuf,0,sizeof(zerobuf));
142 force_vm=force_native=squarepix=0;
143 sync_flip=vo_vsync;
144 blackbar_osd=0;
146 if(arg)while(*arg) {
147 #ifdef CONFIG_VIDIX
148 if(memcmp(arg,"vidix",5)==0) {
149 i=6;
150 while(arg[i] && arg[i]!=':') i++;
151 strncpy(vidix_name, arg+6, i-6);
152 vidix_name[i-5]=0;
153 if(arg[i]==':')i++;
154 arg+=i;
155 vidix_preinit(vidix_name, &video_out_svga);
157 #endif
158 if(!strncmp(arg,"sq",2)) {
159 squarepix=1;
160 arg+=2;
161 if( *arg == ':' ) arg++;
164 if(!strncmp(arg,"native",6)) {
165 force_native=1;
166 arg+=6;
167 if( *arg == ':' ) arg++;
170 if(!strncmp(arg,"bbosd",5)) {
171 blackbar_osd=1;
172 arg+=5;
173 if( *arg == ':' ) arg++;
176 if(!strncmp(arg,"retrace",7)) {
177 sync_flip=1;
178 arg+=7;
179 if( *arg == ':' ) arg++;
182 if(*arg) {
183 i=0;
184 while(arg[i] && arg[i]!=':')i++;
185 if(i<64){
186 strncpy(s, arg, i);
187 s[i]=0;
189 force_vm=vga_getmodenumber(s);
190 if(force_vm>0) {
191 if( mp_msg_test(MSGT_VO,MSGL_V) ) mp_msg(MSGT_VO,MSGL_V, "vo_svga: Forcing mode %i\n",force_vm);
192 }else{
193 force_vm = 0;
196 arg+=i;
197 if(*arg==':')arg++;
201 rez = vga_init();
202 if(rez != 0){
203 mp_msg(MSGT_VO,MSGL_ERR, "vo_svga: vga_init() returned error=%d\n",rez);
205 return !!rez;
208 static void svga_clear_box(int x,int y,int w,int h){
209 uint8_t * rgbplane;
210 int i;
212 if (mode_capabilities&CAP_ACCEL_CLEAR){
213 if( mp_msg_test(MSGT_VO,MSGL_DBG3) )
214 mp_msg(MSGT_VO,MSGL_DBG3, "vo_svga: clearing box %d,%d - %d,%d with HW acceleration\n",
215 x,y,w,h);
216 if(mode_capabilities&CAP_ACCEL_BACKGR)
217 vga_accel(ACCEL_SYNC);
218 vga_accel(ACCEL_SETFGCOLOR,0);//black
219 vga_accel(ACCEL_FILLBOX,x,y,w,h);
220 return;
222 if (mode_capabilities & CAP_LINEAR){
223 if( mp_msg_test(MSGT_VO,MSGL_DBG3) )
224 mp_msg(MSGT_VO,MSGL_DBG3, "vo_svga: clearing box %d,%d - %d,%d with memset\n",x,y,w,h);
225 rgbplane=PageStore[0].vbase + (y*mode_stride) + (x*modeinfo->bytesperpixel);
226 for(i=0;i<h;i++){
227 //i'm afraid that memcpy is better optimized than memset;)
228 fast_memcpy(rgbplane,zerobuf,w*modeinfo->bytesperpixel);
229 // memset(rgbplane,0,w*modeinfo->bytesperpixel);
230 rgbplane+=mode_stride;
232 return;
234 //native
235 if( mp_msg_test(MSGT_VO,MSGL_DBG3) )
236 mp_msg(MSGT_VO,MSGL_DBG3, "vo_svga: clearing box %d,%d - %d,%d with native draw \n",x,y,w,h);
237 if(modeinfo->bytesperpixel!=0) w*=modeinfo->bytesperpixel;
238 for(i=0;i<h;i++){
239 vga_drawscansegment(zerobuf,x,y+i,w);
243 static uint32_t svga_draw_image(mp_image_t *mpi){
244 int i,x,y,w,h;
245 int stride;
246 uint8_t *rgbplane, *base;
247 int bytesperline;
248 int page;
250 if(mpi->flags & MP_IMGFLAG_DIRECT){
251 if( mp_msg_test(MSGT_VO,MSGL_DBG3) )
252 mp_msg(MSGT_VO,MSGL_DBG3, "vo_svga: drawing direct rendered surface\n");
253 cpage=(uint32_t)mpi->priv;
254 assert((cpage>=0)&&(cpage<max_pages));
255 return VO_TRUE; //it's already done
257 // if (mpi->flags&MP_IMGFLAGS_DRAWBACK)
258 // return VO_TRUE;//direct render method 2
260 //find a free page to draw into
261 //if there is no one then use the current one
262 page = page_find_free();
263 if(page>=0) cpage=page;
264 PageStore[cpage].locks=PAGE_BUSY;
266 // these variables are used in loops
267 x = mpi->x;
268 y = mpi->y;
269 w = mpi->w;
270 h = mpi->h;
271 stride = mpi->stride[0];
272 rgbplane = mpi->planes[0] + y*stride + (x*mpi->bpp)/8;
273 x+=x_pos;//center
274 y+=y_pos;
276 if(mpi->bpp >= 8){//for modes<8 use only native
277 if( (mode_capabilities&CAP_ACCEL_PUTIMAGE) && (x==0) && (w==mpi->width) &&
278 (stride == mode_stride) ){ //only monolite image can be accelerated
279 w=(stride*8)/mpi->bpp;//we transfer pixels in the stride so the source
280 //ACCELERATE
281 if( mp_msg_test(MSGT_VO,MSGL_DBG3) )
282 mp_msg(MSGT_VO,MSGL_DBG3, "vo_svga: using HW PutImage (x=%d,y=%d,w=%d,h=%d)\n",x,y,w,h);
283 if(mode_capabilities & CAP_ACCEL_BACKGR)
284 vga_accel(ACCEL_SYNC);
286 vga_accel(ACCEL_PUTIMAGE,x,y+PageStore[cpage].yoffset,w,h,rgbplane);
287 return VO_TRUE;
290 if( mode_capabilities&CAP_LINEAR){
291 //DIRECT
292 if( mp_msg_test(MSGT_VO,MSGL_DBG3) )
293 mp_msg(MSGT_VO,MSGL_DBG3, "vo_svga: using Direct memcpy (x=%d,y=%d,w=%d,h=%d)\n",x,y,w,h);
294 bytesperline=(w*mpi->bpp)/8;
295 base=PageStore[cpage].vbase + (y*mode_stride) + (x*mpi->bpp)/8;
297 for(i=0;i<h;i++){
298 mem2agpcpy(base,rgbplane,bytesperline);
299 base+=mode_stride;
300 rgbplane+=stride;
302 return VO_TRUE;
304 }//(modebpp>=8
307 //NATIVE
309 int length;
310 length=(w*mpi->bpp)/8;
311 //one byte per pixel! svgalib innovation
312 if(mpi->imgfmt==IMGFMT_RG4B || mpi->imgfmt==IMGFMT_BG4B) length=w;
314 if( mp_msg_test(MSGT_VO,MSGL_DBG3) )
315 mp_msg(MSGT_VO,MSGL_DBG3, "vo_svga: using Native vga_draw(x=%d,y=%d,w=%d,h=%d)\n",x,y,w,h);
316 y+=PageStore[cpage].yoffset;//y position of the page beggining
317 for(i=0;i<h;i++){
318 vga_drawscansegment(rgbplane,x,y+i,length);
319 rgbplane+=stride;
322 return VO_TRUE;
325 static int bpp_from_vminfo(vga_modeinfo *vminfo){
326 switch(vminfo->colors){
327 case 2: return 1;
328 case 16: return 4;
329 case 256: return 8;
330 case 32768: return 15;
331 case 65536: return 16;
332 case 1<<24: return 8*vminfo->bytesperpixel;
334 return 0;
337 static int find_best_svga_mode(int req_w,int req_h, int req_bpp){
338 int badness,prev_badness;
339 int bestmode,lastmode;
340 int i;
341 vga_modeinfo *vminfo;
342 //int best aspect mode // best linear mode // best normal mode (no modeX)
344 prev_badness = 0;//take care of special case below
345 bestmode = 0; //0 is the TEXT mode
346 lastmode = vga_lastmodenumber();
347 for(i=1;i<=lastmode;i++){
348 vminfo = vga_getmodeinfo(i);
349 if( vminfo == NULL ) continue;
350 if( mp_msg_test(MSGT_VO,MSGL_DBG4) )
351 mp_msg(MSGT_VO,MSGL_DBG4, "vo_svga: testing mode %d (%s)\n",i,vga_getmodename(i));
352 if( vga_hasmode(i) == 0 ) continue;
353 if( req_bpp != bpp_from_vminfo(vminfo) )continue;
354 if( (vminfo->width < req_w) || (vminfo->height < req_h) ) continue;
355 badness=(vminfo->width * vminfo->height) - (req_h * req_w);
356 //put here aspect calculations
357 if(squarepix)
358 if( vminfo->width*3 != vminfo->height*4 ) continue;
360 if( bestmode==0 || prev_badness >= badness ){//modeX etc...
361 prev_badness=badness;
362 bestmode=i;
363 if( mp_msg_test(MSGT_VO,MSGL_DBG4) )
364 mp_msg(MSGT_VO,MSGL_DBG4, "vo_svga: found good mode %d with badness %d\n",i,badness);
367 return bestmode;
370 static int control(uint32_t request, void *data, ...)
372 switch (request) {
373 case VOCTRL_QUERY_FORMAT:
374 return query_format(*((uint32_t*)data));
375 case VOCTRL_DRAW_IMAGE:
376 return svga_draw_image( (mp_image_t *)data);
377 case VOCTRL_GET_IMAGE:
378 return get_image(data);
381 #ifdef CONFIG_VIDIX
382 if (vidix_name[0]) {
383 switch (request) {
384 case VOCTRL_SET_EQUALIZER:
386 va_list ap;
387 int value;
389 va_start(ap, data);
390 value = va_arg(ap, int);
391 va_end(ap);
393 return vidix_control(request, data, value);
395 case VOCTRL_GET_EQUALIZER:
397 va_list ap;
398 int *value;
400 va_start(ap, data);
401 value = va_arg(ap, int*);
402 va_end(ap);
404 return vidix_control(request, data, value);
407 return vidix_control(request, data);
409 #endif
411 return VO_NOTIMPL;
415 // This function is called to init the video driver for specific mode
417 static int config(uint32_t width, uint32_t height, uint32_t d_width,
418 uint32_t d_height, uint32_t flags, char *title,
419 uint32_t format) {
420 int32_t req_w = width;// (d_width > 0 ? d_width : width);
421 int32_t req_h = height;// (d_height > 0 ? d_height : height);
422 uint16_t vid_mode = 0;
423 int32_t req_bpp;
425 uint32_t accflags;
426 if( mp_msg_test(MSGT_VO,MSGL_V) )
427 mp_msg(MSGT_VO,MSGL_V, "vo_svga: config(%i, %i, %i, %i, %08x, %s, %08x)\n", width, height,
428 d_width, d_height, flags, title, format);
429 //Only RGB modes supported
430 if (!IMGFMT_IS_RGB(format) && !IMGFMT_IS_BGR(format)) {assert(0);return -1;}
431 req_bpp = IMGFMT_BGR_DEPTH(format);
433 if( vo_dbpp!=0 && vo_dbpp!=req_bpp) {assert(0);return-1;}
435 if(!force_vm) {
436 if ( mp_msg_test(MSGT_VO,MSGL_V) ) {
437 mp_msg(MSGT_VO,MSGL_V, "vo_svga: Looking for the best resolution...\n");
438 mp_msg(MSGT_VO,MSGL_V, "vo_svga: req_w: %d, req_h: %d, bpp: %d\n",req_w,req_h,req_bpp);
440 vid_mode=find_best_svga_mode(req_w,req_h,req_bpp);
441 if(vid_mode==0)
442 return 1;
443 modeinfo=vga_getmodeinfo(vid_mode);
444 }else{//force_vm
445 vid_mode=force_vm;
446 if(vga_hasmode(vid_mode) == 0){
447 mp_msg(MSGT_VO,MSGL_ERR, MSGTR_LIBVO_SVGA_ForcedVidmodeNotAvailable,
448 vid_mode,vga_getmodename(vid_mode));
449 return 1; //error;
451 modeinfo=vga_getmodeinfo(vid_mode);
452 if( (modeinfo->width < req_w) || (modeinfo->height < req_h) ){
453 mp_msg(MSGT_VO,MSGL_ERR, MSGTR_LIBVO_SVGA_ForcedVidmodeTooSmall,
454 vid_mode,vga_getmodename(vid_mode));
455 return 1;
458 mode_bpp=bpp_from_vminfo(modeinfo);
460 mp_msg(MSGT_VO,MSGL_INFO, MSGTR_LIBVO_SVGA_Vidmode,
461 vid_mode,modeinfo->width,modeinfo->height,mode_bpp);
463 if (vga_setmode(vid_mode) == -1) {
464 mp_msg(MSGT_VO,MSGL_ERR, MSGTR_LIBVO_SVGA_VgasetmodeFailed,vid_mode);
465 uninit();
466 return 1; // error
468 /* set 332 palette for 8 bpp */
469 if(mode_bpp==8){
470 int i;
471 for(i=0; i<256; i++)
472 vga_setpalette(i, ((i>>5)&7)*9, ((i>>2)&7)*9, (i&3)*21);
474 /* set 121 palette for 4 bpp */
475 else if(mode_bpp==4){
476 int i;
477 for(i=0; i<16; i++)
478 vga_setpalette(i, ((i>>3)&1)*63, ((i>>1)&3)*21, (i&1)*63);
480 //if we change the logical width, we should know the granularity
481 stride_granularity=8;//according to man vga_logicalwidth
482 if(modeinfo->flags & EXT_INFO_AVAILABLE){
483 stride_granularity=modeinfo->linewidth_unit;
485 //look for hardware acceleration
486 mode_capabilities=0;//NATIVE;
487 if(!force_native){//if we want to use only native drawers
488 if(modeinfo->flags & HAVE_EXT_SET){//support for hwaccel interface
489 accflags=vga_ext_set(VGA_EXT_AVAILABLE,VGA_AVAIL_ACCEL);
490 if(accflags & ACCELFLAG_FILLBOX) // clear screen
491 mode_capabilities|=CAP_ACCEL_CLEAR;
492 if(accflags & ACCELFLAG_PUTIMAGE)//support for mem->vid transfer
493 mode_capabilities|=CAP_ACCEL_PUTIMAGE;
494 if((accflags & ACCELFLAG_SETMODE) && (accflags & ACCELFLAG_SYNC)){
495 vga_accel(ACCEL_SETMODE,BLITS_IN_BACKGROUND);
496 mode_capabilities|=CAP_ACCEL_BACKGR;//can draw in backgraund
499 if(modeinfo->flags & IS_LINEAR){
500 mode_capabilities|=CAP_LINEAR; //don't use bank & vga_draw
502 else{
503 if(modeinfo->flags & CAPABLE_LINEAR){
504 int vid_mem_size;
505 vid_mem_size = vga_setlinearaddressing();
506 if(vid_mem_size != -1){
507 modeinfo=vga_getmodeinfo(vid_mode);//sometimes they change parameters
508 mode_capabilities|=CAP_LINEAR;
512 }//fi force native
513 if(mode_capabilities&CAP_LINEAR){
514 mp_msg(MSGT_VO,MSGL_INFO, MSGTR_LIBVO_SVGA_VideoModeIsLinearAndMemcpyCouldBeUsed);
516 if(mode_capabilities&CAP_ACCEL_PUTIMAGE){
517 mp_msg(MSGT_VO,MSGL_INFO, MSGTR_LIBVO_SVGA_VideoModeHasHardwareAcceleration);
518 mp_msg(MSGT_VO,MSGL_INFO, MSGTR_LIBVO_SVGA_IfItWorksForYouIWouldLikeToKnow);
521 //here is the place to handle strides for accel_ modes;
522 mode_stride=modeinfo->linewidth;
523 //we may try to set a bigger stride for video mode that will match the mpi->stride,
524 //this way we will transfer more data, but HW put_image can do it in backgraund!
526 //now let's see how many pages we can use
527 max_pages = modeinfo->maxpixels/(modeinfo->height * modeinfo->width);
528 if(max_pages > MAXPAGES) max_pages = MAXPAGES;
529 if(!vo_doublebuffering) max_pages=1;
530 //fill PageStore structs
532 int i;
533 uint8_t * GRAPH_MEM;
534 int dof;
535 GRAPH_MEM=vga_getgraphmem();
536 for(i=0;i<max_pages;i++){
537 //calculate display offset
538 dof = i * modeinfo->height * modeinfo->width;
539 if(modeinfo->bytesperpixel != 0) dof*=modeinfo->bytesperpixel;
540 //check video chip limitations
541 if( dof != (dof & modeinfo->startaddressrange) ){
542 max_pages=i;//page 0 will never come here
543 break;
545 PageStore[i].yoffset = i * modeinfo->height;//starting y offset
546 PageStore[i].vbase = GRAPH_MEM + i*modeinfo->height*mode_stride; //memory base address
547 PageStore[i].doffset = dof; //display offset
548 PageStore[i].locks = PAGE_EMPTY;
551 assert(max_pages>0);
552 mp_msg(MSGT_VO,MSGL_INFO, MSGTR_LIBVO_SVGA_VideoModeHas,max_pages);
553 //15bpp
554 if(modeinfo->bytesperpixel!=0)
555 vga_claimvideomemory(max_pages * modeinfo->height * modeinfo->width * modeinfo->bytesperpixel);
556 else
557 vga_claimvideomemory(max_pages * modeinfo->height * modeinfo->width * mode_bpp / 8);
559 cpage=old_page=0;
560 svga_clear_box(0,0,modeinfo->width,modeinfo->height * max_pages);
562 image_height=req_h;
563 image_width=req_w;
564 x_pos = (modeinfo->width - req_w) / 2;
565 y_pos = (modeinfo->height - req_h) / 2;
566 x_pos &= ~(15); //align x offset position to 16 pixels
567 mp_msg(MSGT_VO,MSGL_INFO, MSGTR_LIBVO_SVGA_CenteringImageStartAt,x_pos,y_pos);
569 #ifdef CONFIG_VIDIX
571 if(vidix_name[0]){
572 vidix_init(width, height, x_pos, y_pos, modeinfo->width, modeinfo->height,
573 format, mode_bpp, modeinfo->width,modeinfo->height);
574 mp_msg(MSGT_VO,MSGL_INFO, MSGTR_LIBVO_SVGA_UsingVidix,width,height,
575 modeinfo->width,modeinfo->height);
576 vidix_start();
577 /*set colorkey*/
578 if(vidix_grkey_support()){
579 vidix_grkey_get(&gr_key);
580 gr_key.key_op = KEYS_PUT;
581 if (!(vo_colorkey & 0xFF000000)) {
582 gr_key.ckey.op = CKEY_TRUE;
583 gr_key.ckey.red = (vo_colorkey & 0x00FF0000) >> 16;
584 gr_key.ckey.green = (vo_colorkey & 0x0000FF00) >> 8;
585 gr_key.ckey.blue = vo_colorkey & 0x000000FF;
586 } else
587 gr_key.ckey.op = CKEY_FALSE;
588 vidix_grkey_set(&gr_key);
591 #endif
593 vga_setdisplaystart(0);
594 return 0;
597 static int draw_slice(uint8_t *image[],int stride[],
598 int w, int h, int x, int y) {
599 assert(0);
600 UNUSED(image);UNUSED(stride);
601 UNUSED(w);UNUSED(h);
602 UNUSED(x);UNUSED(y);
604 return VO_ERROR;//this is yv12 only -> vf_scale should do all transforms
607 static int draw_frame(uint8_t *src[]) {
608 assert(0);
609 UNUSED(src);
610 return VO_ERROR;//this one should not be called
613 static void draw_osd(void)
615 if( mp_msg_test(MSGT_VO,MSGL_DBG4) )
616 mp_msg(MSGT_VO,MSGL_DBG4, "vo_svga: draw_osd()\n");
617 //only modes with bytesperpixel>0 can draw OSD
618 if(modeinfo->bytesperpixel==0) return;
619 if(!(mode_capabilities&CAP_LINEAR)) return;//force_native will remove OSD
621 if(blackbar_osd){
622 //111
623 //3 4
624 //222
625 svga_clear_box(0,0 + PageStore[cpage].yoffset,
626 modeinfo->width, y_pos);
627 svga_clear_box(0, image_height + y_pos + PageStore[cpage].yoffset,
628 modeinfo->width, modeinfo->height-(image_height+ y_pos));
629 svga_clear_box(0, y_pos + PageStore[cpage].yoffset,
630 x_pos, image_height);
631 svga_clear_box(image_width + x_pos, y_pos + PageStore[cpage].yoffset,
632 modeinfo->width-(x_pos+image_width), image_height);
633 // vo_remove_text(modeinfo->width, modeinfo->height, clear_alpha);
634 vo_draw_text(modeinfo->width, modeinfo->height, draw_alpha);
635 }else{
636 vo_draw_text(image_width, image_height, draw_alpha);
640 static void flip_page(void) {
641 PageStore[old_page].locks=PAGE_EMPTY;
642 PageStore[cpage].locks=PAGE_BUSY;
644 if( mp_msg_test(MSGT_VO,MSGL_DBG3) )
645 mp_msg(MSGT_VO,MSGL_DBG3, "vo_svga: viewing page %d\n",cpage);
646 if(sync_flip && old_page!=cpage){
647 if( mp_msg_test(MSGT_VO,MSGL_DBG3) ) mp_msg(MSGT_VO,MSGL_DBG3, "vo_svga:vga_waitretrace\n");
648 vga_waitretrace();
650 vga_setdisplaystart(PageStore[cpage].doffset);
652 old_page=cpage;//cpage will be overwriten on next draw_image
655 static void check_events(void) {
658 static void uninit(void) {
660 #ifdef CONFIG_VIDIX
661 if(vidix_name[0])vidix_term();
662 #endif
663 vga_setmode(TEXT);
666 /* --------------------------------------------------------------------- */
667 static int query_format(uint32_t format) {
668 int32_t req_bpp,flags;
669 int i,lastmode;
670 vga_modeinfo * vminfo;
672 if ( mp_msg_test(MSGT_VO,MSGL_DBG4) )
673 mp_msg(MSGT_VO,MSGL_DBG4, "vo_svga: query_format=%X \n",format);
674 //only RGB modes supported
675 if( (!IMGFMT_IS_RGB(format)) && (!IMGFMT_IS_BGR(format)) ) return 0;
677 // Reject different endian
678 #ifdef WORDS_BIGENDIAN
679 if (IMGFMT_IS_BGR(format)) return 0;
680 #else
681 if (IMGFMT_IS_RGB(format)) return 0;
682 #endif
684 //svgalib supports only BG4B! if we want BGR4 we have to emulate it (sw)
685 if( format==IMGFMT_BGR4 || format==IMGFMT_RGB4) return 0;
686 req_bpp = IMGFMT_RGB_DEPTH(format);
687 if( vo_dbpp>0 && vo_dbpp!=req_bpp ) return 0; //support -bpp options
688 //scan all modes
689 lastmode = vga_lastmodenumber();
690 for(i=1;i<=lastmode;i++){
691 vminfo = vga_getmodeinfo(i);
692 if( vminfo == NULL ) continue;
693 if( vga_hasmode(i) == 0 ) continue;
694 if( req_bpp != bpp_from_vminfo(vminfo) ) continue;
695 if( (force_vm > 0) && (force_vm != i) ) continue;//quick hack
696 flags = VFCAP_CSP_SUPPORTED|
697 VFCAP_CSP_SUPPORTED_BY_HW|
698 VFCAP_ACCEPT_STRIDE|
700 if(req_bpp>8) flags|=VFCAP_OSD;
701 return flags;
703 return 0;
706 static void draw_alpha(int x0, int y0, int w, int h, unsigned char *src,
707 unsigned char *srca, int stride) {
708 char* base;
710 if( mp_msg_test(MSGT_VO,MSGL_DBG3) )
711 mp_msg(MSGT_VO,MSGL_DBG3, "vo_svga: draw_alpha(x0=%d,y0=%d,w=%d,h=%d,src=%p,srca=%p,stride=%d\n",
712 x0,y0,w,h,src,srca,stride);
713 if(!blackbar_osd) {
714 //drawing in the image, so place the stuff there
715 x0+=x_pos;
716 y0+=y_pos;
719 if( mp_msg_test(MSGT_VO,MSGL_DBG4) )
720 mp_msg(MSGT_VO,MSGL_DBG4, "vo_svga: OSD draw in page %d\n",cpage);
721 base=PageStore[cpage].vbase + y0*mode_stride + x0*modeinfo->bytesperpixel;
722 switch (mode_bpp) {
723 case 32:
724 vo_draw_alpha_rgb32(w, h, src, srca, stride, base, mode_stride);
725 break;
726 case 24:
727 vo_draw_alpha_rgb24(w, h, src, srca, stride, base, mode_stride);
728 break;
729 case 16:
730 vo_draw_alpha_rgb16(w, h, src, srca, stride, base, mode_stride);
731 break;
732 case 15:
733 vo_draw_alpha_rgb15(w, h, src, srca, stride, base, mode_stride);
734 break;
738 static uint32_t get_image(mp_image_t *mpi){
739 int page;
741 if(!IMGFMT_IS_BGR(mpi->imgfmt) && !IMGFMT_IS_RGB(mpi->imgfmt) ){
742 assert(0);//should never happen
743 return VO_FALSE;
746 if (
747 ( (mpi->type != MP_IMGTYPE_STATIC) && (mpi->type != MP_IMGTYPE_TEMP)) ||
748 (mpi->flags & MP_IMGFLAG_PLANAR) ||
749 (mpi->flags & MP_IMGFLAG_YUV)
751 return VO_FALSE;
753 //reading from video memory is horribly slow
754 if( !(mpi->flags & MP_IMGFLAG_READABLE) && vo_directrendering &&
755 (mode_capabilities & CAP_LINEAR) ){
757 //find free page and reserve it
758 page=page_find_free();
759 if(page >= 0){
760 PageStore[page].locks=PAGE_BUSY;
762 mpi->flags |= MP_IMGFLAG_DIRECT;
763 mpi->stride[0] = mode_stride;
764 mpi->planes[0] = PageStore[page].vbase +
765 y_pos*mode_stride + (x_pos*mpi->bpp)/8;
766 mpi->priv=(void *)page;
767 if( mp_msg_test(MSGT_VO,MSGL_DBG3) )
768 mp_msg(MSGT_VO,MSGL_DBG3, "vo_svga: direct render allocated! page=%d\n",page);
769 return VO_TRUE;
773 return VO_FALSE;