Как я могу решить EXC_BAD_ACCESS, который возникает в AudioToolBox IOS

Мне нужна помощь в решении проблемы, из-за которой я иногда испытываю сбой EXC_BAD_ACCESS, который происходит в AudioToolBox (AVAudioSessionPropertyListener) во время перехода между MPMoviePlayerController и MPMusicPlayerController. Следующее - трассировка звонка:

OS Version:      iPhone OS 5.1 (9B176)
Report Version:  104

Exception Type:  EXC_BAD_ACCESS (SIGSEGV)
Exception Codes: KERN_INVALID_ADDRESS at 0x00000009
Crashed Thread:  0

Thread 0 name:  Dispatch queue: com.apple.main-thread
Thread 0 Crashed:
0   libobjc.A.dylib                 0x3676ef78 objc_msgSend + 16
1   AVFoundation                    0x31e60b1c _ZL30AVAudioSessionPropertyListenerPvmmPKv + 236
2   AudioToolbox                    0x3630b300 AudioSessionPropertyListeners::CallPropertyListenersImp(unsigned long, unsigned long, void const*) + 268
3   AudioToolbox                    0x3630b5de AudioSessionPropertyListeners::CallPropertyListeners(unsigned long, unsigned long, void const*) + 234
4   AudioToolbox                    0x3630925a SSServer_AudioSessionInterruptionListenerMessage + 50
5   AudioToolbox                    0x362b0d2c _XAudioSessionInterruptionListenerMessage + 56
6   AudioToolbox                    0x36245cdc mshMIGPerform + 368
7   CoreFoundation                  0x3532151c __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE1_PERFORM_FUNCTION__ + 32
8   CoreFoundation                  0x353214be __CFRunLoopDoSource1 + 134
9   CoreFoundation                  0x3532030c __CFRunLoopRun + 1364
10  CoreFoundation                  0x352a349e CFRunLoopRunSpecific + 294
11  CoreFoundation                  0x352a3366 CFRunLoopRunInMode + 98
12  GraphicsServices                0x3659f432 GSEventRunModal + 130
13  UIKit                           0x32399e76 UIApplicationMain + 1074
14  MyGreatApp                      0x00054986 main (main.m:14)
15  MyGreatApp                      0x00054944 start + 32

Мое приложение чередует использование музыкального проигрывателя и проигрывателя фильмов. Проигрыватель фильмов используется для воспроизведения подкастов. Проигрыватель фильмов автоматически запускается каждый раз, когда воспроизводится новый подкаст. Когда я закончу с проигрывателем фильмов, я удаляю всех установленных мною наблюдателей, убеждаюсь, что проигрыватель фильмов остановлен, и перезапускаю музыкальный проигрыватель.

Вот соответствующий код для инициализации использования проигрывателя фильмов:

[musicPlayer pause];

self.moviePlayer = [[[MPMoviePlayerController alloc] initWithContentURL: address] autorelease];   // Release the old moviePlayer and create a new one.

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(moviePreloadDidFinish:) name:MPMoviePlayerLoadStateDidChangeNotification object:self.moviePlayer];    

// Register to receive a notification when the movie has finished playing. 
[[NSNotificationCenter defaultCenter] addObserver:self  selector:@selector(moviePlayBackDidFinish:) name:MPMoviePlayerPlaybackDidFinishNotification object:self.moviePlayer];

// Register to receive a notification when the movie playback state changes (specifically looking for the state of interrupted). 
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(moviePlayBackStateChanged:) name:MPMoviePlayerPlaybackStateDidChangeNotification object:self.moviePlayer];

[self.moviePlayer play];

Фильм может либо закончиться после воспроизведения, либо пользователь может завершить фильм, нажав кнопку быстрой перемотки вперед. Вот соответствующий код для обработки этих двух сценариев:

-(IBAction) fastForwardMovie: (id) sender {

        // The self.moviePlayer is still playing, but the user wants to skip the rest of the movie/podcast. 
        // Lets fake a movie playback did finish event
        NSNotification *notification = [NSNotification notificationWithName:MPMoviePlayerPlaybackDidFinishNotification object:nil];
        [self moviePlayBackDidFinish: notification];    // Fake a finished playback so that the music player can be restarted
        return;
}


