blob: 856450c3daa25df26f420e7a182fdef478fc8e05 [file] [log] [blame]
Glenn Kasten99e53b82012-01-19 08:59:58 -08001/*
Mathias Agopian65ab4712010-07-14 17:59:35 -07002**
3** Copyright 2007, The Android Open Source Project
4**
5** Licensed under the Apache License, Version 2.0 (the "License");
6** you may not use this file except in compliance with the License.
7** You may obtain a copy of the License at
8**
9** http://www.apache.org/licenses/LICENSE-2.0
10**
11** Unless required by applicable law or agreed to in writing, software
12** distributed under the License is distributed on an "AS IS" BASIS,
13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14** See the License for the specific language governing permissions and
15** limitations under the License.
16*/
17
18#ifndef ANDROID_AUDIO_MIXER_H
19#define ANDROID_AUDIO_MIXER_H
20
21#include <stdint.h>
22#include <sys/types.h>
23
24#include "AudioBufferProvider.h"
25#include "AudioResampler.h"
26
27namespace android {
28
29// ----------------------------------------------------------------------------
30
Mathias Agopian65ab4712010-07-14 17:59:35 -070031class AudioMixer
32{
33public:
Glenn Kasten5c94b6c2012-03-20 17:01:29 -070034 AudioMixer(size_t frameCount, uint32_t sampleRate,
35 uint32_t maxNumTracks = MAX_NUM_TRACKS);
Mathias Agopian65ab4712010-07-14 17:59:35 -070036
Glenn Kastenc19e2242012-01-30 14:54:39 -080037 /*virtual*/ ~AudioMixer(); // non-virtual saves a v-table, restore if sub-classed
Mathias Agopian65ab4712010-07-14 17:59:35 -070038
39 static const uint32_t MAX_NUM_TRACKS = 32;
40 static const uint32_t MAX_NUM_CHANNELS = 2;
41
42 static const uint16_t UNITY_GAIN = 0x1000;
43
44 enum { // names
45
Glenn Kasten9c56d4a2011-12-19 15:06:39 -080046 // track names (MAX_NUM_TRACKS units)
Mathias Agopian65ab4712010-07-14 17:59:35 -070047 TRACK0 = 0x1000,
48
Glenn Kasten1c48c3c2011-12-15 14:54:01 -080049 // 0x2000 is unused
Mathias Agopian65ab4712010-07-14 17:59:35 -070050
51 // setParameter targets
52 TRACK = 0x3000,
53 RESAMPLE = 0x3001,
54 RAMP_VOLUME = 0x3002, // ramp to new volume
55 VOLUME = 0x3003, // don't ramp
56
57 // set Parameter names
58 // for target TRACK
Jean-Michel Trivi0d255b22011-05-24 15:53:33 -070059 CHANNEL_MASK = 0x4000,
Mathias Agopian65ab4712010-07-14 17:59:35 -070060 FORMAT = 0x4001,
61 MAIN_BUFFER = 0x4002,
62 AUX_BUFFER = 0x4003,
Glenn Kasten362c4e62011-12-14 10:28:06 -080063 // for target RESAMPLE
Mathias Agopian65ab4712010-07-14 17:59:35 -070064 SAMPLE_RATE = 0x4100,
Eric Laurent243f5f92011-02-28 16:52:51 -080065 RESET = 0x4101,
Glenn Kasten362c4e62011-12-14 10:28:06 -080066 // for target RAMP_VOLUME and VOLUME (8 channels max)
Mathias Agopian65ab4712010-07-14 17:59:35 -070067 VOLUME0 = 0x4200,
68 VOLUME1 = 0x4201,
69 AUXLEVEL = 0x4210,
70 };
71
72
Glenn Kasten9c56d4a2011-12-19 15:06:39 -080073 // For all APIs with "name": TRACK0 <= name < TRACK0 + MAX_NUM_TRACKS
Glenn Kasten17a736c2012-02-14 08:52:15 -080074
75 // Allocate a track name. Returns new track name if successful, -1 on failure.
Mathias Agopian65ab4712010-07-14 17:59:35 -070076 int getTrackName();
Glenn Kasten17a736c2012-02-14 08:52:15 -080077
78 // Free an allocated track by name
Mathias Agopian65ab4712010-07-14 17:59:35 -070079 void deleteTrackName(int name);
80
Glenn Kasten17a736c2012-02-14 08:52:15 -080081 // Enable or disable an allocated track by name
Glenn Kasten9c56d4a2011-12-19 15:06:39 -080082 void enable(int name);
83 void disable(int name);
Mathias Agopian65ab4712010-07-14 17:59:35 -070084
Glenn Kasten9c56d4a2011-12-19 15:06:39 -080085 void setParameter(int name, int target, int param, void *value);
Mathias Agopian65ab4712010-07-14 17:59:35 -070086
Glenn Kasten9c56d4a2011-12-19 15:06:39 -080087 void setBufferProvider(int name, AudioBufferProvider* bufferProvider);
John Grossman4ff14ba2012-02-08 16:37:41 -080088 void process(int64_t pts);
Mathias Agopian65ab4712010-07-14 17:59:35 -070089
90 uint32_t trackNames() const { return mTrackNames; }
91
Glenn Kastenc59c0042012-02-02 14:06:11 -080092 size_t getUnreleasedFrames(int name) const;
Eric Laurent071ccd52011-12-22 16:08:41 -080093
Mathias Agopian65ab4712010-07-14 17:59:35 -070094private:
95
96 enum {
97 NEEDS_CHANNEL_COUNT__MASK = 0x00000003,
98 NEEDS_FORMAT__MASK = 0x000000F0,
99 NEEDS_MUTE__MASK = 0x00000100,
100 NEEDS_RESAMPLE__MASK = 0x00001000,
101 NEEDS_AUX__MASK = 0x00010000,
102 };
103
104 enum {
105 NEEDS_CHANNEL_1 = 0x00000000,
106 NEEDS_CHANNEL_2 = 0x00000001,
107
108 NEEDS_FORMAT_16 = 0x00000010,
109
110 NEEDS_MUTE_DISABLED = 0x00000000,
111 NEEDS_MUTE_ENABLED = 0x00000100,
112
113 NEEDS_RESAMPLE_DISABLED = 0x00000000,
114 NEEDS_RESAMPLE_ENABLED = 0x00001000,
115
116 NEEDS_AUX_DISABLED = 0x00000000,
117 NEEDS_AUX_ENABLED = 0x00010000,
118 };
119
Mathias Agopian65ab4712010-07-14 17:59:35 -0700120 struct state_t;
121 struct track_t;
122
Mathias Agopian65ab4712010-07-14 17:59:35 -0700123 typedef void (*hook_t)(track_t* t, int32_t* output, size_t numOutFrames, int32_t* temp, int32_t* aux);
124 static const int BLOCKSIZE = 16; // 4 cache lines
125
126 struct track_t {
127 uint32_t needs;
128
129 union {
Glenn Kastenbf71f1e2011-12-13 11:52:35 -0800130 int16_t volume[MAX_NUM_CHANNELS]; // [0]3.12 fixed point
Mathias Agopian65ab4712010-07-14 17:59:35 -0700131 int32_t volumeRL;
132 };
133
Glenn Kastenbf71f1e2011-12-13 11:52:35 -0800134 int32_t prevVolume[MAX_NUM_CHANNELS];
Mathias Agopian65ab4712010-07-14 17:59:35 -0700135
Glenn Kasten3b81aca2012-01-27 15:26:23 -0800136 // 16-byte boundary
137
Glenn Kastenbf71f1e2011-12-13 11:52:35 -0800138 int32_t volumeInc[MAX_NUM_CHANNELS];
Mathias Agopian65ab4712010-07-14 17:59:35 -0700139 int32_t auxInc;
140 int32_t prevAuxLevel;
141
Glenn Kasten3b81aca2012-01-27 15:26:23 -0800142 // 16-byte boundary
143
144 int16_t auxLevel; // 0 <= auxLevel <= MAX_GAIN_INT, but signed for mul performance
Mathias Agopian65ab4712010-07-14 17:59:35 -0700145 uint16_t frameCount;
146
Glenn Kasten3b81aca2012-01-27 15:26:23 -0800147 uint8_t channelCount; // 1 or 2, redundant with (needs & NEEDS_CHANNEL_COUNT__MASK)
148 uint8_t format; // always 16
149 uint16_t enabled; // actually bool
150 uint32_t channelMask; // currently under-used
Mathias Agopian65ab4712010-07-14 17:59:35 -0700151
152 AudioBufferProvider* bufferProvider;
Glenn Kasten3b81aca2012-01-27 15:26:23 -0800153
154 // 16-byte boundary
155
156 mutable AudioBufferProvider::Buffer buffer; // 8 bytes
Mathias Agopian65ab4712010-07-14 17:59:35 -0700157
158 hook_t hook;
Glenn Kasten54c3b662012-01-06 07:46:30 -0800159 const void* in; // current location in buffer
Mathias Agopian65ab4712010-07-14 17:59:35 -0700160
Glenn Kasten3b81aca2012-01-27 15:26:23 -0800161 // 16-byte boundary
162
Mathias Agopian65ab4712010-07-14 17:59:35 -0700163 AudioResampler* resampler;
164 uint32_t sampleRate;
165 int32_t* mainBuffer;
166 int32_t* auxBuffer;
167
Glenn Kasten3b81aca2012-01-27 15:26:23 -0800168 // 16-byte boundary
169
John Grossman4ff14ba2012-02-08 16:37:41 -0800170 uint64_t localTimeFreq;
171
Glenn Kasten3b81aca2012-01-27 15:26:23 -0800172 int64_t padding;
173
174 // 16-byte boundary
175
Mathias Agopian65ab4712010-07-14 17:59:35 -0700176 bool setResampler(uint32_t sampleRate, uint32_t devSampleRate);
Glenn Kastenc59c0042012-02-02 14:06:11 -0800177 bool doesResample() const { return resampler != NULL; }
178 void resetResampler() { if (resampler != NULL) resampler->reset(); }
Mathias Agopian65ab4712010-07-14 17:59:35 -0700179 void adjustVolumeRamp(bool aux);
Glenn Kastenc59c0042012-02-02 14:06:11 -0800180 size_t getUnreleasedFrames() const { return resampler != NULL ?
181 resampler->getUnreleasedFrames() : 0; };
Mathias Agopian65ab4712010-07-14 17:59:35 -0700182 };
183
184 // pad to 32-bytes to fill cache line
185 struct state_t {
186 uint32_t enabledTracks;
187 uint32_t needsChanged;
188 size_t frameCount;
Glenn Kastena1117922012-01-26 10:53:32 -0800189 void (*hook)(state_t* state, int64_t pts); // one of process__*, never NULL
Mathias Agopian65ab4712010-07-14 17:59:35 -0700190 int32_t *outputTemp;
191 int32_t *resampleTemp;
192 int32_t reserved[2];
Glenn Kasten5c94b6c2012-03-20 17:01:29 -0700193 // FIXME allocate dynamically to save some memory when maxNumTracks < MAX_NUM_TRACKS
Glenn Kastenbf71f1e2011-12-13 11:52:35 -0800194 track_t tracks[MAX_NUM_TRACKS]; __attribute__((aligned(32)));
Mathias Agopian65ab4712010-07-14 17:59:35 -0700195 };
196
Glenn Kasten9c56d4a2011-12-19 15:06:39 -0800197 // bitmask of allocated track names, where bit 0 corresponds to TRACK0 etc.
Mathias Agopian65ab4712010-07-14 17:59:35 -0700198 uint32_t mTrackNames;
Glenn Kasten5c94b6c2012-03-20 17:01:29 -0700199
200 // bitmask of configured track names; ~0 if maxNumTracks == MAX_NUM_TRACKS,
201 // but will have fewer bits set if maxNumTracks < MAX_NUM_TRACKS
202 const uint32_t mConfiguredNames;
203
Mathias Agopian65ab4712010-07-14 17:59:35 -0700204 const uint32_t mSampleRate;
205
206 state_t mState __attribute__((aligned(32)));
207
208 void invalidateState(uint32_t mask);
209
210 static void track__genericResample(track_t* t, int32_t* out, size_t numFrames, int32_t* temp, int32_t* aux);
211 static void track__nop(track_t* t, int32_t* out, size_t numFrames, int32_t* temp, int32_t* aux);
212 static void track__16BitsStereo(track_t* t, int32_t* out, size_t numFrames, int32_t* temp, int32_t* aux);
213 static void track__16BitsMono(track_t* t, int32_t* out, size_t numFrames, int32_t* temp, int32_t* aux);
214 static void volumeRampStereo(track_t* t, int32_t* out, size_t frameCount, int32_t* temp, int32_t* aux);
215 static void volumeStereo(track_t* t, int32_t* out, size_t frameCount, int32_t* temp, int32_t* aux);
216
John Grossman4ff14ba2012-02-08 16:37:41 -0800217 static void process__validate(state_t* state, int64_t pts);
218 static void process__nop(state_t* state, int64_t pts);
219 static void process__genericNoResampling(state_t* state, int64_t pts);
220 static void process__genericResampling(state_t* state, int64_t pts);
221 static void process__OneTrack16BitsStereoNoResampling(state_t* state,
222 int64_t pts);
Glenn Kasten81a028f2011-12-15 09:53:12 -0800223#if 0
John Grossman4ff14ba2012-02-08 16:37:41 -0800224 static void process__TwoTracks16BitsStereoNoResampling(state_t* state,
225 int64_t pts);
Glenn Kasten81a028f2011-12-15 09:53:12 -0800226#endif
John Grossman4ff14ba2012-02-08 16:37:41 -0800227
228 static int64_t calculateOutputPTS(const track_t& t, int64_t basePTS,
229 int outputFrameIndex);
Mathias Agopian65ab4712010-07-14 17:59:35 -0700230};
231
232// ----------------------------------------------------------------------------
233}; // namespace android
234
235#endif // ANDROID_AUDIO_MIXER_H