iOS:CGImageCreateWith [PNG或JPEG] DataProvider导致分段错误

时间:2012-09-11 06:19:26

标签: ios objective-c core-graphics jailbreak zbar-sdk

我面临一个奇怪的问题。我正在使用libzbar开发一个iOS命令行条形码扫描器实用程序(是的,这是用于越狱设备)。一切顺利,除非我尝试使用CGImageCreateWithPNGDataProvider()CGImageCreateWithJPEGDataProvider()方法从文件中获取CGImageRef - 因为这两个函数在我的5.1.1 iPad上抛出了一个段错误。问题不在我的自定义类ZBarScanner中,因为如果我使用UIImage获取图像数据,请使用类似

的内容
UIImage *uiImage = [UIImage imageWithContentsOfFile:fname];
CGImageRef image = uiImage.CGImage;

然后它工作正常并打印存储在条形码中的数据。此外,PNG和JPEG图像格式正确 - 我可以使用设备本身的文件浏览器查看它们,我也尝试了其他几个图像。我甚至试图省略所有CFRelease()函数调用和release消息,以避免有悬空指针。这是我的代码:

#define LOG() NSLog(@"Reached line %d", __LINE__)

int main(int argc, char **argv)
{
    if (argc != 2)
            return 1;

    NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];

    LOG(); // line 21

    NSString *fname = [[NSString stringWithUTF8String:argv[1]] retain]; // added an extra retain just in case

    LOG(); // line 25

    CFDataRef data = (CFDataRef)[NSData dataWithContentsOfFile:fname];
    CGDataProviderRef dProv = CGDataProviderCreateWithCFData(data);
    // I also tried using
    // dProv = CGDataProviderCreateWithFilename(argv[1]);
    // that made no difference either. The data and data provider are
    // valid, but the CGImage constructors always segfault.
    if (dProv == NULL) {
        fprintf(stderr, "Invalid CGDataProvider\n");
        abort();
    }

    LOG(); // line 34

    CGImageRef image = NULL;

    if ([[fname pathExtension] isEqualToString:@"png"]) {
        LOG(); // line 39
        NSLog(@"Function pointer: %p", CGImageCreateWithPNGDataProvider);
        image = CGImageCreateWithPNGDataProvider(dProv, NULL, false, kCGRenderingIntentDefault); // This function segfaults, or...
        LOG();
    } else if ([[fname pathExtension] isEqualToString:@"jpg"]
        || [[fname pathExtension] isEqualToString:@"jpeg"]) {
        LOG();
        image = CGImageCreateWithJPEGDataProvider(dProv, NULL, true, kCGRenderingIntentDefault); // ... or this one.
        LOG();
    } else {
        fprintf(stderr, "File '%s' is neither a PNG nor a JPEG file!\n", argv[1]);
        LOG();
        abort();
    }

    LOG();
    // CFRelease(dProv);
    LOG();
    ZBarScanner *scanner = [ZBarScanner zbarScannerWithCGImage:image];
    // CFRelease(image);
    LOG();
    NSArray *arr = [scanner scan];
    NSLog(@"The result of the scanning is:\n%@", arr);
    LOG();
    [pool drain];

    return 0;
}

如果我在调试器中运行它(为清楚起见,删除了GDB和NSLog杂乱):

gdb ./scanner
(gdb) run ./barcode1.png
Reached line 21
Reached line 25
Reached line 34
Reached line 39
Function pointer: 0x37c5b535

Program received signal EXC_BAD_ACCESS, Could not access memory.
Reason: KERN_INVALID_ADDRESS at address: 0x00000000
0x00000000 in ?? ()
(gdb) backtrace
#0 0x00000000 in ?? ()
(gdb)

所以即使是回溯也没有显示任何明显错误/有用的东西......似乎某些东西某个地方。我甚至怀疑由于我的工具链是一个非官方的4.0版本,这些功能可能在iOS 5.1.1中不可用,所以构建成功,因为CGImageCreateWith[...]DataProvider符号在开发中 sysroot但不在iOS'实际动态库中,但如果是这种情况,我NSLogged的函数指针将是NULL,对吧?但是,NS和CG对象以及函数似乎都不是NULL - 我传递给CGImage构造函数的唯一NULL是decodeArray参数,但在Apple的文档中明确提到它可以为NULL ... (更新:我尝试传递一个有效的非NULL数组来查明文档是否错误,但我仍然遇到同样的错误。)

请问关于此次崩溃的任何指示(双关语)?我在这里错过了什么?到目前为止我发现的所有教程和参考建议使用CGDataProvider和CGImage就像这样。

1 个答案:

答案 0 :(得分:5)

使用文件名时,Apple文档中的示例代码使用了您在评论中提到的尝试的组合,加上值kCGRenderingIntentPerceptual而不是默认值:

    CGDataProviderRef pngDP = CGDataProviderCreateWithFilename([filePath fileSystemRepresentation]);
    if (pngDP) {
        CGImageRef img = CGImageCreateWithPNGDataProvider(pngDP, NULL, true, kCGRenderingIntentPerceptual); // true for interpolate, false for not-interpolate
执行此操作应该使您不必将数据保留在程序中,并且可以防止您看到的段错误。

(至少,或许可以在官方iOS文档中找到CoreTextPageViewer的示例代码,构建该项目,并尝试弄清楚你所做的事情有何不同。)