2010-01-30 17:57:40 +01: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-05 00:12:49 +02:00
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2012-12-12 00:43:36 +01:00
|
|
|
#include <assert.h>
|
2007-08-05 00:12:49 +02:00
|
|
|
|
2011-10-06 20:46:01 +02:00
|
|
|
#include "talloc.h"
|
2007-08-05 00:12:49 +02:00
|
|
|
|
2012-11-09 01:06:43 +01:00
|
|
|
#include "video/img_format.h"
|
|
|
|
#include "video/mp_image.h"
|
|
|
|
#include "video/sws_utils.h"
|
2012-12-12 00:43:36 +01:00
|
|
|
#include "video/filter/vf.h"
|
2007-08-05 00:12:49 +02:00
|
|
|
|
2012-11-09 01:06:43 +01:00
|
|
|
#include "video/memcpy_pic.h"
|
2010-04-20 06:33:00 +02:00
|
|
|
#include "libavutil/mem.h"
|
2012-10-17 13:46:46 +02:00
|
|
|
#include "libavutil/common.h"
|
2007-08-05 00:12:49 +02:00
|
|
|
|
2012-12-12 00:43:36 +01:00
|
|
|
struct m_refcount {
|
|
|
|
void *arg;
|
|
|
|
// free() is called if refcount reaches 0.
|
|
|
|
void (*free)(void *arg);
|
|
|
|
// External refcounted object (such as libavcodec DR buffers). This assumes
|
|
|
|
// that the actual data is managed by the external object, not by
|
|
|
|
// m_refcount. The .ext_* calls use that external object's refcount
|
|
|
|
// primitives. It usually doesn't make sense to set both .free and .ext_*.
|
|
|
|
void (*ext_ref)(void *arg);
|
|
|
|
void (*ext_unref)(void *arg);
|
|
|
|
bool (*ext_is_unique)(void *arg);
|
|
|
|
// Native refcount (there may be additional references if .ext_* are set)
|
|
|
|
int refcount;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Only for checking API usage
|
|
|
|
static int m_refcount_destructor(void *ptr)
|
|
|
|
{
|
|
|
|
struct m_refcount *ref = ptr;
|
|
|
|
assert(ref->refcount == 0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Starts out with refcount==1, caller can set .arg and .free and .ext_*
|
|
|
|
static struct m_refcount *m_refcount_new(void)
|
|
|
|
{
|
|
|
|
struct m_refcount *ref = talloc_ptrtype(NULL, ref);
|
|
|
|
*ref = (struct m_refcount) { .refcount = 1 };
|
|
|
|
talloc_set_destructor(ref, m_refcount_destructor);
|
|
|
|
return ref;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void m_refcount_ref(struct m_refcount *ref)
|
|
|
|
{
|
|
|
|
ref->refcount++;
|
|
|
|
if (ref->ext_ref)
|
|
|
|
ref->ext_ref(ref->arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void m_refcount_unref(struct m_refcount *ref)
|
|
|
|
{
|
|
|
|
assert(ref->refcount > 0);
|
|
|
|
if (ref->ext_unref)
|
|
|
|
ref->ext_unref(ref->arg);
|
|
|
|
ref->refcount--;
|
|
|
|
if (ref->refcount == 0) {
|
|
|
|
if (ref->free)
|
|
|
|
ref->free(ref->arg);
|
|
|
|
talloc_free(ref);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool m_refcount_is_unique(struct m_refcount *ref)
|
|
|
|
{
|
|
|
|
if (ref->refcount > 1)
|
|
|
|
return false;
|
|
|
|
if (ref->ext_is_unique)
|
|
|
|
return ref->ext_is_unique(ref->arg); // referenced only by us
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-01 00:09:35 +01:00
|
|
|
void mp_image_alloc_planes(mp_image_t *mpi) {
|
2012-12-12 00:43:36 +01:00
|
|
|
assert(!mpi->refcount);
|
2007-08-05 00:12:49 +02:00
|
|
|
// IF09 - allocate space for 4. plane delta info - unused
|
2010-01-01 00:09:35 +01:00
|
|
|
if (mpi->imgfmt == IMGFMT_IF09) {
|
2010-04-20 06:33:00 +02:00
|
|
|
mpi->planes[0]=av_malloc(mpi->bpp*mpi->width*(mpi->height+2)/8+
|
2010-01-01 00:09:35 +01:00
|
|
|
mpi->chroma_width*mpi->chroma_height);
|
|
|
|
} else
|
2010-04-20 06:33:00 +02:00
|
|
|
mpi->planes[0]=av_malloc(mpi->bpp*mpi->width*(mpi->height+2)/8);
|
2011-10-06 20:46:01 +02:00
|
|
|
if (!mpi->planes[0])
|
|
|
|
abort(); //out of memory
|
2010-01-01 00:09:35 +01:00
|
|
|
if (mpi->flags&MP_IMGFLAG_PLANAR) {
|
2012-10-24 19:05:49 +02: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-05 00:12:49 +02:00
|
|
|
// YV12/I420/YVU9/IF09. feel free to add other planar formats here...
|
2010-01-01 00:45:07 +01:00
|
|
|
mpi->stride[0]=mpi->stride[3]=bpp*mpi->width;
|
2010-01-01 00:09:35 +01: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;
|
2010-01-01 00:45:07 +01:00
|
|
|
if (mpi->num_planes > 3)
|
|
|
|
mpi->planes[3]=mpi->planes[2]+mpi->stride[2]*mpi->chroma_height;
|
2010-01-01 00:09:35 +01: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;
|
2010-01-01 00:45:07 +01:00
|
|
|
if (mpi->num_planes > 3)
|
|
|
|
mpi->planes[3]=mpi->planes[1]+mpi->stride[1]*mpi->chroma_height;
|
2010-01-01 00:09:35 +01:00
|
|
|
}
|
2007-08-05 00:12:49 +02:00
|
|
|
} else {
|
2010-01-01 00:09:35 +01:00
|
|
|
// NV12/NV21
|
|
|
|
mpi->stride[1]=mpi->chroma_width;
|
|
|
|
mpi->planes[1]=mpi->planes[0]+mpi->stride[0]*mpi->height;
|
2007-08-05 00:12:49 +02:00
|
|
|
}
|
|
|
|
} else {
|
2010-01-01 00:09:35 +01:00
|
|
|
mpi->stride[0]=mpi->width*mpi->bpp/8;
|
|
|
|
if (mpi->flags & MP_IMGFLAG_RGB_PALETTE)
|
2010-04-20 06:33:00 +02:00
|
|
|
mpi->planes[1] = av_malloc(1024);
|
2007-08-05 00:12:49 +02:00
|
|
|
}
|
|
|
|
mpi->flags|=MP_IMGFLAG_ALLOCATED;
|
2010-01-01 00:09:35 +01:00
|
|
|
}
|
|
|
|
|
2012-12-12 00:43:36 +01:00
|
|
|
void mp_image_copy(struct mp_image *dmpi, struct mp_image *mpi)
|
|
|
|
{
|
2007-08-05 00:12:49 +02:00
|
|
|
if(mpi->flags&MP_IMGFLAG_PLANAR){
|
2012-10-24 19:05:49 +02:00
|
|
|
memcpy_pic(dmpi->planes[0],mpi->planes[0], MP_IMAGE_BYTES_PER_ROW_ON_PLANE(mpi, 0), mpi->h,
|
2007-08-05 00:12:49 +02:00
|
|
|
dmpi->stride[0],mpi->stride[0]);
|
2012-10-24 19:05:49 +02:00
|
|
|
memcpy_pic(dmpi->planes[1],mpi->planes[1], MP_IMAGE_BYTES_PER_ROW_ON_PLANE(mpi, 1), mpi->chroma_height,
|
2007-08-05 00:12:49 +02:00
|
|
|
dmpi->stride[1],mpi->stride[1]);
|
2012-10-24 19:05:49 +02:00
|
|
|
memcpy_pic(dmpi->planes[2], mpi->planes[2], MP_IMAGE_BYTES_PER_ROW_ON_PLANE(mpi, 2), mpi->chroma_height,
|
2007-08-05 00:12:49 +02:00
|
|
|
dmpi->stride[2],mpi->stride[2]);
|
|
|
|
} else {
|
2009-05-13 04:58:57 +02:00
|
|
|
memcpy_pic(dmpi->planes[0],mpi->planes[0],
|
2012-10-24 19:05:49 +02:00
|
|
|
MP_IMAGE_BYTES_PER_ROW_ON_PLANE(mpi, 0), mpi->h,
|
2007-08-05 00:12:49 +02:00
|
|
|
dmpi->stride[0],mpi->stride[0]);
|
|
|
|
}
|
|
|
|
}
|
2010-04-15 07:39:36 +02:00
|
|
|
|
2012-12-12 00:43:36 +01:00
|
|
|
void mp_image_copy_attributes(struct mp_image *dmpi, struct mp_image *mpi)
|
|
|
|
{
|
|
|
|
vf_clone_mpi_attributes(dmpi, mpi);
|
|
|
|
}
|
|
|
|
|
2010-04-15 07:39:36 +02: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 22:52:22 +02:00
|
|
|
if(IMGFMT_IS_HWACCEL(out_fmt)){
|
2010-04-15 07:39:36 +02: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-21 01:10:32 +02:00
|
|
|
switch (out_fmt) {
|
|
|
|
case IMGFMT_BGR0:
|
|
|
|
mpi->bpp = 32;
|
|
|
|
return;
|
|
|
|
}
|
2010-04-15 07:39:36 +02:00
|
|
|
mpi->num_planes=3;
|
2012-03-25 23:12:25 +02:00
|
|
|
if (out_fmt == IMGFMT_GBRP) {
|
2011-10-23 22:40:31 +02: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 13:30:16 +01:00
|
|
|
mpi->chroma_x_shift = 0;
|
|
|
|
mpi->chroma_y_shift = 0;
|
|
|
|
mpi->chroma_width=mpi->width;
|
|
|
|
mpi->chroma_height=mpi->height;
|
2011-10-23 22:40:31 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
mpi->flags|=MP_IMGFLAG_YUV;
|
2011-05-10 19:51:39 +02:00
|
|
|
if (mp_get_chroma_shift(out_fmt, NULL, NULL, NULL)) {
|
2010-04-15 07:39:36 +02:00
|
|
|
mpi->flags|=MP_IMGFLAG_PLANAR;
|
2011-05-10 19:51:39 +02:00
|
|
|
mpi->bpp = mp_get_chroma_shift(out_fmt, &mpi->chroma_x_shift, &mpi->chroma_y_shift, NULL);
|
2010-04-15 07:39:36 +02: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:
|
2012-11-27 10:39:09 +01:00
|
|
|
case IMGFMT_444P14_LE:
|
|
|
|
case IMGFMT_444P14_BE:
|
|
|
|
case IMGFMT_444P12_LE:
|
|
|
|
case IMGFMT_444P12_BE:
|
2011-06-26 00:22:53 +02:00
|
|
|
case IMGFMT_444P10_LE:
|
|
|
|
case IMGFMT_444P10_BE:
|
|
|
|
case IMGFMT_444P9_LE:
|
|
|
|
case IMGFMT_444P9_BE:
|
2010-04-15 07:39:36 +02:00
|
|
|
case IMGFMT_422P16_LE:
|
|
|
|
case IMGFMT_422P16_BE:
|
2012-11-27 10:39:09 +01:00
|
|
|
case IMGFMT_422P14_LE:
|
|
|
|
case IMGFMT_422P14_BE:
|
|
|
|
case IMGFMT_422P12_LE:
|
|
|
|
case IMGFMT_422P12_BE:
|
2011-06-26 00:22:53 +02:00
|
|
|
case IMGFMT_422P10_LE:
|
|
|
|
case IMGFMT_422P10_BE:
|
2012-01-05 21:32:10 +01:00
|
|
|
case IMGFMT_422P9_LE:
|
|
|
|
case IMGFMT_422P9_BE:
|
2010-04-15 07:39:36 +02:00
|
|
|
case IMGFMT_420P16_LE:
|
|
|
|
case IMGFMT_420P16_BE:
|
2012-11-27 10:39:09 +01:00
|
|
|
case IMGFMT_420P14_LE:
|
|
|
|
case IMGFMT_420P14_BE:
|
|
|
|
case IMGFMT_420P12_LE:
|
|
|
|
case IMGFMT_420P12_BE:
|
2011-06-26 00:22:53 +02:00
|
|
|
case IMGFMT_420P10_LE:
|
|
|
|
case IMGFMT_420P10_BE:
|
|
|
|
case IMGFMT_420P9_LE:
|
|
|
|
case IMGFMT_420P9_BE:
|
2010-04-15 07:39:36 +02:00
|
|
|
return;
|
|
|
|
case IMGFMT_Y800:
|
|
|
|
case IMGFMT_Y8:
|
2012-11-14 11:19:04 +01:00
|
|
|
case IMGFMT_Y16LE:
|
|
|
|
case IMGFMT_Y16BE:
|
2010-04-15 07:39:36 +02:00
|
|
|
/* they're planar ones, but for easier handling use them as packed */
|
2010-05-09 02:18:26 +02:00
|
|
|
mpi->flags&=~MP_IMGFLAG_PLANAR;
|
2010-04-15 07:39:36 +02:00
|
|
|
mpi->num_planes=1;
|
|
|
|
return;
|
|
|
|
case IMGFMT_UYVY:
|
|
|
|
mpi->flags|=MP_IMGFLAG_SWAPPED;
|
|
|
|
case IMGFMT_YUY2:
|
2011-11-22 20:31:29 +01:00
|
|
|
mpi->chroma_x_shift = 1;
|
2012-10-21 17:08:54 +02:00
|
|
|
mpi->chroma_y_shift = 1;
|
|
|
|
mpi->chroma_width=(mpi->width>>1);
|
|
|
|
mpi->chroma_height=(mpi->height>>1);
|
2010-04-15 07:39:36 +02: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 20:46:01 +02:00
|
|
|
static int mp_image_destructor(void *ptr)
|
|
|
|
{
|
|
|
|
mp_image_t *mpi = ptr;
|
|
|
|
|
2012-12-12 00:43:36 +01:00
|
|
|
if (mpi->refcount) {
|
|
|
|
m_refcount_unref(mpi->refcount);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mpi->flags & MP_IMGFLAG_ALLOCATED) {
|
2011-10-06 20:46:01 +02:00
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
|
2012-12-12 00:43:36 +01:00
|
|
|
// Image without format or allocated image data
|
|
|
|
struct mp_image *mp_image_new_empty(int w, int h)
|
|
|
|
{
|
|
|
|
struct mp_image *mpi = talloc_zero(NULL, struct mp_image);
|
2011-10-06 20:46:01 +02:00
|
|
|
talloc_set_destructor(mpi, mp_image_destructor);
|
2010-04-15 07:39:36 +02:00
|
|
|
mpi->width=mpi->w=w;
|
|
|
|
mpi->height=mpi->h=h;
|
|
|
|
return mpi;
|
|
|
|
}
|
|
|
|
|
2012-12-12 00:43:36 +01:00
|
|
|
struct mp_image *mp_image_alloc(unsigned int imgfmt, int w, int h)
|
|
|
|
{
|
|
|
|
struct mp_image *mpi = mp_image_new_empty(w, h);
|
|
|
|
|
|
|
|
mpi->width = FFALIGN(w, MP_STRIDE_ALIGNMENT);
|
|
|
|
mp_image_setfmt(mpi, imgfmt);
|
|
|
|
mp_image_alloc_planes(mpi);
|
|
|
|
mpi->width = w;
|
|
|
|
mp_image_setfmt(mpi, imgfmt); // reset chroma size
|
|
|
|
|
|
|
|
mpi->flags &= ~MP_IMGFLAG_ALLOCATED;
|
|
|
|
mpi->refcount = m_refcount_new();
|
|
|
|
mpi->refcount->free = av_free;
|
|
|
|
mpi->refcount->arg = mpi->planes[0];
|
|
|
|
// NOTE: palette isn't free'd. Palette handling should be fixed instead.
|
|
|
|
|
|
|
|
return mpi;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct mp_image *mp_image_new_copy(struct mp_image *img)
|
|
|
|
{
|
|
|
|
struct mp_image *new = mp_image_alloc(img->imgfmt, img->w, img->h);
|
|
|
|
mp_image_copy(new, img);
|
|
|
|
mp_image_copy_attributes(new, img);
|
|
|
|
|
|
|
|
// Normally these are covered by the reference to the original image data
|
|
|
|
// (like the AVFrame in vd_lavc.c), but we can't manage it on our own.
|
|
|
|
new->qscale = NULL;
|
|
|
|
new->qstride = 0;
|
|
|
|
|
|
|
|
return new;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make dst take over the image data of src, and free src.
|
|
|
|
// This is basically a safe version of *dst = *src; free(src);
|
|
|
|
// Only works with ref-counted images, and can't change image size/format.
|
|
|
|
void mp_image_steal_data(struct mp_image *dst, struct mp_image *src)
|
|
|
|
{
|
|
|
|
assert(dst->imgfmt == src->imgfmt && dst->w == src->w && dst->h == src->h);
|
|
|
|
assert(dst->refcount && src->refcount);
|
|
|
|
|
|
|
|
for (int p = 0; p < MP_MAX_PLANES; p++) {
|
|
|
|
dst->planes[p] = src->planes[p];
|
|
|
|
dst->stride[p] = src->stride[p];
|
|
|
|
}
|
|
|
|
mp_image_copy_attributes(dst, src);
|
|
|
|
|
|
|
|
m_refcount_unref(dst->refcount);
|
|
|
|
dst->refcount = src->refcount;
|
|
|
|
talloc_set_destructor(src, NULL);
|
|
|
|
talloc_free(src);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return a new reference to img. The returned reference is owned by the caller,
|
|
|
|
// while img is left untouched.
|
|
|
|
struct mp_image *mp_image_new_ref(struct mp_image *img)
|
|
|
|
{
|
|
|
|
if (!img->refcount)
|
|
|
|
return mp_image_new_copy(img);
|
|
|
|
|
|
|
|
struct mp_image *new = talloc_ptrtype(NULL, new);
|
|
|
|
talloc_set_destructor(new, mp_image_destructor);
|
|
|
|
*new = *img;
|
|
|
|
|
|
|
|
m_refcount_ref(new->refcount);
|
|
|
|
return new;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return a reference counted reference to img. If the reference count reaches
|
|
|
|
// 0, call free(free_arg). The data passed by img must not be free'd before
|
|
|
|
// that. The new reference will be writeable.
|
|
|
|
struct mp_image *mp_image_new_custom_ref(struct mp_image *img, void *free_arg,
|
|
|
|
void (*free)(void *arg))
|
|
|
|
{
|
|
|
|
struct mp_image *new = talloc_ptrtype(NULL, new);
|
|
|
|
talloc_set_destructor(new, mp_image_destructor);
|
|
|
|
*new = *img;
|
|
|
|
|
|
|
|
new->flags &= ~MP_IMGFLAG_ALLOCATED;
|
|
|
|
new->refcount = m_refcount_new();
|
|
|
|
new->refcount->free = free;
|
|
|
|
new->refcount->arg = free_arg;
|
|
|
|
return new;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return a reference counted reference to img. ref/unref/is_unique are used to
|
|
|
|
// connect to an external refcounting API. It is assumed that the new object
|
|
|
|
// has an initial reference to that external API.
|
|
|
|
struct mp_image *mp_image_new_external_ref(struct mp_image *img, void *arg,
|
|
|
|
void (*ref)(void *arg),
|
|
|
|
void (*unref)(void *arg),
|
|
|
|
bool (*is_unique)(void *arg))
|
|
|
|
{
|
|
|
|
struct mp_image *new = talloc_ptrtype(NULL, new);
|
|
|
|
talloc_set_destructor(new, mp_image_destructor);
|
|
|
|
*new = *img;
|
|
|
|
|
|
|
|
new->flags &= ~MP_IMGFLAG_ALLOCATED;
|
|
|
|
new->refcount = m_refcount_new();
|
|
|
|
new->refcount->ext_ref = ref;
|
|
|
|
new->refcount->ext_unref = unref;
|
|
|
|
new->refcount->ext_is_unique = is_unique;
|
|
|
|
new->refcount->arg = arg;
|
|
|
|
return new;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool mp_image_is_writeable(struct mp_image *img)
|
|
|
|
{
|
|
|
|
// if non ref-counted, it's writeable if the caller allocated the image
|
|
|
|
if (!img->refcount)
|
|
|
|
return img->flags & MP_IMGFLAG_ALLOCATED;
|
|
|
|
return m_refcount_is_unique(img->refcount);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make the image data referenced by img writeable. This allocates new data
|
|
|
|
// if the data wasn't already writeable, and img->planes[] and img->stride[]
|
|
|
|
// will be set to the copy.
|
|
|
|
void mp_image_make_writeable(struct mp_image *img)
|
|
|
|
{
|
|
|
|
if (mp_image_is_writeable(img))
|
|
|
|
return;
|
|
|
|
|
|
|
|
mp_image_steal_data(img, mp_image_new_copy(img));
|
|
|
|
assert(mp_image_is_writeable(img));
|
|
|
|
}
|
|
|
|
|
|
|
|
void mp_image_setrefp(struct mp_image **p_img, struct mp_image *new_value)
|
|
|
|
{
|
|
|
|
if (*p_img != new_value) {
|
|
|
|
talloc_free(*p_img);
|
|
|
|
*p_img = new_value ? mp_image_new_ref(new_value) : NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mere helper function (mp_image can be directly free'd with talloc_free)
|
|
|
|
void mp_image_unrefp(struct mp_image **p_img)
|
|
|
|
{
|
|
|
|
talloc_free(*p_img);
|
|
|
|
*p_img = NULL;
|
2010-04-15 07:39:36 +02:00
|
|
|
}
|
|
|
|
|
2012-10-27 18:01:51 +02: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;
|
|
|
|
}
|
|
|
|
}
|