More cosmetics
[FFMpeg-mirror/DVCPRO-HD.git] / libavcodec / ppc / dsputil_ppc.c
blob168f8d8b6d0dd62284bf4c95a2119e2bb5c30307
1 /*
2 * Copyright (c) 2002 Brian Foley
3 * Copyright (c) 2002 Dieter Shirley
4 * Copyright (c) 2003-2004 Romain Dolbeau <romain@dolbeau.org>
6 * This file is part of FFmpeg.
8 * FFmpeg is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
13 * FFmpeg is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with FFmpeg; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
23 #include "libavcodec/dsputil.h"
25 #include "dsputil_ppc.h"
27 #ifdef HAVE_ALTIVEC
28 #include "dsputil_altivec.h"
30 extern void fdct_altivec(int16_t *block);
31 extern void gmc1_altivec(uint8_t *dst, uint8_t *src, int stride, int h,
32 int x16, int y16, int rounder);
33 extern void idct_put_altivec(uint8_t *dest, int line_size, int16_t *block);
34 extern void idct_add_altivec(uint8_t *dest, int line_size, int16_t *block);
36 void dsputil_h264_init_ppc(DSPContext* c, AVCodecContext *avctx);
38 void dsputil_init_altivec(DSPContext* c, AVCodecContext *avctx);
39 void vc1dsp_init_altivec(DSPContext* c, AVCodecContext *avctx);
40 void snow_init_altivec(DSPContext* c, AVCodecContext *avctx);
41 void float_init_altivec(DSPContext* c, AVCodecContext *avctx);
42 void int_init_altivec(DSPContext* c, AVCodecContext *avctx);
44 #endif
46 int mm_flags = 0;
48 int mm_support(void)
50 int result = 0;
51 #ifdef HAVE_ALTIVEC
52 if (has_altivec()) {
53 result |= MM_ALTIVEC;
55 #endif /* result */
56 return result;
59 #ifdef CONFIG_POWERPC_PERF
60 unsigned long long perfdata[POWERPC_NUM_PMC_ENABLED][powerpc_perf_total][powerpc_data_total];
61 /* list below must match enum in dsputil_ppc.h */
62 static unsigned char* perfname[] = {
63 "ff_fft_calc_altivec",
64 "gmc1_altivec",
65 "dct_unquantize_h263_altivec",
66 "fdct_altivec",
67 "idct_add_altivec",
68 "idct_put_altivec",
69 "put_pixels16_altivec",
70 "avg_pixels16_altivec",
71 "avg_pixels8_altivec",
72 "put_pixels8_xy2_altivec",
73 "put_no_rnd_pixels8_xy2_altivec",
74 "put_pixels16_xy2_altivec",
75 "put_no_rnd_pixels16_xy2_altivec",
76 "hadamard8_diff8x8_altivec",
77 "hadamard8_diff16_altivec",
78 "avg_pixels8_xy2_altivec",
79 "clear_blocks_dcbz32_ppc",
80 "clear_blocks_dcbz128_ppc",
81 "put_h264_chroma_mc8_altivec",
82 "avg_h264_chroma_mc8_altivec",
83 "put_h264_qpel16_h_lowpass_altivec",
84 "avg_h264_qpel16_h_lowpass_altivec",
85 "put_h264_qpel16_v_lowpass_altivec",
86 "avg_h264_qpel16_v_lowpass_altivec",
87 "put_h264_qpel16_hv_lowpass_altivec",
88 "avg_h264_qpel16_hv_lowpass_altivec",
91 #include <stdio.h>
92 #endif
94 #ifdef CONFIG_POWERPC_PERF
95 void powerpc_display_perf_report(void)
97 int i, j;
98 av_log(NULL, AV_LOG_INFO, "PowerPC performance report\n Values are from the PMC registers, and represent whatever the registers are set to record.\n");
99 for(i = 0 ; i < powerpc_perf_total ; i++)
101 for (j = 0; j < POWERPC_NUM_PMC_ENABLED ; j++)
103 if (perfdata[j][i][powerpc_data_num] != (unsigned long long)0)
104 av_log(NULL, AV_LOG_INFO,
105 " Function \"%s\" (pmc%d):\n\tmin: %"PRIu64"\n\tmax: %"PRIu64"\n\tavg: %1.2lf (%"PRIu64")\n",
106 perfname[i],
107 j+1,
108 perfdata[j][i][powerpc_data_min],
109 perfdata[j][i][powerpc_data_max],
110 (double)perfdata[j][i][powerpc_data_sum] /
111 (double)perfdata[j][i][powerpc_data_num],
112 perfdata[j][i][powerpc_data_num]);
116 #endif /* CONFIG_POWERPC_PERF */
118 /* ***** WARNING ***** WARNING ***** WARNING ***** */
120 clear_blocks_dcbz32_ppc will not work properly
121 on PowerPC processors with a cache line size
122 not equal to 32 bytes.
123 Fortunately all processor used by Apple up to
124 at least the 7450 (aka second generation G4)
125 use 32 bytes cache line.
126 This is due to the use of the 'dcbz' instruction.
127 It simply clear to zero a single cache line,
128 so you need to know the cache line size to use it !
129 It's absurd, but it's fast...
131 update 24/06/2003 : Apple released yesterday the G5,
132 with a PPC970. cache line size : 128 bytes. Oups.
133 The semantic of dcbz was changed, it always clear
134 32 bytes. so the function below will work, but will
135 be slow. So I fixed check_dcbz_effect to use dcbzl,
136 which is defined to clear a cache line (as dcbz before).
137 So we still can distinguish, and use dcbz (32 bytes)
138 or dcbzl (one cache line) as required.
140 see <http://developer.apple.com/technotes/tn/tn2087.html>
141 and <http://developer.apple.com/technotes/tn/tn2086.html>
143 void clear_blocks_dcbz32_ppc(DCTELEM *blocks)
145 POWERPC_PERF_DECLARE(powerpc_clear_blocks_dcbz32, 1);
146 register int misal = ((unsigned long)blocks & 0x00000010);
147 register int i = 0;
148 POWERPC_PERF_START_COUNT(powerpc_clear_blocks_dcbz32, 1);
149 #if 1
150 if (misal) {
151 ((unsigned long*)blocks)[0] = 0L;
152 ((unsigned long*)blocks)[1] = 0L;
153 ((unsigned long*)blocks)[2] = 0L;
154 ((unsigned long*)blocks)[3] = 0L;
155 i += 16;
157 for ( ; i < sizeof(DCTELEM)*6*64-31 ; i += 32) {
158 asm volatile("dcbz %0,%1" : : "b" (blocks), "r" (i) : "memory");
160 if (misal) {
161 ((unsigned long*)blocks)[188] = 0L;
162 ((unsigned long*)blocks)[189] = 0L;
163 ((unsigned long*)blocks)[190] = 0L;
164 ((unsigned long*)blocks)[191] = 0L;
165 i += 16;
167 #else
168 memset(blocks, 0, sizeof(DCTELEM)*6*64);
169 #endif
170 POWERPC_PERF_STOP_COUNT(powerpc_clear_blocks_dcbz32, 1);
173 /* same as above, when dcbzl clear a whole 128B cache line
174 i.e. the PPC970 aka G5 */
175 #ifdef HAVE_DCBZL
176 void clear_blocks_dcbz128_ppc(DCTELEM *blocks)
178 POWERPC_PERF_DECLARE(powerpc_clear_blocks_dcbz128, 1);
179 register int misal = ((unsigned long)blocks & 0x0000007f);
180 register int i = 0;
181 POWERPC_PERF_START_COUNT(powerpc_clear_blocks_dcbz128, 1);
182 #if 1
183 if (misal) {
184 // we could probably also optimize this case,
185 // but there's not much point as the machines
186 // aren't available yet (2003-06-26)
187 memset(blocks, 0, sizeof(DCTELEM)*6*64);
189 else
190 for ( ; i < sizeof(DCTELEM)*6*64 ; i += 128) {
191 asm volatile("dcbzl %0,%1" : : "b" (blocks), "r" (i) : "memory");
193 #else
194 memset(blocks, 0, sizeof(DCTELEM)*6*64);
195 #endif
196 POWERPC_PERF_STOP_COUNT(powerpc_clear_blocks_dcbz128, 1);
198 #else
199 void clear_blocks_dcbz128_ppc(DCTELEM *blocks)
201 memset(blocks, 0, sizeof(DCTELEM)*6*64);
203 #endif
205 #ifdef HAVE_DCBZL
206 /* check dcbz report how many bytes are set to 0 by dcbz */
207 /* update 24/06/2003 : replace dcbz by dcbzl to get
208 the intended effect (Apple "fixed" dcbz)
209 unfortunately this cannot be used unless the assembler
210 knows about dcbzl ... */
211 long check_dcbzl_effect(void)
213 register char *fakedata = av_malloc(1024);
214 register char *fakedata_middle;
215 register long zero = 0;
216 register long i = 0;
217 long count = 0;
219 if (!fakedata)
221 return 0L;
224 fakedata_middle = (fakedata + 512);
226 memset(fakedata, 0xFF, 1024);
228 /* below the constraint "b" seems to mean "Address base register"
229 in gcc-3.3 / RS/6000 speaks. seems to avoid using r0, so.... */
230 asm volatile("dcbzl %0, %1" : : "b" (fakedata_middle), "r" (zero));
232 for (i = 0; i < 1024 ; i ++)
234 if (fakedata[i] == (char)0)
235 count++;
238 av_free(fakedata);
240 return count;
242 #else
243 long check_dcbzl_effect(void)
245 return 0;
247 #endif
249 static void prefetch_ppc(void *mem, int stride, int h)
251 register const uint8_t *p = mem;
252 do {
253 asm volatile ("dcbt 0,%0" : : "r" (p));
254 p+= stride;
255 } while(--h);
258 void dsputil_init_ppc(DSPContext* c, AVCodecContext *avctx)
260 // Common optimizations whether AltiVec is available or not
261 c->prefetch = prefetch_ppc;
262 switch (check_dcbzl_effect()) {
263 case 32:
264 c->clear_blocks = clear_blocks_dcbz32_ppc;
265 break;
266 case 128:
267 c->clear_blocks = clear_blocks_dcbz128_ppc;
268 break;
269 default:
270 break;
273 #ifdef HAVE_ALTIVEC
274 if(ENABLE_H264_DECODER) dsputil_h264_init_ppc(c, avctx);
276 if (has_altivec()) {
277 mm_flags |= MM_ALTIVEC;
279 dsputil_init_altivec(c, avctx);
280 if(ENABLE_SNOW_DECODER) snow_init_altivec(c, avctx);
281 if(ENABLE_VC1_DECODER || ENABLE_WMV3_DECODER)
282 vc1dsp_init_altivec(c, avctx);
283 float_init_altivec(c, avctx);
284 int_init_altivec(c, avctx);
285 c->gmc1 = gmc1_altivec;
287 #ifdef CONFIG_ENCODERS
288 if (avctx->dct_algo == FF_DCT_AUTO ||
289 avctx->dct_algo == FF_DCT_ALTIVEC)
291 c->fdct = fdct_altivec;
293 #endif //CONFIG_ENCODERS
295 if (avctx->lowres==0)
297 if ((avctx->idct_algo == FF_IDCT_AUTO) ||
298 (avctx->idct_algo == FF_IDCT_ALTIVEC))
300 c->idct_put = idct_put_altivec;
301 c->idct_add = idct_add_altivec;
302 c->idct_permutation_type = FF_TRANSPOSE_IDCT_PERM;
306 #ifdef CONFIG_POWERPC_PERF
308 int i, j;
309 for (i = 0 ; i < powerpc_perf_total ; i++)
311 for (j = 0; j < POWERPC_NUM_PMC_ENABLED ; j++)
313 perfdata[j][i][powerpc_data_min] = 0xFFFFFFFFFFFFFFFFULL;
314 perfdata[j][i][powerpc_data_max] = 0x0000000000000000ULL;
315 perfdata[j][i][powerpc_data_sum] = 0x0000000000000000ULL;
316 perfdata[j][i][powerpc_data_num] = 0x0000000000000000ULL;
320 #endif /* CONFIG_POWERPC_PERF */
322 #endif /* HAVE_ALTIVEC */