iOS 8.3 - EXC_BAD_ACCESS в CoreGraphics Рисование изображения
В моем приложении у меня есть модуль рисования изображений CoreGraphics. Приложение случайно падает при вызове функции CGContextAddPath(). Путь - это объект CGPathRef, созданный с помощью вызова функции CGPathCreateMutable(). Пример кода приведен ниже:
-(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
if (touching) return;
//start a new path
path = CGPathCreateMutable();
//set the path's starting point
UITouch *touch = (UITouch *)[touches anyObject];
CGPathMoveToPoint(path, NULL, [touch locationInView:m_view].x, [touch locationInView:m_view].y);
touching = YES;
}
-(void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event {
if (touching){
//get the current location of the touch event
UITouch *theTouch = (UITouch *)[touches anyObject];
pathPoint = [theTouch locationInView:m_view];
CGPathAddLineToPoint(path, NULL, pathPoint.x, pathPoint.y);
[canvasLayer setNeedsDisplay];
}
}
-(void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event {
if (!touching) return;
//create a new image context
UIGraphicsBeginImageContext(CGSizeMake(backgroundLayer.bounds.size.width, backgroundLayer.bounds.size.height));
//grab a reference to the new image context
CGContextRef ctx = UIGraphicsGetCurrentContext();
//push the image context to the top of the drawing stack
UIGraphicsPushContext(ctx);
//set the blend mode to prevent white pixels from
//covering up the lines that have already been drawn
CGContextSetBlendMode(ctx, kCGBlendModeDarken);
if (cacheImage != nil) {
//draw the cached state of the image to the image context and release it
[cacheImage drawInRect:CGRectMake(0, 0, backgroundLayer.bounds.size.width, backgroundLayer.bounds.size.height)];
}
//blend the drawing layer into the image context
[canvasLayer drawInContext:ctx];
//we're done drawing to the image context
UIGraphicsPopContext();
//store the image context so we can add to it again later
cacheImage = UIGraphicsGetImageFromCurrentImageContext();
//we're finished with the image context altogether
UIGraphicsEndImageContext();
touching = NO;
//release the path
CGPathRelease(path);
//update the background layer (we'll need to draw the cached image to the background)
[backgroundLayer setNeedsDisplay];
}
- (void)drawLayer:(CATiledLayer *)layer inContext:(CGContextRef)ctx {
//this method is handling multiple layers, so first
//determine which layer we're drawing to
if (layer == canvasLayer) {
if (!touching) return;
//add the path to the context
CGContextAddPath(ctx, path); /***Exception points to this line***/
//set a line width and draw the path
CGContextSetLineWidth(ctx, 6.0f);//---thikness of line
CGContextStrokePath(ctx);
}
else if (layer == backgroundLayer) {
//remember the current state of the context
CGContextSaveGState(ctx);
//the cached image coordinate system is upside down, so do a backflip
CGContextTranslateCTM(ctx, 0, backgroundLayer.bounds.size.height);
CGContextScaleCTM(ctx, 1.0, -1.0);
//draw the image
CGImageRef ref = cacheImage.CGImage;
CGContextDrawImage(ctx, backgroundLayer.bounds, ref);
//restore the context to its pre-flipped state
CGContextRestoreGState(ctx);
}
}
Отчет о сбое диагностики, полученный с устройства, показывает:
Exception Type: EXC_BAD_ACCESS (SIGBUS)
Exception Subtype: KERN_PROTECTION_FAILURE at 0x090d3330
Triggered by Thread: 13
Thread 13 name: Dispatch queue: com.apple.root.default-qos
Thread 13 Crashed:
0 ??? 0x090d3330 0 + 151860016
1 CoreGraphics 0x22886b6c 0x22874000 + 76652
2 CoreGraphics 0x228a036e 0x22874000 + 181102
3 MyApplication 0x001c8062 0xb6000 + 1122402
4 QuartzCore 0x256d08fc 0x25695000 + 243964
5 QuartzCore 0x257a9e86 0x25695000 + 1134214
6 QuartzCore 0x256fdd14 0x25695000 + 429332
7 libdispatch.dylib 0x30f99866 0x30f8b000 + 59494
8 libdispatch.dylib 0x30f9a89e 0x30f8b000 + 63646
9 libsystem_pthread.dylib 0x3110eda6 0x3110e000 + 3494
10 libsystem_pthread.dylib 0x3110eaf8 0x3110e000 + 2808
Когда я символизирую адреса памяти в отчете о сбое, исключение указывает на CGContextAddPath(ctx, path);, Это вызвано явным освобождением пути с использованием CGPathRelease (path);? Если я закомментирую функцию релиза, приложение не аварийно завершит работу, однако произойдет ли утечка памяти?
1 ответ
Вы уничтожаете путь в touchEnded, но после этого может вызываться drawLayer.