mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-04-14 00:58:38 +02:00
utvideoenc: Avoid writing into the input picture
Signed-off-by: Michael Niedermayer <michaelni@gmx.at>
This commit is contained in:
parent
11d957fbd8
commit
bbefd27e52
@ -75,7 +75,7 @@ typedef struct UtvideoContext {
|
|||||||
int interlaced;
|
int interlaced;
|
||||||
int frame_pred;
|
int frame_pred;
|
||||||
|
|
||||||
uint8_t *slice_bits, *slice_buffer;
|
uint8_t *slice_bits, *slice_buffer[4];
|
||||||
int slice_bits_size;
|
int slice_bits_size;
|
||||||
} UtvideoContext;
|
} UtvideoContext;
|
||||||
|
|
||||||
|
@ -44,10 +44,12 @@ static int huff_cmp_sym(const void *a, const void *b)
|
|||||||
static av_cold int utvideo_encode_close(AVCodecContext *avctx)
|
static av_cold int utvideo_encode_close(AVCodecContext *avctx)
|
||||||
{
|
{
|
||||||
UtvideoContext *c = avctx->priv_data;
|
UtvideoContext *c = avctx->priv_data;
|
||||||
|
int i;
|
||||||
|
|
||||||
av_freep(&avctx->coded_frame);
|
av_freep(&avctx->coded_frame);
|
||||||
av_freep(&c->slice_bits);
|
av_freep(&c->slice_bits);
|
||||||
av_freep(&c->slice_buffer);
|
for (i = 0; i < 4; i++)
|
||||||
|
av_freep(&c->slice_buffer[i]);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -55,7 +57,7 @@ static av_cold int utvideo_encode_close(AVCodecContext *avctx)
|
|||||||
static av_cold int utvideo_encode_init(AVCodecContext *avctx)
|
static av_cold int utvideo_encode_init(AVCodecContext *avctx)
|
||||||
{
|
{
|
||||||
UtvideoContext *c = avctx->priv_data;
|
UtvideoContext *c = avctx->priv_data;
|
||||||
|
int i;
|
||||||
uint32_t original_format;
|
uint32_t original_format;
|
||||||
|
|
||||||
c->avctx = avctx;
|
c->avctx = avctx;
|
||||||
@ -142,13 +144,14 @@ static av_cold int utvideo_encode_init(AVCodecContext *avctx)
|
|||||||
return AVERROR(ENOMEM);
|
return AVERROR(ENOMEM);
|
||||||
}
|
}
|
||||||
|
|
||||||
c->slice_buffer = av_malloc(avctx->width * avctx->height +
|
for (i = 0; i < c->planes; i++) {
|
||||||
FF_INPUT_BUFFER_PADDING_SIZE);
|
c->slice_buffer[i] = av_malloc(avctx->width * (avctx->height + 1) +
|
||||||
|
FF_INPUT_BUFFER_PADDING_SIZE);
|
||||||
if (!c->slice_buffer) {
|
if (!c->slice_buffer[i]) {
|
||||||
av_log(avctx, AV_LOG_ERROR, "Cannot allocate temporary buffer 1.\n");
|
av_log(avctx, AV_LOG_ERROR, "Cannot allocate temporary buffer 1.\n");
|
||||||
utvideo_encode_close(avctx);
|
utvideo_encode_close(avctx);
|
||||||
return AVERROR(ENOMEM);
|
return AVERROR(ENOMEM);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -193,20 +196,33 @@ static av_cold int utvideo_encode_init(AVCodecContext *avctx)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void mangle_rgb_planes(uint8_t *src, int step, int stride, int width,
|
static void mangle_rgb_planes(uint8_t *dst[4], uint8_t *src, int step,
|
||||||
int height)
|
int stride, int width, int height)
|
||||||
{
|
{
|
||||||
int i, j;
|
int i, j;
|
||||||
uint8_t r, g, b;
|
int k = width;
|
||||||
|
unsigned int g;
|
||||||
|
|
||||||
for (j = 0; j < height; j++) {
|
for (j = 0; j < height; j++) {
|
||||||
for (i = 0; i < width * step; i += step) {
|
if (step == 3) {
|
||||||
r = src[i];
|
for (i = 0; i < width * step; i += step) {
|
||||||
g = src[i + 1];
|
g = src[i + 1];
|
||||||
b = src[i + 2];
|
dst[0][k] = g;
|
||||||
|
g += 0x80;
|
||||||
src[i] = r - g + 0x80;
|
dst[1][k] = src[i + 2] - g;
|
||||||
src[i + 2] = b - g + 0x80;
|
dst[2][k] = src[i + 0] - g;
|
||||||
|
k++;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (i = 0; i < width * step; i += step) {
|
||||||
|
g = src[i + 1];
|
||||||
|
dst[0][k] = g;
|
||||||
|
g += 0x80;
|
||||||
|
dst[1][k] = src[i + 2] - g;
|
||||||
|
dst[2][k] = src[i + 0] - g;
|
||||||
|
dst[3][k] = src[i + 3];
|
||||||
|
k++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
src += stride;
|
src += stride;
|
||||||
}
|
}
|
||||||
@ -535,16 +551,16 @@ static int utvideo_encode_frame(AVCodecContext *avctx, AVPacket *pkt,
|
|||||||
|
|
||||||
/* In case of RGB, mangle the planes to Ut Video's format */
|
/* In case of RGB, mangle the planes to Ut Video's format */
|
||||||
if (avctx->pix_fmt == PIX_FMT_RGBA || avctx->pix_fmt == PIX_FMT_RGB24)
|
if (avctx->pix_fmt == PIX_FMT_RGBA || avctx->pix_fmt == PIX_FMT_RGB24)
|
||||||
mangle_rgb_planes(pic->data[0], c->planes, pic->linesize[0], width,
|
mangle_rgb_planes(c->slice_buffer, pic->data[0], c->planes,
|
||||||
height);
|
pic->linesize[0], width, height);
|
||||||
|
|
||||||
/* Deal with the planes */
|
/* Deal with the planes */
|
||||||
switch (avctx->pix_fmt) {
|
switch (avctx->pix_fmt) {
|
||||||
case PIX_FMT_RGB24:
|
case PIX_FMT_RGB24:
|
||||||
case PIX_FMT_RGBA:
|
case PIX_FMT_RGBA:
|
||||||
for (i = 0; i < c->planes; i++) {
|
for (i = 0; i < c->planes; i++) {
|
||||||
ret = encode_plane(avctx, pic->data[0] + ff_ut_rgb_order[i],
|
ret = encode_plane(avctx, c->slice_buffer[i] + width,
|
||||||
c->slice_buffer, c->planes, pic->linesize[0],
|
c->slice_buffer[i], 1, width,
|
||||||
width, height, &pb);
|
width, height, &pb);
|
||||||
|
|
||||||
if (ret) {
|
if (ret) {
|
||||||
@ -555,7 +571,7 @@ static int utvideo_encode_frame(AVCodecContext *avctx, AVPacket *pkt,
|
|||||||
break;
|
break;
|
||||||
case PIX_FMT_YUV422P:
|
case PIX_FMT_YUV422P:
|
||||||
for (i = 0; i < c->planes; i++) {
|
for (i = 0; i < c->planes; i++) {
|
||||||
ret = encode_plane(avctx, pic->data[i], c->slice_buffer, 1,
|
ret = encode_plane(avctx, pic->data[i], c->slice_buffer[0], 1,
|
||||||
pic->linesize[i], width >> !!i, height, &pb);
|
pic->linesize[i], width >> !!i, height, &pb);
|
||||||
|
|
||||||
if (ret) {
|
if (ret) {
|
||||||
@ -566,7 +582,7 @@ static int utvideo_encode_frame(AVCodecContext *avctx, AVPacket *pkt,
|
|||||||
break;
|
break;
|
||||||
case PIX_FMT_YUV420P:
|
case PIX_FMT_YUV420P:
|
||||||
for (i = 0; i < c->planes; i++) {
|
for (i = 0; i < c->planes; i++) {
|
||||||
ret = encode_plane(avctx, pic->data[i], c->slice_buffer, 1,
|
ret = encode_plane(avctx, pic->data[i], c->slice_buffer[0], 1,
|
||||||
pic->linesize[i], width >> !!i, height >> !!i,
|
pic->linesize[i], width >> !!i, height >> !!i,
|
||||||
&pb);
|
&pb);
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user