mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-01-24 13:56:33 +02:00
Merge commit '62256010e9bc8879e2bf7f3b94af8ff85e239082'
* commit '62256010e9bc8879e2bf7f3b94af8ff85e239082': indeo: Refactor ff_ivi_init_tiles and ivi_decode_blocks Conflicts: libavcodec/ivi_common.c Merged-by: Michael Niedermayer <michaelni@gmx.at>
This commit is contained in:
commit
9b10440dcd
@ -288,11 +288,50 @@ av_cold int ff_ivi_init_planes(IVIPlaneDesc *planes, const IVIPicConfig *cfg)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
av_cold int ff_ivi_init_tiles(IVIPlaneDesc *planes, int tile_width, int tile_height)
|
static int ivi_init_tiles(IVIBandDesc *band, IVITile *ref_tile,
|
||||||
|
int p, int b, int t_height, int t_width)
|
||||||
{
|
{
|
||||||
int p, b, x, y, x_tiles, y_tiles, t_width, t_height;
|
int x, y;
|
||||||
|
IVITile *tile = band->tiles;
|
||||||
|
|
||||||
|
for (y = 0; y < band->height; y += t_height) {
|
||||||
|
for (x = 0; x < band->width; x += t_width) {
|
||||||
|
tile->xpos = x;
|
||||||
|
tile->ypos = y;
|
||||||
|
tile->mb_size = band->mb_size;
|
||||||
|
tile->width = FFMIN(band->width - x, t_width);
|
||||||
|
tile->height = FFMIN(band->height - y, t_height);
|
||||||
|
tile->is_empty = tile->data_size = 0;
|
||||||
|
/* calculate number of macroblocks */
|
||||||
|
tile->num_MBs = IVI_MBs_PER_TILE(tile->width, tile->height,
|
||||||
|
band->mb_size);
|
||||||
|
|
||||||
|
av_freep(&tile->mbs);
|
||||||
|
tile->mbs = av_malloc(tile->num_MBs * sizeof(IVIMbInfo));
|
||||||
|
if (!tile->mbs)
|
||||||
|
return AVERROR(ENOMEM);
|
||||||
|
|
||||||
|
tile->ref_mbs = 0;
|
||||||
|
if (p || b) {
|
||||||
|
if (tile->num_MBs <= ref_tile->num_MBs) {
|
||||||
|
tile->ref_mbs = ref_tile->mbs;
|
||||||
|
}else
|
||||||
|
av_log(NULL, AV_LOG_DEBUG, "Cannot use ref_tile, too few mbs\n");
|
||||||
|
|
||||||
|
ref_tile++;
|
||||||
|
}
|
||||||
|
tile++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
av_cold int ff_ivi_init_tiles(IVIPlaneDesc *planes,
|
||||||
|
int tile_width, int tile_height)
|
||||||
|
{
|
||||||
|
int p, b, x_tiles, y_tiles, t_width, t_height, ret;
|
||||||
IVIBandDesc *band;
|
IVIBandDesc *band;
|
||||||
IVITile *tile, *ref_tile;
|
|
||||||
|
|
||||||
for (p = 0; p < 3; p++) {
|
for (p = 0; p < 3; p++) {
|
||||||
t_width = !p ? tile_width : (tile_width + 3) >> 2;
|
t_width = !p ? tile_width : (tile_width + 3) >> 2;
|
||||||
@ -316,44 +355,14 @@ av_cold int ff_ivi_init_tiles(IVIPlaneDesc *planes, int tile_width, int tile_hei
|
|||||||
if (!band->tiles)
|
if (!band->tiles)
|
||||||
return AVERROR(ENOMEM);
|
return AVERROR(ENOMEM);
|
||||||
|
|
||||||
tile = band->tiles;
|
|
||||||
|
|
||||||
/* use the first luma band as reference for motion vectors
|
/* use the first luma band as reference for motion vectors
|
||||||
* and quant */
|
* and quant */
|
||||||
ref_tile = planes[0].bands[0].tiles;
|
ret = ivi_init_tiles(band, planes[0].bands[0].tiles,
|
||||||
|
p, b, t_height, t_width);
|
||||||
for (y = 0; y < band->height; y += t_height) {
|
if (ret < 0)
|
||||||
for (x = 0; x < band->width; x += t_width) {
|
return ret;
|
||||||
tile->xpos = x;
|
}
|
||||||
tile->ypos = y;
|
}
|
||||||
tile->mb_size = band->mb_size;
|
|
||||||
tile->width = FFMIN(band->width - x, t_width);
|
|
||||||
tile->height = FFMIN(band->height - y, t_height);
|
|
||||||
tile->is_empty = tile->data_size = 0;
|
|
||||||
/* calculate number of macroblocks */
|
|
||||||
tile->num_MBs = IVI_MBs_PER_TILE(tile->width, tile->height,
|
|
||||||
band->mb_size);
|
|
||||||
|
|
||||||
av_freep(&tile->mbs);
|
|
||||||
tile->mbs = av_malloc(tile->num_MBs * sizeof(IVIMbInfo));
|
|
||||||
if (!tile->mbs)
|
|
||||||
return AVERROR(ENOMEM);
|
|
||||||
|
|
||||||
tile->ref_mbs = 0;
|
|
||||||
if (p || b) {
|
|
||||||
if (tile->num_MBs <= ref_tile->num_MBs) {
|
|
||||||
tile->ref_mbs = ref_tile->mbs;
|
|
||||||
}else
|
|
||||||
av_log(NULL, AV_LOG_DEBUG, "Cannot use ref_tile, too few mbs\n");
|
|
||||||
ref_tile++;
|
|
||||||
}
|
|
||||||
|
|
||||||
tile++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}// for b
|
|
||||||
}// for p
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -385,6 +394,99 @@ static int ivi_dec_tile_data_size(GetBitContext *gb)
|
|||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int ivi_decode_coded_blocks(GetBitContext *gb, IVIBandDesc *band,
|
||||||
|
ivi_mc_func mc, int mv_x, int mv_y,
|
||||||
|
int *prev_dc, int is_intra, int mc_type,
|
||||||
|
uint32_t quant, int offs,
|
||||||
|
AVCodecContext *avctx)
|
||||||
|
{
|
||||||
|
const uint16_t *base_tab = is_intra ? band->intra_base : band->inter_base;
|
||||||
|
RVMapDesc *rvmap = band->rv_map;
|
||||||
|
uint8_t col_flags[8];
|
||||||
|
int32_t trvec[64];
|
||||||
|
uint32_t sym = 0, lo, hi, q;
|
||||||
|
int pos, run, val;
|
||||||
|
int blk_size = band->blk_size;
|
||||||
|
int num_coeffs = blk_size * blk_size;
|
||||||
|
int col_mask = blk_size - 1;
|
||||||
|
int scan_pos = -1;
|
||||||
|
|
||||||
|
if (!band->scan) {
|
||||||
|
av_log(avctx, AV_LOG_ERROR, "Scan pattern is not set.\n");
|
||||||
|
return AVERROR_INVALIDDATA;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* zero transform vector */
|
||||||
|
memset(trvec, 0, num_coeffs * sizeof(trvec[0]));
|
||||||
|
/* zero column flags */
|
||||||
|
memset(col_flags, 0, sizeof(col_flags));
|
||||||
|
while (scan_pos <= num_coeffs) {
|
||||||
|
sym = get_vlc2(gb, band->blk_vlc.tab->table,
|
||||||
|
IVI_VLC_BITS, 1);
|
||||||
|
if (sym == rvmap->eob_sym)
|
||||||
|
break; /* End of block */
|
||||||
|
|
||||||
|
/* Escape - run/val explicitly coded using 3 vlc codes */
|
||||||
|
if (sym == rvmap->esc_sym) {
|
||||||
|
run = get_vlc2(gb, band->blk_vlc.tab->table, IVI_VLC_BITS, 1) + 1;
|
||||||
|
lo = get_vlc2(gb, band->blk_vlc.tab->table, IVI_VLC_BITS, 1);
|
||||||
|
hi = get_vlc2(gb, band->blk_vlc.tab->table, IVI_VLC_BITS, 1);
|
||||||
|
/* merge them and convert into signed val */
|
||||||
|
val = IVI_TOSIGNED((hi << 6) | lo);
|
||||||
|
} else {
|
||||||
|
if (sym >= 256U) {
|
||||||
|
av_log(avctx, AV_LOG_ERROR, "Invalid sym encountered: %d.\n", sym);
|
||||||
|
return AVERROR_INVALIDDATA;
|
||||||
|
}
|
||||||
|
run = rvmap->runtab[sym];
|
||||||
|
val = rvmap->valtab[sym];
|
||||||
|
}
|
||||||
|
|
||||||
|
/* de-zigzag and dequantize */
|
||||||
|
scan_pos += run;
|
||||||
|
if (scan_pos >= (unsigned)num_coeffs)
|
||||||
|
break;
|
||||||
|
pos = band->scan[scan_pos];
|
||||||
|
|
||||||
|
if (!val)
|
||||||
|
av_dlog(avctx, "Val = 0 encountered!\n");
|
||||||
|
|
||||||
|
q = (base_tab[pos] * quant) >> 9;
|
||||||
|
if (q > 1)
|
||||||
|
val = val * q + FFSIGN(val) * (((q ^ 1) - 1) >> 1);
|
||||||
|
trvec[pos] = val;
|
||||||
|
/* track columns containing non-zero coeffs */
|
||||||
|
col_flags[pos & col_mask] |= !!val;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (scan_pos >= num_coeffs && sym != rvmap->eob_sym)
|
||||||
|
return AVERROR_INVALIDDATA; /* corrupt block data */
|
||||||
|
|
||||||
|
/* undoing DC coeff prediction for intra-blocks */
|
||||||
|
if (is_intra && band->is_2d_trans) {
|
||||||
|
*prev_dc += trvec[0];
|
||||||
|
trvec[0] = *prev_dc;
|
||||||
|
col_flags[0] |= !!*prev_dc;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(band->transform_size > band->blk_size){
|
||||||
|
av_log(NULL, AV_LOG_ERROR, "Too large transform\n");
|
||||||
|
return AVERROR_INVALIDDATA;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* apply inverse transform */
|
||||||
|
band->inv_transform(trvec, band->buf + offs,
|
||||||
|
band->pitch, col_flags);
|
||||||
|
|
||||||
|
/* apply motion compensation */
|
||||||
|
if (!is_intra)
|
||||||
|
mc(band->buf + offs,
|
||||||
|
band->ref_buf + offs + mv_y * band->pitch + mv_x,
|
||||||
|
band->pitch, mc_type);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
/*
|
/*
|
||||||
* Decode block data:
|
* Decode block data:
|
||||||
* extract huffman-coded transform coefficients from the bitstream,
|
* extract huffman-coded transform coefficients from the bitstream,
|
||||||
@ -396,26 +498,22 @@ static int ivi_dec_tile_data_size(GetBitContext *gb)
|
|||||||
* @param[in] tile pointer to the tile descriptor
|
* @param[in] tile pointer to the tile descriptor
|
||||||
* @return result code: 0 - OK, -1 = error (corrupted blocks data)
|
* @return result code: 0 - OK, -1 = error (corrupted blocks data)
|
||||||
*/
|
*/
|
||||||
static int ivi_decode_blocks(GetBitContext *gb, IVIBandDesc *band, IVITile *tile,
|
static int ivi_decode_blocks(GetBitContext *gb, IVIBandDesc *band,
|
||||||
AVCodecContext *avctx)
|
IVITile *tile, AVCodecContext *avctx)
|
||||||
{
|
{
|
||||||
int mbn, blk, num_blocks, num_coeffs, blk_size, scan_pos, run, val,
|
int mbn, blk, num_blocks, blk_size, ret, is_intra, mc_type = 0;
|
||||||
pos, is_intra, mc_type = 0, av_uninit(mv_x), av_uninit(mv_y), col_mask;
|
int mv_x = 0, mv_y = 0;
|
||||||
uint8_t col_flags[8];
|
int32_t prev_dc;
|
||||||
int32_t prev_dc, trvec[64];
|
uint32_t cbp, quant, buf_offs;
|
||||||
uint32_t cbp, av_uninit(sym), lo, hi, quant, buf_offs, q;
|
IVIMbInfo *mb;
|
||||||
IVIMbInfo *mb;
|
|
||||||
RVMapDesc *rvmap = band->rv_map;
|
|
||||||
ivi_mc_func mc_with_delta_func, mc_no_delta_func;
|
ivi_mc_func mc_with_delta_func, mc_no_delta_func;
|
||||||
const uint16_t *base_tab;
|
const uint8_t *scale_tab;
|
||||||
const uint8_t *scale_tab;
|
|
||||||
|
|
||||||
prev_dc = 0; /* init intra prediction for the DC coefficient */
|
|
||||||
|
|
||||||
|
/* init intra prediction for the DC coefficient */
|
||||||
|
prev_dc = 0;
|
||||||
blk_size = band->blk_size;
|
blk_size = band->blk_size;
|
||||||
col_mask = blk_size - 1; /* column mask for tracking non-zero coeffs */
|
/* number of blocks per mb */
|
||||||
num_blocks = (band->mb_size != blk_size) ? 4 : 1; /* number of blocks per mb */
|
num_blocks = (band->mb_size != blk_size) ? 4 : 1;
|
||||||
num_coeffs = blk_size * blk_size;
|
|
||||||
if (blk_size == 8) {
|
if (blk_size == 8) {
|
||||||
mc_with_delta_func = ff_ivi_mc_8x8_delta;
|
mc_with_delta_func = ff_ivi_mc_8x8_delta;
|
||||||
mc_no_delta_func = ff_ivi_mc_8x8_no_delta;
|
mc_no_delta_func = ff_ivi_mc_8x8_no_delta;
|
||||||
@ -435,7 +533,6 @@ static int ivi_decode_blocks(GetBitContext *gb, IVIBandDesc *band, IVITile *tile
|
|||||||
else
|
else
|
||||||
quant = av_clip(quant, 0, 23);
|
quant = av_clip(quant, 0, 23);
|
||||||
|
|
||||||
base_tab = is_intra ? band->intra_base : band->inter_base;
|
|
||||||
scale_tab = is_intra ? band->intra_scale : band->inter_scale;
|
scale_tab = is_intra ? band->intra_scale : band->inter_scale;
|
||||||
if (scale_tab)
|
if (scale_tab)
|
||||||
quant = scale_tab[quant];
|
quant = scale_tab[quant];
|
||||||
@ -456,10 +553,10 @@ static int ivi_decode_blocks(GetBitContext *gb, IVIBandDesc *band, IVITile *tile
|
|||||||
cx = mb->mv_x & band->is_halfpel;
|
cx = mb->mv_x & band->is_halfpel;
|
||||||
cy = mb->mv_y & band->is_halfpel;
|
cy = mb->mv_y & band->is_halfpel;
|
||||||
|
|
||||||
if ( mb->xpos + dmv_x < 0
|
if (mb->xpos + dmv_x < 0 ||
|
||||||
|| mb->xpos + dmv_x + band->mb_size + cx > band->pitch
|
mb->xpos + dmv_x + band->mb_size + cx > band->pitch ||
|
||||||
|| mb->ypos + dmv_y < 0
|
mb->ypos + dmv_y < 0 ||
|
||||||
|| mb->ypos + dmv_y + band->mb_size + cy > band->aheight) {
|
mb->ypos + dmv_y + band->mb_size + cy > band->aheight) {
|
||||||
return AVERROR_INVALIDDATA;
|
return AVERROR_INVALIDDATA;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -475,72 +572,11 @@ static int ivi_decode_blocks(GetBitContext *gb, IVIBandDesc *band, IVITile *tile
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (cbp & 1) { /* block coded ? */
|
if (cbp & 1) { /* block coded ? */
|
||||||
if (!band->scan) {
|
ret = ivi_decode_coded_blocks(gb, band, mc_with_delta_func,
|
||||||
av_log(avctx, AV_LOG_ERROR, "Scan pattern is not set.\n");
|
mv_x, mv_y, &prev_dc, is_intra,
|
||||||
return AVERROR_INVALIDDATA;
|
mc_type, quant, buf_offs, avctx);
|
||||||
}
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
scan_pos = -1;
|
|
||||||
memset(trvec, 0, num_coeffs*sizeof(trvec[0])); /* zero transform vector */
|
|
||||||
memset(col_flags, 0, sizeof(col_flags)); /* zero column flags */
|
|
||||||
|
|
||||||
while (scan_pos <= num_coeffs) {
|
|
||||||
sym = get_vlc2(gb, band->blk_vlc.tab->table, IVI_VLC_BITS, 1);
|
|
||||||
if (sym == rvmap->eob_sym)
|
|
||||||
break; /* End of block */
|
|
||||||
|
|
||||||
if (sym == rvmap->esc_sym) { /* Escape - run/val explicitly coded using 3 vlc codes */
|
|
||||||
run = get_vlc2(gb, band->blk_vlc.tab->table, IVI_VLC_BITS, 1) + 1;
|
|
||||||
lo = get_vlc2(gb, band->blk_vlc.tab->table, IVI_VLC_BITS, 1);
|
|
||||||
hi = get_vlc2(gb, band->blk_vlc.tab->table, IVI_VLC_BITS, 1);
|
|
||||||
val = IVI_TOSIGNED((hi << 6) | lo); /* merge them and convert into signed val */
|
|
||||||
} else {
|
|
||||||
if (sym >= 256U) {
|
|
||||||
av_log(avctx, AV_LOG_ERROR, "Invalid sym encountered: %d.\n", sym);
|
|
||||||
return AVERROR_INVALIDDATA;
|
|
||||||
}
|
|
||||||
run = rvmap->runtab[sym];
|
|
||||||
val = rvmap->valtab[sym];
|
|
||||||
}
|
|
||||||
|
|
||||||
/* de-zigzag and dequantize */
|
|
||||||
scan_pos += run;
|
|
||||||
if (scan_pos >= (unsigned)num_coeffs)
|
|
||||||
break;
|
|
||||||
pos = band->scan[scan_pos];
|
|
||||||
|
|
||||||
if (!val)
|
|
||||||
av_dlog(avctx, "Val = 0 encountered!\n");
|
|
||||||
|
|
||||||
q = (base_tab[pos] * quant) >> 9;
|
|
||||||
if (q > 1)
|
|
||||||
val = val * q + FFSIGN(val) * (((q ^ 1) - 1) >> 1);
|
|
||||||
trvec[pos] = val;
|
|
||||||
col_flags[pos & col_mask] |= !!val; /* track columns containing non-zero coeffs */
|
|
||||||
}// while
|
|
||||||
|
|
||||||
if (scan_pos >= num_coeffs && sym != rvmap->eob_sym)
|
|
||||||
return AVERROR_INVALIDDATA; /* corrupt block data */
|
|
||||||
|
|
||||||
/* undoing DC coeff prediction for intra-blocks */
|
|
||||||
if (is_intra && band->is_2d_trans) {
|
|
||||||
prev_dc += trvec[0];
|
|
||||||
trvec[0] = prev_dc;
|
|
||||||
col_flags[0] |= !!prev_dc;
|
|
||||||
}
|
|
||||||
if(band->transform_size > band->blk_size){
|
|
||||||
av_log(NULL, AV_LOG_ERROR, "Too large transform\n");
|
|
||||||
return AVERROR_INVALIDDATA;
|
|
||||||
}
|
|
||||||
/* apply inverse transform */
|
|
||||||
band->inv_transform(trvec, band->buf + buf_offs,
|
|
||||||
band->pitch, col_flags);
|
|
||||||
|
|
||||||
/* apply motion compensation */
|
|
||||||
if (!is_intra)
|
|
||||||
mc_with_delta_func(band->buf + buf_offs,
|
|
||||||
band->ref_buf + buf_offs + mv_y * band->pitch + mv_x,
|
|
||||||
band->pitch, mc_type);
|
|
||||||
} else {
|
} else {
|
||||||
/* block not coded */
|
/* block not coded */
|
||||||
/* for intra blocks apply the dc slant transform */
|
/* for intra blocks apply the dc slant transform */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user