xref: /third_party/ffmpeg/libavcodec/h263dsp.c (revision cabdff1a)
1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * This file is part of FFmpeg.
3cabdff1aSopenharmony_ci *
4cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or
5cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public
6cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either
7cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version.
8cabdff1aSopenharmony_ci *
9cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful,
10cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of
11cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12cabdff1aSopenharmony_ci * Lesser General Public License for more details.
13cabdff1aSopenharmony_ci *
14cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public
15cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software
16cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17cabdff1aSopenharmony_ci */
18cabdff1aSopenharmony_ci
19cabdff1aSopenharmony_ci#include <stdint.h>
20cabdff1aSopenharmony_ci
21cabdff1aSopenharmony_ci#include "libavutil/attributes.h"
22cabdff1aSopenharmony_ci#include "libavutil/common.h"
23cabdff1aSopenharmony_ci#include "config.h"
24cabdff1aSopenharmony_ci#include "h263dsp.h"
25cabdff1aSopenharmony_ci
26cabdff1aSopenharmony_ciconst uint8_t ff_h263_loop_filter_strength[32] = {
27cabdff1aSopenharmony_ci    0, 1, 1, 2, 2, 3, 3,  4,  4,  4,  5,  5,  6,  6,  7, 7,
28cabdff1aSopenharmony_ci    7, 8, 8, 8, 9, 9, 9, 10, 10, 10, 11, 11, 11, 12, 12, 12
29cabdff1aSopenharmony_ci};
30cabdff1aSopenharmony_ci
31cabdff1aSopenharmony_cistatic void h263_h_loop_filter_c(uint8_t *src, int stride, int qscale)
32cabdff1aSopenharmony_ci{
33cabdff1aSopenharmony_ci    int y;
34cabdff1aSopenharmony_ci    const int strength = ff_h263_loop_filter_strength[qscale];
35cabdff1aSopenharmony_ci
36cabdff1aSopenharmony_ci    for (y = 0; y < 8; y++) {
37cabdff1aSopenharmony_ci        int d1, d2, ad1;
38cabdff1aSopenharmony_ci        int p0 = src[y * stride - 2];
39cabdff1aSopenharmony_ci        int p1 = src[y * stride - 1];
40cabdff1aSopenharmony_ci        int p2 = src[y * stride + 0];
41cabdff1aSopenharmony_ci        int p3 = src[y * stride + 1];
42cabdff1aSopenharmony_ci        int d  = (p0 - p3 + 4 * (p2 - p1)) / 8;
43cabdff1aSopenharmony_ci
44cabdff1aSopenharmony_ci        if (d < -2 * strength)
45cabdff1aSopenharmony_ci            d1 = 0;
46cabdff1aSopenharmony_ci        else if (d < -strength)
47cabdff1aSopenharmony_ci            d1 = -2 * strength - d;
48cabdff1aSopenharmony_ci        else if (d < strength)
49cabdff1aSopenharmony_ci            d1 = d;
50cabdff1aSopenharmony_ci        else if (d < 2 * strength)
51cabdff1aSopenharmony_ci            d1 = 2 * strength - d;
52cabdff1aSopenharmony_ci        else
53cabdff1aSopenharmony_ci            d1 = 0;
54cabdff1aSopenharmony_ci
55cabdff1aSopenharmony_ci        p1 += d1;
56cabdff1aSopenharmony_ci        p2 -= d1;
57cabdff1aSopenharmony_ci        if (p1 & 256)
58cabdff1aSopenharmony_ci            p1 = ~(p1 >> 31);
59cabdff1aSopenharmony_ci        if (p2 & 256)
60cabdff1aSopenharmony_ci            p2 = ~(p2 >> 31);
61cabdff1aSopenharmony_ci
62cabdff1aSopenharmony_ci        src[y * stride - 1] = p1;
63cabdff1aSopenharmony_ci        src[y * stride + 0] = p2;
64cabdff1aSopenharmony_ci
65cabdff1aSopenharmony_ci        ad1 = FFABS(d1) >> 1;
66cabdff1aSopenharmony_ci
67cabdff1aSopenharmony_ci        d2 = av_clip((p0 - p3) / 4, -ad1, ad1);
68cabdff1aSopenharmony_ci
69cabdff1aSopenharmony_ci        src[y * stride - 2] = p0 - d2;
70cabdff1aSopenharmony_ci        src[y * stride + 1] = p3 + d2;
71cabdff1aSopenharmony_ci    }
72cabdff1aSopenharmony_ci}
73cabdff1aSopenharmony_ci
74cabdff1aSopenharmony_cistatic void h263_v_loop_filter_c(uint8_t *src, int stride, int qscale)
75cabdff1aSopenharmony_ci{
76cabdff1aSopenharmony_ci    int x;
77cabdff1aSopenharmony_ci    const int strength = ff_h263_loop_filter_strength[qscale];
78cabdff1aSopenharmony_ci
79cabdff1aSopenharmony_ci    for (x = 0; x < 8; x++) {
80cabdff1aSopenharmony_ci        int d1, d2, ad1;
81cabdff1aSopenharmony_ci        int p0 = src[x - 2 * stride];
82cabdff1aSopenharmony_ci        int p1 = src[x - 1 * stride];
83cabdff1aSopenharmony_ci        int p2 = src[x + 0 * stride];
84cabdff1aSopenharmony_ci        int p3 = src[x + 1 * stride];
85cabdff1aSopenharmony_ci        int d  = (p0 - p3 + 4 * (p2 - p1)) / 8;
86cabdff1aSopenharmony_ci
87cabdff1aSopenharmony_ci        if (d < -2 * strength)
88cabdff1aSopenharmony_ci            d1 = 0;
89cabdff1aSopenharmony_ci        else if (d < -strength)
90cabdff1aSopenharmony_ci            d1 = -2 * strength - d;
91cabdff1aSopenharmony_ci        else if (d < strength)
92cabdff1aSopenharmony_ci            d1 = d;
93cabdff1aSopenharmony_ci        else if (d < 2 * strength)
94cabdff1aSopenharmony_ci            d1 = 2 * strength - d;
95cabdff1aSopenharmony_ci        else
96cabdff1aSopenharmony_ci            d1 = 0;
97cabdff1aSopenharmony_ci
98cabdff1aSopenharmony_ci        p1 += d1;
99cabdff1aSopenharmony_ci        p2 -= d1;
100cabdff1aSopenharmony_ci        if (p1 & 256)
101cabdff1aSopenharmony_ci            p1 = ~(p1 >> 31);
102cabdff1aSopenharmony_ci        if (p2 & 256)
103cabdff1aSopenharmony_ci            p2 = ~(p2 >> 31);
104cabdff1aSopenharmony_ci
105cabdff1aSopenharmony_ci        src[x - 1 * stride] = p1;
106cabdff1aSopenharmony_ci        src[x + 0 * stride] = p2;
107cabdff1aSopenharmony_ci
108cabdff1aSopenharmony_ci        ad1 = FFABS(d1) >> 1;
109cabdff1aSopenharmony_ci
110cabdff1aSopenharmony_ci        d2 = av_clip((p0 - p3) / 4, -ad1, ad1);
111cabdff1aSopenharmony_ci
112cabdff1aSopenharmony_ci        src[x - 2 * stride] = p0 - d2;
113cabdff1aSopenharmony_ci        src[x + stride]     = p3 + d2;
114cabdff1aSopenharmony_ci    }
115cabdff1aSopenharmony_ci}
116cabdff1aSopenharmony_ci
117cabdff1aSopenharmony_ciav_cold void ff_h263dsp_init(H263DSPContext *ctx)
118cabdff1aSopenharmony_ci{
119cabdff1aSopenharmony_ci    ctx->h263_h_loop_filter = h263_h_loop_filter_c;
120cabdff1aSopenharmony_ci    ctx->h263_v_loop_filter = h263_v_loop_filter_c;
121cabdff1aSopenharmony_ci
122cabdff1aSopenharmony_ci#if ARCH_X86
123cabdff1aSopenharmony_ci    ff_h263dsp_init_x86(ctx);
124cabdff1aSopenharmony_ci#elif ARCH_MIPS
125cabdff1aSopenharmony_ci    ff_h263dsp_init_mips(ctx);
126cabdff1aSopenharmony_ci#endif
127cabdff1aSopenharmony_ci}
128