swift2 - iOS CVPixelBufferCreate在swift 2中泄漏内存

标签 swift2 avfoundation avassetwriter cvpixelbuffer

我正在尝试将图像转换为视频,正确的方法似乎是使用带有 AVAssetWriterInputPixelBufferAdaptor 的 AVAssetWriter,它运行良好,但会泄漏内存。

当我将 CGImage 转换为 CVPixelBuffer 时,我调用了 CVPixelBufferCreate,它永远不会释放它的内存。

func CGImageToPixelBuffer(image: CGImageRef, frameSize: CGSize) -> CVPixelBuffer {

    // stupid CFDictionary stuff
    let keys: [CFStringRef] = [kCVPixelBufferCGImageCompatibilityKey, kCVPixelBufferCGBitmapContextCompatibilityKey]
    let values: [CFTypeRef] = [kCFBooleanTrue, kCFBooleanTrue]
    let keysPointer = UnsafeMutablePointer<UnsafePointer<Void>>.alloc(1)
    let valuesPointer =  UnsafeMutablePointer<UnsafePointer<Void>>.alloc(1)
    keysPointer.initialize(keys)
    valuesPointer.initialize(values)
    let options = CFDictionaryCreate(kCFAllocatorDefault, keysPointer, valuesPointer, keys.count,
        UnsafePointer<CFDictionaryKeyCallBacks>(), UnsafePointer<CFDictionaryValueCallBacks>())

    let buffer = UnsafeMutablePointer<CVPixelBuffer?>.alloc(1)

    // here's the leak >:[
    let status = CVPixelBufferCreate(kCFAllocatorDefault, Int(frameSize.width), Int(frameSize.height),
        kCVPixelFormatType_32ARGB, options, buffer)

    CVPixelBufferLockBaseAddress(pixelBuffer.memory!, 0);
    let bufferData = CVPixelBufferGetBaseAddress(buffer.memory!);

    let rgbColorSpace = CGColorSpaceCreateDeviceRGB();
    let context = CGBitmapContextCreate(bufferData, Int(frameSize.width),
        Int(frameSize.height), 8, 4*Int(frameSize.width), rgbColorSpace,
        CGImageAlphaInfo.NoneSkipFirst.rawValue);

    CGContextDrawImage(context, CGRectMake(0, 0,
        CGFloat(CGImageGetWidth(image)),
        CGFloat(CGImageGetHeight(image))), image);

    CVPixelBufferUnlockBaseAddress(pixelBuffer.memory!, 0);
    return buffer.memory!
}

这是调用 CGImageToPixelBuffer 的代码
func saveImageAsVideoFile(path: NSURL, image: UIImage, duration: Double) {
    let writer = try! AVAssetWriter(URL: path, fileType: AVFileTypeQuickTimeMovie)

    let videoSettings: NSDictionary = [
        AVVideoCodecKey : AVVideoCodecH264,
        AVVideoWidthKey : NSNumber(integer: Int(image.size.width)),
        AVVideoHeightKey : NSNumber(integer: Int(image.size.height))
    ]
    let input = AVAssetWriterInput(mediaType: AVMediaTypeVideo, outputSettings: videoSettings as? [String : AnyObject])
    let inputAdaptor = AVAssetWriterInputPixelBufferAdaptor(assetWriterInput: input, sourcePixelBufferAttributes: nil)
    input.expectsMediaDataInRealTime = false
    writer.addInput(input)
    writer.startWriting()
    writer.startSessionAtSourceTime(kCMTimeZero)

    // leak starts here
    let buffer = CGImageToPixelBuffer(image.CGImage!, frameSize: image.size)

    // append 30 frames to AVAssetWriter thing
    for i in 0..<30
        if input.readyForMoreMediaData {
            inputAdaptor.appendPixelBuffer(buffer, withPresentationTime: CMTimeMake(i, 30))
        }
    }

    //CVPixelBufferRelease(buffer) ??

    input.markAsFinished()
    writer.finishWritingWithCompletionHandler({ () -> Void in
        print("yay")
    })
}

CVPixelBufferRelease 在 swift 2 中不可用

CVPixelBufferCreate 不会在 swift 2 中返回非托管指针,所以我不能使用这个 guys code

我尝试在 unsafepointer 上手动调用 destroy 和 dealloc,但无济于事。

每次调用它都会增加内存使用量,并且如果调用足够多的话会导致设备崩溃
memory profile

任何帮助或建议将不胜感激。

最佳答案

您的 let buffer = UnsafeMutablePointer<CVPixelBuffer?>.alloc(1)不平衡 dealloc ,但是添加这似乎并不能解决泄漏问题。
只需使用 CVPixelBuffer?确实如此,而且看起来更简单:

var buffer: CVPixelBuffer?

// TODO: handle status != noErr
let status = CVPixelBufferCreate(kCFAllocatorDefault, Int(frameSize.width), Int(frameSize.height),
    kCVPixelFormatType_32ARGB, options, &buffer)

CVPixelBufferLockBaseAddress(buffer!, 0);
let bufferData = CVPixelBufferGetBaseAddress(buffer!);

let rgbColorSpace = CGColorSpaceCreateDeviceRGB();
let context = CGBitmapContextCreate(bufferData, Int(frameSize.width),
    Int(frameSize.height), 8, 4*Int(frameSize.width), rgbColorSpace,
    CGImageAlphaInfo.NoneSkipFirst.rawValue);

CGContextDrawImage(context, CGRectMake(0, 0,
    CGFloat(CGImageGetWidth(image)),
    CGFloat(CGImageGetHeight(image))), image);

CVPixelBufferUnlockBaseAddress(buffer!, 0);

你也应该处理 status != noErr并尽快解开可选缓冲区,以避免所有这些 buffer! s。

关于swift2 - iOS CVPixelBufferCreate在swift 2中泄漏内存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33746584/

相关文章:

ios - 在后台检查服务器端应用程序通知,在 Swift 中

swift - indexPath swift 的索引超出范围

objective-c - 使用 AVFoundation 修改音频元数据

ios - 如何在 iOS Swift 中为 CMSampleBufferGetFormatDescription 创建一个AudioSampleBuffer

swift - 调用 AVAudioRecorder 的初始化程序 - Swift 2.0

ios - 将 MPVolumeView slider 拇指位置设置为设备音量

iOS AVAudioSession 中断通知未按预期工作

iOS - 混合两个音频文件

ios - AVCaptureSession 和 AVAssetWriter 不同的设置

iphone - iPhone 上的 OpenGL 到视频