2010-01-30 16:57:40 +00:00
|
|
|
/*
|
|
|
|
* This file is part of MPlayer.
|
|
|
|
*
|
|
|
|
* MPlayer 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.
|
|
|
|
*
|
|
|
|
* MPlayer 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 MPlayer; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
2007-08-04 22:12:49 +00:00
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2011-10-06 18:46:01 +00:00
|
|
|
#include "talloc.h"
|
2007-08-04 22:12:49 +00:00
|
|
|
|
2012-11-09 00:06:43 +00:00
|
|
|
#include "video/img_format.h"
|
|
|
|
#include "video/mp_image.h"
|
|
|
|
#include "video/sws_utils.h"
|
2007-08-04 22:12:49 +00:00
|
|
|
|
2012-11-09 00:06:43 +00:00
|
|
|
#include "video/memcpy_pic.h"
|
2010-04-20 04:33:00 +00:00
|
|
|
#include "libavutil/mem.h"
|
2012-10-17 11:46:46 +00:00
|
|
|
#include "libavutil/common.h"
|
2007-08-04 22:12:49 +00:00
|
|
|
|
2009-12-31 23:09:35 +00:00
|
|
|
void mp_image_alloc_planes(mp_image_t *mpi) {
|
2007-08-04 22:12:49 +00:00
|
|
|
// IF09 - allocate space for 4. plane delta info - unused
|
2009-12-31 23:09:35 +00:00
|
|
|
if (mpi->imgfmt == IMGFMT_IF09) {
|
2010-04-20 04:33:00 +00:00
|
|
|
mpi->planes[0]=av_malloc(mpi->bpp*mpi->width*(mpi->height+2)/8+
|
2009-12-31 23:09:35 +00:00
|
|
|
mpi->chroma_width*mpi->chroma_height);
|
|
|
|
} else
|
2010-04-20 04:33:00 +00:00
|
|
|
mpi->planes[0]=av_malloc(mpi->bpp*mpi->width*(mpi->height+2)/8);
|
2011-10-06 18:46:01 +00:00
|
|
|
if (!mpi->planes[0])
|
|
|
|
abort(); //out of memory
|
2009-12-31 23:09:35 +00:00
|
|
|
if (mpi->flags&MP_IMGFLAG_PLANAR) {
|
2012-10-24 17:05:49 +00:00
|
|
|
// FIXME this code only supports same bpp for all planes, and bpp divisible
|
|
|
|
// by 8. Currently the case for all planar formats.
|
|
|
|
int bpp = MP_IMAGE_PLANAR_BITS_PER_PIXEL_ON_PLANE(mpi, 0) / 8;
|
2007-08-04 22:12:49 +00:00
|
|
|
// YV12/I420/YVU9/IF09. feel free to add other planar formats here...
|
2009-12-31 23:45:07 +00:00
|
|
|
mpi->stride[0]=mpi->stride[3]=bpp*mpi->width;
|
2009-12-31 23:09:35 +00:00
|
|
|
if(mpi->num_planes > 2){
|
|
|
|
mpi->stride[1]=mpi->stride[2]=bpp*mpi->chroma_width;
|
|
|
|
if(mpi->flags&MP_IMGFLAG_SWAPPED){
|
|
|
|
// I420/IYUV (Y,U,V)
|
|
|
|
mpi->planes[1]=mpi->planes[0]+mpi->stride[0]*mpi->height;
|
|
|
|
mpi->planes[2]=mpi->planes[1]+mpi->stride[1]*mpi->chroma_height;
|
2009-12-31 23:45:07 +00:00
|
|
|
if (mpi->num_planes > 3)
|
|
|
|
mpi->planes[3]=mpi->planes[2]+mpi->stride[2]*mpi->chroma_height;
|
2009-12-31 23:09:35 +00:00
|
|
|
} else {
|
|
|
|
// YV12,YVU9,IF09 (Y,V,U)
|
|
|
|
mpi->planes[2]=mpi->planes[0]+mpi->stride[0]*mpi->height;
|
|
|
|
mpi->planes[1]=mpi->planes[2]+mpi->stride[1]*mpi->chroma_height;
|
2009-12-31 23:45:07 +00:00
|
|
|
if (mpi->num_planes > 3)
|
|
|
|
mpi->planes[3]=mpi->planes[1]+mpi->stride[1]*mpi->chroma_height;
|
2009-12-31 23:09:35 +00:00
|
|
|
}
|
2007-08-04 22:12:49 +00:00
|
|
|
} else {
|
2009-12-31 23:09:35 +00:00
|
|
|
// NV12/NV21
|
|
|
|
mpi->stride[1]=mpi->chroma_width;
|
|
|
|
mpi->planes[1]=mpi->planes[0]+mpi->stride[0]*mpi->height;
|
2007-08-04 22:12:49 +00:00
|
|
|
}
|
|
|
|
} else {
|
2009-12-31 23:09:35 +00:00
|
|
|
mpi->stride[0]=mpi->width*mpi->bpp/8;
|
|
|
|
if (mpi->flags & MP_IMGFLAG_RGB_PALETTE)
|
2010-04-20 04:33:00 +00:00
|
|
|
mpi->planes[1] = av_malloc(1024);
|
2007-08-04 22:12:49 +00:00
|
|
|
}
|
|
|
|
mpi->flags|=MP_IMGFLAG_ALLOCATED;
|
2009-12-31 23:09:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
mp_image_t* alloc_mpi(int w, int h, unsigned long int fmt) {
|
|
|
|
mp_image_t* mpi = new_mp_image(w,h);
|
|
|
|
|
2012-11-22 18:22:38 +00:00
|
|
|
mpi->width=FFALIGN(w, MP_STRIDE_ALIGNMENT);
|
2009-12-31 23:09:35 +00:00
|
|
|
mp_image_setfmt(mpi,fmt);
|
|
|
|
mp_image_alloc_planes(mpi);
|
2012-11-22 18:22:38 +00:00
|
|
|
mpi->width=w;
|
|
|
|
mp_image_setfmt(mpi,fmt); // reset chroma size
|
2009-05-13 02:58:57 +00:00
|
|
|
|
2007-08-04 22:12:49 +00:00
|
|
|
return mpi;
|
|
|
|
}
|
|
|
|
|
|
|
|
void copy_mpi(mp_image_t *dmpi, mp_image_t *mpi) {
|
|
|
|
if(mpi->flags&MP_IMGFLAG_PLANAR){
|
2012-10-24 17:05:49 +00:00
|
|
|
memcpy_pic(dmpi->planes[0],mpi->planes[0], MP_IMAGE_BYTES_PER_ROW_ON_PLANE(mpi, 0), mpi->h,
|
2007-08-04 22:12:49 +00:00
|
|
|
dmpi->stride[0],mpi->stride[0]);
|
2012-10-24 17:05:49 +00:00
|
|
|
memcpy_pic(dmpi->planes[1],mpi->planes[1], MP_IMAGE_BYTES_PER_ROW_ON_PLANE(mpi, 1), mpi->chroma_height,
|
2007-08-04 22:12:49 +00:00
|
|
|
dmpi->stride[1],mpi->stride[1]);
|
2012-10-24 17:05:49 +00:00
|
|
|
memcpy_pic(dmpi->planes[2], mpi->planes[2], MP_IMAGE_BYTES_PER_ROW_ON_PLANE(mpi, 2), mpi->chroma_height,
|
2007-08-04 22:12:49 +00:00
|
|
|
dmpi->stride[2],mpi->stride[2]);
|
|
|
|
} else {
|
2009-05-13 02:58:57 +00:00
|
|
|
memcpy_pic(dmpi->planes[0],mpi->planes[0],
|
2012-10-24 17:05:49 +00:00
|
|
|
MP_IMAGE_BYTES_PER_ROW_ON_PLANE(mpi, 0), mpi->h,
|
2007-08-04 22:12:49 +00:00
|
|
|
dmpi->stride[0],mpi->stride[0]);
|
|
|
|
}
|
|
|
|
}
|
2010-04-15 05:39:36 +00:00
|
|
|
|
|
|
|
void mp_image_setfmt(mp_image_t* mpi,unsigned int out_fmt){
|
|
|
|
mpi->flags&=~(MP_IMGFLAG_PLANAR|MP_IMGFLAG_YUV|MP_IMGFLAG_SWAPPED);
|
|
|
|
mpi->imgfmt=out_fmt;
|
|
|
|
// compressed formats
|
2012-10-22 20:52:22 +00:00
|
|
|
if(IMGFMT_IS_HWACCEL(out_fmt)){
|
2010-04-15 05:39:36 +00:00
|
|
|
mpi->bpp=0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
mpi->num_planes=1;
|
|
|
|
if (IMGFMT_IS_RGB(out_fmt)) {
|
|
|
|
if (IMGFMT_RGB_DEPTH(out_fmt) < 8 && !(out_fmt&128))
|
|
|
|
mpi->bpp = IMGFMT_RGB_DEPTH(out_fmt);
|
|
|
|
else
|
|
|
|
mpi->bpp=(IMGFMT_RGB_DEPTH(out_fmt)+7)&(~7);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (IMGFMT_IS_BGR(out_fmt)) {
|
|
|
|
if (IMGFMT_BGR_DEPTH(out_fmt) < 8 && !(out_fmt&128))
|
|
|
|
mpi->bpp = IMGFMT_BGR_DEPTH(out_fmt);
|
|
|
|
else
|
|
|
|
mpi->bpp=(IMGFMT_BGR_DEPTH(out_fmt)+7)&(~7);
|
|
|
|
mpi->flags|=MP_IMGFLAG_SWAPPED;
|
|
|
|
return;
|
|
|
|
}
|
2012-10-20 23:10:32 +00:00
|
|
|
switch (out_fmt) {
|
|
|
|
case IMGFMT_BGR0:
|
|
|
|
mpi->bpp = 32;
|
|
|
|
return;
|
|
|
|
}
|
2010-04-15 05:39:36 +00:00
|
|
|
mpi->num_planes=3;
|
2012-03-25 21:12:25 +00:00
|
|
|
if (out_fmt == IMGFMT_GBRP) {
|
2011-10-23 20:40:31 +00:00
|
|
|
mpi->bpp=24;
|
|
|
|
mpi->flags|=MP_IMGFLAG_PLANAR;
|
draw_bmp: add RGB rendering to fix image quality issues
As pointed out in commit ed01df, the quality loss due to frequent
conversion between RGB and YUV is too much when drawing OSD and
subtitles.
Fix this by staying in the same colorspace when drawing subtitles.
Render directly to RGB, without converting to YUV first.
The bad thing about packed RGB is that there are many pixel formats,
which would all require special code for blending. It's also completely
incompatible to planar YUV. Use planar RGB instead, which allows us to
reuse all code originally written for planar YUV. The only thing that
needs to be changed is the color conversion in the libass case. (In
exchange for simpler code, the image has to be copied, but this is
still much better than converting to YUV.)
Unfortunately, libswscale doesn't support planar RGB output. Add a hack
to sws_utils.c to handle conversion to planar RGB. In the common case,
when converting 32 bit per pixel RGB, calling swscale can be avoided
entirely.
The change in mp_image.c is needed to allocate GBRP images correctly.
(The issue with vo_x11 could be easily solved by always backing up the
same bounding box as the bitmap drawing RGB<->YUV conversion does, but
this commit is probably the better fix.)
2012-11-22 12:30:16 +00:00
|
|
|
mpi->chroma_x_shift = 0;
|
|
|
|
mpi->chroma_y_shift = 0;
|
|
|
|
mpi->chroma_width=mpi->width;
|
|
|
|
mpi->chroma_height=mpi->height;
|
2011-10-23 20:40:31 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
mpi->flags|=MP_IMGFLAG_YUV;
|
2011-05-10 17:51:39 +00:00
|
|
|
if (mp_get_chroma_shift(out_fmt, NULL, NULL, NULL)) {
|
2010-04-15 05:39:36 +00:00
|
|
|
mpi->flags|=MP_IMGFLAG_PLANAR;
|
2011-05-10 17:51:39 +00:00
|
|
|
mpi->bpp = mp_get_chroma_shift(out_fmt, &mpi->chroma_x_shift, &mpi->chroma_y_shift, NULL);
|
2010-04-15 05:39:36 +00:00
|
|
|
mpi->chroma_width = mpi->width >> mpi->chroma_x_shift;
|
|
|
|
mpi->chroma_height = mpi->height >> mpi->chroma_y_shift;
|
|
|
|
}
|
|
|
|
switch(out_fmt){
|
|
|
|
case IMGFMT_I420:
|
|
|
|
case IMGFMT_IYUV:
|
|
|
|
mpi->flags|=MP_IMGFLAG_SWAPPED;
|
|
|
|
case IMGFMT_YV12:
|
|
|
|
return;
|
|
|
|
case IMGFMT_420A:
|
|
|
|
case IMGFMT_IF09:
|
|
|
|
mpi->num_planes=4;
|
|
|
|
case IMGFMT_YVU9:
|
|
|
|
case IMGFMT_444P:
|
|
|
|
case IMGFMT_422P:
|
|
|
|
case IMGFMT_411P:
|
|
|
|
case IMGFMT_440P:
|
|
|
|
case IMGFMT_444P16_LE:
|
|
|
|
case IMGFMT_444P16_BE:
|
2011-06-25 22:22:53 +00:00
|
|
|
case IMGFMT_444P10_LE:
|
|
|
|
case IMGFMT_444P10_BE:
|
|
|
|
case IMGFMT_444P9_LE:
|
|
|
|
case IMGFMT_444P9_BE:
|
2010-04-15 05:39:36 +00:00
|
|
|
case IMGFMT_422P16_LE:
|
|
|
|
case IMGFMT_422P16_BE:
|
2011-06-25 22:22:53 +00:00
|
|
|
case IMGFMT_422P10_LE:
|
|
|
|
case IMGFMT_422P10_BE:
|
2012-01-05 20:32:10 +00:00
|
|
|
case IMGFMT_422P9_LE:
|
|
|
|
case IMGFMT_422P9_BE:
|
2010-04-15 05:39:36 +00:00
|
|
|
case IMGFMT_420P16_LE:
|
|
|
|
case IMGFMT_420P16_BE:
|
2011-06-25 22:22:53 +00:00
|
|
|
case IMGFMT_420P10_LE:
|
|
|
|
case IMGFMT_420P10_BE:
|
|
|
|
case IMGFMT_420P9_LE:
|
|
|
|
case IMGFMT_420P9_BE:
|
2010-04-15 05:39:36 +00:00
|
|
|
return;
|
|
|
|
case IMGFMT_Y800:
|
|
|
|
case IMGFMT_Y8:
|
2012-11-14 10:19:04 +00:00
|
|
|
case IMGFMT_Y16LE:
|
|
|
|
case IMGFMT_Y16BE:
|
2010-04-15 05:39:36 +00:00
|
|
|
/* they're planar ones, but for easier handling use them as packed */
|
2010-05-09 00:18:26 +00:00
|
|
|
mpi->flags&=~MP_IMGFLAG_PLANAR;
|
2010-04-15 05:39:36 +00:00
|
|
|
mpi->num_planes=1;
|
|
|
|
return;
|
|
|
|
case IMGFMT_UYVY:
|
|
|
|
mpi->flags|=MP_IMGFLAG_SWAPPED;
|
|
|
|
case IMGFMT_YUY2:
|
2011-11-22 19:31:29 +00:00
|
|
|
mpi->chroma_x_shift = 1;
|
2012-10-21 15:08:54 +00:00
|
|
|
mpi->chroma_y_shift = 1;
|
|
|
|
mpi->chroma_width=(mpi->width>>1);
|
|
|
|
mpi->chroma_height=(mpi->height>>1);
|
2010-04-15 05:39:36 +00:00
|
|
|
mpi->bpp=16;
|
|
|
|
mpi->num_planes=1;
|
|
|
|
return;
|
|
|
|
case IMGFMT_NV12:
|
|
|
|
mpi->flags|=MP_IMGFLAG_SWAPPED;
|
|
|
|
case IMGFMT_NV21:
|
|
|
|
mpi->flags|=MP_IMGFLAG_PLANAR;
|
|
|
|
mpi->bpp=12;
|
|
|
|
mpi->num_planes=2;
|
|
|
|
mpi->chroma_width=(mpi->width>>0);
|
|
|
|
mpi->chroma_height=(mpi->height>>1);
|
|
|
|
mpi->chroma_x_shift=0;
|
|
|
|
mpi->chroma_y_shift=1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
mp_msg(MSGT_DECVIDEO,MSGL_WARN,"mp_image: unknown out_fmt: 0x%X\n",out_fmt);
|
|
|
|
mpi->bpp=0;
|
|
|
|
}
|
|
|
|
|
2011-10-06 18:46:01 +00:00
|
|
|
static int mp_image_destructor(void *ptr)
|
|
|
|
{
|
|
|
|
mp_image_t *mpi = ptr;
|
|
|
|
|
|
|
|
if(mpi->flags&MP_IMGFLAG_ALLOCATED){
|
|
|
|
/* because we allocate the whole image at once */
|
|
|
|
av_free(mpi->planes[0]);
|
|
|
|
if (mpi->flags & MP_IMGFLAG_RGB_PALETTE)
|
|
|
|
av_free(mpi->planes[1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-04-15 05:39:36 +00:00
|
|
|
mp_image_t* new_mp_image(int w,int h){
|
2011-10-06 18:46:01 +00:00
|
|
|
mp_image_t* mpi = talloc_zero(NULL, mp_image_t);
|
|
|
|
talloc_set_destructor(mpi, mp_image_destructor);
|
2010-04-15 05:39:36 +00:00
|
|
|
mpi->width=mpi->w=w;
|
|
|
|
mpi->height=mpi->h=h;
|
|
|
|
return mpi;
|
|
|
|
}
|
|
|
|
|
|
|
|
void free_mp_image(mp_image_t* mpi){
|
2011-10-06 18:46:01 +00:00
|
|
|
talloc_free(mpi);
|
2010-04-15 05:39:36 +00:00
|
|
|
}
|
|
|
|
|
2012-10-27 16:01:51 +00:00
|
|
|
enum mp_csp mp_image_csp(struct mp_image *img)
|
|
|
|
{
|
|
|
|
if (img->colorspace != MP_CSP_AUTO)
|
|
|
|
return img->colorspace;
|
|
|
|
return (img->flags & MP_IMGFLAG_YUV) ? MP_CSP_BT_601 : MP_CSP_RGB;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum mp_csp_levels mp_image_levels(struct mp_image *img)
|
|
|
|
{
|
|
|
|
if (img->levels != MP_CSP_LEVELS_AUTO)
|
|
|
|
return img->levels;
|
|
|
|
return (img->flags & MP_IMGFLAG_YUV) ? MP_CSP_LEVELS_TV : MP_CSP_LEVELS_PC;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mp_image_set_colorspace_details(struct mp_image *image,
|
|
|
|
struct mp_csp_details *csp)
|
|
|
|
{
|
|
|
|
if (image->flags & MP_IMGFLAG_YUV) {
|
|
|
|
image->colorspace = csp->format;
|
|
|
|
if (image->colorspace == MP_CSP_AUTO)
|
|
|
|
image->colorspace = MP_CSP_BT_601;
|
|
|
|
image->levels = csp->levels_in;
|
|
|
|
if (image->levels == MP_CSP_LEVELS_AUTO)
|
|
|
|
image->levels = MP_CSP_LEVELS_TV;
|
|
|
|
} else {
|
|
|
|
image->colorspace = MP_CSP_RGB;
|
|
|
|
image->levels = MP_CSP_LEVELS_PC;
|
|
|
|
}
|
|
|
|
}
|