From d0ec6489529dcd262ab6ef5639fe2c03035b6e41 Mon Sep 17 00:00:00 2001 From: rogerman Date: Mon, 9 Mar 2015 21:49:22 +0000 Subject: [PATCH] Cocoa Port: - Fix some UI bugs with mic level checks where the mic status icon wasn't showing the correct color under certain conditions. - Fix UI bug where the mic mute control wasn't being respected under certain conditions. - The mic status icon now includes a tooltip that reports the name and sample rate of the current input device. - Handle mic hardware state changes more gracefully when the input device is changed externally. --- desmume/src/cocoa/cocoa_core.h | 4 +- desmume/src/cocoa/cocoa_core.mm | 19 +- desmume/src/cocoa/cocoa_globals.h | 4 +- desmume/src/cocoa/cocoa_input.h | 25 + desmume/src/cocoa/cocoa_input.mm | 145 ++++- desmume/src/cocoa/coreaudiosound.cpp | 216 +++--- desmume/src/cocoa/coreaudiosound.h | 45 +- .../English.lproj/DisplayWindow.xib | 613 ++---------------- .../userinterface/EmuControllerDelegate.h | 7 +- .../userinterface/EmuControllerDelegate.mm | 90 +-- 10 files changed, 433 insertions(+), 735 deletions(-) diff --git a/desmume/src/cocoa/cocoa_core.h b/desmume/src/cocoa/cocoa_core.h index 360fbc75c..50067f31a 100644 --- a/desmume/src/cocoa/cocoa_core.h +++ b/desmume/src/cocoa/cocoa_core.h @@ -23,6 +23,7 @@ #import "cocoa_util.h" +@class CocoaDSCore; @class CocoaDSController; @class CocoaDSFirmware; @class CocoaDSGPU; @@ -32,8 +33,7 @@ typedef void *gdbstub_handle_t; typedef struct { - void *cdsCore; - void *cdsController; + CocoaDSCore *cdsCore; int state; bool isFrameSkipEnabled; size_t frameCount; diff --git a/desmume/src/cocoa/cocoa_core.mm b/desmume/src/cocoa/cocoa_core.mm index c517c6410..3dce82f0d 100644 --- a/desmume/src/cocoa/cocoa_core.mm +++ b/desmume/src/cocoa/cocoa_core.mm @@ -178,7 +178,6 @@ volatile bool execute = true; _slot1R4Path = ""; threadParam.cdsCore = self; - threadParam.cdsController = cdsController; threadParam.state = CORESTATE_PAUSE; threadParam.isFrameSkipEnabled = true; threadParam.frameCount = 0; @@ -289,7 +288,6 @@ volatile bool execute = true; pthread_mutex_lock(&threadParam.mutexThreadExecute); [cdsController release]; cdsController = theController; - threadParam.cdsController = theController; pthread_mutex_unlock(&threadParam.mutexThreadExecute); OSSpinLockUnlock(&spinlockCdsController); @@ -1093,13 +1091,14 @@ static void* RunCoreThread(void *arg) timeBudget = param->timeBudgetMachAbsTime; } + CocoaDSController *cdsController = [cdsCore cdsController]; if (param->state != CORESTATE_FRAMEJUMP) { - [(CocoaDSController *)param->cdsController flush]; + [cdsController flush]; } else { - [(CocoaDSController *)param->cdsController flushEmpty]; + [cdsController flushEmpty]; } NDS_beginProcessingInput(); @@ -1127,6 +1126,18 @@ static void* RunCoreThread(void *arg) param->frameCount++; } + // Make sure that the mic level is updated at least once per frame, regardless + // of whether the NDS actually reads the mic or not. + if (![cdsController isHardwareMicReadThisFrame]) + { + [cdsController updateHardwareMicLevelWithSample:MIC_NULL_SAMPLE_VALUE]; + [[cdsController delegate] doMicSamplesReadFromController:cdsController inSample:MIC_NULL_SAMPLE_VALUE]; + } + else + { + [cdsController setIsHardwareMicReadThisFrame:NO]; + } + pthread_mutex_lock(¶m->mutexOutputList); switch (param->state) diff --git a/desmume/src/cocoa/cocoa_globals.h b/desmume/src/cocoa/cocoa_globals.h index bc64e85da..7bd1d40af 100644 --- a/desmume/src/cocoa/cocoa_globals.h +++ b/desmume/src/cocoa/cocoa_globals.h @@ -249,9 +249,9 @@ #define MIC_SAMPLE_SIZE ((MIC_SAMPLE_RESOLUTION / 8) * MIC_NUMBER_CHANNELS) // Bytes per sample, multiplied by the number of channels #define MIC_MAX_BUFFER_SAMPLES ((MIC_SAMPLE_RATE / DS_FRAMES_PER_SECOND) * MIC_SAMPLE_SIZE) #define MIC_CAPTURE_FRAMES 192 // The number of audio frames that the NDS microphone should pull. The lower this value, the lower the latency. Ensure that this value is not too low, or else audio frames may be lost. -#define MIC_CLIP_FRAME_THRESHOLD 15 // When the number of clipped mic samples exceeds this amount, the UI will reflect that the mic is clipping. +#define MIC_CLIP_FRAME_THRESHOLD 15 // When the number of clipped mic samples reaches this amount, the UI will reflect that the mic is clipping. #define MIC_CLIP_FRAME_MAX_COUNT 60 // When counting clipped mic samples, don't count past this number. This is to prevent the mic clip state from persisting too long after the mic has stopped clipping. -#define MIC_NULL_FRAME_THRESHOLD 12 // When the number of null mic samples exceeds this amount, the UI will reflect that the mic is null. +#define MIC_NULL_FRAME_THRESHOLD 12 // When the number of null mic samples reaches this amount, the UI will reflect that the mic is null. #define MIC_NULL_FRAME_MAX_COUNT 24 // When counting null mic samples, don't count past this number. This is to prevent the mic null state from persisting too long after the mic has read some samples. #define MIC_NULL_SAMPLE_VALUE 64 diff --git a/desmume/src/cocoa/cocoa_input.h b/desmume/src/cocoa/cocoa_input.h index 209fb0211..a785a1d02 100644 --- a/desmume/src/cocoa/cocoa_input.h +++ b/desmume/src/cocoa/cocoa_input.h @@ -21,6 +21,7 @@ @class CocoaDSController; class CoreAudioInput; +struct CoreAudioInputDeviceInfo; class AudioGenerator; class AudioSampleBlockGenerator; @@ -80,6 +81,11 @@ typedef struct - (uint8_t) doMicSamplesReadFromController:(CocoaDSController *)cdsController inSample:(uint8_t)sampleValue; @optional +- (void) doMicHardwareStateChangedFromController:(CocoaDSController *)cdsController + isEnabled:(BOOL)isHardwareEnabled + isLocked:(BOOL)isHardwareLocked + isMuted:(BOOL)isHardwareMuted; + - (void) doMicHardwareGainChangedFromController:(CocoaDSController *)cdsController gain:(float)gainValue; @end @@ -87,10 +93,13 @@ typedef struct @interface CocoaDSController : NSObject { id delegate; + BOOL isHardwareMicReadThisFrame; BOOL autohold; BOOL isAutoholdCleared; BOOL _useHardwareMic; size_t _availableMicSamples; + uint32_t _hwMicNumberIdleFrames; + uint32_t _hwMicNumberClippedFrames; NSInteger micMode; NSPoint touchLocation; @@ -99,6 +108,7 @@ typedef struct CoreAudioInput *CAInputDevice; AudioGenerator *softwareMicSampleGenerator; NSInteger paddleAdjust; + NSString *hardwareMicInfoString; OSSpinLock spinlockControllerState; } @@ -106,6 +116,9 @@ typedef struct @property (retain) id delegate; @property (assign) BOOL autohold; @property (readonly) BOOL isHardwareMicAvailable; +@property (assign) BOOL isHardwareMicIdle; +@property (assign) BOOL isHardwareMicInClip; +@property (assign) BOOL isHardwareMicReadThisFrame; @property (assign) BOOL hardwareMicEnabled; @property (readonly) BOOL hardwareMicLocked; @property (assign) float hardwareMicGain; @@ -118,6 +131,7 @@ typedef struct @property (readonly) AudioGenerator *softwareMicSampleGenerator; @property (assign) AudioSampleBlockGenerator *selectedAudioFileGenerator; @property (assign) NSInteger paddleAdjust; +@property (retain) NSString *hardwareMicInfoString; - (void) setControllerState:(BOOL)theState controlID:(const NSUInteger)controlID; - (void) setControllerState:(BOOL)theState controlID:(const NSUInteger)controlID turbo:(const BOOL)isTurboEnabled; @@ -126,7 +140,12 @@ typedef struct - (void) clearAutohold; - (void) flush; - (void) flushEmpty; +- (void) updateHardwareMicLevelWithSample:(uint8_t)inSample; - (uint8_t) handleMicSampleRead:(CoreAudioInput *)caInput softwareMic:(AudioGenerator *)sampleGenerator; +- (void) handleMicHardwareStateChanged:(CoreAudioInputDeviceInfo *)deviceInfo + isEnabled:(BOOL)isHardwareEnabled + isLocked:(BOOL)isHardwareLocked + isMuted:(BOOL)isHardwareMuted; - (void) handleMicHardwareGainChanged:(float)gainValue; @end @@ -138,6 +157,12 @@ extern "C" void CAResetCallback(void *inParam1, void *inParam2); uint8_t CASampleReadCallback(void *inParam1, void *inParam2); +void CAHardwareStateChangedCallback(CoreAudioInputDeviceInfo *deviceInfo, + const bool isHardwareEnabled, + const bool isHardwareLocked, + const bool isHardwareMuted, + void *inParam1, + void *inParam2); void CAHardwareGainChangedCallback(float normalizedGain, void *inParam1, void *inParam2); #ifdef __cplusplus diff --git a/desmume/src/cocoa/cocoa_input.mm b/desmume/src/cocoa/cocoa_input.mm index da83eb476..af453f20f 100644 --- a/desmume/src/cocoa/cocoa_input.mm +++ b/desmume/src/cocoa/cocoa_input.mm @@ -36,6 +36,9 @@ SineWaveGenerator sineWaveGenerator(250.0, MIC_SAMPLE_RATE); @synthesize delegate; @dynamic isHardwareMicAvailable; +@dynamic isHardwareMicIdle; +@dynamic isHardwareMicInClip; +@synthesize isHardwareMicReadThisFrame; @dynamic hardwareMicEnabled; @dynamic hardwareMicLocked; @dynamic hardwareMicGain; @@ -49,6 +52,7 @@ SineWaveGenerator sineWaveGenerator(250.0, MIC_SAMPLE_RATE); @synthesize softwareMicSampleGenerator; @synthesize selectedAudioFileGenerator; @synthesize paddleAdjust; +@synthesize hardwareMicInfoString; - (id)init { @@ -70,16 +74,21 @@ SineWaveGenerator sineWaveGenerator(250.0, MIC_SAMPLE_RATE); spinlockControllerState = OS_SPINLOCK_INIT; autohold = NO; isAutoholdCleared = YES; + isHardwareMicReadThisFrame = NO; _useHardwareMic = NO; _availableMicSamples = 0; + _hwMicNumberIdleFrames = MIC_NULL_FRAME_THRESHOLD; + _hwMicNumberClippedFrames = 0; micMode = MICMODE_NONE; selectedAudioFileGenerator = NULL; CAInputDevice = new CoreAudioInput; + CAInputDevice->SetCallbackHardwareStateChanged(&CAHardwareStateChangedCallback, self, NULL); CAInputDevice->SetCallbackHardwareGainChanged(&CAHardwareGainChangedCallback, self, NULL); softwareMicSampleGenerator = &nullSampleGenerator; touchLocation = NSMakePoint(0.0f, 0.0f); paddleAdjust = 0; + hardwareMicInfoString = @"No hardware input detected."; Mic_SetResetCallback(&CAResetCallback, self, NULL); Mic_SetSampleReadCallback(&CASampleReadCallback, self, NULL); @@ -100,6 +109,42 @@ SineWaveGenerator sineWaveGenerator(250.0, MIC_SAMPLE_RATE); !CAInputDevice->GetPauseState() ) ? YES : NO; } +- (void) setIsHardwareMicIdle:(BOOL)theState +{ + if (theState) + { + _hwMicNumberIdleFrames = MIC_NULL_FRAME_THRESHOLD; + _hwMicNumberClippedFrames = 0; + } + else + { + _hwMicNumberIdleFrames = 0; + } +} + +- (BOOL) isHardwareMicIdle +{ + return (_hwMicNumberIdleFrames >= MIC_NULL_FRAME_THRESHOLD); +} + +- (void) setIsHardwareMicInClip:(BOOL)theState +{ + if (theState) + { + _hwMicNumberIdleFrames = 0; + _hwMicNumberClippedFrames = MIC_CLIP_FRAME_THRESHOLD; + } + else + { + _hwMicNumberClippedFrames = 0; + } +} + +- (BOOL) isHardwareMicInClip +{ + return (_hwMicNumberClippedFrames >= MIC_CLIP_FRAME_THRESHOLD); +} + - (void) setHardwareMicEnabled:(BOOL)micEnabled { if (micEnabled) @@ -140,20 +185,18 @@ SineWaveGenerator sineWaveGenerator(250.0, MIC_SAMPLE_RATE); - (BOOL) hardwareMicMute { - BOOL isMicMuted = (CAInputDevice->GetMuteState()) ? YES : NO; - return isMicMuted; + return (CAInputDevice->GetMuteState()) ? YES : NO; } - (void) setHardwareMicPause:(BOOL)isMicPaused { bool pauseState = (isMicPaused) ? true : false; - CAInputDevice->SetMuteState(pauseState); + CAInputDevice->SetPauseState(pauseState); } - (BOOL) hardwareMicPause { - BOOL isMicPaused = (CAInputDevice->GetPauseState()) ? YES : NO; - return isMicPaused; + return (CAInputDevice->GetPauseState()) ? YES : NO; } - (void) setSoftwareMicState:(BOOL)theState @@ -483,6 +526,45 @@ SineWaveGenerator sineWaveGenerator(250.0, MIC_SAMPLE_RATE); NDS_setMic(false); } +- (void) updateHardwareMicLevelWithSample:(uint8_t)inSample +{ + switch (inSample) + { + case MIC_NULL_SAMPLE_VALUE: + { + if (_hwMicNumberIdleFrames < MIC_NULL_FRAME_MAX_COUNT) + { + _hwMicNumberIdleFrames++; + } + break; + } + + case 0: + case 127: + { + if (_hwMicNumberClippedFrames < MIC_CLIP_FRAME_MAX_COUNT) + { + _hwMicNumberClippedFrames++; + } + break; + } + + default: + { + if (_hwMicNumberIdleFrames > 0) + { + _hwMicNumberIdleFrames--; + } + + if (_hwMicNumberClippedFrames > 0) + { + _hwMicNumberClippedFrames--; + } + break; + } + } +} + - (uint8_t) handleMicSampleRead:(CoreAudioInput *)caInput softwareMic:(AudioGenerator *)sampleGenerator { uint8_t theSample = MIC_NULL_SAMPLE_VALUE; @@ -503,6 +585,9 @@ SineWaveGenerator sineWaveGenerator(250.0, MIC_SAMPLE_RATE); caInput->_samplesConverted->read(&theSample, 1); theSample >>= 1; // Samples from CoreAudio are 8-bit, so we need to convert the sample to 7-bit _availableMicSamples--; + + [self updateHardwareMicLevelWithSample:theSample]; + [self setIsHardwareMicReadThisFrame:YES]; } } else @@ -513,14 +598,42 @@ SineWaveGenerator sineWaveGenerator(250.0, MIC_SAMPLE_RATE); return [[self delegate] doMicSamplesReadFromController:self inSample:theSample]; } -- (void) handleMicHardwareGainChanged:(float)gainValue +- (void) handleMicHardwareStateChanged:(CoreAudioInputDeviceInfo *)deviceInfo + isEnabled:(BOOL)isHardwareEnabled + isLocked:(BOOL)isHardwareLocked + isMuted:(BOOL)isHardwareMuted { - if (delegate == nil || ![delegate respondsToSelector:@selector(doMicHardwareGainChangedFromController:gain:)]) + NSString *newHWMicInfoString; + if (deviceInfo->objectID == kAudioObjectUnknown) { - return; + newHWMicInfoString = @"No hardware input detected."; + } + else + { + newHWMicInfoString = [NSString stringWithFormat:@"%@\nSample Rate: %1.1f Hz", + (NSString *)deviceInfo->name, + (double)deviceInfo->sampleRate]; } - [[self delegate] doMicHardwareGainChangedFromController:self gain:gainValue]; + [self setHardwareMicInfoString:newHWMicInfoString]; + [self setIsHardwareMicIdle:YES]; + [self setIsHardwareMicInClip:NO]; + + if (delegate != nil && [delegate respondsToSelector:@selector(doMicHardwareStateChangedFromController:isEnabled:isLocked:isMuted:)]) + { + [[self delegate] doMicHardwareStateChangedFromController:self + isEnabled:isHardwareEnabled + isLocked:isHardwareLocked + isMuted:isHardwareMuted]; + } +} + +- (void) handleMicHardwareGainChanged:(float)gainValue +{ + if (delegate != nil && [delegate respondsToSelector:@selector(doMicHardwareGainChangedFromController:gain:)]) + { + [[self delegate] doMicHardwareGainChangedFromController:self gain:gainValue]; + } } @end @@ -537,6 +650,20 @@ uint8_t CASampleReadCallback(void *inParam1, void *inParam2) return [cdsController handleMicSampleRead:[cdsController CAInputDevice] softwareMic:[cdsController softwareMicSampleGenerator]]; } +void CAHardwareStateChangedCallback(CoreAudioInputDeviceInfo *deviceInfo, + const bool isHardwareEnabled, + const bool isHardwareLocked, + const bool isHardwareMuted, + void *inParam1, + void *inParam2) +{ + CocoaDSController *cdsController = (CocoaDSController *)inParam1; + [cdsController handleMicHardwareStateChanged:(CoreAudioInputDeviceInfo *)deviceInfo + isEnabled:((isHardwareEnabled) ? YES : NO) + isLocked:((isHardwareLocked) ? YES : NO) + isMuted:((isHardwareMuted) ? YES : NO)]; +} + void CAHardwareGainChangedCallback(float normalizedGain, void *inParam1, void *inParam2) { CocoaDSController *cdsController = (CocoaDSController *)inParam1; diff --git a/desmume/src/cocoa/coreaudiosound.cpp b/desmume/src/cocoa/coreaudiosound.cpp index e7877a76a..4d04eccfa 100644 --- a/desmume/src/cocoa/coreaudiosound.cpp +++ b/desmume/src/cocoa/coreaudiosound.cpp @@ -29,6 +29,10 @@ CoreAudioInput::CoreAudioInput() _spinlockAUHAL = (OSSpinLock *)malloc(sizeof(OSSpinLock)); *_spinlockAUHAL = OS_SPINLOCK_INIT; + _hwStateChangedCallbackFunc = &CoreAudioInputDefaultHardwareStateChangedCallback; + _hwStateChangedCallbackParam1 = NULL; + _hwStateChangedCallbackParam2 = NULL; + _hwGainChangedCallbackFunc = &CoreAudioInputDefaultHardwareGainChangedCallback; _hwGainChangedCallbackParam1 = NULL; _hwGainChangedCallbackParam2 = NULL; @@ -37,6 +41,12 @@ CoreAudioInput::CoreAudioInput() _inputElement = 0; _isMuted = false; + _hwDeviceInfo.objectID = kAudioObjectUnknown; + _hwDeviceInfo.name = CFSTR("Unknown Device"); + _hwDeviceInfo.manufacturer = CFSTR("Unknown Manufacturer"); + _hwDeviceInfo.deviceUID = CFSTR(""); + _hwDeviceInfo.modelUID = CFSTR(""); + _hwDeviceInfo.sampleRate = 0.0; _isPaused = false; _isHardwareEnabled = false; _isHardwareLocked = true; @@ -269,6 +279,11 @@ CoreAudioInput::~CoreAudioInput() free(_spinlockAUHAL); _spinlockAUHAL = NULL; + + CFRelease(this->_hwDeviceInfo.name); + CFRelease(this->_hwDeviceInfo.manufacturer); + CFRelease(this->_hwDeviceInfo.deviceUID); + CFRelease(this->_hwDeviceInfo.modelUID); } OSStatus CoreAudioInput::InitInputAUHAL(UInt32 deviceID) @@ -277,6 +292,7 @@ OSStatus CoreAudioInput::InitInputAUHAL(UInt32 deviceID) static const AudioUnitScope inputBus = 1; UInt32 propertySize = 0; + this->_hwDeviceInfo.objectID = kAudioObjectUnknown; if (deviceID == kAudioObjectUnknown) { error = kAudioHardwareUnspecifiedError; @@ -286,47 +302,47 @@ OSStatus CoreAudioInput::InitInputAUHAL(UInt32 deviceID) // Get some information about the device before attempting to attach it to the AUHAL. // Currently, this information is only available in debug mode. However, if there is // a need to actually do something with this information, then we've already got it. - CFStringRef devicePropertyString; AudioObjectPropertyAddress deviceProperty; UInt32 dataSize = 0; deviceProperty.mScope = kAudioObjectPropertyScopeGlobal; deviceProperty.mElement = kAudioObjectPropertyElementMaster; printf("\nInput Device Information:\n"); + printf(" Object ID: %d\n", deviceID); deviceProperty.mSelector = kAudioObjectPropertyName; error = AudioObjectGetPropertyDataSize(deviceID, &deviceProperty, 0, NULL, &dataSize); if (error == noErr) { - error = AudioObjectGetPropertyData(deviceID, &deviceProperty, 0, NULL, &dataSize, &devicePropertyString); - printf(" Name: %s\n", CFStringGetCStringPtr(devicePropertyString, kCFStringEncodingUTF8)); - CFRelease(devicePropertyString); + CFRelease(this->_hwDeviceInfo.name); + error = AudioObjectGetPropertyData(deviceID, &deviceProperty, 0, NULL, &dataSize, &this->_hwDeviceInfo.name); + printf(" Name: %s\n", CFStringGetCStringPtr(this->_hwDeviceInfo.name, kCFStringEncodingUTF8)); } deviceProperty.mSelector = kAudioObjectPropertyManufacturer; error = AudioObjectGetPropertyDataSize(deviceID, &deviceProperty, 0, NULL, &dataSize); if (error == noErr) { - error = AudioObjectGetPropertyData(deviceID, &deviceProperty, 0, NULL, &dataSize, &devicePropertyString); - printf(" Manufacturer: %s\n", CFStringGetCStringPtr(devicePropertyString, kCFStringEncodingUTF8)); - CFRelease(devicePropertyString); + CFRelease(this->_hwDeviceInfo.manufacturer); + error = AudioObjectGetPropertyData(deviceID, &deviceProperty, 0, NULL, &dataSize, &this->_hwDeviceInfo.manufacturer); + printf(" Manufacturer: %s\n", CFStringGetCStringPtr(this->_hwDeviceInfo.manufacturer, kCFStringEncodingUTF8)); } deviceProperty.mSelector = kAudioDevicePropertyDeviceUID; error = AudioObjectGetPropertyDataSize(deviceID, &deviceProperty, 0, NULL, &dataSize); if (error == noErr) { - error = AudioObjectGetPropertyData(deviceID, &deviceProperty, 0, NULL, &dataSize, &devicePropertyString); - printf(" Device UID: %s\n", CFStringGetCStringPtr(devicePropertyString, kCFStringEncodingUTF8)); - CFRelease(devicePropertyString); + CFRelease(this->_hwDeviceInfo.deviceUID); + error = AudioObjectGetPropertyData(deviceID, &deviceProperty, 0, NULL, &dataSize, &this->_hwDeviceInfo.deviceUID); + printf(" Device UID: %s\n", CFStringGetCStringPtr(this->_hwDeviceInfo.deviceUID, kCFStringEncodingUTF8)); } deviceProperty.mSelector = kAudioDevicePropertyModelUID; error = AudioObjectGetPropertyDataSize(deviceID, &deviceProperty, 0, NULL, &dataSize); if (error == noErr) { - error = AudioObjectGetPropertyData(deviceID, &deviceProperty, 0, NULL, &dataSize, &devicePropertyString); - printf(" Model UID: %s\n", CFStringGetCStringPtr(devicePropertyString, kCFStringEncodingUTF8)); - CFRelease(devicePropertyString); + CFRelease(this->_hwDeviceInfo.modelUID); + error = AudioObjectGetPropertyData(deviceID, &deviceProperty, 0, NULL, &dataSize, &this->_hwDeviceInfo.modelUID); + printf(" Model UID: %s\n", CFStringGetCStringPtr(this->_hwDeviceInfo.modelUID, kCFStringEncodingUTF8)); } deviceProperty.mSelector = kAudioDevicePropertyNominalSampleRate; @@ -334,9 +350,8 @@ OSStatus CoreAudioInput::InitInputAUHAL(UInt32 deviceID) error = AudioObjectGetPropertyDataSize(deviceID, &deviceProperty, 0, NULL, &dataSize); if (error == noErr) { - Float64 deviceSampleRate = 0.0; - error = AudioObjectGetPropertyData(deviceID, &deviceProperty, 0, NULL, &dataSize, &deviceSampleRate); - printf(" Sample Rate: %1.1f\n\n", deviceSampleRate); + error = AudioObjectGetPropertyData(deviceID, &deviceProperty, 0, NULL, &dataSize, &this->_hwDeviceInfo.sampleRate); + printf(" Sample Rate: %1.1f\n\n", this->_hwDeviceInfo.sampleRate); } // Before attaching the HAL input device, stop everything first. @@ -444,6 +459,8 @@ OSStatus CoreAudioInput::InitInputAUHAL(UInt32 deviceID) return error; } + this->_hwDeviceInfo.objectID = deviceID; + return error; } @@ -489,8 +506,7 @@ void CoreAudioInput::Start() if (error == noErr) { this->_inputElement = elementNumber; - this->SetGain_ValueOnly(theGain); - this->RunHardwareGainChangedCallback(theGain); + this->UpdateHardwareGain(theGain); break; } } @@ -505,7 +521,7 @@ void CoreAudioInput::Start() this->_isHardwareEnabled = false; } - if (this->_isHardwareEnabled && !this->IsHardwareLocked() && !this->_isPaused) + if (this->IsHardwareEnabled() && !this->IsHardwareLocked() && !this->GetPauseState()) { error = AudioOutputUnitStart(this->_auHALInputDevice); } @@ -513,13 +529,19 @@ void CoreAudioInput::Start() OSSpinLockUnlock(this->_spinlockAUHAL); error = AUGraphInitialize(_auGraph); - if (!this->_isPaused) + if (!this->GetPauseState()) { error = AUGraphStart(this->_auGraph); } this->_samplesCaptured->clear(); this->_samplesConverted->clear(); + this->_hwStateChangedCallbackFunc(&this->_hwDeviceInfo, + this->IsHardwareEnabled(), + this->IsHardwareLocked(), + this->_isMuted, + this->_hwStateChangedCallbackParam1, + this->_hwStateChangedCallbackParam2); } void CoreAudioInput::Stop() @@ -569,8 +591,14 @@ bool CoreAudioInput::GetMuteState() const void CoreAudioInput::SetMuteState(bool muteState) { - this->SetPauseState(muteState); this->_isMuted = muteState; + this->SetPauseState(muteState); + this->_hwStateChangedCallbackFunc(&this->_hwDeviceInfo, + this->IsHardwareEnabled(), + this->IsHardwareLocked(), + this->_isMuted, + this->_hwStateChangedCallbackParam1, + this->_hwStateChangedCallbackParam2); } bool CoreAudioInput::GetPauseState() const @@ -580,14 +608,14 @@ bool CoreAudioInput::GetPauseState() const void CoreAudioInput::SetPauseState(bool pauseState) { - if (pauseState && !this->_isPaused) + if (pauseState && !this->GetPauseState()) { OSSpinLockLock(this->_spinlockAUHAL); AudioOutputUnitStop(this->_auHALInputDevice); OSSpinLockUnlock(this->_spinlockAUHAL); AUGraphStop(this->_auGraph); } - else if (!pauseState && this->_isPaused && !this->IsHardwareLocked()) + else if (!pauseState && this->GetPauseState() && !this->IsHardwareLocked() && !this->GetMuteState()) { OSSpinLockLock(this->_spinlockAUHAL); AudioOutputUnitStart(this->_auHALInputDevice); @@ -595,7 +623,7 @@ void CoreAudioInput::SetPauseState(bool pauseState) AUGraphStart(this->_auGraph); } - this->_isPaused = pauseState; + this->_isPaused = (this->IsHardwareLocked() || this->GetMuteState()) ? true : pauseState; } float CoreAudioInput::GetGain() const @@ -619,9 +647,10 @@ void CoreAudioInput::SetGain(float normalizedGain) OSSpinLockUnlock(this->_spinlockAUHAL); } -void CoreAudioInput::SetGain_ValueOnly(float normalizedGain) +void CoreAudioInput::UpdateHardwareGain(float normalizedGain) { this->_inputGainNormalized = normalizedGain; + this->_hwGainChangedCallbackFunc(this->_inputGainNormalized, this->_hwGainChangedCallbackParam1, this->_hwGainChangedCallbackParam2); } void CoreAudioInput::UpdateHardwareLock() @@ -629,65 +658,78 @@ void CoreAudioInput::UpdateHardwareLock() OSStatus error = noErr; bool hardwareLocked = false; - if (!this->IsHardwareEnabled()) + if (this->IsHardwareEnabled()) + { + // Check if another application has exclusive access to the hardware. + pid_t hogMode = 0; + UInt32 propertySize = sizeof(hogMode); + error = AudioUnitGetProperty(this->_auHALInputDevice, + kAudioDevicePropertyHogMode, + kAudioUnitScope_Input, + 1, + &hogMode, + &propertySize); + if (error == noErr) + { + if (hogMode != -1) + { + hardwareLocked = true; + } + } + else + { + // If this property is not supported, then always assume that + // the hardware device is shared. + hogMode = -1; + } + + // Check if the hardware device is plugged in. + UInt32 isJackConnected = 0; + propertySize = sizeof(isJackConnected); + error = AudioUnitGetProperty(this->_auHALInputDevice, + kAudioDevicePropertyJackIsConnected, + kAudioUnitScope_Input, + 1, + &hogMode, + &propertySize); + if (error == noErr) + { + if (isJackConnected == 0) + { + hardwareLocked = true; + } + } + else + { + // If this property is not supported, then always assume that + // the hardware device is always plugged in. + isJackConnected = 1; + } + } + else { hardwareLocked = true; - this->_isHardwareLocked = hardwareLocked; - return; } - // Check if another application has exclusive access to the hardware. - pid_t hogMode = 0; - UInt32 propertySize = sizeof(hogMode); - error = AudioUnitGetProperty(this->_auHALInputDevice, - kAudioDevicePropertyHogMode, - kAudioUnitScope_Input, - 1, - &hogMode, - &propertySize); - if (error == noErr) - { - if (hogMode != -1) - { - hardwareLocked = true; - } - } - else - { - // If this property is not supported, then always assume that - // the hardware device is shared. - hogMode = -1; - } - - // Check if the hardware device is plugged in. - UInt32 isJackConnected = 0; - propertySize = sizeof(isJackConnected); - error = AudioUnitGetProperty(this->_auHALInputDevice, - kAudioDevicePropertyJackIsConnected, - kAudioUnitScope_Input, - 1, - &hogMode, - &propertySize); - if (error == noErr) - { - if (isJackConnected == 0) - { - hardwareLocked = true; - } - } - else - { - // If this property is not supported, then always assume that - // the hardware device is always plugged in. - isJackConnected = 1; - } - - if (hardwareLocked && !this->GetPauseState()) + this->_isHardwareLocked = hardwareLocked; + if (this->_isHardwareLocked && !this->GetPauseState()) { this->SetPauseState(true); } - this->_isHardwareLocked = hardwareLocked; + this->_hwStateChangedCallbackFunc(&this->_hwDeviceInfo, + this->IsHardwareEnabled(), + this->IsHardwareLocked(), + this->_isMuted, + this->_hwStateChangedCallbackParam1, + this->_hwStateChangedCallbackParam2); +} + +void CoreAudioInput::SetCallbackHardwareStateChanged(CoreAudioInputHardwareStateChangedCallback callbackFunc, void *inParam1, void *inParam2) +{ + this->_hwStateChangedCallbackFunc = callbackFunc; + this->_hwStateChangedCallbackParam1 = inParam1; + this->_hwStateChangedCallbackParam2 = inParam2; } void CoreAudioInput::SetCallbackHardwareGainChanged(CoreAudioInputHardwareGainChangedCallback callbackFunc, void *inParam1, void *inParam2) @@ -697,11 +739,6 @@ void CoreAudioInput::SetCallbackHardwareGainChanged(CoreAudioInputHardwareGainCh this->_hwGainChangedCallbackParam2 = inParam2; } -void CoreAudioInput::RunHardwareGainChangedCallback(float normalizedGain) -{ - this->_hwGainChangedCallbackFunc(normalizedGain, this->_hwGainChangedCallbackParam1, this->_hwGainChangedCallbackParam2); -} - OSStatus CoreAudioInputCaptureCallback(void *inRefCon, AudioUnitRenderActionFlags *ioActionFlags, const AudioTimeStamp *inTimeStamp, @@ -808,28 +845,33 @@ void CoreAudioInputAUHALChanged(void *inRefCon, inElement, &gainValue, &propertySize); - if (error == noErr) { - caInput->SetGain_ValueOnly(gainValue); - caInput->RunHardwareGainChangedCallback(gainValue); + caInput->UpdateHardwareGain(gainValue); } - break; } case kAudioDevicePropertyHogMode: case kAudioDevicePropertyJackIsConnected: - { caInput->UpdateHardwareLock(); break; - } default: break; } } +void CoreAudioInputDefaultHardwareStateChangedCallback(CoreAudioInputDeviceInfo *deviceInfo, + const bool isHardwareEnabled, + const bool isHardwareLocked, + const bool isHardwareMuted, + void *inParam1, + void *inParam2) +{ + // Do nothing. +} + void CoreAudioInputDefaultHardwareGainChangedCallback(float normalizedGain, void *inParam1, void *inParam2) { // Do nothing. diff --git a/desmume/src/cocoa/coreaudiosound.h b/desmume/src/cocoa/coreaudiosound.h index f27ddc6a6..302817db6 100644 --- a/desmume/src/cocoa/coreaudiosound.h +++ b/desmume/src/cocoa/coreaudiosound.h @@ -25,6 +25,24 @@ #include "ringbuffer.h" +struct CoreAudioInputDeviceInfo +{ + AudioObjectID objectID; + CFStringRef name; + CFStringRef manufacturer; + CFStringRef deviceUID; + CFStringRef modelUID; + Float64 sampleRate; +}; +typedef CoreAudioInputDeviceInfo CoreAudioInputDeviceInfo; + +typedef void (*CoreAudioInputHardwareStateChangedCallback)(CoreAudioInputDeviceInfo *deviceInfo, + const bool isHardwareEnabled, + const bool isHardwareLocked, + const bool isHardwareMuted, + void *inParam1, + void *inParam2); + typedef void (*CoreAudioInputHardwareGainChangedCallback)(float normalizedGain, void *inParam1, void *inParam2); class CoreAudioInput @@ -32,6 +50,10 @@ class CoreAudioInput private: OSSpinLock *_spinlockAUHAL; + CoreAudioInputHardwareStateChangedCallback _hwStateChangedCallbackFunc; + void *_hwStateChangedCallbackParam1; + void *_hwStateChangedCallbackParam2; + CoreAudioInputHardwareGainChangedCallback _hwGainChangedCallbackFunc; void *_hwGainChangedCallbackParam1; void *_hwGainChangedCallbackParam2; @@ -39,26 +61,26 @@ private: AUGraph _auGraph; AUNode _auFormatConverterNode; AUNode _auOutputNode; - AudioUnit _auFormatConverterUnit; + AudioUnit _auOutputUnit; + AudioBufferList *_convertBufferList; + UInt32 _captureFrames; float _inputGainNormalized; AudioUnitElement _inputElement; - bool _isMuted; bool _isPaused; + CoreAudioInputDeviceInfo _hwDeviceInfo; bool _isHardwareEnabled; bool _isHardwareLocked; + bool _isMuted; OSStatus InitInputAUHAL(UInt32 deviceID); public: AudioUnit _auHALInputDevice; - AudioUnit _auOutputUnit; AudioTimeStamp _timeStamp; AudioBufferList *_captureBufferList; - AudioBufferList *_convertBufferList; - UInt32 _captureFrames; RingBuffer *_samplesCaptured; RingBuffer *_samplesConverted; @@ -68,6 +90,7 @@ public: void Start(); void Stop(); size_t Pull(); + bool IsHardwareEnabled() const; bool IsHardwareLocked() const; bool GetMuteState() const; @@ -76,10 +99,11 @@ public: void SetPauseState(bool pauseState); float GetGain() const; void SetGain(float normalizedGain); - void SetGain_ValueOnly(float normalizedGain); + + void UpdateHardwareGain(float normalizedGain); void UpdateHardwareLock(); + void SetCallbackHardwareStateChanged(CoreAudioInputHardwareStateChangedCallback callbackFunc, void *inParam1, void *inParam2); void SetCallbackHardwareGainChanged(CoreAudioInputHardwareGainChangedCallback callbackFunc, void *inParam1, void *inParam2); - void RunHardwareGainChangedCallback(float normalizedGain); }; class CoreAudioOutput @@ -146,6 +170,13 @@ OSStatus CoreAudioOutputRenderCallback(void *inRefCon, UInt32 inNumberFrames, AudioBufferList *ioData); +void CoreAudioInputDefaultHardwareStateChangedCallback(CoreAudioInputDeviceInfo *deviceInfo, + const bool isHardwareEnabled, + const bool isHardwareLocked, + const bool isHardwareMuted, + void *inParam1, + void *inParam2); + void CoreAudioInputDefaultHardwareGainChangedCallback(float normalizedGain, void *inParam1, void *inParam2); bool CreateAudioUnitInstance(AudioUnit *au, ComponentDescription *auDescription); diff --git a/desmume/src/cocoa/translations/English.lproj/DisplayWindow.xib b/desmume/src/cocoa/translations/English.lproj/DisplayWindow.xib index bda2e0edb..640ad882f 100644 --- a/desmume/src/cocoa/translations/English.lproj/DisplayWindow.xib +++ b/desmume/src/cocoa/translations/English.lproj/DisplayWindow.xib @@ -301,6 +301,7 @@ 294 {{17, 5}, {137, 14}} + YES 68157504 @@ -420,6 +421,7 @@ 289 {{152, -1}, {40, 26}} + _NS:791 YES @@ -611,7 +613,7 @@ NSView - + 268 @@ -619,8 +621,6 @@ 268 {{2, 0}, {49, 18}} - - _NS:682 YES @@ -651,7 +651,6 @@ 268 {{-4, 20}, {66, 11}} - YES @@ -729,12 +728,11 @@ 268 {{17, 31}, {21, 105}} - _NS:768 YES - 67371264 + 604242176 0 _NS:768 @@ -752,14 +750,12 @@ {55, 139} - - _NS:1109 NSView - + 268 @@ -767,8 +763,6 @@ 268 {{-1, 6}, {51, 11}} - - YES 68157504 @@ -829,7 +823,6 @@ 268 {{12, 17}, {21, 105}} - _NS:768 YES @@ -852,8 +845,6 @@ {45, 125} - - _NS:1109 NSView @@ -1154,6 +1145,22 @@ 133 + + + toolTip: emuControl.cdsCoreController.content.cdsController.hardwareMicInfoString + + + + + + toolTip: emuControl.cdsCoreController.content.cdsController.hardwareMicInfoString + toolTip + emuControl.cdsCoreController.content.cdsController.hardwareMicInfoString + 2 + + + 226 + image: emuControl.currentMicStatusIcon @@ -1186,6 +1193,22 @@ 138 + + + enabled: emuControl.isHardwareMicAvailable + + + + + + enabled: emuControl.isHardwareMicAvailable + enabled + emuControl.isHardwareMicAvailable + 2 + + + 218 + value: emuControl.currentMicGainValue @@ -1200,7 +1223,7 @@ 2 - 200 + 230 @@ -1262,6 +1285,22 @@ 167 + + + enabled: emuControl.isHardwareMicAvailable + + + + + + enabled: emuControl.isHardwareMicAvailable + enabled + emuControl.isHardwareMicAvailable + 2 + + + 220 + value: emuControl.currentMicGainValue @@ -1280,23 +1319,23 @@ 2 - 205 + 232 - value: emuControl.isHardwareMicMuted + value: emuControl.cdsCoreController.content.cdsController.hardwareMicMute - value: emuControl.isHardwareMicMuted + value: emuControl.cdsCoreController.content.cdsController.hardwareMicMute value - emuControl.isHardwareMicMuted + emuControl.cdsCoreController.content.cdsController.hardwareMicMute 2 - 199 + 224 @@ -1750,539 +1789,9 @@ - 216 - - - - - DisplayView - NSView - - IBProjectSource - ../userinterface/DisplayWindowController.h - - - - DisplayWindowController - NSWindowController - - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - - - - changeCoreSpeed: - id - - - changeDisplayGap: - id - - - changeDisplayMode: - id - - - changeDisplayOrder: - id - - - changeDisplayOrientation: - id - - - changeHardwareMicGain: - id - - - changeRotation: - id - - - changeRotationRelative: - id - - - changeScale: - id - - - changeVideoOutputFilter: - id - - - changeVideoPixelScaler: - id - - - changeVolume: - id - - - copy: - id - - - openRom: - id - - - reset: - id - - - saveScreenshotAs: - id - - - toggleExecutePause: - id - - - toggleFullScreenDisplay: - id - - - toggleKeepMinDisplaySizeAtNormal: - id - - - toggleStatusBar: - id - - - toggleVideoSourceDeposterize: - id - - - writeDefaultsDisplayGap: - id - - - writeDefaultsDisplayRotation: - id - - - writeDefaultsDisplayVideoSettings: - id - - - writeDefaultsHUDSettings: - id - - - - NSView - NSMenuItem - NSSlider - NSButton - NSView - NSMenuItem - NSView - DisplayView - - - - microphoneGainControlView - NSView - - - microphoneGainMenuItem - NSMenuItem - - - microphoneGainSlider - NSSlider - - - microphoneMuteButton - NSButton - - - outputVolumeControlView - NSView - - - outputVolumeMenuItem - NSMenuItem - - - saveScreenshotPanelAccessoryView - NSView - - - view - DisplayView - - - - - - DisplayWindowController - - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - id - - - - changeCoreSpeed: - id - - - changeDisplayGap: - id - - - changeDisplayMode: - id - - - changeDisplayOrder: - id - - - changeDisplayOrientation: - id - - - changeHardwareMicGain: - id - - - changeRotation: - id - - - changeRotationRelative: - id - - - changeScale: - id - - - changeVideoOutputFilter: - id - - - changeVideoPixelScaler: - id - - - changeVolume: - id - - - copy: - id - - - openRom: - id - - - reset: - id - - - saveScreenshotAs: - id - - - toggleExecutePause: - id - - - toggleFullScreenDisplay: - id - - - toggleKeepMinDisplaySizeAtNormal: - id - - - toggleStatusBar: - id - - - toggleVideoSourceDeposterize: - id - - - writeDefaultsDisplayGap: - id - - - writeDefaultsDisplayRotation: - id - - - writeDefaultsDisplayVideoSettings: - id - - - writeDefaultsHUDSettings: - id - - - - IBProjectSource - ../userinterface/DisplayWindowController.mm - - - - - - NSActionCell - NSCell - - IBFrameworkSource - AppKit.framework/Headers/NSActionCell.h - - - - NSApplication - NSResponder - - IBFrameworkSource - AppKit.framework/Headers/NSApplication.h - - - - NSButton - NSControl - - IBFrameworkSource - AppKit.framework/Headers/NSButton.h - - - - NSButtonCell - NSActionCell - - IBFrameworkSource - AppKit.framework/Headers/NSButtonCell.h - - - - NSCell - NSObject - - IBFrameworkSource - AppKit.framework/Headers/NSCell.h - - - - NSControl - NSView - - IBFrameworkSource - AppKit.framework/Headers/NSControl.h - - - - NSFormatter - NSObject - - IBFrameworkSource - Foundation.framework/Headers/NSFormatter.h - - - - NSMenu - NSObject - - IBFrameworkSource - AppKit.framework/Headers/NSMenu.h - - - - NSMenuItem - NSObject - - IBFrameworkSource - AppKit.framework/Headers/NSMenuItem.h - - - - NSMenuItemCell - NSButtonCell - - IBFrameworkSource - AppKit.framework/Headers/NSMenuItemCell.h - - - - NSNumberFormatter - NSFormatter - - IBFrameworkSource - Foundation.framework/Headers/NSNumberFormatter.h - - - - NSPopUpButton - NSButton - - IBFrameworkSource - AppKit.framework/Headers/NSPopUpButton.h - - - - NSPopUpButtonCell - NSMenuItemCell - - IBFrameworkSource - AppKit.framework/Headers/NSPopUpButtonCell.h - - - - NSProgressIndicator - NSView - - IBFrameworkSource - AppKit.framework/Headers/NSProgressIndicator.h - - - - NSResponder - NSObject - - IBFrameworkSource - AppKit.framework/Headers/NSResponder.h - - - - NSSlider - NSControl - - IBFrameworkSource - AppKit.framework/Headers/NSSlider.h - - - - NSSliderCell - NSActionCell - - IBFrameworkSource - AppKit.framework/Headers/NSSliderCell.h - - - - NSTextField - NSControl - - IBFrameworkSource - AppKit.framework/Headers/NSTextField.h - - - - NSTextFieldCell - NSActionCell - - IBFrameworkSource - AppKit.framework/Headers/NSTextFieldCell.h - - - - NSToolbar - NSObject - - IBFrameworkSource - AppKit.framework/Headers/NSToolbar.h - - - - NSToolbarItem - NSObject - - IBFrameworkSource - AppKit.framework/Headers/NSToolbarItem.h - - - - NSView - NSResponder - - IBFrameworkSource - AppKit.framework/Headers/NSView.h - - - - NSWindow - NSResponder - - IBFrameworkSource - AppKit.framework/Headers/NSWindow.h - - - - NSWindowController - NSResponder - - showWindow: - id - - - showWindow: - - showWindow: - id - - - - IBFrameworkSource - AppKit.framework/Headers/NSWindowController.h - - - + 232 + 0 IBCocoaFramework NO diff --git a/desmume/src/cocoa/userinterface/EmuControllerDelegate.h b/desmume/src/cocoa/userinterface/EmuControllerDelegate.h index 2ef5e9b8b..e1193c988 100644 --- a/desmume/src/cocoa/userinterface/EmuControllerDelegate.h +++ b/desmume/src/cocoa/userinterface/EmuControllerDelegate.h @@ -75,9 +75,8 @@ class AudioSampleBlockGenerator; NSInteger frameJumpToFrame; CGFloat lastSetSpeedScalar; - uint32_t hwMicNumberIdleFrames; - uint32_t hwMicNumberClippedFrames; BOOL isSoftwareMicActive; + BOOL isHardwareMicAvailable; BOOL isHardwareMicIdle; BOOL isHardwareMicInClip; float currentMicGainValue; @@ -142,7 +141,7 @@ class AudioSampleBlockGenerator; @property (assign) BOOL isRomLoading; @property (assign) NSString *statusText; @property (assign) BOOL isSoftwareMicActive; -@property (assign) BOOL isHardwareMicMuted; +@property (assign) BOOL isHardwareMicAvailable; @property (assign) BOOL isHardwareMicIdle; @property (assign) BOOL isHardwareMicInClip; @property (assign) float currentMicGainValue; @@ -261,8 +260,6 @@ class AudioSampleBlockGenerator; - (void) pauseCore; - (void) restoreCoreState; - (void) updateMicStatusIcon; -- (BOOL) isMicSampleIdle:(uint8_t)sampleValue; -- (BOOL) isMicSampleCausingClip:(uint8_t)sampleValue; - (AudioSampleBlockGenerator *) selectedAudioFileGenerator; - (void) setSelectedAudioFileGenerator:(AudioSampleBlockGenerator *)theGenerator; diff --git a/desmume/src/cocoa/userinterface/EmuControllerDelegate.mm b/desmume/src/cocoa/userinterface/EmuControllerDelegate.mm index 0577d2fce..3f30dff09 100644 --- a/desmume/src/cocoa/userinterface/EmuControllerDelegate.mm +++ b/desmume/src/cocoa/userinterface/EmuControllerDelegate.mm @@ -72,7 +72,7 @@ @synthesize isRomLoading; @synthesize statusText; @synthesize isSoftwareMicActive; -@dynamic isHardwareMicMuted; +@synthesize isHardwareMicAvailable; @synthesize isHardwareMicIdle; @synthesize isHardwareMicInClip; @synthesize currentMicGainValue; @@ -127,12 +127,11 @@ frameJumpToFrame = 0; lastSetSpeedScalar = 1.0f; - hwMicNumberIdleFrames = 0; - hwMicNumberClippedFrames = 0; isSoftwareMicActive = NO; + isHardwareMicAvailable = NO; isHardwareMicIdle = YES; isHardwareMicInClip = NO; - currentMicGainValue = 0.5f; + currentMicGainValue = 0.0f; isSoundMuted = NO; lastSetVolumeValue = MAX_VOLUME; @@ -292,19 +291,6 @@ return [cdsCore speedScalar]; } -- (void) setIsHardwareMicMuted:(BOOL)theState -{ - CocoaDSCore *cdsCore = (CocoaDSCore *)[cdsCoreController content]; - [[cdsCore cdsController] setHardwareMicMute:theState]; - [self updateMicStatusIcon]; -} - -- (BOOL) isHardwareMicMuted -{ - CocoaDSCore *cdsCore = (CocoaDSCore *)[cdsCoreController content]; - return [[cdsCore cdsController] hardwareMicMute]; -} - - (void) setCurrentVolumeValue:(float)vol { currentVolumeValue = vol; @@ -1960,46 +1946,6 @@ [self performSelectorOnMainThread:@selector(setCurrentMicStatusIcon:) withObject:micIcon waitUntilDone:NO]; } -- (BOOL) isMicSampleIdle:(uint8_t)sampleValue -{ - if (sampleValue == MIC_NULL_SAMPLE_VALUE) - { - if (hwMicNumberIdleFrames < MIC_NULL_FRAME_MAX_COUNT) - { - hwMicNumberIdleFrames++; - } - } - else - { - if (hwMicNumberIdleFrames > 0) - { - hwMicNumberIdleFrames--; - } - } - - return (hwMicNumberIdleFrames >= MIC_NULL_FRAME_THRESHOLD); -} - -- (BOOL) isMicSampleCausingClip:(uint8_t)sampleValue -{ - if (sampleValue == 0 || sampleValue == 127) - { - if (hwMicNumberClippedFrames < MIC_CLIP_FRAME_MAX_COUNT) - { - hwMicNumberClippedFrames++; - } - } - else - { - if (hwMicNumberClippedFrames > 0) - { - hwMicNumberClippedFrames--; - } - } - - return (hwMicNumberClippedFrames >= MIC_CLIP_FRAME_THRESHOLD); -} - - (AudioSampleBlockGenerator *) selectedAudioFileGenerator { CocoaDSCore *cdsCore = (CocoaDSCore *)[cdsCoreController content]; @@ -2494,21 +2440,21 @@ // sure to check the mic's level state. if (!controllerSoftwareMicState) { - const BOOL isSampleIdle = [self isMicSampleIdle:sampleValue]; - const BOOL isIdle = [self isHardwareMicIdle]; - if (( isSampleIdle && !isIdle) || - (!isSampleIdle && isIdle)) + const BOOL isIdleHardware = [cdsController isHardwareMicIdle]; + const BOOL isIdleEmuControl = [self isHardwareMicIdle]; + if (( isIdleHardware && !isIdleEmuControl) || + (!isIdleHardware && isIdleEmuControl)) { - [self setIsHardwareMicIdle:isSampleIdle]; + [self setIsHardwareMicIdle:isIdleHardware]; needsUpdate = YES; } - const BOOL isCausingClip = [self isMicSampleCausingClip:sampleValue]; - const BOOL isInClip = [self isHardwareMicInClip]; - if (( isCausingClip && !isInClip) || - (!isCausingClip && isInClip)) + const BOOL isClipHardware = [cdsController isHardwareMicInClip]; + const BOOL isClipEmuControl = [self isHardwareMicInClip]; + if (( isClipHardware && !isClipEmuControl) || + (!isClipHardware && isClipEmuControl)) { - [self setIsHardwareMicInClip:isCausingClip]; + [self setIsHardwareMicInClip:isClipHardware]; needsUpdate = YES; } } @@ -2521,6 +2467,16 @@ return sampleValue; } +- (void) doMicHardwareStateChangedFromController:(CocoaDSController *)cdsController + isEnabled:(BOOL)isHardwareEnabled + isLocked:(BOOL)isHardwareLocked + isMuted:(BOOL)isHardwareMuted +{ + const BOOL hwMicAvailable = (isHardwareEnabled && !isHardwareLocked); + [self setIsHardwareMicAvailable:hwMicAvailable]; + [self updateMicStatusIcon]; +} + - (void) doMicHardwareGainChangedFromController:(CocoaDSController *)cdsController gain:(float)gainValue { [self setCurrentMicGainValue:gainValue];