Channel Mixer: Rewrite upmixing, changed HRIR
Simple upmixing algorithms now use Accelerate framework functions instead of complex loops, and the HRIR filter now supports forcing stereo output to any channel output configuration that has at least front stereo speakers. Signed-off-by: Christopher Snowhill <kode54@gmail.com>CQTexperiment
parent
5bcedab274
commit
6b148fef11
|
@ -158,87 +158,53 @@ static void downmix_to_mono(const float *inBuffer, int channels, uint32_t config
|
||||||
|
|
||||||
static void upmix(const float *inBuffer, int inchannels, uint32_t inconfig, float *outBuffer, int outchannels, uint32_t outconfig, size_t count) {
|
static void upmix(const float *inBuffer, int inchannels, uint32_t inconfig, float *outBuffer, int outchannels, uint32_t outconfig, size_t count) {
|
||||||
if(inconfig == AudioConfigMono && outconfig == AudioConfigStereo) {
|
if(inconfig == AudioConfigMono && outconfig == AudioConfigStereo) {
|
||||||
for(size_t i = 0; i < count; ++i) {
|
cblas_scopy((int)count, inBuffer, 1, outBuffer, 2);
|
||||||
// upmix mono to stereo
|
cblas_scopy((int)count, inBuffer, 1, outBuffer + 1, 2);
|
||||||
float sample = inBuffer[i];
|
|
||||||
outBuffer[i * 2 + 0] = sample;
|
|
||||||
outBuffer[i * 2 + 1] = sample;
|
|
||||||
}
|
|
||||||
} else if(inconfig == AudioConfigMono && outconfig == AudioConfig4Point0) {
|
} else if(inconfig == AudioConfigMono && outconfig == AudioConfig4Point0) {
|
||||||
for(size_t i = 0; i < count; ++i) {
|
cblas_scopy((int)count, inBuffer, 1, outBuffer, 4);
|
||||||
// upmix mono to quad
|
cblas_scopy((int)count, inBuffer, 1, outBuffer + 1, 4);
|
||||||
float sample = inBuffer[i];
|
vDSP_vclr(outBuffer + 2, 4, count);
|
||||||
outBuffer[i * 4 + 0] = sample;
|
vDSP_vclr(outBuffer + 3, 4, count);
|
||||||
outBuffer[i * 4 + 1] = sample;
|
|
||||||
outBuffer[i * 4 + 2] = 0;
|
|
||||||
outBuffer[i * 4 + 3] = 0;
|
|
||||||
}
|
|
||||||
} else if(inconfig == AudioConfigMono && (outconfig & AudioChannelFrontCenter)) {
|
} else if(inconfig == AudioConfigMono && (outconfig & AudioChannelFrontCenter)) {
|
||||||
uint32_t cIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontCenter];
|
uint32_t cIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontCenter];
|
||||||
for(size_t i = 0; i < count; ++i) {
|
cblas_scopy((int)count, inBuffer, 1, outBuffer + cIndex, outchannels);
|
||||||
// upmix mono to center channel
|
for(size_t i = 0; i < cIndex; ++i) {
|
||||||
float sample = inBuffer[i];
|
vDSP_vclr(outBuffer + i, outchannels, (int)count);
|
||||||
outBuffer[i * outchannels + cIndex] = sample;
|
}
|
||||||
for(int j = 0; j < cIndex; ++j) {
|
for(size_t i = cIndex + 1; i < outchannels; ++i) {
|
||||||
outBuffer[i * outchannels + j] = 0;
|
vDSP_vclr(outBuffer + i, outchannels, (int)count);
|
||||||
}
|
|
||||||
for(int j = cIndex + 1; j < outchannels; ++j) {
|
|
||||||
outBuffer[i * outchannels + j] = 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else if(inconfig == AudioConfig4Point0 && outchannels >= 5) {
|
} else if(inconfig == AudioConfig4Point0 && outchannels >= 5) {
|
||||||
uint32_t flIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontLeft];
|
uint32_t flIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontLeft];
|
||||||
uint32_t frIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontRight];
|
uint32_t frIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontRight];
|
||||||
uint32_t blIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelBackLeft];
|
uint32_t blIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelBackLeft];
|
||||||
uint32_t brIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelBackRight];
|
uint32_t brIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelBackRight];
|
||||||
for(size_t i = 0; i < count; ++i) {
|
vDSP_vclr(outBuffer, 1, count * outchannels);
|
||||||
float fl = inBuffer[i * 4 + 0];
|
if(flIndex != ~0)
|
||||||
float fr = inBuffer[i * 4 + 1];
|
cblas_scopy((int)count, inBuffer + 0, 4, outBuffer + flIndex, outchannels);
|
||||||
float bl = inBuffer[i * 4 + 2];
|
if(frIndex != ~0)
|
||||||
float br = inBuffer[i * 4 + 3];
|
cblas_scopy((int)count, inBuffer + 1, 4, outBuffer + frIndex, outchannels);
|
||||||
memset(outBuffer + i * outchannels, 0, sizeof(float) * outchannels);
|
if(blIndex != ~0)
|
||||||
if(flIndex != ~0) {
|
cblas_scopy((int)count, inBuffer + 2, 4, outBuffer + blIndex, outchannels);
|
||||||
outBuffer[i * outchannels + flIndex] = fl;
|
if(brIndex != ~0)
|
||||||
}
|
cblas_scopy((int)count, inBuffer + 3, 4, outBuffer + brIndex, outchannels);
|
||||||
if(frIndex != ~0) {
|
|
||||||
outBuffer[i * outchannels + frIndex] = fr;
|
|
||||||
}
|
|
||||||
if(blIndex != ~0) {
|
|
||||||
outBuffer[i * outchannels + blIndex] = bl;
|
|
||||||
}
|
|
||||||
if(brIndex != ~0) {
|
|
||||||
outBuffer[i * outchannels + brIndex] = br;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if(inconfig == AudioConfig5Point0 && outchannels >= 6) {
|
} else if(inconfig == AudioConfig5Point0 && outchannels >= 6) {
|
||||||
uint32_t flIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontLeft];
|
uint32_t flIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontLeft];
|
||||||
uint32_t frIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontRight];
|
uint32_t frIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontRight];
|
||||||
uint32_t cIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontCenter];
|
uint32_t cIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontCenter];
|
||||||
uint32_t blIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelBackLeft];
|
uint32_t blIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelBackLeft];
|
||||||
uint32_t brIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelBackRight];
|
uint32_t brIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelBackRight];
|
||||||
for(size_t i = 0; i < count; ++i) {
|
vDSP_vclr(outBuffer, 1, count * outchannels);
|
||||||
float fl = inBuffer[i * 5 + 0];
|
if(flIndex != ~0)
|
||||||
float fr = inBuffer[i * 5 + 1];
|
cblas_scopy((int)count, inBuffer + 0, 5, outBuffer + flIndex, outchannels);
|
||||||
float c = inBuffer[i * 5 + 2];
|
if(frIndex != ~0)
|
||||||
float bl = inBuffer[i * 5 + 3];
|
cblas_scopy((int)count, inBuffer + 1, 5, outBuffer + frIndex, outchannels);
|
||||||
float br = inBuffer[i * 5 + 4];
|
if(cIndex != ~0)
|
||||||
memset(outBuffer + i * outchannels, 0, sizeof(float) * outchannels);
|
cblas_scopy((int)count, inBuffer + 2, 5, outBuffer + cIndex, outchannels);
|
||||||
if(flIndex != ~0) {
|
if(blIndex != ~0)
|
||||||
outBuffer[i * outchannels + flIndex] = fl;
|
cblas_scopy((int)count, inBuffer + 3, 5, outBuffer + blIndex, outchannels);
|
||||||
}
|
if(brIndex != ~0)
|
||||||
if(frIndex != ~0) {
|
cblas_scopy((int)count, inBuffer + 4, 5, outBuffer + brIndex, outchannels);
|
||||||
outBuffer[i * outchannels + frIndex] = fr;
|
|
||||||
}
|
|
||||||
if(cIndex != ~0) {
|
|
||||||
outBuffer[i * outchannels + cIndex] = c;
|
|
||||||
}
|
|
||||||
if(blIndex != ~0) {
|
|
||||||
outBuffer[i * outchannels + blIndex] = bl;
|
|
||||||
}
|
|
||||||
if(brIndex != ~0) {
|
|
||||||
outBuffer[i * outchannels + brIndex] = br;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if(inconfig == AudioConfig6Point1 && outchannels >= 8) {
|
} else if(inconfig == AudioConfig6Point1 && outchannels >= 8) {
|
||||||
uint32_t flIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontLeft];
|
uint32_t flIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontLeft];
|
||||||
uint32_t frIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontRight];
|
uint32_t frIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelFrontRight];
|
||||||
|
@ -249,58 +215,34 @@ static void upmix(const float *inBuffer, int inchannels, uint32_t inconfig, floa
|
||||||
uint32_t bcIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelBackCenter];
|
uint32_t bcIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelBackCenter];
|
||||||
uint32_t slIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelSideLeft];
|
uint32_t slIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelSideLeft];
|
||||||
uint32_t srIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelSideRight];
|
uint32_t srIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:AudioChannelSideRight];
|
||||||
for(size_t i = 0; i < count; ++i) {
|
vDSP_vclr(outBuffer, 1, count * outchannels);
|
||||||
float fl = inBuffer[i * 7 + 0];
|
if(flIndex != ~0)
|
||||||
float fr = inBuffer[i * 7 + 1];
|
cblas_scopy((int)count, inBuffer + 0, 7, outBuffer + flIndex, outchannels);
|
||||||
float c = inBuffer[i * 7 + 2];
|
if(frIndex != ~0)
|
||||||
float lfe = inBuffer[i * 7 + 3];
|
cblas_scopy((int)count, inBuffer + 1, 7, outBuffer + frIndex, outchannels);
|
||||||
float sl = inBuffer[i * 7 + 4];
|
if(cIndex != ~0)
|
||||||
float sr = inBuffer[i * 7 + 5];
|
cblas_scopy((int)count, inBuffer + 2, 7, outBuffer + cIndex, outchannels);
|
||||||
float bc = inBuffer[i * 7 + 6];
|
if(lfeIndex != ~0)
|
||||||
memset(outBuffer + i * outchannels, 0, sizeof(float) * outchannels);
|
cblas_scopy((int)count, inBuffer + 3, 7, outBuffer + lfeIndex, outchannels);
|
||||||
if(flIndex != ~0) {
|
if(slIndex != ~0)
|
||||||
outBuffer[i * outchannels + flIndex] = fl;
|
cblas_scopy((int)count, inBuffer + 4, 7, outBuffer + slIndex, outchannels);
|
||||||
}
|
if(srIndex != ~0)
|
||||||
if(frIndex != ~0) {
|
cblas_scopy((int)count, inBuffer + 5, 7, outBuffer + srIndex, outchannels);
|
||||||
outBuffer[i * outchannels + frIndex] = fr;
|
if(bcIndex != ~0)
|
||||||
}
|
cblas_scopy((int)count, inBuffer + 6, 7, outBuffer + bcIndex, outchannels);
|
||||||
if(cIndex != ~0) {
|
else {
|
||||||
outBuffer[i * outchannels + cIndex] = c;
|
if(blIndex != ~0)
|
||||||
}
|
cblas_scopy((int)count, inBuffer + 6, 7, outBuffer + blIndex, outchannels);
|
||||||
if(lfeIndex != ~0) {
|
if(brIndex != ~0)
|
||||||
outBuffer[i * outchannels + lfeIndex] = lfe;
|
cblas_scopy((int)count, inBuffer + 6, 7, outBuffer + brIndex, outchannels);
|
||||||
}
|
|
||||||
if(slIndex != ~0) {
|
|
||||||
outBuffer[i * outchannels + slIndex] = sl;
|
|
||||||
}
|
|
||||||
if(srIndex != ~0) {
|
|
||||||
outBuffer[i * outchannels + srIndex] = sr;
|
|
||||||
}
|
|
||||||
if(bcIndex != ~0) {
|
|
||||||
outBuffer[i * outchannels + bcIndex] = bc;
|
|
||||||
} else {
|
|
||||||
if(blIndex != ~0) {
|
|
||||||
outBuffer[i * outchannels + blIndex] = bc;
|
|
||||||
}
|
|
||||||
if(brIndex != ~0) {
|
|
||||||
outBuffer[i * outchannels + brIndex] = bc;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
uint32_t outIndexes[inchannels];
|
vDSP_vclr(outBuffer, 1, count * outchannels);
|
||||||
for(int i = 0; i < inchannels; ++i) {
|
for(int i = 0; i < inchannels; ++i) {
|
||||||
uint32_t channelFlag = [AudioChunk extractChannelFlag:i fromConfig:inconfig];
|
uint32_t channelFlag = [AudioChunk extractChannelFlag:i fromConfig:inconfig];
|
||||||
outIndexes[i] = [AudioChunk channelIndexFromConfig:outconfig forFlag:channelFlag];
|
uint32_t outIndex = [AudioChunk channelIndexFromConfig:outconfig forFlag:channelFlag];
|
||||||
}
|
if(outIndex != ~0)
|
||||||
for(size_t i = 0; i < count; ++i) {
|
cblas_scopy((int)count, inBuffer + i, inchannels, outBuffer + outIndex, outchannels);
|
||||||
// upmix N channels to N channels plus silence the empty channels
|
|
||||||
memset(outBuffer + i * outchannels, 0, sizeof(float) * outchannels);
|
|
||||||
for(int j = 0; j < inchannels; ++j) {
|
|
||||||
if(outIndexes[j] != ~0) {
|
|
||||||
outBuffer[i * outchannels + outIndexes[j]] = inBuffer[i * inchannels + j];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -353,8 +295,8 @@ static void upmix(const float *inBuffer, int inchannels, uint32_t inconfig, floa
|
||||||
BOOL hVirt = [[[NSUserDefaultsController sharedUserDefaultsController] defaults] boolForKey:@"headphoneVirtualization"];
|
BOOL hVirt = [[[NSUserDefaultsController sharedUserDefaultsController] defaults] boolForKey:@"headphoneVirtualization"];
|
||||||
|
|
||||||
if(hVirt &&
|
if(hVirt &&
|
||||||
outputFormat.mChannelsPerFrame == 2 &&
|
outputFormat.mChannelsPerFrame >= 2 &&
|
||||||
outConfig == AudioConfigStereo &&
|
(outConfig & AudioConfigStereo) == AudioConfigStereo &&
|
||||||
inputFormat.mChannelsPerFrame >= 1 &&
|
inputFormat.mChannelsPerFrame >= 1 &&
|
||||||
(inConfig & (AudioConfig7Point1 | AudioChannelBackCenter)) != 0) {
|
(inConfig & (AudioConfig7Point1 | AudioChannelBackCenter)) != 0) {
|
||||||
NSString *userPreset = [[[NSUserDefaultsController sharedUserDefaultsController] defaults] stringForKey:@"hrirPath"];
|
NSString *userPreset = [[[NSUserDefaultsController sharedUserDefaultsController] defaults] stringForKey:@"hrirPath"];
|
||||||
|
@ -396,7 +338,18 @@ static void upmix(const float *inBuffer, int inchannels, uint32_t inconfig, floa
|
||||||
- (void)process:(const void *)inBuffer frameCount:(size_t)frames output:(void *)outBuffer {
|
- (void)process:(const void *)inBuffer frameCount:(size_t)frames output:(void *)outBuffer {
|
||||||
@synchronized(hFilter) {
|
@synchronized(hFilter) {
|
||||||
if(hFilter) {
|
if(hFilter) {
|
||||||
[hFilter process:(const float *)inBuffer sampleCount:frames toBuffer:(float *)outBuffer];
|
uint32_t outChannels = outputFormat.mChannelsPerFrame;
|
||||||
|
if(outChannels > 2) {
|
||||||
|
float tempBuffer[frames * 2];
|
||||||
|
[hFilter process:(const float *)inBuffer sampleCount:frames toBuffer:&tempBuffer[0]];
|
||||||
|
cblas_scopy((int)frames, tempBuffer, 2, (float *)outBuffer, outChannels);
|
||||||
|
cblas_scopy((int)frames, tempBuffer + 1, 2, ((float *)outBuffer) + 1, outChannels);
|
||||||
|
for(size_t i = 2; i < outChannels; ++i) {
|
||||||
|
vDSP_vclr(((float *)outBuffer) + i, outChannels, (int)frames);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
[hFilter process:(const float *)inBuffer sampleCount:frames toBuffer:(float *)outBuffer];
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue