Also shorten HuffYUV context member names to avoid clutter.tags/n2.3
| @@ -1542,6 +1542,7 @@ CONFIG_EXTRA=" | |||||
| hpeldsp | hpeldsp | ||||
| huffman | huffman | ||||
| huffyuvdsp | huffyuvdsp | ||||
| huffyuvencdsp | |||||
| intrax8 | intrax8 | ||||
| lgplv3 | lgplv3 | ||||
| lpc | lpc | ||||
| @@ -1773,7 +1774,7 @@ h264_decoder_select="cabac golomb h264chroma h264dsp h264pred h264qpel videodsp" | |||||
| h264_decoder_suggest="error_resilience" | h264_decoder_suggest="error_resilience" | ||||
| hevc_decoder_select="cabac dsputil golomb videodsp" | hevc_decoder_select="cabac dsputil golomb videodsp" | ||||
| huffyuv_decoder_select="dsputil huffyuvdsp" | huffyuv_decoder_select="dsputil huffyuvdsp" | ||||
| huffyuv_encoder_select="dsputil huffman" | |||||
| huffyuv_encoder_select="dsputil huffman huffyuvencdsp" | |||||
| iac_decoder_select="imc_decoder" | iac_decoder_select="imc_decoder" | ||||
| imc_decoder_select="dsputil fft mdct sinewin" | imc_decoder_select="dsputil fft mdct sinewin" | ||||
| indeo3_decoder_select="hpeldsp" | indeo3_decoder_select="hpeldsp" | ||||
| @@ -1826,7 +1827,7 @@ on2avc_decoder_select="mdct" | |||||
| opus_decoder_deps="avresample" | opus_decoder_deps="avresample" | ||||
| png_decoder_deps="zlib" | png_decoder_deps="zlib" | ||||
| png_encoder_deps="zlib" | png_encoder_deps="zlib" | ||||
| png_encoder_select="dsputil" | |||||
| png_encoder_select="huffyuvencdsp" | |||||
| prores_decoder_select="dsputil" | prores_decoder_select="dsputil" | ||||
| prores_encoder_select="dsputil" | prores_encoder_select="dsputil" | ||||
| qcelp_decoder_select="lsp" | qcelp_decoder_select="lsp" | ||||
| @@ -1857,7 +1858,7 @@ truespeech_decoder_select="dsputil" | |||||
| tscc_decoder_deps="zlib" | tscc_decoder_deps="zlib" | ||||
| twinvq_decoder_select="mdct lsp sinewin" | twinvq_decoder_select="mdct lsp sinewin" | ||||
| utvideo_decoder_select="dsputil" | utvideo_decoder_select="dsputil" | ||||
| utvideo_encoder_select="dsputil huffman" | |||||
| utvideo_encoder_select="dsputil huffman huffyuvencdsp" | |||||
| vble_decoder_select="huffyuvdsp" | vble_decoder_select="huffyuvdsp" | ||||
| vc1_decoder_select="error_resilience h263_decoder h264chroma h264qpel intrax8" | vc1_decoder_select="error_resilience h263_decoder h264chroma h264qpel intrax8" | ||||
| vc1image_decoder_select="vc1_decoder" | vc1image_decoder_select="vc1_decoder" | ||||
| @@ -47,6 +47,7 @@ OBJS-$(CONFIG_H264QPEL) += h264qpel.o | |||||
| OBJS-$(CONFIG_HPELDSP) += hpeldsp.o | OBJS-$(CONFIG_HPELDSP) += hpeldsp.o | ||||
| OBJS-$(CONFIG_HUFFMAN) += huffman.o | OBJS-$(CONFIG_HUFFMAN) += huffman.o | ||||
| OBJS-$(CONFIG_HUFFYUVDSP) += huffyuvdsp.o | OBJS-$(CONFIG_HUFFYUVDSP) += huffyuvdsp.o | ||||
| OBJS-$(CONFIG_HUFFYUVENCDSP) += huffyuvencdsp.o | |||||
| OBJS-$(CONFIG_INTRAX8) += intrax8.o intrax8dsp.o | OBJS-$(CONFIG_INTRAX8) += intrax8.o intrax8dsp.o | ||||
| OBJS-$(CONFIG_LIBXVID) += libxvid_rc.o | OBJS-$(CONFIG_LIBXVID) += libxvid_rc.o | ||||
| OBJS-$(CONFIG_LPC) += lpc.o | OBJS-$(CONFIG_LPC) += lpc.o | ||||
| @@ -53,10 +53,6 @@ uint32_t ff_square_tab[512] = { 0, }; | |||||
| #include "dsputil_template.c" | #include "dsputil_template.c" | ||||
| #include "dsputilenc_template.c" | #include "dsputilenc_template.c" | ||||
| // 0x7f7f7f7f or 0x7f7f7f7f7f7f7f7f or whatever, depending on the cpu's native arithmetic size | |||||
| #define pb_7f (~0UL / 255 * 0x7f) | |||||
| #define pb_80 (~0UL / 255 * 0x80) | |||||
| const uint8_t ff_alternate_horizontal_scan[64] = { | const uint8_t ff_alternate_horizontal_scan[64] = { | ||||
| 0, 1, 2, 3, 8, 9, 16, 17, | 0, 1, 2, 3, 8, 9, 16, 17, | ||||
| 10, 11, 4, 5, 6, 7, 15, 14, | 10, 11, 4, 5, 6, 7, 15, 14, | ||||
| @@ -1663,55 +1659,6 @@ void ff_set_cmp(DSPContext *c, me_cmp_func *cmp, int type) | |||||
| } | } | ||||
| } | } | ||||
| static void diff_bytes_c(uint8_t *dst, uint8_t *src1, uint8_t *src2, int w) | |||||
| { | |||||
| long i; | |||||
| #if !HAVE_FAST_UNALIGNED | |||||
| if ((long) src2 & (sizeof(long) - 1)) { | |||||
| for (i = 0; i + 7 < w; i += 8) { | |||||
| dst[i + 0] = src1[i + 0] - src2[i + 0]; | |||||
| dst[i + 1] = src1[i + 1] - src2[i + 1]; | |||||
| dst[i + 2] = src1[i + 2] - src2[i + 2]; | |||||
| dst[i + 3] = src1[i + 3] - src2[i + 3]; | |||||
| dst[i + 4] = src1[i + 4] - src2[i + 4]; | |||||
| dst[i + 5] = src1[i + 5] - src2[i + 5]; | |||||
| dst[i + 6] = src1[i + 6] - src2[i + 6]; | |||||
| dst[i + 7] = src1[i + 7] - src2[i + 7]; | |||||
| } | |||||
| } else | |||||
| #endif | |||||
| for (i = 0; i <= w - (int) sizeof(long); i += sizeof(long)) { | |||||
| long a = *(long *) (src1 + i); | |||||
| long b = *(long *) (src2 + i); | |||||
| *(long *) (dst + i) = ((a | pb_80) - (b & pb_7f)) ^ | |||||
| ((a ^ b ^ pb_80) & pb_80); | |||||
| } | |||||
| for (; i < w; i++) | |||||
| dst[i + 0] = src1[i + 0] - src2[i + 0]; | |||||
| } | |||||
| static void sub_hfyu_median_prediction_c(uint8_t *dst, const uint8_t *src1, | |||||
| const uint8_t *src2, int w, | |||||
| int *left, int *left_top) | |||||
| { | |||||
| int i; | |||||
| uint8_t l, lt; | |||||
| l = *left; | |||||
| lt = *left_top; | |||||
| for (i = 0; i < w; i++) { | |||||
| const int pred = mid_pred(l, src1[i], (l + src1[i] - lt) & 0xFF); | |||||
| lt = src1[i]; | |||||
| l = src2[i]; | |||||
| dst[i] = l - pred; | |||||
| } | |||||
| *left = l; | |||||
| *left_top = lt; | |||||
| } | |||||
| #define BUTTERFLY2(o1, o2, i1, i2) \ | #define BUTTERFLY2(o1, o2, i1, i2) \ | ||||
| o1 = (i1) + (i2); \ | o1 = (i1) + (i2); \ | ||||
| o2 = (i1) - (i2); | o2 = (i1) - (i2); | ||||
| @@ -2485,9 +2432,6 @@ av_cold void ff_dsputil_init(DSPContext *c, AVCodecContext *avctx) | |||||
| c->ssd_int8_vs_int16 = ssd_int8_vs_int16_c; | c->ssd_int8_vs_int16 = ssd_int8_vs_int16_c; | ||||
| c->diff_bytes = diff_bytes_c; | |||||
| c->sub_hfyu_median_prediction = sub_hfyu_median_prediction_c; | |||||
| c->bswap_buf = bswap_buf; | c->bswap_buf = bswap_buf; | ||||
| c->bswap16_buf = bswap16_buf; | c->bswap16_buf = bswap16_buf; | ||||
| @@ -185,19 +185,6 @@ typedef struct DSPContext { | |||||
| me_cmp_func pix_abs[2][4]; | me_cmp_func pix_abs[2][4]; | ||||
| /* HuffYUV specific */ | |||||
| void (*diff_bytes)(uint8_t *dst /* align 16 */, | |||||
| uint8_t *src1 /* align 16 */, | |||||
| uint8_t *src2 /* align 1 */, | |||||
| int w); | |||||
| /** | |||||
| * Subtract HuffYUV's variant of median prediction. | |||||
| * Note, this might read from src1[-1], src2[-1]. | |||||
| */ | |||||
| void (*sub_hfyu_median_prediction)(uint8_t *dst, const uint8_t *src1, | |||||
| const uint8_t *src2, int w, | |||||
| int *left, int *left_top); | |||||
| void (*bswap_buf)(uint32_t *dst, const uint32_t *src, int w); | void (*bswap_buf)(uint32_t *dst, const uint32_t *src, int w); | ||||
| void (*bswap16_buf)(uint16_t *dst, const uint16_t *src, int len); | void (*bswap16_buf)(uint16_t *dst, const uint16_t *src, int len); | ||||
| @@ -35,6 +35,7 @@ | |||||
| #include "dsputil.h" | #include "dsputil.h" | ||||
| #include "get_bits.h" | #include "get_bits.h" | ||||
| #include "huffyuvdsp.h" | #include "huffyuvdsp.h" | ||||
| #include "huffyuvencdsp.h" | |||||
| #include "put_bits.h" | #include "put_bits.h" | ||||
| #define VLC_BITS 11 | #define VLC_BITS 11 | ||||
| @@ -83,6 +84,7 @@ typedef struct HYuvContext { | |||||
| unsigned int bitstream_buffer_size; | unsigned int bitstream_buffer_size; | ||||
| DSPContext dsp; | DSPContext dsp; | ||||
| HuffYUVDSPContext hdsp; | HuffYUVDSPContext hdsp; | ||||
| HuffYUVEncDSPContext hencdsp; | |||||
| } HYuvContext; | } HYuvContext; | ||||
| void ff_huffyuv_common_init(AVCodecContext *s); | void ff_huffyuv_common_init(AVCodecContext *s); | ||||
| @@ -29,6 +29,7 @@ | |||||
| #include "avcodec.h" | #include "avcodec.h" | ||||
| #include "huffyuv.h" | #include "huffyuv.h" | ||||
| #include "huffman.h" | #include "huffman.h" | ||||
| #include "huffyuvencdsp.h" | |||||
| #include "put_bits.h" | #include "put_bits.h" | ||||
| static inline int sub_left_prediction(HYuvContext *s, uint8_t *dst, | static inline int sub_left_prediction(HYuvContext *s, uint8_t *dst, | ||||
| @@ -48,7 +49,7 @@ static inline int sub_left_prediction(HYuvContext *s, uint8_t *dst, | |||||
| dst[i] = temp - left; | dst[i] = temp - left; | ||||
| left = temp; | left = temp; | ||||
| } | } | ||||
| s->dsp.diff_bytes(dst + 16, src + 16, src + 15, w - 16); | |||||
| s->hencdsp.diff_bytes(dst + 16, src + 16, src + 15, w - 16); | |||||
| return src[w-1]; | return src[w-1]; | ||||
| } | } | ||||
| } | } | ||||
| @@ -80,7 +81,7 @@ static inline void sub_left_prediction_bgr32(HYuvContext *s, uint8_t *dst, | |||||
| a = at; | a = at; | ||||
| } | } | ||||
| s->dsp.diff_bytes(dst + 16, src + 16, src + 12, w * 4 - 16); | |||||
| s->hencdsp.diff_bytes(dst + 16, src + 16, src + 12, w * 4 - 16); | |||||
| *red = src[(w - 1) * 4 + R]; | *red = src[(w - 1) * 4 + R]; | ||||
| *green = src[(w - 1) * 4 + G]; | *green = src[(w - 1) * 4 + G]; | ||||
| @@ -109,7 +110,7 @@ static inline void sub_left_prediction_rgb24(HYuvContext *s, uint8_t *dst, | |||||
| b = bt; | b = bt; | ||||
| } | } | ||||
| s->dsp.diff_bytes(dst + 48, src + 48, src + 48 - 3, w * 3 - 48); | |||||
| s->hencdsp.diff_bytes(dst + 48, src + 48, src + 48 - 3, w * 3 - 48); | |||||
| *red = src[(w - 1) * 3 + 0]; | *red = src[(w - 1) * 3 + 0]; | ||||
| *green = src[(w - 1) * 3 + 1]; | *green = src[(w - 1) * 3 + 1]; | ||||
| @@ -146,6 +147,7 @@ static av_cold int encode_init(AVCodecContext *avctx) | |||||
| int i, j; | int i, j; | ||||
| ff_huffyuv_common_init(avctx); | ff_huffyuv_common_init(avctx); | ||||
| ff_huffyuvencdsp_init(&s->hencdsp); | |||||
| avctx->extradata = av_mallocz(1024*30); // 256*3+4 == 772 | avctx->extradata = av_mallocz(1024*30); // 256*3+4 == 772 | ||||
| avctx->stats_out = av_mallocz(1024*30); // 21*256*3(%llu ) + 3(\n) + 1(0) = 16132 | avctx->stats_out = av_mallocz(1024*30); // 21*256*3(%llu ) + 3(\n) + 1(0) = 16132 | ||||
| @@ -503,9 +505,9 @@ static int encode_frame(AVCodecContext *avctx, AVPacket *pkt, | |||||
| lefttopy = p->data[0][3]; | lefttopy = p->data[0][3]; | ||||
| lefttopu = p->data[1][1]; | lefttopu = p->data[1][1]; | ||||
| lefttopv = p->data[2][1]; | lefttopv = p->data[2][1]; | ||||
| s->dsp.sub_hfyu_median_prediction(s->temp[0], p->data[0]+4, p->data[0] + fake_ystride + 4, width - 4 , &lefty, &lefttopy); | |||||
| s->dsp.sub_hfyu_median_prediction(s->temp[1], p->data[1]+2, p->data[1] + fake_ustride + 2, width2 - 2, &leftu, &lefttopu); | |||||
| s->dsp.sub_hfyu_median_prediction(s->temp[2], p->data[2]+2, p->data[2] + fake_vstride + 2, width2 - 2, &leftv, &lefttopv); | |||||
| s->hencdsp.sub_hfyu_median_pred(s->temp[0], p->data[0] + 4, p->data[0] + fake_ystride + 4, width - 4, &lefty, &lefttopy); | |||||
| s->hencdsp.sub_hfyu_median_pred(s->temp[1], p->data[1] + 2, p->data[1] + fake_ustride + 2, width2 - 2, &leftu, &lefttopu); | |||||
| s->hencdsp.sub_hfyu_median_pred(s->temp[2], p->data[2] + 2, p->data[2] + fake_vstride + 2, width2 - 2, &leftv, &lefttopv); | |||||
| encode_422_bitstream(s, 0, width - 4); | encode_422_bitstream(s, 0, width - 4); | ||||
| y++; cy++; | y++; cy++; | ||||
| @@ -515,7 +517,7 @@ static int encode_frame(AVCodecContext *avctx, AVPacket *pkt, | |||||
| if (s->bitstream_bpp == 12) { | if (s->bitstream_bpp == 12) { | ||||
| while (2 * cy > y) { | while (2 * cy > y) { | ||||
| ydst = p->data[0] + p->linesize[0] * y; | ydst = p->data[0] + p->linesize[0] * y; | ||||
| s->dsp.sub_hfyu_median_prediction(s->temp[0], ydst - fake_ystride, ydst, width , &lefty, &lefttopy); | |||||
| s->hencdsp.sub_hfyu_median_pred(s->temp[0], ydst - fake_ystride, ydst, width, &lefty, &lefttopy); | |||||
| encode_gray_bitstream(s, width); | encode_gray_bitstream(s, width); | ||||
| y++; | y++; | ||||
| } | } | ||||
| @@ -525,9 +527,9 @@ static int encode_frame(AVCodecContext *avctx, AVPacket *pkt, | |||||
| udst = p->data[1] + p->linesize[1] * cy; | udst = p->data[1] + p->linesize[1] * cy; | ||||
| vdst = p->data[2] + p->linesize[2] * cy; | vdst = p->data[2] + p->linesize[2] * cy; | ||||
| s->dsp.sub_hfyu_median_prediction(s->temp[0], ydst - fake_ystride, ydst, width , &lefty, &lefttopy); | |||||
| s->dsp.sub_hfyu_median_prediction(s->temp[1], udst - fake_ustride, udst, width2, &leftu, &lefttopu); | |||||
| s->dsp.sub_hfyu_median_prediction(s->temp[2], vdst - fake_vstride, vdst, width2, &leftv, &lefttopv); | |||||
| s->hencdsp.sub_hfyu_median_pred(s->temp[0], ydst - fake_ystride, ydst, width, &lefty, &lefttopy); | |||||
| s->hencdsp.sub_hfyu_median_pred(s->temp[1], udst - fake_ustride, udst, width2, &leftu, &lefttopu); | |||||
| s->hencdsp.sub_hfyu_median_pred(s->temp[2], vdst - fake_vstride, vdst, width2, &leftv, &lefttopv); | |||||
| encode_422_bitstream(s, 0, width); | encode_422_bitstream(s, 0, width); | ||||
| } | } | ||||
| @@ -540,7 +542,7 @@ static int encode_frame(AVCodecContext *avctx, AVPacket *pkt, | |||||
| ydst = p->data[0] + p->linesize[0] * y; | ydst = p->data[0] + p->linesize[0] * y; | ||||
| if (s->predictor == PLANE && s->interlaced < y) { | if (s->predictor == PLANE && s->interlaced < y) { | ||||
| s->dsp.diff_bytes(s->temp[1], ydst, ydst - fake_ystride, width); | |||||
| s->hencdsp.diff_bytes(s->temp[1], ydst, ydst - fake_ystride, width); | |||||
| lefty = sub_left_prediction(s, s->temp[0], s->temp[1], width , lefty); | lefty = sub_left_prediction(s, s->temp[0], s->temp[1], width , lefty); | ||||
| } else { | } else { | ||||
| @@ -556,9 +558,9 @@ static int encode_frame(AVCodecContext *avctx, AVPacket *pkt, | |||||
| vdst = p->data[2] + p->linesize[2] * cy; | vdst = p->data[2] + p->linesize[2] * cy; | ||||
| if (s->predictor == PLANE && s->interlaced < cy) { | if (s->predictor == PLANE && s->interlaced < cy) { | ||||
| s->dsp.diff_bytes(s->temp[1], ydst, ydst - fake_ystride, width); | |||||
| s->dsp.diff_bytes(s->temp[2], udst, udst - fake_ustride, width2); | |||||
| s->dsp.diff_bytes(s->temp[2] + width2, vdst, vdst - fake_vstride, width2); | |||||
| s->hencdsp.diff_bytes(s->temp[1], ydst, ydst - fake_ystride, width); | |||||
| s->hencdsp.diff_bytes(s->temp[2], udst, udst - fake_ustride, width2); | |||||
| s->hencdsp.diff_bytes(s->temp[2] + width2, vdst, vdst - fake_vstride, width2); | |||||
| lefty = sub_left_prediction(s, s->temp[0], s->temp[1], width , lefty); | lefty = sub_left_prediction(s, s->temp[0], s->temp[1], width , lefty); | ||||
| leftu = sub_left_prediction(s, s->temp[1], s->temp[2], width2, leftu); | leftu = sub_left_prediction(s, s->temp[1], s->temp[2], width2, leftu); | ||||
| @@ -591,7 +593,7 @@ static int encode_frame(AVCodecContext *avctx, AVPacket *pkt, | |||||
| for (y = 1; y < s->height; y++) { | for (y = 1; y < s->height; y++) { | ||||
| uint8_t *dst = data + y*stride; | uint8_t *dst = data + y*stride; | ||||
| if (s->predictor == PLANE && s->interlaced < y) { | if (s->predictor == PLANE && s->interlaced < y) { | ||||
| s->dsp.diff_bytes(s->temp[1], dst, dst - fake_stride, width * 4); | |||||
| s->hencdsp.diff_bytes(s->temp[1], dst, dst - fake_stride, width * 4); | |||||
| sub_left_prediction_bgr32(s, s->temp[0], s->temp[1], width, | sub_left_prediction_bgr32(s, s->temp[0], s->temp[1], width, | ||||
| &leftr, &leftg, &leftb, &lefta); | &leftr, &leftg, &leftb, &lefta); | ||||
| } else { | } else { | ||||
| @@ -619,8 +621,8 @@ static int encode_frame(AVCodecContext *avctx, AVPacket *pkt, | |||||
| for (y = 1; y < s->height; y++) { | for (y = 1; y < s->height; y++) { | ||||
| uint8_t *dst = data + y * stride; | uint8_t *dst = data + y * stride; | ||||
| if (s->predictor == PLANE && s->interlaced < y) { | if (s->predictor == PLANE && s->interlaced < y) { | ||||
| s->dsp.diff_bytes(s->temp[1], dst, dst - fake_stride, | |||||
| width * 3); | |||||
| s->hencdsp.diff_bytes(s->temp[1], dst, dst - fake_stride, | |||||
| width * 3); | |||||
| sub_left_prediction_rgb24(s, s->temp[0], s->temp[1], width, | sub_left_prediction_rgb24(s, s->temp[0], s->temp[1], width, | ||||
| &leftr, &leftg, &leftb); | &leftr, &leftg, &leftb); | ||||
| } else { | } else { | ||||
| @@ -0,0 +1,84 @@ | |||||
| /* | |||||
| * This file is part of Libav. | |||||
| * | |||||
| * Libav is free software; you can redistribute it and/or | |||||
| * modify it under the terms of the GNU Lesser General Public | |||||
| * License as published by the Free Software Foundation; either | |||||
| * version 2.1 of the License, or (at your option) any later version. | |||||
| * | |||||
| * Libav 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 | |||||
| * Lesser General Public License for more details. | |||||
| * | |||||
| * You should have received a copy of the GNU Lesser General Public | |||||
| * License along with Libav; if not, write to the Free Software | |||||
| * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | |||||
| */ | |||||
| #include "config.h" | |||||
| #include "libavutil/attributes.h" | |||||
| #include "huffyuvencdsp.h" | |||||
| #include "mathops.h" | |||||
| // 0x7f7f7f7f or 0x7f7f7f7f7f7f7f7f or whatever, depending on the cpu's native arithmetic size | |||||
| #define pb_7f (~0UL / 255 * 0x7f) | |||||
| #define pb_80 (~0UL / 255 * 0x80) | |||||
| static void diff_bytes_c(uint8_t *dst, uint8_t *src1, uint8_t *src2, int w) | |||||
| { | |||||
| long i; | |||||
| #if !HAVE_FAST_UNALIGNED | |||||
| if ((long) src2 & (sizeof(long) - 1)) { | |||||
| for (i = 0; i + 7 < w; i += 8) { | |||||
| dst[i + 0] = src1[i + 0] - src2[i + 0]; | |||||
| dst[i + 1] = src1[i + 1] - src2[i + 1]; | |||||
| dst[i + 2] = src1[i + 2] - src2[i + 2]; | |||||
| dst[i + 3] = src1[i + 3] - src2[i + 3]; | |||||
| dst[i + 4] = src1[i + 4] - src2[i + 4]; | |||||
| dst[i + 5] = src1[i + 5] - src2[i + 5]; | |||||
| dst[i + 6] = src1[i + 6] - src2[i + 6]; | |||||
| dst[i + 7] = src1[i + 7] - src2[i + 7]; | |||||
| } | |||||
| } else | |||||
| #endif | |||||
| for (i = 0; i <= w - (int) sizeof(long); i += sizeof(long)) { | |||||
| long a = *(long *) (src1 + i); | |||||
| long b = *(long *) (src2 + i); | |||||
| *(long *) (dst + i) = ((a | pb_80) - (b & pb_7f)) ^ | |||||
| ((a ^ b ^ pb_80) & pb_80); | |||||
| } | |||||
| for (; i < w; i++) | |||||
| dst[i + 0] = src1[i + 0] - src2[i + 0]; | |||||
| } | |||||
| static void sub_hfyu_median_pred_c(uint8_t *dst, const uint8_t *src1, | |||||
| const uint8_t *src2, int w, | |||||
| int *left, int *left_top) | |||||
| { | |||||
| int i; | |||||
| uint8_t l, lt; | |||||
| l = *left; | |||||
| lt = *left_top; | |||||
| for (i = 0; i < w; i++) { | |||||
| const int pred = mid_pred(l, src1[i], (l + src1[i] - lt) & 0xFF); | |||||
| lt = src1[i]; | |||||
| l = src2[i]; | |||||
| dst[i] = l - pred; | |||||
| } | |||||
| *left = l; | |||||
| *left_top = lt; | |||||
| } | |||||
| av_cold void ff_huffyuvencdsp_init(HuffYUVEncDSPContext *c) | |||||
| { | |||||
| c->diff_bytes = diff_bytes_c; | |||||
| c->sub_hfyu_median_pred = sub_hfyu_median_pred_c; | |||||
| if (ARCH_X86) | |||||
| ff_huffyuvencdsp_init_x86(c); | |||||
| } | |||||
| @@ -0,0 +1,41 @@ | |||||
| /* | |||||
| * This file is part of Libav. | |||||
| * | |||||
| * Libav is free software; you can redistribute it and/or | |||||
| * modify it under the terms of the GNU Lesser General Public | |||||
| * License as published by the Free Software Foundation; either | |||||
| * version 2.1 of the License, or (at your option) any later version. | |||||
| * | |||||
| * Libav 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 | |||||
| * Lesser General Public License for more details. | |||||
| * | |||||
| * You should have received a copy of the GNU Lesser General Public | |||||
| * License along with Libav; if not, write to the Free Software | |||||
| * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | |||||
| */ | |||||
| #ifndef AVCODEC_HUFFYUVENCDSP_H | |||||
| #define AVCODEC_HUFFYUVENCDSP_H | |||||
| #include <stdint.h> | |||||
| typedef struct HuffYUVEncDSPContext { | |||||
| void (*diff_bytes)(uint8_t *dst /* align 16 */, | |||||
| uint8_t *src1 /* align 16 */, | |||||
| uint8_t *src2 /* align 1 */, | |||||
| int w); | |||||
| /** | |||||
| * Subtract HuffYUV's variant of median prediction. | |||||
| * Note, this might read from src1[-1], src2[-1]. | |||||
| */ | |||||
| void (*sub_hfyu_median_pred)(uint8_t *dst, const uint8_t *src1, | |||||
| const uint8_t *src2, int w, | |||||
| int *left, int *left_top); | |||||
| } HuffYUVEncDSPContext; | |||||
| void ff_huffyuvencdsp_init(HuffYUVEncDSPContext *c); | |||||
| void ff_huffyuvencdsp_init_x86(HuffYUVEncDSPContext *c); | |||||
| #endif /* AVCODEC_HUFFYUVENCDSP_H */ | |||||
| @@ -21,7 +21,7 @@ | |||||
| #include "avcodec.h" | #include "avcodec.h" | ||||
| #include "bytestream.h" | #include "bytestream.h" | ||||
| #include "dsputil.h" | |||||
| #include "huffyuvencdsp.h" | |||||
| #include "png.h" | #include "png.h" | ||||
| /* TODO: | /* TODO: | ||||
| @@ -33,7 +33,7 @@ | |||||
| #define IOBUF_SIZE 4096 | #define IOBUF_SIZE 4096 | ||||
| typedef struct PNGEncContext { | typedef struct PNGEncContext { | ||||
| DSPContext dsp; | |||||
| HuffYUVEncDSPContext hdsp; | |||||
| uint8_t *bytestream; | uint8_t *bytestream; | ||||
| uint8_t *bytestream_start; | uint8_t *bytestream_start; | ||||
| @@ -111,7 +111,7 @@ static void sub_png_paeth_prediction(uint8_t *dst, uint8_t *src, uint8_t *top, | |||||
| } | } | ||||
| } | } | ||||
| static void png_filter_row(DSPContext *dsp, uint8_t *dst, int filter_type, | |||||
| static void png_filter_row(PNGEncContext *c, uint8_t *dst, int filter_type, | |||||
| uint8_t *src, uint8_t *top, int size, int bpp) | uint8_t *src, uint8_t *top, int size, int bpp) | ||||
| { | { | ||||
| int i; | int i; | ||||
| @@ -121,11 +121,11 @@ static void png_filter_row(DSPContext *dsp, uint8_t *dst, int filter_type, | |||||
| memcpy(dst, src, size); | memcpy(dst, src, size); | ||||
| break; | break; | ||||
| case PNG_FILTER_VALUE_SUB: | case PNG_FILTER_VALUE_SUB: | ||||
| dsp->diff_bytes(dst, src, src - bpp, size); | |||||
| c->hdsp.diff_bytes(dst, src, src - bpp, size); | |||||
| memcpy(dst, src, bpp); | memcpy(dst, src, bpp); | ||||
| break; | break; | ||||
| case PNG_FILTER_VALUE_UP: | case PNG_FILTER_VALUE_UP: | ||||
| dsp->diff_bytes(dst, src, top, size); | |||||
| c->hdsp.diff_bytes(dst, src, top, size); | |||||
| break; | break; | ||||
| case PNG_FILTER_VALUE_AVG: | case PNG_FILTER_VALUE_AVG: | ||||
| for (i = 0; i < bpp; i++) | for (i = 0; i < bpp; i++) | ||||
| @@ -153,7 +153,7 @@ static uint8_t *png_choose_filter(PNGEncContext *s, uint8_t *dst, | |||||
| int cost, bcost = INT_MAX; | int cost, bcost = INT_MAX; | ||||
| uint8_t *buf1 = dst, *buf2 = dst + size + 16; | uint8_t *buf1 = dst, *buf2 = dst + size + 16; | ||||
| for (pred = 0; pred < 5; pred++) { | for (pred = 0; pred < 5; pred++) { | ||||
| png_filter_row(&s->dsp, buf1 + 1, pred, src, top, size, bpp); | |||||
| png_filter_row(s, buf1 + 1, pred, src, top, size, bpp); | |||||
| buf1[0] = pred; | buf1[0] = pred; | ||||
| cost = 0; | cost = 0; | ||||
| for (i = 0; i <= size; i++) | for (i = 0; i <= size; i++) | ||||
| @@ -165,7 +165,7 @@ static uint8_t *png_choose_filter(PNGEncContext *s, uint8_t *dst, | |||||
| } | } | ||||
| return buf2; | return buf2; | ||||
| } else { | } else { | ||||
| png_filter_row(&s->dsp, dst + 1, pred, src, top, size, bpp); | |||||
| png_filter_row(s, dst + 1, pred, src, top, size, bpp); | |||||
| dst[0] = pred; | dst[0] = pred; | ||||
| return dst; | return dst; | ||||
| } | } | ||||
| @@ -462,7 +462,7 @@ static av_cold int png_enc_init(AVCodecContext *avctx) | |||||
| avctx->coded_frame->pict_type = AV_PICTURE_TYPE_I; | avctx->coded_frame->pict_type = AV_PICTURE_TYPE_I; | ||||
| avctx->coded_frame->key_frame = 1; | avctx->coded_frame->key_frame = 1; | ||||
| ff_dsputil_init(&s->dsp, avctx); | |||||
| ff_huffyuvencdsp_init(&s->hdsp); | |||||
| s->filter_type = av_clip(avctx->prediction_method, | s->filter_type = av_clip(avctx->prediction_method, | ||||
| PNG_FILTER_VALUE_NONE, | PNG_FILTER_VALUE_NONE, | ||||
| @@ -30,6 +30,7 @@ | |||||
| #include "libavutil/common.h" | #include "libavutil/common.h" | ||||
| #include "avcodec.h" | #include "avcodec.h" | ||||
| #include "dsputil.h" | #include "dsputil.h" | ||||
| #include "huffyuvencdsp.h" | |||||
| enum { | enum { | ||||
| PRED_NONE = 0, | PRED_NONE = 0, | ||||
| @@ -66,6 +67,7 @@ extern const int ff_ut_rgb_order[4]; | |||||
| typedef struct UtvideoContext { | typedef struct UtvideoContext { | ||||
| AVCodecContext *avctx; | AVCodecContext *avctx; | ||||
| DSPContext dsp; | DSPContext dsp; | ||||
| HuffYUVEncDSPContext hdsp; | |||||
| uint32_t frame_info_size, flags, frame_info; | uint32_t frame_info_size, flags, frame_info; | ||||
| int planes; | int planes; | ||||
| @@ -31,6 +31,7 @@ | |||||
| #include "bytestream.h" | #include "bytestream.h" | ||||
| #include "put_bits.h" | #include "put_bits.h" | ||||
| #include "dsputil.h" | #include "dsputil.h" | ||||
| #include "huffyuvencdsp.h" | |||||
| #include "mathops.h" | #include "mathops.h" | ||||
| #include "utvideo.h" | #include "utvideo.h" | ||||
| #include "huffman.h" | #include "huffman.h" | ||||
| @@ -109,6 +110,7 @@ static av_cold int utvideo_encode_init(AVCodecContext *avctx) | |||||
| } | } | ||||
| ff_dsputil_init(&c->dsp, avctx); | ff_dsputil_init(&c->dsp, avctx); | ||||
| ff_huffyuvencdsp_init(&c->hdsp); | |||||
| /* Check the prediction method, and error out if unsupported */ | /* Check the prediction method, and error out if unsupported */ | ||||
| if (avctx->prediction_method < 0 || avctx->prediction_method > 4) { | if (avctx->prediction_method < 0 || avctx->prediction_method > 4) { | ||||
| @@ -312,7 +314,7 @@ static void median_predict(UtvideoContext *c, uint8_t *src, uint8_t *dst, int st | |||||
| /* Rest of the coded part uses median prediction */ | /* Rest of the coded part uses median prediction */ | ||||
| for (j = 1; j < height; j++) { | for (j = 1; j < height; j++) { | ||||
| c->dsp.sub_hfyu_median_prediction(dst, src - stride, src, width, &A, &B); | |||||
| c->hdsp.sub_hfyu_median_pred(dst, src - stride, src, width, &A, &B); | |||||
| dst += width; | dst += width; | ||||
| src += stride; | src += stride; | ||||
| } | } | ||||
| @@ -15,6 +15,7 @@ OBJS-$(CONFIG_H264PRED) += x86/h264_intrapred_init.o | |||||
| OBJS-$(CONFIG_H264QPEL) += x86/h264_qpel.o | OBJS-$(CONFIG_H264QPEL) += x86/h264_qpel.o | ||||
| OBJS-$(CONFIG_HPELDSP) += x86/hpeldsp_init.o | OBJS-$(CONFIG_HPELDSP) += x86/hpeldsp_init.o | ||||
| OBJS-$(CONFIG_HUFFYUVDSP) += x86/huffyuvdsp_init.o | OBJS-$(CONFIG_HUFFYUVDSP) += x86/huffyuvdsp_init.o | ||||
| OBJS-$(CONFIG_HUFFYUVENCDSP) += x86/huffyuvencdsp_mmx.o | |||||
| OBJS-$(CONFIG_LPC) += x86/lpc.o | OBJS-$(CONFIG_LPC) += x86/lpc.o | ||||
| OBJS-$(CONFIG_MPEGAUDIODSP) += x86/mpegaudiodsp.o | OBJS-$(CONFIG_MPEGAUDIODSP) += x86/mpegaudiodsp.o | ||||
| OBJS-$(CONFIG_MPEGVIDEO) += x86/mpegvideo.o | OBJS-$(CONFIG_MPEGVIDEO) += x86/mpegvideo.o | ||||
| @@ -29,7 +29,6 @@ | |||||
| #include "libavcodec/dct.h" | #include "libavcodec/dct.h" | ||||
| #include "libavcodec/dsputil.h" | #include "libavcodec/dsputil.h" | ||||
| #include "libavcodec/mpegvideo.h" | #include "libavcodec/mpegvideo.h" | ||||
| #include "libavcodec/mathops.h" | |||||
| #include "dsputil_x86.h" | #include "dsputil_x86.h" | ||||
| void ff_get_pixels_mmx(int16_t *block, const uint8_t *pixels, int line_size); | void ff_get_pixels_mmx(int16_t *block, const uint8_t *pixels, int line_size); | ||||
| @@ -694,70 +693,6 @@ static int vsad16_mmxext(MpegEncContext *v, uint8_t *pix1, uint8_t *pix2, | |||||
| } | } | ||||
| #undef SUM | #undef SUM | ||||
| static void diff_bytes_mmx(uint8_t *dst, uint8_t *src1, uint8_t *src2, int w) | |||||
| { | |||||
| x86_reg i = 0; | |||||
| __asm__ volatile ( | |||||
| "1: \n\t" | |||||
| "movq (%2, %0), %%mm0 \n\t" | |||||
| "movq (%1, %0), %%mm1 \n\t" | |||||
| "psubb %%mm0, %%mm1 \n\t" | |||||
| "movq %%mm1, (%3, %0) \n\t" | |||||
| "movq 8(%2, %0), %%mm0 \n\t" | |||||
| "movq 8(%1, %0), %%mm1 \n\t" | |||||
| "psubb %%mm0, %%mm1 \n\t" | |||||
| "movq %%mm1, 8(%3, %0) \n\t" | |||||
| "add $16, %0 \n\t" | |||||
| "cmp %4, %0 \n\t" | |||||
| " jb 1b \n\t" | |||||
| : "+r" (i) | |||||
| : "r" (src1), "r" (src2), "r" (dst), "r" ((x86_reg) w - 15)); | |||||
| for (; i < w; i++) | |||||
| dst[i + 0] = src1[i + 0] - src2[i + 0]; | |||||
| } | |||||
| static void sub_hfyu_median_prediction_mmxext(uint8_t *dst, const uint8_t *src1, | |||||
| const uint8_t *src2, int w, | |||||
| int *left, int *left_top) | |||||
| { | |||||
| x86_reg i = 0; | |||||
| uint8_t l, lt; | |||||
| __asm__ volatile ( | |||||
| "movq (%1, %0), %%mm0 \n\t" // LT | |||||
| "psllq $8, %%mm0 \n\t" | |||||
| "1: \n\t" | |||||
| "movq (%1, %0), %%mm1 \n\t" // T | |||||
| "movq -1(%2, %0), %%mm2 \n\t" // L | |||||
| "movq (%2, %0), %%mm3 \n\t" // X | |||||
| "movq %%mm2, %%mm4 \n\t" // L | |||||
| "psubb %%mm0, %%mm2 \n\t" | |||||
| "paddb %%mm1, %%mm2 \n\t" // L + T - LT | |||||
| "movq %%mm4, %%mm5 \n\t" // L | |||||
| "pmaxub %%mm1, %%mm4 \n\t" // max(T, L) | |||||
| "pminub %%mm5, %%mm1 \n\t" // min(T, L) | |||||
| "pminub %%mm2, %%mm4 \n\t" | |||||
| "pmaxub %%mm1, %%mm4 \n\t" | |||||
| "psubb %%mm4, %%mm3 \n\t" // dst - pred | |||||
| "movq %%mm3, (%3, %0) \n\t" | |||||
| "add $8, %0 \n\t" | |||||
| "movq -1(%1, %0), %%mm0 \n\t" // LT | |||||
| "cmp %4, %0 \n\t" | |||||
| " jb 1b \n\t" | |||||
| : "+r" (i) | |||||
| : "r" (src1), "r" (src2), "r" (dst), "r" ((x86_reg) w)); | |||||
| l = *left; | |||||
| lt = *left_top; | |||||
| dst[0] = src2[0] - mid_pred(l, src1[0], (l + src1[0] - lt) & 0xFF); | |||||
| *left_top = src1[w - 1]; | |||||
| *left = src2[w - 1]; | |||||
| } | |||||
| #define MMABS_MMX(a,z) \ | #define MMABS_MMX(a,z) \ | ||||
| "pxor " #z ", " #z " \n\t" \ | "pxor " #z ", " #z " \n\t" \ | ||||
| "pcmpgtw " #a ", " #z " \n\t" \ | "pcmpgtw " #a ", " #z " \n\t" \ | ||||
| @@ -1010,7 +945,6 @@ av_cold void ff_dsputilenc_init_mmx(DSPContext *c, AVCodecContext *avctx, | |||||
| (dct_algo == FF_DCT_AUTO || dct_algo == FF_DCT_MMX)) | (dct_algo == FF_DCT_AUTO || dct_algo == FF_DCT_MMX)) | ||||
| c->fdct = ff_fdct_mmx; | c->fdct = ff_fdct_mmx; | ||||
| c->diff_bytes = diff_bytes_mmx; | |||||
| c->sum_abs_dctelem = sum_abs_dctelem_mmx; | c->sum_abs_dctelem = sum_abs_dctelem_mmx; | ||||
| c->sse[0] = sse16_mmx; | c->sse[0] = sse16_mmx; | ||||
| @@ -1046,8 +980,6 @@ av_cold void ff_dsputilenc_init_mmx(DSPContext *c, AVCodecContext *avctx, | |||||
| if (!(avctx->flags & CODEC_FLAG_BITEXACT)) { | if (!(avctx->flags & CODEC_FLAG_BITEXACT)) { | ||||
| c->vsad[0] = vsad16_mmxext; | c->vsad[0] = vsad16_mmxext; | ||||
| } | } | ||||
| c->sub_hfyu_median_prediction = sub_hfyu_median_prediction_mmxext; | |||||
| } | } | ||||
| if (INLINE_SSE2(cpu_flags)) { | if (INLINE_SSE2(cpu_flags)) { | ||||
| @@ -0,0 +1,113 @@ | |||||
| /* | |||||
| * SIMD-optimized HuffYUV encoding functions | |||||
| * Copyright (c) 2000, 2001 Fabrice Bellard | |||||
| * Copyright (c) 2002-2004 Michael Niedermayer <michaelni@gmx.at> | |||||
| * | |||||
| * MMX optimization by Nick Kurshev <nickols_k@mail.ru> | |||||
| * | |||||
| * This file is part of Libav. | |||||
| * | |||||
| * Libav is free software; you can redistribute it and/or | |||||
| * modify it under the terms of the GNU Lesser General Public | |||||
| * License as published by the Free Software Foundation; either | |||||
| * version 2.1 of the License, or (at your option) any later version. | |||||
| * | |||||
| * Libav 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 | |||||
| * Lesser General Public License for more details. | |||||
| * | |||||
| * You should have received a copy of the GNU Lesser General Public | |||||
| * License along with Libav; if not, write to the Free Software | |||||
| * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | |||||
| */ | |||||
| #include "libavutil/attributes.h" | |||||
| #include "libavutil/cpu.h" | |||||
| #include "libavutil/x86/asm.h" | |||||
| #include "libavutil/x86/cpu.h" | |||||
| #include "libavcodec/huffyuvencdsp.h" | |||||
| #include "libavcodec/mathops.h" | |||||
| #if HAVE_INLINE_ASM | |||||
| static void diff_bytes_mmx(uint8_t *dst, uint8_t *src1, uint8_t *src2, int w) | |||||
| { | |||||
| x86_reg i = 0; | |||||
| __asm__ volatile ( | |||||
| "1: \n\t" | |||||
| "movq (%2, %0), %%mm0 \n\t" | |||||
| "movq (%1, %0), %%mm1 \n\t" | |||||
| "psubb %%mm0, %%mm1 \n\t" | |||||
| "movq %%mm1, (%3, %0) \n\t" | |||||
| "movq 8(%2, %0), %%mm0 \n\t" | |||||
| "movq 8(%1, %0), %%mm1 \n\t" | |||||
| "psubb %%mm0, %%mm1 \n\t" | |||||
| "movq %%mm1, 8(%3, %0) \n\t" | |||||
| "add $16, %0 \n\t" | |||||
| "cmp %4, %0 \n\t" | |||||
| " jb 1b \n\t" | |||||
| : "+r" (i) | |||||
| : "r" (src1), "r" (src2), "r" (dst), "r" ((x86_reg) w - 15)); | |||||
| for (; i < w; i++) | |||||
| dst[i + 0] = src1[i + 0] - src2[i + 0]; | |||||
| } | |||||
| static void sub_hfyu_median_pred_mmxext(uint8_t *dst, const uint8_t *src1, | |||||
| const uint8_t *src2, int w, | |||||
| int *left, int *left_top) | |||||
| { | |||||
| x86_reg i = 0; | |||||
| uint8_t l, lt; | |||||
| __asm__ volatile ( | |||||
| "movq (%1, %0), %%mm0 \n\t" // LT | |||||
| "psllq $8, %%mm0 \n\t" | |||||
| "1: \n\t" | |||||
| "movq (%1, %0), %%mm1 \n\t" // T | |||||
| "movq -1(%2, %0), %%mm2 \n\t" // L | |||||
| "movq (%2, %0), %%mm3 \n\t" // X | |||||
| "movq %%mm2, %%mm4 \n\t" // L | |||||
| "psubb %%mm0, %%mm2 \n\t" | |||||
| "paddb %%mm1, %%mm2 \n\t" // L + T - LT | |||||
| "movq %%mm4, %%mm5 \n\t" // L | |||||
| "pmaxub %%mm1, %%mm4 \n\t" // max(T, L) | |||||
| "pminub %%mm5, %%mm1 \n\t" // min(T, L) | |||||
| "pminub %%mm2, %%mm4 \n\t" | |||||
| "pmaxub %%mm1, %%mm4 \n\t" | |||||
| "psubb %%mm4, %%mm3 \n\t" // dst - pred | |||||
| "movq %%mm3, (%3, %0) \n\t" | |||||
| "add $8, %0 \n\t" | |||||
| "movq -1(%1, %0), %%mm0 \n\t" // LT | |||||
| "cmp %4, %0 \n\t" | |||||
| " jb 1b \n\t" | |||||
| : "+r" (i) | |||||
| : "r" (src1), "r" (src2), "r" (dst), "r" ((x86_reg) w)); | |||||
| l = *left; | |||||
| lt = *left_top; | |||||
| dst[0] = src2[0] - mid_pred(l, src1[0], (l + src1[0] - lt) & 0xFF); | |||||
| *left_top = src1[w - 1]; | |||||
| *left = src2[w - 1]; | |||||
| } | |||||
| #endif /* HAVE_INLINE_ASM */ | |||||
| av_cold void ff_huffyuvencdsp_init_x86(HuffYUVEncDSPContext *c) | |||||
| { | |||||
| #if HAVE_INLINE_ASM | |||||
| int cpu_flags = av_get_cpu_flags(); | |||||
| if (INLINE_MMX(cpu_flags)) { | |||||
| c->diff_bytes = diff_bytes_mmx; | |||||
| } | |||||
| if (INLINE_MMXEXT(cpu_flags)) { | |||||
| c->sub_hfyu_median_pred = sub_hfyu_median_pred_mmxext; | |||||
| } | |||||
| #endif /* HAVE_INLINE_ASM */ | |||||
| } | |||||