annotate libaf/af_scaletempo.c @ 35955:7d9406af9f09

configure: Fix mingw vs. mingw32 typo in libdvdcss check.
author diego
date Sat, 23 Mar 2013 18:29:58 +0000
parents a93891202051
children a88581872033
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
1 /*
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
2 * scaletempo audio filter
26740
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
3 *
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
4 * scale tempo while maintaining pitch
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
5 * (WSOLA technique with cross correlation)
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
6 * inspired by SoundTouch library by Olli Parviainen
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
7 *
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
8 * basic algorithm
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
9 * - produce 'stride' output samples per loop
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
10 * - consume stride*scale input samples per loop
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
11 *
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
12 * to produce smoother transitions between strides, blend next overlap
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
13 * samples from last stride with correlated samples of current input
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
14 *
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
15 * Copyright (c) 2007 Robert Juliano
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
16 *
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
17 * This file is part of MPlayer.
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
18 *
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
19 * MPlayer is free software; you can redistribute it and/or modify
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
20 * it under the terms of the GNU General Public License as published by
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
21 * the Free Software Foundation; either version 2 of the License, or
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
22 * (at your option) any later version.
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
23 *
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
24 * MPlayer is distributed in the hope that it will be useful,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
25 * but WITHOUT ANY WARRANTY; without even the implied warranty of
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
26 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
27 * GNU General Public License for more details.
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
28 *
26740
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
29 * You should have received a copy of the GNU General Public License along
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
30 * with MPlayer; if not, write to the Free Software Foundation, Inc.,
b3a38b361fef Use standard license headers with standard formatting.
diego
parents: 25188
diff changeset
31 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
32 */
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
33
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
34 #include <stdlib.h>
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
35 #include <string.h>
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
36 #include <limits.h>
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
37
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
38 #include "af.h"
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
39 #include "libavutil/common.h"
34174
a93891202051 Add missing mp_msg.h #includes, remove some unnecessary ones.
diego
parents: 29049
diff changeset
40 #include "mp_msg.h"
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
41 #include "subopt-helper.h"
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
42 #include "help_mp.h"
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
43
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
44 // Data for specific instances of this filter
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
45 typedef struct af_scaletempo_s
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
46 {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
47 // stride
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
48 float scale;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
49 float speed;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
50 float frames_stride_scaled;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
51 float frames_stride_error;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
52 int bytes_per_frame;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
53 int bytes_stride;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
54 float bytes_stride_scaled;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
55 int bytes_queue;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
56 int bytes_queued;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
57 int bytes_to_slide;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
58 int8_t* buf_queue;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
59 // overlap
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
60 int samples_overlap;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
61 int samples_standing;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
62 int bytes_overlap;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
63 int bytes_standing;
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
64 void* buf_overlap;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
65 void* table_blend;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
66 void (*output_overlap)(struct af_scaletempo_s* s, void* out_buf, int bytes_off);
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
67 // best overlap
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
68 int frames_search;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
69 int num_channels;
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
70 void* buf_pre_corr;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
71 void* table_window;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
72 int (*best_overlap_offset)(struct af_scaletempo_s* s);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
73 // command line
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
74 float scale_nominal;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
75 float ms_stride;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
76 float percent_overlap;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
77 float ms_search;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
78 short speed_tempo;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
79 short speed_pitch;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
80 } af_scaletempo_t;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
81
24899
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
82 static int fill_queue(struct af_instance_s* af, af_data_t* data, int offset)
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
83 {
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
84 af_scaletempo_t* s = af->setup;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
85 int bytes_in = data->len - offset;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
86 int offset_unchanged = offset;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
87
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
88 if (s->bytes_to_slide > 0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
89 if (s->bytes_to_slide < s->bytes_queued) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
90 int bytes_move = s->bytes_queued - s->bytes_to_slide;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
91 memmove(s->buf_queue,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
92 s->buf_queue + s->bytes_to_slide,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
93 bytes_move);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
94 s->bytes_to_slide = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
95 s->bytes_queued = bytes_move;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
96 } else {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
97 int bytes_skip;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
98 s->bytes_to_slide -= s->bytes_queued;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
99 bytes_skip = FFMIN(s->bytes_to_slide, bytes_in);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
100 s->bytes_queued = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
101 s->bytes_to_slide -= bytes_skip;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
102 offset += bytes_skip;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
103 bytes_in -= bytes_skip;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
104 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
105 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
106
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
107 if (bytes_in > 0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
108 int bytes_copy = FFMIN(s->bytes_queue - s->bytes_queued, bytes_in);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
109 memcpy(s->buf_queue + s->bytes_queued,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
110 (int8_t*)data->audio + offset,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
111 bytes_copy);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
112 s->bytes_queued += bytes_copy;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
113 offset += bytes_copy;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
114 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
115
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
116 return offset - offset_unchanged;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
117 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
118
25076
dfd3c5702d32 Add padding and unroll loop 4x for at least another 10% speedup
reimar
parents: 25075
diff changeset
119 #define UNROLL_PADDING (4*4)
dfd3c5702d32 Add padding and unroll loop 4x for at least another 10% speedup
reimar
parents: 25075
diff changeset
120
24899
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
121 static int best_overlap_offset_float(af_scaletempo_t* s)
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
122 {
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
123 float *pw, *po, *ppc, *search_start;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
124 float best_corr = INT_MIN;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
125 int best_off = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
126 int i, off;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
127
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
128 pw = s->table_window;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
129 po = s->buf_overlap;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
130 po += s->num_channels;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
131 ppc = s->buf_pre_corr;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
132 for (i=s->num_channels; i<s->samples_overlap; i++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
133 *ppc++ = *pw++ * *po++;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
134 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
135
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
136 search_start = (float*)s->buf_queue + s->num_channels;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
137 for (off=0; off<s->frames_search; off++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
138 float corr = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
139 float* ps = search_start;
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
140 ppc = s->buf_pre_corr;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
141 for (i=s->num_channels; i<s->samples_overlap; i++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
142 corr += *ppc++ * *ps++;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
143 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
144 if (corr > best_corr) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
145 best_corr = corr;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
146 best_off = off;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
147 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
148 search_start += s->num_channels;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
149 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
150
24897
d4a8f31df9ba af_scaletempo: Fix audio copy position
uau
parents: 24896
diff changeset
151 return best_off * 4 * s->num_channels;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
152 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
153
24899
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
154 static int best_overlap_offset_s16(af_scaletempo_t* s)
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
155 {
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
156 int32_t *pw, *ppc;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
157 int16_t *po, *search_start;
25075
c18499a044ce Change to a 64 bit accumulation variable instead of shifting.
reimar
parents: 25074
diff changeset
158 int64_t best_corr = INT64_MIN;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
159 int best_off = 0;
25073
21227667b70a Use "long" instead of "int" for innermost loop variable.
reimar
parents: 25072
diff changeset
160 int off;
21227667b70a Use "long" instead of "int" for innermost loop variable.
reimar
parents: 25072
diff changeset
161 long i;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
162
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
163 pw = s->table_window;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
164 po = s->buf_overlap;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
165 po += s->num_channels;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
166 ppc = s->buf_pre_corr;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
167 for (i=s->num_channels; i<s->samples_overlap; i++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
168 *ppc++ = ( *pw++ * *po++ ) >> 15;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
169 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
170
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
171 search_start = (int16_t*)s->buf_queue + s->num_channels;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
172 for (off=0; off<s->frames_search; off++) {
25075
c18499a044ce Change to a 64 bit accumulation variable instead of shifting.
reimar
parents: 25074
diff changeset
173 int64_t corr = 0;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
174 int16_t* ps = search_start;
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
175 ppc = s->buf_pre_corr;
25075
c18499a044ce Change to a 64 bit accumulation variable instead of shifting.
reimar
parents: 25074
diff changeset
176 ppc += s->samples_overlap - s->num_channels;
c18499a044ce Change to a 64 bit accumulation variable instead of shifting.
reimar
parents: 25074
diff changeset
177 ps += s->samples_overlap - s->num_channels;
25072
6c3f9215d606 Rearrange scaletempo inner loop.
reimar
parents: 25070
diff changeset
178 i = -(s->samples_overlap - s->num_channels);
6c3f9215d606 Rearrange scaletempo inner loop.
reimar
parents: 25070
diff changeset
179 do {
25076
dfd3c5702d32 Add padding and unroll loop 4x for at least another 10% speedup
reimar
parents: 25075
diff changeset
180 corr += ppc[i+0] * ps[i+0];
dfd3c5702d32 Add padding and unroll loop 4x for at least another 10% speedup
reimar
parents: 25075
diff changeset
181 corr += ppc[i+1] * ps[i+1];
dfd3c5702d32 Add padding and unroll loop 4x for at least another 10% speedup
reimar
parents: 25075
diff changeset
182 corr += ppc[i+2] * ps[i+2];
dfd3c5702d32 Add padding and unroll loop 4x for at least another 10% speedup
reimar
parents: 25075
diff changeset
183 corr += ppc[i+3] * ps[i+3];
dfd3c5702d32 Add padding and unroll loop 4x for at least another 10% speedup
reimar
parents: 25075
diff changeset
184 i += 4;
dfd3c5702d32 Add padding and unroll loop 4x for at least another 10% speedup
reimar
parents: 25075
diff changeset
185 } while (i < 0);
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
186 if (corr > best_corr) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
187 best_corr = corr;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
188 best_off = off;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
189 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
190 search_start += s->num_channels;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
191 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
192
24897
d4a8f31df9ba af_scaletempo: Fix audio copy position
uau
parents: 24896
diff changeset
193 return best_off * 2 * s->num_channels;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
194 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
195
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
196 static void output_overlap_float(af_scaletempo_t* s, void* buf_out,
24899
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
197 int bytes_off)
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
198 {
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
199 float* pout = buf_out;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
200 float* pb = s->table_blend;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
201 float* po = s->buf_overlap;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
202 float* pin = (float*)(s->buf_queue + bytes_off);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
203 int i;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
204 for (i=0; i<s->samples_overlap; i++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
205 *pout++ = *po - *pb++ * ( *po - *pin++ ); po++;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
206 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
207 }
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
208 static void output_overlap_s16(af_scaletempo_t* s, void* buf_out,
24899
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
209 int bytes_off)
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
210 {
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
211 int16_t* pout = buf_out;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
212 int32_t* pb = s->table_blend;
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
213 int16_t* po = s->buf_overlap;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
214 int16_t* pin = (int16_t*)(s->buf_queue + bytes_off);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
215 int i;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
216 for (i=0; i<s->samples_overlap; i++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
217 *pout++ = *po - ( ( *pb++ * ( *po - *pin++ ) ) >> 16 ); po++;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
218 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
219 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
220
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
221 // Filter data through filter
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
222 static af_data_t* play(struct af_instance_s* af, af_data_t* data)
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
223 {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
224 af_scaletempo_t* s = af->setup;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
225 int offset_in;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
226 int max_bytes_out;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
227 int8_t* pout;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
228
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
229 if (s->scale == 1.0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
230 return data;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
231 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
232
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
233 // RESIZE_LOCAL_BUFFER - can't use macro
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
234 max_bytes_out = ((int)(data->len / s->bytes_stride_scaled) + 1) * s->bytes_stride;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
235 if (max_bytes_out > af->data->len) {
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
236 mp_msg(MSGT_AFILTER, MSGL_V, "[libaf] Reallocating memory in module %s, "
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
237 "old len = %i, new len = %i\n",af->info->name,af->data->len,max_bytes_out);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
238 af->data->audio = realloc(af->data->audio, max_bytes_out);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
239 if (!af->data->audio) {
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
240 mp_msg(MSGT_AFILTER, MSGL_FATAL, "[libaf] Could not allocate memory\n");
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
241 return NULL;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
242 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
243 af->data->len = max_bytes_out;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
244 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
245
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
246 offset_in = fill_queue(af, data, 0);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
247 pout = af->data->audio;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
248 while (s->bytes_queued >= s->bytes_queue) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
249 int ti;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
250 float tf;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
251 int bytes_off = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
252
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
253 // output stride
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
254 if (s->output_overlap) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
255 if (s->best_overlap_offset)
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
256 bytes_off = s->best_overlap_offset(s);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
257 s->output_overlap(s, pout, bytes_off);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
258 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
259 memcpy(pout + s->bytes_overlap,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
260 s->buf_queue + bytes_off + s->bytes_overlap,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
261 s->bytes_standing);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
262 pout += s->bytes_stride;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
263
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
264 // input stride
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
265 memcpy(s->buf_overlap,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
266 s->buf_queue + bytes_off + s->bytes_stride,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
267 s->bytes_overlap);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
268 tf = s->frames_stride_scaled + s->frames_stride_error;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
269 ti = (int)tf;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
270 s->frames_stride_error = tf - ti;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
271 s->bytes_to_slide = ti * s->bytes_per_frame;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
272
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
273 offset_in += fill_queue(af, data, offset_in);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
274 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
275
24900
9079c9745ff9 A/V sync: take audio filter buffers into account
uau
parents: 24899
diff changeset
276 // This filter can have a negative delay when scale > 1:
9079c9745ff9 A/V sync: take audio filter buffers into account
uau
parents: 24899
diff changeset
277 // output corresponding to some length of input can be decided and written
9079c9745ff9 A/V sync: take audio filter buffers into account
uau
parents: 24899
diff changeset
278 // after receiving only a part of that input.
9079c9745ff9 A/V sync: take audio filter buffers into account
uau
parents: 24899
diff changeset
279 af->delay = s->bytes_queued - s->bytes_to_slide;
9079c9745ff9 A/V sync: take audio filter buffers into account
uau
parents: 24899
diff changeset
280
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
281 data->audio = af->data->audio;
24899
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
282 data->len = pout - (int8_t *)af->data->audio;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
283 return data;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
284 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
285
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
286 // Initialization and runtime control
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
287 static int control(struct af_instance_s* af, int cmd, void* arg)
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
288 {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
289 af_scaletempo_t* s = af->setup;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
290 switch(cmd){
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
291 case AF_CONTROL_REINIT:{
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
292 af_data_t* data = (af_data_t*)arg;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
293 float srate = data->rate / 1000;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
294 int nch = data->nch;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
295 int bps;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
296 int use_int = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
297 int frames_stride, frames_overlap;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
298 int i, j;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
299
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
300 mp_msg(MSGT_AFILTER, MSGL_V,
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
301 "[scaletempo] %.3f speed * %.3f scale_nominal = %.3f\n",
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
302 s->speed, s->scale_nominal, s->scale);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
303
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
304 if (s->scale == 1.0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
305 if (s->speed_tempo && s->speed_pitch)
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
306 return AF_DETACH;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
307 memcpy(af->data, data, sizeof(af_data_t));
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
308 return af_test_output(af, data);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
309 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
310
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
311 af->data->rate = data->rate;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
312 af->data->nch = data->nch;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
313 if ( data->format == AF_FORMAT_S16_LE
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
314 || data->format == AF_FORMAT_S16_BE ) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
315 use_int = 1;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
316 af->data->format = AF_FORMAT_S16_NE;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
317 af->data->bps = bps = 2;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
318 } else {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
319 af->data->format = AF_FORMAT_FLOAT_NE;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
320 af->data->bps = bps = 4;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
321 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
322
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
323 frames_stride = srate * s->ms_stride;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
324 s->bytes_stride = frames_stride * bps * nch;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
325 s->bytes_stride_scaled = s->scale * s->bytes_stride;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
326 s->frames_stride_scaled = s->scale * frames_stride;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
327 s->frames_stride_error = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
328 af->mul = (double)s->bytes_stride / s->bytes_stride_scaled;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
329
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
330 frames_overlap = frames_stride * s->percent_overlap;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
331 if (frames_overlap <= 0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
332 s->bytes_standing = s->bytes_stride;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
333 s->samples_standing = s->bytes_standing / bps;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
334 s->output_overlap = NULL;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
335 } else {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
336 s->samples_overlap = frames_overlap * nch;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
337 s->bytes_overlap = frames_overlap * nch * bps;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
338 s->bytes_standing = s->bytes_stride - s->bytes_overlap;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
339 s->samples_standing = s->bytes_standing / bps;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
340 s->buf_overlap = realloc(s->buf_overlap, s->bytes_overlap);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
341 s->table_blend = realloc(s->table_blend, s->bytes_overlap * 4);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
342 if(!s->buf_overlap || !s->table_blend) {
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
343 mp_msg(MSGT_AFILTER, MSGL_FATAL, "[scaletempo] Out of memory\n");
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
344 return AF_ERROR;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
345 }
25188
32b0428fbd5a 100l, bzero is deprecated, use memset instead
reimar
parents: 25076
diff changeset
346 memset(s->buf_overlap, 0, s->bytes_overlap);
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
347 if (use_int) {
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
348 int32_t* pb = s->table_blend;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
349 int64_t blend = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
350 for (i=0; i<frames_overlap; i++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
351 int32_t v = blend / frames_overlap;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
352 for (j=0; j<nch; j++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
353 *pb++ = v;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
354 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
355 blend += 65536; // 2^16
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
356 }
24899
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
357 s->output_overlap = output_overlap_s16;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
358 } else {
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
359 float* pb = s->table_blend;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
360 for (i=0; i<frames_overlap; i++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
361 float v = i / (float)frames_overlap;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
362 for (j=0; j<nch; j++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
363 *pb++ = v;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
364 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
365 }
24899
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
366 s->output_overlap = output_overlap_float;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
367 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
368 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
369
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
370 s->frames_search = (frames_overlap > 1) ? srate * s->ms_search : 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
371 if (s->frames_search <= 0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
372 s->best_overlap_offset = NULL;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
373 } else {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
374 if (use_int) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
375 int64_t t = frames_overlap;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
376 int32_t n = 8589934588LL / (t * t); // 4 * (2^31 - 1) / t^2
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
377 int32_t* pw;
25076
dfd3c5702d32 Add padding and unroll loop 4x for at least another 10% speedup
reimar
parents: 25075
diff changeset
378 s->buf_pre_corr = realloc(s->buf_pre_corr, s->bytes_overlap * 2 + UNROLL_PADDING);
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
379 s->table_window = realloc(s->table_window, s->bytes_overlap * 2 - nch * bps * 2);
25070
60b34bf9a41e 100l, someone mixed up && and ||, so if allocation of only one buffers failed
reimar
parents: 25069
diff changeset
380 if(!s->buf_pre_corr || !s->table_window) {
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
381 mp_msg(MSGT_AFILTER, MSGL_FATAL, "[scaletempo] Out of memory\n");
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
382 return AF_ERROR;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
383 }
25076
dfd3c5702d32 Add padding and unroll loop 4x for at least another 10% speedup
reimar
parents: 25075
diff changeset
384 memset((char *)s->buf_pre_corr + s->bytes_overlap * 2, 0, UNROLL_PADDING);
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
385 pw = s->table_window;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
386 for (i=1; i<frames_overlap; i++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
387 int32_t v = ( i * (t - i) * n ) >> 15;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
388 for (j=0; j<nch; j++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
389 *pw++ = v;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
390 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
391 }
24899
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
392 s->best_overlap_offset = best_overlap_offset_s16;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
393 } else {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
394 float* pw;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
395 s->buf_pre_corr = realloc(s->buf_pre_corr, s->bytes_overlap);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
396 s->table_window = realloc(s->table_window, s->bytes_overlap - nch * bps);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
397 if(!s->buf_pre_corr || !s->table_window) {
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
398 mp_msg(MSGT_AFILTER, MSGL_FATAL, "[scaletempo] Out of memory\n");
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
399 return AF_ERROR;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
400 }
25069
d2f7027f0556 Avoid some casts by changing int8_t* to void* in af_scaletempo
reimar
parents: 24900
diff changeset
401 pw = s->table_window;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
402 for (i=1; i<frames_overlap; i++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
403 float v = i * (frames_overlap - i);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
404 for (j=0; j<nch; j++) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
405 *pw++ = v;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
406 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
407 }
24899
3245638ab801 af_scaletempo: code cleanup
uau
parents: 24898
diff changeset
408 s->best_overlap_offset = best_overlap_offset_float;
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
409 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
410 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
411
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
412 s->bytes_per_frame = bps * nch;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
413 s->num_channels = nch;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
414
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
415 s->bytes_queue
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
416 = (s->frames_search + frames_stride + frames_overlap) * bps * nch;
25076
dfd3c5702d32 Add padding and unroll loop 4x for at least another 10% speedup
reimar
parents: 25075
diff changeset
417 s->buf_queue = realloc(s->buf_queue, s->bytes_queue + UNROLL_PADDING);
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
418 if(!s->buf_queue) {
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
419 mp_msg(MSGT_AFILTER, MSGL_FATAL, "[scaletempo] Out of memory\n");
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
420 return AF_ERROR;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
421 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
422
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
423 mp_msg (MSGT_AFILTER, MSGL_DBG2, "[scaletempo] "
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
424 "%.2f stride_in, %i stride_out, %i standing, "
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
425 "%i overlap, %i search, %i queue, %s mode\n",
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
426 s->frames_stride_scaled,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
427 (int)(s->bytes_stride / nch / bps),
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
428 (int)(s->bytes_standing / nch / bps),
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
429 (int)(s->bytes_overlap / nch / bps),
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
430 s->frames_search,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
431 (int)(s->bytes_queue / nch / bps),
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
432 (use_int?"s16":"float"));
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
433
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
434 return af_test_output(af, (af_data_t*)arg);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
435 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
436 case AF_CONTROL_PLAYBACK_SPEED | AF_CONTROL_SET:{
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
437 if (s->speed_tempo) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
438 if (s->speed_pitch) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
439 break;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
440 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
441 s->speed = *(float*)arg;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
442 s->scale = s->speed * s->scale_nominal;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
443 } else {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
444 if (s->speed_pitch) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
445 s->speed = 1 / *(float*)arg;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
446 s->scale = s->speed * s->scale_nominal;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
447 break;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
448 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
449 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
450 return AF_OK;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
451 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
452 case AF_CONTROL_SCALETEMPO_AMOUNT | AF_CONTROL_SET:{
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
453 s->scale = *(float*)arg;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
454 s->scale = s->speed * s->scale_nominal;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
455 return AF_OK;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
456 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
457 case AF_CONTROL_SCALETEMPO_AMOUNT | AF_CONTROL_GET:
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
458 *(float*)arg = s->scale;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
459 return AF_OK;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
460 case AF_CONTROL_COMMAND_LINE:{
24898
0701957d2bf7 af_scaletempo: Fix crash in option parsing
uau
parents: 24897
diff changeset
461 strarg_t speed = {};
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
462 opt_t subopts[] = {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
463 {"scale", OPT_ARG_FLOAT, &s->scale_nominal, NULL},
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
464 {"stride", OPT_ARG_FLOAT, &s->ms_stride, NULL},
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
465 {"overlap", OPT_ARG_FLOAT, &s->percent_overlap, NULL},
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
466 {"search", OPT_ARG_FLOAT, &s->ms_search, NULL},
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
467 {"speed", OPT_ARG_STR, &speed, NULL},
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
468 {NULL},
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
469 };
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
470 if (subopt_parse(arg, subopts) != 0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
471 return AF_ERROR;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
472 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
473 if (s->scale_nominal <= 0) {
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
474 mp_msg(MSGT_AFILTER, MSGL_ERR, "[scaletempo] "
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
475 MSGTR_ErrorParsingCommandLine ": " MSGTR_AF_ValueOutOfRange
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
476 ": scale > 0\n");
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
477 return AF_ERROR;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
478 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
479 if (s->ms_stride <= 0) {
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
480 mp_msg(MSGT_AFILTER, MSGL_ERR, "[scaletempo] "
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
481 MSGTR_ErrorParsingCommandLine ": " MSGTR_AF_ValueOutOfRange
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
482 ": stride > 0\n");
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
483 return AF_ERROR;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
484 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
485 if (s->percent_overlap < 0 || s->percent_overlap > 1) {
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
486 mp_msg(MSGT_AFILTER, MSGL_ERR, "[scaletempo] "
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
487 MSGTR_ErrorParsingCommandLine ": " MSGTR_AF_ValueOutOfRange
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
488 ": 0 <= overlap <= 1\n");
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
489 return AF_ERROR;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
490 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
491 if (s->ms_search < 0) {
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
492 mp_msg(MSGT_AFILTER, MSGL_ERR, "[scaletempo] "
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
493 MSGTR_ErrorParsingCommandLine ": " MSGTR_AF_ValueOutOfRange
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
494 ": search >= 0\n");
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
495 return AF_ERROR;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
496 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
497 if (speed.len > 0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
498 if (strcmp(speed.str, "pitch") == 0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
499 s->speed_tempo = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
500 s->speed_pitch = 1;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
501 } else if (strcmp(speed.str, "tempo") == 0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
502 s->speed_tempo = 1;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
503 s->speed_pitch = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
504 } else if (strcmp(speed.str, "none") == 0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
505 s->speed_tempo = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
506 s->speed_pitch = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
507 } else if (strcmp(speed.str, "both") == 0) {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
508 s->speed_tempo = 1;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
509 s->speed_pitch = 1;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
510 } else {
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
511 mp_msg(MSGT_AFILTER, MSGL_ERR, "[scaletempo] "
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
512 MSGTR_ErrorParsingCommandLine ": " MSGTR_AF_ValueOutOfRange
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
513 ": speed=[pitch|tempo|none|both]\n");
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
514 return AF_ERROR;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
515 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
516 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
517 s->scale = s->speed * s->scale_nominal;
29049
8c706ce21c6f Remove af_msg special-casing API in libaf.
bircoph
parents: 26740
diff changeset
518 mp_msg(MSGT_AFILTER, MSGL_DBG2, "[scaletempo] %6.3f scale, %6.2f stride, %6.2f overlap, %6.2f search, speed = %s\n", s->scale_nominal, s->ms_stride, s->percent_overlap, s->ms_search, (s->speed_tempo?(s->speed_pitch?"tempo and speed":"tempo"):(s->speed_pitch?"pitch":"none")));
24896
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
519 return AF_OK;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
520 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
521 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
522 return AF_UNKNOWN;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
523 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
524
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
525 // Deallocate memory
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
526 static void uninit(struct af_instance_s* af)
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
527 {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
528 af_scaletempo_t* s = af->setup;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
529 free(af->data->audio);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
530 free(af->data);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
531 free(s->buf_queue);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
532 free(s->buf_overlap);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
533 free(s->buf_pre_corr);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
534 free(s->table_blend);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
535 free(s->table_window);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
536 free(af->setup);
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
537 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
538
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
539 // Allocate memory and set function pointers
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
540 static int af_open(af_instance_t* af){
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
541 af_scaletempo_t* s;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
542
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
543 af->control = control;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
544 af->uninit = uninit;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
545 af->play = play;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
546 af->mul = 1;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
547 af->data = calloc(1,sizeof(af_data_t));
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
548 af->setup = calloc(1,sizeof(af_scaletempo_t));
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
549 if(af->data == NULL || af->setup == NULL)
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
550 return AF_ERROR;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
551
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
552 s = af->setup;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
553 s->scale = s->speed = s->scale_nominal = 1.0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
554 s->speed_tempo = 1;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
555 s->speed_pitch = 0;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
556 s->ms_stride = 60;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
557 s->percent_overlap = .20;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
558 s->ms_search = 14;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
559
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
560 return AF_OK;
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
561 }
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
562
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
563 // Description of this filter
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
564 af_info_t af_info_scaletempo = {
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
565 "Scale audio tempo while maintaining pitch",
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
566 "scaletempo",
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
567 "Robert Juliano",
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
568 "",
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
569 AF_FLAGS_REENTRANT,
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
570 af_open
8133163bd1dd Add audio filter scaletempo
uau
parents:
diff changeset
571 };