1
0
mirror of https://github.com/FFmpeg/FFmpeg.git synced 2025-01-03 05:10:03 +02:00

libavcodec/j2kenc: Encoding up to 16 bits

This patch allows the JPEG2000 encoder to
encode images to up to 16 bits.

Signed-off-by: Michael Niedermayer <michael@niedermayer.cc>
This commit is contained in:
Gautam Ramakrishnan 2020-07-29 21:19:10 +05:30 committed by Michael Niedermayer
parent bd4f37f2eb
commit 9d302efdf2

View File

@ -298,7 +298,7 @@ static int put_siz(Jpeg2000EncoderContext *s)
bytestream_put_be16(&s->buf, s->ncomponents); // CSiz bytestream_put_be16(&s->buf, s->ncomponents); // CSiz
for (i = 0; i < s->ncomponents; i++){ // Ssiz_i XRsiz_i, YRsiz_i for (i = 0; i < s->ncomponents; i++){ // Ssiz_i XRsiz_i, YRsiz_i
bytestream_put_byte(&s->buf, 7); bytestream_put_byte(&s->buf, s->cbps[i] - 1);
bytestream_put_byte(&s->buf, i?1<<s->chroma_shift[0]:1); bytestream_put_byte(&s->buf, i?1<<s->chroma_shift[0]:1);
bytestream_put_byte(&s->buf, i?1<<s->chroma_shift[1]:1); bytestream_put_byte(&s->buf, i?1<<s->chroma_shift[1]:1);
} }
@ -447,43 +447,49 @@ static int init_tiles(Jpeg2000EncoderContext *s)
return 0; return 0;
} }
static void copy_frame(Jpeg2000EncoderContext *s) #define COPY_FRAME(D, PIXEL) \
{ static void copy_frame_ ##D(Jpeg2000EncoderContext *s) \
int tileno, compno, i, y, x; { \
uint8_t *line; int tileno, compno, i, y, x; \
for (tileno = 0; tileno < s->numXtiles * s->numYtiles; tileno++){ PIXEL *line; \
Jpeg2000Tile *tile = s->tile + tileno; for (tileno = 0; tileno < s->numXtiles * s->numYtiles; tileno++){ \
if (s->planar){ Jpeg2000Tile *tile = s->tile + tileno; \
for (compno = 0; compno < s->ncomponents; compno++){ if (s->planar){ \
Jpeg2000Component *comp = tile->comp + compno; for (compno = 0; compno < s->ncomponents; compno++){ \
int *dst = comp->i_data; Jpeg2000Component *comp = tile->comp + compno; \
line = s->picture->data[compno] int *dst = comp->i_data; \
+ comp->coord[1][0] * s->picture->linesize[compno] int cbps = s->cbps[compno]; \
+ comp->coord[0][0]; line = (PIXEL*)s->picture->data[compno] \
for (y = comp->coord[1][0]; y < comp->coord[1][1]; y++){ + comp->coord[1][0] * (s->picture->linesize[compno] / sizeof(PIXEL)) \
uint8_t *ptr = line; + comp->coord[0][0]; \
for (x = comp->coord[0][0]; x < comp->coord[0][1]; x++) for (y = comp->coord[1][0]; y < comp->coord[1][1]; y++){ \
*dst++ = *ptr++ - (1 << 7); PIXEL *ptr = line; \
line += s->picture->linesize[compno]; for (x = comp->coord[0][0]; x < comp->coord[0][1]; x++) \
*dst++ = *ptr++ - (1 << (cbps - 1)); \
line += s->picture->linesize[compno] / sizeof(PIXEL); \
} \
} \
} else{ \
line = (PIXEL*)s->picture->data[0] + tile->comp[0].coord[1][0] * (s->picture->linesize[0] / sizeof(PIXEL)) \
+ tile->comp[0].coord[0][0] * s->ncomponents; \
\
i = 0; \
for (y = tile->comp[0].coord[1][0]; y < tile->comp[0].coord[1][1]; y++){ \
PIXEL *ptr = line; \
for (x = tile->comp[0].coord[0][0]; x < tile->comp[0].coord[0][1]; x++, i++){ \
for (compno = 0; compno < s->ncomponents; compno++){ \
int cbps = s->cbps[compno]; \
tile->comp[compno].i_data[i] = *ptr++ - (1 << (cbps - 1)); \
} \
} \
line += s->picture->linesize[0] / sizeof(PIXEL); \
} \
} \
} \
} }
}
} else{
line = s->picture->data[0] + tile->comp[0].coord[1][0] * s->picture->linesize[0]
+ tile->comp[0].coord[0][0] * s->ncomponents;
i = 0; COPY_FRAME(8, uint8_t)
for (y = tile->comp[0].coord[1][0]; y < tile->comp[0].coord[1][1]; y++){ COPY_FRAME(16, uint16_t)
uint8_t *ptr = line;
for (x = tile->comp[0].coord[0][0]; x < tile->comp[0].coord[0][1]; x++, i++){
for (compno = 0; compno < s->ncomponents; compno++){
tile->comp[compno].i_data[i] = *ptr++ - (1 << 7);
}
}
line += s->picture->linesize[0];
}
}
}
}
static void init_quantization(Jpeg2000EncoderContext *s) static void init_quantization(Jpeg2000EncoderContext *s)
{ {
@ -1015,7 +1021,11 @@ static int encode_frame(AVCodecContext *avctx, AVPacket *pkt,
s->lambda = s->picture->quality * LAMBDA_SCALE; s->lambda = s->picture->quality * LAMBDA_SCALE;
copy_frame(s); if (avctx->pix_fmt == AV_PIX_FMT_BGR48 || avctx->pix_fmt == AV_PIX_FMT_GRAY16)
copy_frame_16(s);
else
copy_frame_8(s);
reinit(s); reinit(s);
if (s->format == CODEC_JP2) { if (s->format == CODEC_JP2) {
@ -1180,12 +1190,16 @@ FF_ENABLE_DEPRECATION_WARNINGS
s->width = avctx->width; s->width = avctx->width;
s->height = avctx->height; s->height = avctx->height;
for (i = 0; i < 3; i++) for (i = 0; i < 3; i++) {
if (avctx->pix_fmt == AV_PIX_FMT_GRAY16 || avctx->pix_fmt == AV_PIX_FMT_RGB48)
s->cbps[i] = 16;
else
s->cbps[i] = 8; s->cbps[i] = 8;
}
if (avctx->pix_fmt == AV_PIX_FMT_RGB24){ if (avctx->pix_fmt == AV_PIX_FMT_RGB24 || avctx->pix_fmt == AV_PIX_FMT_RGB48){
s->ncomponents = 3; s->ncomponents = 3;
} else if (avctx->pix_fmt == AV_PIX_FMT_GRAY8 || avctx->pix_fmt == AV_PIX_FMT_PAL8){ } else if (avctx->pix_fmt == AV_PIX_FMT_GRAY8 || avctx->pix_fmt == AV_PIX_FMT_PAL8 || avctx->pix_fmt == AV_PIX_FMT_GRAY16){
s->ncomponents = 1; s->ncomponents = 1;
} else{ // planar YUV } else{ // planar YUV
s->planar = 1; s->planar = 1;
@ -1255,6 +1269,7 @@ AVCodec ff_jpeg2000_encoder = {
AV_PIX_FMT_YUV420P, AV_PIX_FMT_YUV422P, AV_PIX_FMT_YUV420P, AV_PIX_FMT_YUV422P,
AV_PIX_FMT_YUV410P, AV_PIX_FMT_YUV411P, AV_PIX_FMT_YUV410P, AV_PIX_FMT_YUV411P,
AV_PIX_FMT_PAL8, AV_PIX_FMT_PAL8,
AV_PIX_FMT_RGB48, AV_PIX_FMT_GRAY16,
AV_PIX_FMT_NONE AV_PIX_FMT_NONE
}, },
.priv_class = &j2k_class, .priv_class = &j2k_class,