28

我在苹果商店有一个应用程序,在 iOS6 更新后,我收到了数百个崩溃报告MKMapView。我无法在我的设备上重现崩溃。看起来有问题EAGLContextMKMapView我们的应用程序中没有使用 OpenGL,但我们在不同的控制器中有多个实例。我在这里发现了一个类似的问题,iOS 6 app 在显示地图时在 EAGLContext 中崩溃,但它们使用 OpenGL。

这里有回溯:

Exception Type:  SIGSEGV
Exception Codes: SEGV_ACCERR at 0x1
Crashed Thread:  0

Thread 0 Crashed:
0   libGPUSupportMercury.dylib          0x00000e22 gpus_ReturnNotPermittedKillClient + 10
1   libGPUSupportMercury.dylib          0x3bccc5fb gldCreateContext + 190
2   GLEngine                            0x344c2b15 gliCreateContextWithShared + 676
3   OpenGLES                            0x0000491d -[EAGLContext initWithAPI:properties:] + 1433
4   OpenGLES                            0x000042d7 -[EAGLContext initWithAPI:sharedWithCompute:] + 143
5   VectorKit                           0x00011c81 -[VGLGPU init] + 105
6   VectorKit                           0x000d4659 __24+[VGLGPU sharedInstance]_block_invoke_0 + 49
7   libdispatch.dylib                   0x000014b7 _dispatch_client_callout + 23
8   libdispatch.dylib                   0x000073f7 dispatch_once_f$VARIANT$mp + 43
9   VectorKit                           0x00011c13 +[VGLGPU sharedInstance] + 39
10  VectorKit                           0x00001db1 -[VKMainLoop updateLinkState] + 485
11  VectorKit                           0x00001955 -[VKScreenCanvas _updateDisplayStatus:] + 109
12  UIKit                               0x0001c371 -[UIView initWithFrame:] + 129
13  VectorKit                           0x00010ca5 -[VGLScreenCanvas initWithFrame:context:] + 53
14  VectorKit                           0x00010a7d -[VKScreenCanvas initWithFrame:context:] + 57
15  VectorKit                           0x00010a3f -[VKScreenCanvas initWithFrame:] + 39
16  VectorKit                           0x000106bd -[VKMapCanvas initWithFrame:shouldRasterize:] + 65
17  VectorKit                           0x000104bb -[VKMapView initWithFrame:andGlobe:shouldRasterize:] + 647
18  MapKit                              0x0000dc95 -[MKMapView _commonInitAndEnableLoading:fromIB:] + 725
19  MapKit                              0x0000d811 -[MKMapView initWithFrame:] + 257
.....
4

4 回答 4

24

当用户在我们的应用程序后台运行时,我们遇到了类似的问题,就像我们弹出一个包含地图子视图的窗口一样。崩溃似乎是由于我们在后台使用 openGL 调用的地图。我们必须将地图子视图的创建包装在一个检查中,如下所示:

UIApplicationState appState = [[UIApplication sharedApplication] applicationState];
    if( (appState != UIApplicationStateBackground) && (appState != UIApplicationStateInactive))
    {
        // Do map subview initialization...
    }
    else
    {
        self.attemptedToLoadMap = YES;
    }

我们保存了布尔值,以便如果应用程序回到前台,我们可以添加子视图以进行显示。

每当您以导致重新绘制操作的方式(例如,添加注释)操作地图时,您都必须这样做。

于 2012-10-25T21:30:53.097 回答
3

http://developer.apple.com/library/ios/#qa/qa1766/_index.html

此技术 QA 解决了此问题

于 2013-01-04T21:42:32.593 回答
3

我们在我们的应用程序中找到了这个问题的原因,所以我想发布解决方案,以防它对其他人有所帮助。我们的主视图控制器在显示时监视显着的位置变化,并在隐藏时停止监视。我们的一些用户在此屏幕上遇到了不相关的崩溃,这使应用程序处于监控状态。当应用程序注册了重大位置更改更新时,如果应用程序没有运行,iOS 实际上会将应用程序启动到后台以告知新位置。由于我们的应用程序在首次启动时会显示地图,这导致了崩溃。Apple 支持与我们确认,运行 iOS 8.x 的 32 位设备上存在一个错误,如果在应用程序处于后台时更新 MapView(或其他 OpenGL 上下文),可能会导致崩溃。

我们更改了代码,以便如果应用程序由于位置发生重大变化而启动,我们会立即停止监控并引发异常以使应用程序崩溃。这对用户来说是完全不可见的,因此他们不会注意到崩溃,并且可以防止进一步的崩溃发生。

- (void)validateLaunchWithOptions:(NSDictionary *)launchOptions
{
    if (launchOptions[@"UIApplicationLaunchOptionsLocationKey"]) {
        // the app was launched due to a significant location change, which is not valid and causes crashes
        // prevent this from happening again by disabling significant location monitoring
        CLLocationManager *locationManager = [[CLLocationManager alloc] init];
        [locationManager stopMonitoringSignificantLocationChanges];

        // intentionally crashing the app here; it is not in a good state and it needs to be prevented from
        // getting to any code that would re-enable significant location monitoring
        @throw [NSException exceptionWithName:@"com.redacted.significantLocationLaunch"
                                       reason:@"app may not be launched due to significant location changes"
                                     userInfo:@{}];
    }
}

我们看到了数千次这样的崩溃,但在我们找出原因之前无法在我们的测试设备上复制它。如果您想复制它(并确认修复),请在您的应用开始监控重大位置更改后立即@throw 异常。应用程序崩溃后,去开车。一旦您的手机切换信号塔,iOS 将在后台启动该应用程序,您将遇到崩溃。我们能够拿到我们的一部用户手机并检查崩溃日志。所有的车祸都发生在她上下班的路上。

于 2014-12-16T21:00:57.327 回答
2

我面临着类似的堆栈跟踪。我注意到在控制台中它提供了有关实际问题的更多详细信息:您不能在后台使用 GPU。iOS 5 的地图是基于图块的,所以我假设没有使用 GPU,但 iOS 6 中的新地图使用矢量图形,因此使用了 GPU。因此,任何曾经在后台的地图作品都不再可以。

于 2012-10-08T02:59:01.717 回答