1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * Copyright (c) 2016 William Ma, Sofia Kim, Dustin Woo
3cabdff1aSopenharmony_ci *
4cabdff1aSopenharmony_ci * This file is part of FFmpeg.
5cabdff1aSopenharmony_ci *
6cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or
7cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public
8cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either
9cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version.
10cabdff1aSopenharmony_ci *
11cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful,
12cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of
13cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14cabdff1aSopenharmony_ci * Lesser General Public License for more details.
15cabdff1aSopenharmony_ci *
16cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public
17cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software
18cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19cabdff1aSopenharmony_ci */
20cabdff1aSopenharmony_ci
21cabdff1aSopenharmony_ci/**
22cabdff1aSopenharmony_ci * @file
23cabdff1aSopenharmony_ci * Optimal Huffman Encoding tests.
24cabdff1aSopenharmony_ci */
25cabdff1aSopenharmony_ci
26cabdff1aSopenharmony_ci#include "libavcodec/avcodec.h"
27cabdff1aSopenharmony_ci#include <stdlib.h>
28cabdff1aSopenharmony_ci#include "libavcodec/mjpegenc.h"
29cabdff1aSopenharmony_ci#include "libavcodec/mjpegenc_huffman.h"
30cabdff1aSopenharmony_ci#include "libavcodec/mjpegenc_common.h"
31cabdff1aSopenharmony_ci#include "libavcodec/mpegvideo.h"
32cabdff1aSopenharmony_ci
33cabdff1aSopenharmony_ci// Validate the computed lengths satisfy the JPEG restrictions and is optimal.
34cabdff1aSopenharmony_cistatic int check_lengths(int L, int expected_length,
35cabdff1aSopenharmony_ci                         const int *probs, int nprobs)
36cabdff1aSopenharmony_ci{
37cabdff1aSopenharmony_ci    HuffTable lengths[256];
38cabdff1aSopenharmony_ci    PTable val_counts[256];
39cabdff1aSopenharmony_ci    int actual_length = 0, i, j, k, prob, length;
40cabdff1aSopenharmony_ci    int ret = 0;
41cabdff1aSopenharmony_ci    double cantor_measure = 0;
42cabdff1aSopenharmony_ci    av_assert0(nprobs <= 256);
43cabdff1aSopenharmony_ci
44cabdff1aSopenharmony_ci    for (i = 0; i < nprobs; i++) {
45cabdff1aSopenharmony_ci        val_counts[i] = (PTable){.value = i, .prob = probs[i]};
46cabdff1aSopenharmony_ci    }
47cabdff1aSopenharmony_ci
48cabdff1aSopenharmony_ci    ff_mjpegenc_huffman_compute_bits(val_counts, lengths, nprobs, L);
49cabdff1aSopenharmony_ci
50cabdff1aSopenharmony_ci    for (i = 0; i < nprobs; i++) {
51cabdff1aSopenharmony_ci        // Find the value's prob and length
52cabdff1aSopenharmony_ci        for (j = 0; j < nprobs; j++)
53cabdff1aSopenharmony_ci            if (val_counts[j].value == i) break;
54cabdff1aSopenharmony_ci        for (k = 0; k < nprobs; k++)
55cabdff1aSopenharmony_ci            if (lengths[k].code == i) break;
56cabdff1aSopenharmony_ci        if (!(j < nprobs && k < nprobs)) return 1;
57cabdff1aSopenharmony_ci        prob = val_counts[j].prob;
58cabdff1aSopenharmony_ci        length = lengths[k].length;
59cabdff1aSopenharmony_ci
60cabdff1aSopenharmony_ci        if (prob) {
61cabdff1aSopenharmony_ci            actual_length += prob * length;
62cabdff1aSopenharmony_ci            cantor_measure += 1. / (1 << length);
63cabdff1aSopenharmony_ci        }
64cabdff1aSopenharmony_ci
65cabdff1aSopenharmony_ci        if (length > L || length < 1) return 1;
66cabdff1aSopenharmony_ci    }
67cabdff1aSopenharmony_ci    // Check that the codes can be prefix-free.
68cabdff1aSopenharmony_ci    if (cantor_measure > 1) ret = 1;
69cabdff1aSopenharmony_ci    // Check that the total length is optimal
70cabdff1aSopenharmony_ci    if (actual_length != expected_length) ret = 1;
71cabdff1aSopenharmony_ci
72cabdff1aSopenharmony_ci    if (ret == 1) {
73cabdff1aSopenharmony_ci      fprintf(stderr,
74cabdff1aSopenharmony_ci              "Cantor measure: %f\n"
75cabdff1aSopenharmony_ci              "Actual length: %d\n"
76cabdff1aSopenharmony_ci              "Expected length: %d\n",
77cabdff1aSopenharmony_ci              cantor_measure, actual_length, expected_length);
78cabdff1aSopenharmony_ci    }
79cabdff1aSopenharmony_ci
80cabdff1aSopenharmony_ci    return ret;
81cabdff1aSopenharmony_ci}
82cabdff1aSopenharmony_ci
83cabdff1aSopenharmony_cistatic const int probs_zeroes[] = {
84cabdff1aSopenharmony_ci    6, 6, 0, 0, 0
85cabdff1aSopenharmony_ci};
86cabdff1aSopenharmony_ci
87cabdff1aSopenharmony_cistatic const int probs_skewed[] = {
88cabdff1aSopenharmony_ci    2, 0, 0, 0, 0, 1, 0, 0, 20, 0, 2, 0, 10, 5, 1, 1, 9, 1, 1, 6, 0, 5, 0, 1, 0, 7, 6,
89cabdff1aSopenharmony_ci    1, 1, 5, 0, 0, 0, 0, 11, 0, 0, 0, 51, 1, 0, 20, 0, 1, 0, 0, 0, 0, 6, 106, 1, 0, 1,
90cabdff1aSopenharmony_ci    0, 2, 1, 16, 0, 0, 5, 0, 0, 0, 4, 3, 15, 4, 4, 0, 0, 0, 3, 0, 0, 1, 0, 3, 0, 3, 2,
91cabdff1aSopenharmony_ci    2, 0, 0, 4, 3, 40, 1, 2, 0, 22, 0, 0, 0, 9, 0, 0, 0, 0, 1, 1, 0, 1, 6, 11, 4, 10,
92cabdff1aSopenharmony_ci    28, 6, 1, 0, 0, 9, 9, 4, 0, 0, 0, 0, 8, 33844, 2, 0, 2, 1, 1, 5, 0, 0, 1, 9, 1, 0,
93cabdff1aSopenharmony_ci    4, 14, 4, 0, 0, 3, 8, 0, 51, 9, 6, 1, 1, 2, 2, 3, 1, 5, 5, 29, 0, 0, 0, 0, 14, 29,
94cabdff1aSopenharmony_ci    6, 4, 13, 12, 2, 3, 1, 0, 5, 4, 1, 1, 0, 0, 29, 1, 0, 0, 0, 0, 4, 0, 0, 1, 0, 1,
95cabdff1aSopenharmony_ci    7, 0, 42, 0, 0, 0, 0, 0, 2, 0, 3, 9, 0, 0, 0, 2, 1, 0, 0, 6, 5, 6, 1, 2, 3, 0, 0,
96cabdff1aSopenharmony_ci    0, 3, 0, 0, 28, 0, 2, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 23, 0, 0, 0, 0,
97cabdff1aSopenharmony_ci    0, 21, 1, 0, 3, 24, 2, 0, 0, 7, 0, 0, 1, 5, 1, 2, 0, 5
98cabdff1aSopenharmony_ci};
99cabdff1aSopenharmony_ci
100cabdff1aSopenharmony_cistatic const int probs_sat[] = {
101cabdff1aSopenharmony_ci    74, 8, 14, 7, 9345, 40, 0, 2014, 2, 1, 115, 0, 2, 1, 194, 388, 20, 0, 0, 2, 1, 121,
102cabdff1aSopenharmony_ci    1, 1583, 0, 16, 21, 2, 132, 2, 15, 9, 13, 1, 0, 2293, 2, 8, 5, 2, 30, 0, 0, 4, 54,
103cabdff1aSopenharmony_ci    783, 4, 1, 2, 4, 0, 22, 93, 1, 143, 19, 0, 36, 32, 4, 6, 33, 3, 45, 0, 8, 1, 0, 18,
104cabdff1aSopenharmony_ci    17, 1, 0, 1, 0, 0, 1, 1004, 38, 3, 8, 90, 23, 0, 2819, 3, 0, 970, 158, 9, 6, 4, 48,
105cabdff1aSopenharmony_ci    4, 0, 1, 0, 0, 60, 3, 62, 0, 2, 2, 2, 279, 66, 16, 1, 20, 0, 7, 9, 32, 1411, 6, 3,
106cabdff1aSopenharmony_ci    27, 1, 5, 49, 0, 0, 0, 0, 0, 2, 10, 1, 1, 2, 3, 801, 3, 25, 5, 1, 1, 0, 632, 0, 14,
107cabdff1aSopenharmony_ci    18, 5, 8, 200, 4, 4, 22, 12, 0, 4, 1, 0, 2, 4, 9, 3, 16, 7, 2, 2, 213, 0, 2, 620,
108cabdff1aSopenharmony_ci    39303, 0, 1, 0, 2, 1, 183781, 1, 0, 0, 0, 94, 7, 3, 4, 0, 4, 306, 43, 352, 76, 34,
109cabdff1aSopenharmony_ci    13, 11, 0, 51, 1, 13, 19, 0, 26, 0, 7276, 4, 207, 31, 1, 2, 4, 6, 19, 8, 17, 4, 6,
110cabdff1aSopenharmony_ci    0, 1085, 0, 0, 0, 3, 489, 36, 1, 0, 1, 9420, 294, 28, 0, 57, 5, 0, 9, 2, 0, 1, 2,
111cabdff1aSopenharmony_ci    2, 0, 0, 9, 2, 29, 2, 2, 7, 0, 5, 490, 0, 7, 5, 0, 1, 8, 0, 0, 23255, 0, 1
112cabdff1aSopenharmony_ci};
113cabdff1aSopenharmony_ci
114cabdff1aSopenharmony_ci// Test the example given on @see
115cabdff1aSopenharmony_ci// http://guru.multimedia.cx/small-tasks-for-ffmpeg/
116cabdff1aSopenharmony_ciint main(int argc, char **argv)
117cabdff1aSopenharmony_ci{
118cabdff1aSopenharmony_ci    int i, ret = 0;
119cabdff1aSopenharmony_ci    // Probabilities of symbols 0..4
120cabdff1aSopenharmony_ci    PTable val_counts[] = {
121cabdff1aSopenharmony_ci        {.value = 0, .prob = 1},
122cabdff1aSopenharmony_ci        {.value = 1, .prob = 2},
123cabdff1aSopenharmony_ci        {.value = 2, .prob = 5},
124cabdff1aSopenharmony_ci        {.value = 3, .prob = 10},
125cabdff1aSopenharmony_ci        {.value = 4, .prob = 21},
126cabdff1aSopenharmony_ci    };
127cabdff1aSopenharmony_ci    // Expected code lengths for each symbol
128cabdff1aSopenharmony_ci    static const HuffTable expected[] = {
129cabdff1aSopenharmony_ci        {.code = 0, .length = 3},
130cabdff1aSopenharmony_ci        {.code = 1, .length = 3},
131cabdff1aSopenharmony_ci        {.code = 2, .length = 3},
132cabdff1aSopenharmony_ci        {.code = 3, .length = 3},
133cabdff1aSopenharmony_ci        {.code = 4, .length = 1},
134cabdff1aSopenharmony_ci    };
135cabdff1aSopenharmony_ci    // Actual code lengths
136cabdff1aSopenharmony_ci    HuffTable distincts[5];
137cabdff1aSopenharmony_ci
138cabdff1aSopenharmony_ci    // Build optimal huffman tree using an internal function, to allow for
139cabdff1aSopenharmony_ci    // smaller-than-normal test cases. This mutates val_counts by sorting.
140cabdff1aSopenharmony_ci    ff_mjpegenc_huffman_compute_bits(val_counts, distincts,
141cabdff1aSopenharmony_ci                                     FF_ARRAY_ELEMS(distincts), 3);
142cabdff1aSopenharmony_ci
143cabdff1aSopenharmony_ci    for (i = 0; i < FF_ARRAY_ELEMS(distincts); i++) {
144cabdff1aSopenharmony_ci        if (distincts[i].code != expected[i].code ||
145cabdff1aSopenharmony_ci            distincts[i].length != expected[i].length) {
146cabdff1aSopenharmony_ci            fprintf(stderr,
147cabdff1aSopenharmony_ci                    "Built huffman does not equal expectations. "
148cabdff1aSopenharmony_ci                    "Expected: code %d probability %d, "
149cabdff1aSopenharmony_ci                    "Actual: code %d probability %d\n",
150cabdff1aSopenharmony_ci                    expected[i].code, expected[i].length,
151cabdff1aSopenharmony_ci                    distincts[i].code, distincts[i].length);
152cabdff1aSopenharmony_ci            ret = 1;
153cabdff1aSopenharmony_ci        }
154cabdff1aSopenharmony_ci    }
155cabdff1aSopenharmony_ci
156cabdff1aSopenharmony_ci    // Check handling of zero probabilities
157cabdff1aSopenharmony_ci    if (check_lengths(16, 18, probs_zeroes, FF_ARRAY_ELEMS(probs_zeroes)))
158cabdff1aSopenharmony_ci        ret = 1;
159cabdff1aSopenharmony_ci    // Check skewed distribution over 256 without saturated lengths
160cabdff1aSopenharmony_ci    if (check_lengths(16, 41282, probs_skewed, FF_ARRAY_ELEMS(probs_skewed)))
161cabdff1aSopenharmony_ci        ret = 1;
162cabdff1aSopenharmony_ci    // Check skewed distribution over 256 with saturated lengths
163cabdff1aSopenharmony_ci    if (check_lengths(16, 669904, probs_sat, FF_ARRAY_ELEMS(probs_sat)))
164cabdff1aSopenharmony_ci        ret = 1;
165cabdff1aSopenharmony_ci
166cabdff1aSopenharmony_ci    return ret;
167cabdff1aSopenharmony_ci}
168