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.
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
35 - denoise3d fails to find common colorspace, use -vf denoise3d,scale
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
54 #include "video_out.h"
55 #include "video_out_internal.h"
56 #include "fastmemcpy.h"
57 #include "osdep/getch2.h"
59 #include "vosub_vidix.h"
66 //#include "mp_image.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
);
82 #define CAP_ACCEL_CLEAR 8
83 #define CAP_ACCEL_PUTIMAGE 4
84 #define CAP_ACCEL_BACKGR 2
87 static uint8_t zerobuf
[8192];//used when clear screen with vga_draw
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
100 static int mode_capabilities
;
102 static int image_width
,image_height
; // used by OSD
103 static int x_pos
, y_pos
;
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
110 }PageStore
[MAXPAGES
];
112 static const vo_info_t info
= {
115 "Ivan Kalvachev <iive@users.sf.net>",
120 static char vidix_name
[32] = "";
121 static vidix_grkey_t gr_key
;
127 //return number of 1'st free page or -1 if no free one
128 static inline int page_find_free(void){
130 for(i
=0;i
<max_pages
;i
++)
131 if(PageStore
[i
].locks
== PAGE_EMPTY
) return i
;
135 static int preinit(const char *arg
)
141 memset(zerobuf
,0,sizeof(zerobuf
));
142 force_vm
=force_native
=squarepix
=0;
148 if(memcmp(arg
,"vidix",5)==0) {
150 while(arg
[i
] && arg
[i
]!=':') i
++;
151 strncpy(vidix_name
, arg
+6, i
-6);
155 vidix_preinit(vidix_name
, &video_out_svga
);
158 if(!strncmp(arg
,"sq",2)) {
161 if( *arg
== ':' ) arg
++;
164 if(!strncmp(arg
,"native",6)) {
167 if( *arg
== ':' ) arg
++;
170 if(!strncmp(arg
,"bbosd",5)) {
173 if( *arg
== ':' ) arg
++;
176 if(!strncmp(arg
,"retrace",7)) {
179 if( *arg
== ':' ) arg
++;
184 while(arg
[i
] && arg
[i
]!=':')i
++;
189 force_vm
=vga_getmodenumber(s
);
191 if( mp_msg_test(MSGT_VO
,MSGL_V
) ) mp_msg(MSGT_VO
,MSGL_V
, "vo_svga: Forcing mode %i\n",force_vm
);
203 mp_msg(MSGT_VO
,MSGL_ERR
, "vo_svga: vga_init() returned error=%d\n",rez
);
208 static void svga_clear_box(int x
,int y
,int w
,int h
){
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",
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
);
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
);
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
;
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
;
239 vga_drawscansegment(zerobuf
,x
,y
+i
,w
);
243 static uint32_t svga_draw_image(mp_image_t
*mpi
){
246 uint8_t *rgbplane
, *base
;
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
271 stride
= mpi
->stride
[0];
272 rgbplane
= mpi
->planes
[0] + y
*stride
+ (x
*mpi
->bpp
)/8;
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
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
);
290 if( mode_capabilities
&CAP_LINEAR
){
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;
298 mem2agpcpy(base
,rgbplane
,bytesperline
);
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
318 vga_drawscansegment(rgbplane
,x
,y
+i
,length
);
325 static int bpp_from_vminfo(vga_modeinfo
*vminfo
){
326 switch(vminfo
->colors
){
330 case 32768: return 15;
331 case 65536: return 16;
332 case 1<<24: return 8*vminfo
->bytesperpixel
;
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
;
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
358 if( vminfo
->width
*3 != vminfo
->height
*4 ) continue;
360 if( bestmode
==0 || prev_badness
>= badness
){//modeX etc...
361 prev_badness
=badness
;
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
);
370 static int control(uint32_t request
, void *data
, ...)
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
);
384 case VOCTRL_SET_EQUALIZER
:
390 value
= va_arg(ap
, int);
393 return vidix_control(request
, data
, value
);
395 case VOCTRL_GET_EQUALIZER
:
401 value
= va_arg(ap
, int*);
404 return vidix_control(request
, data
, value
);
407 return vidix_control(request
, data
);
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
,
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;
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;}
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
);
443 modeinfo
=vga_getmodeinfo(vid_mode
);
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
));
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
));
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
);
468 /* set 332 palette for 8 bpp */
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){
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
503 if(modeinfo
->flags
& CAPABLE_LINEAR
){
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
;
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
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
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
;
552 mp_msg(MSGT_VO
,MSGL_INFO
, MSGTR_LIBVO_SVGA_VideoModeHas
,max_pages
);
554 if(modeinfo
->bytesperpixel
!=0)
555 vga_claimvideomemory(max_pages
* modeinfo
->height
* modeinfo
->width
* modeinfo
->bytesperpixel
);
557 vga_claimvideomemory(max_pages
* modeinfo
->height
* modeinfo
->width
* mode_bpp
/ 8);
560 svga_clear_box(0,0,modeinfo
->width
,modeinfo
->height
* max_pages
);
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
);
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
);
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;
587 gr_key
.ckey
.op
= CKEY_FALSE
;
588 vidix_grkey_set(&gr_key
);
593 vga_setdisplaystart(0);
597 static int draw_slice(uint8_t *image
[],int stride
[],
598 int w
, int h
, int x
, int y
) {
600 UNUSED(image
);UNUSED(stride
);
604 return VO_ERROR
;//this is yv12 only -> vf_scale should do all transforms
607 static int draw_frame(uint8_t *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
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
);
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");
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) {
661 if(vidix_name
[0])vidix_term();
666 /* --------------------------------------------------------------------- */
667 static int query_format(uint32_t format
) {
668 int32_t req_bpp
,flags
;
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;
681 if (IMGFMT_IS_RGB(format
)) return 0;
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
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
|
700 if(req_bpp
>8) flags
|=VFCAP_OSD
;
706 static void draw_alpha(int x0
, int y0
, int w
, int h
, unsigned char *src
,
707 unsigned char *srca
, int stride
) {
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
);
714 //drawing in the image, so place the stuff there
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
;
724 vo_draw_alpha_rgb32(w
, h
, src
, srca
, stride
, base
, mode_stride
);
727 vo_draw_alpha_rgb24(w
, h
, src
, srca
, stride
, base
, mode_stride
);
730 vo_draw_alpha_rgb16(w
, h
, src
, srca
, stride
, base
, mode_stride
);
733 vo_draw_alpha_rgb15(w
, h
, src
, srca
, stride
, base
, mode_stride
);
738 static uint32_t get_image(mp_image_t
*mpi
){
741 if(!IMGFMT_IS_BGR(mpi
->imgfmt
) && !IMGFMT_IS_RGB(mpi
->imgfmt
) ){
742 assert(0);//should never happen
747 ( (mpi
->type
!= MP_IMGTYPE_STATIC
) && (mpi
->type
!= MP_IMGTYPE_TEMP
)) ||
748 (mpi
->flags
& MP_IMGFLAG_PLANAR
) ||
749 (mpi
->flags
& MP_IMGFLAG_YUV
)
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();
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
);