//  Notification called when the movie FINISHED playing.
- (void) moviePlayBackDidFinish:(NSNotification*)notification { 
    // Remove further notifications until the next time we need the movie player
    [[NSNotificationCenter defaultCenter] removeObserver:self name:MPMoviePlayerLoadStateDidChangeNotification object:self.moviePlayer] ;
    [[NSNotificationCenter defaultCenter] removeObserver:self name:MPMoviePlayerPlaybackDidFinishNotification object:self.moviePlayer] ;
    [[NSNotificationCenter defaultCenter] removeObserver:self name:MPMoviePlayerPlaybackStateDidChangeNotification object:self.moviePlayer] ;


    // Get rid of the movie player
    self.moviePlayer.initialPlaybackTime = -1;  // seems that the best way to kill the movie player is first setting the playback time to -1
    [self.moviePlayer stop];                    // followed by stopping the player

    [musicPlayer play];
}

После завершения фильма (подкаста) музыка будет воспроизводиться даже в случае сбоя (но приложение закрывается). Это потому, что я использую музыкальный проигрыватель iPod, но это может указывать на то, что проблема связана с проигрывателем фильмов.

И последнее замечание. Я попытался тщательно следовать рекомендациям Руководства по программированию аудиосеансов от Apple, где говорится, что, когда желаемая конфигурация должна иметь весь аудиомикс, вы должны «Настроить аудиосеанс с использованием конфигурации смешиваемых категорий» и «Воспользуйтесь преимуществами фильма. значение по умолчанию для useApplicationAudioSession плеера - YES ".

Может ли кто-нибудь помочь мне выяснить, что я делаю неправильно, что приводит к сбою AVAudioSessionPropertyListener с EXC_BAD_ACCESS? Или, по крайней мере, вы можете предложить мне, как изолировать эту проблему, чтобы определить основную причину?


person JeffB6688    schedule 22.04.2012    source источник
comment
Я также случайно получаю эти сбои в моем приложении Lucid Reality Check. II имеют очень похожую настройку видео / аудио и очень похожий отчет о сбое, ура   -  person manitu    schedule 24.04.2012
comment
Я также столкнулся с этим падением в своем приложении. на самом деле он занимает первое место во всех сбоях. Но я вообще понятия не имею. К вашему сведению: на данный момент количество пользователей составляет около 790 из 100000 (новая версия приложения выпущена на 20 дней).   -  person LetBulletFlies    schedule 29.10.2012


Ответы (2)


Попробуйте вместо этого позвонить [musicPlayer performSelectorOnMainThread:@selector(play) ...]?

person user1330054    schedule 12.06.2012
comment
Спасибо за это предложение. Сейчас тестирую вашу идею. Поскольку это не часто встречающаяся проблема, у меня может уйти месяц, чтобы проверить предположение. Если проблема больше не появится, я отмечу это как правильный ответ. Между тем, я заметил, что ваше предложение не имеет негативных последствий. Все идет нормально. Спасибо - person JeffB6688; 20.08.2012
comment
Я только что снова просмотрел этот отчет о сбое с предложенным изменением. К сожалению, предложенное изменение не устранило проблему. - person JeffB6688; 05.09.2012

Мы только что отслеживали очень похожий сбой.

Наши оказываются такими, как описано здесь:

https://github.com/mattgallagher/AudioStreamer/issues/6

В частности, MyAudioSessionInterruptionListener (или имя обратного вызова, переданного в AudioSessionInitialize) и его inClientData не могут быть изменены после регистрации, поэтому обратный вызов всегда должен делать что-то разумное, даже если базовый объект был освобожден.

Решение, предлагаемое для AudioStream, состоит в том, чтобы использовать статическую переменную и убедиться, что она указывает на объект, который в данный момент заинтересован в обратном вызове, и никогда не указывает на освобожденный объект - важно не использовать inClientData.

person JosephH    schedule 04.12.2012