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
Christopher Snowhill 2022-02-11 23:51:41 -08:00
parent 5bcedab274
commit 6b148fef11
1 changed files with 70 additions and 117 deletions

View File

@ -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;
} }
} }