1

我正在读取带有内容的视频数据,AVAssetReader并通过再次读取内容并修改时间戳来“循环”内容。相关代码为:

  private func readFrame() -> CMSampleBuffer? {
    let buffer = output?.copyNextSampleBuffer()
    guard buffer != nil else {
      do {
        reader?.cancelReading()
        try loop()
      } catch let e {
        return nil
      }
      return readFrame()
    }
    
    // if we've looped, add previous loop times
    // MEMORY LEAK IS IN HERE
    if CMTimeCompare(offsetTime, .zero) > 0 {
      var newTiming = [CMSampleTimingInfo(
        duration: CMSampleBufferGetDuration(buffer!),
        presentationTimeStamp: CMTimeAdd(CMSampleBufferGetOutputPresentationTimeStamp(buffer!), offsetTime),
        decodeTimeStamp: CMTimeAdd(CMSampleBufferGetOutputDecodeTimeStamp(buffer!), offsetTime)
      )]
      
      CMSampleBufferCreateCopyWithNewTiming(allocator: kCFAllocatorDefault, sampleBuffer: buffer!, sampleTimingEntryCount: 1, sampleTimingArray: &newTiming, sampleBufferOut: &loopedBuffer)
      return loopedBuffer
    }
    
    return buffer
  }

  private func loop() throws {
    guard (reader?.status == .some(.completed)) else {
      throw FileError.cannotLoop(reader?.error)
    }
    offsetTime = CMTimeAdd(offsetTime, assetDuration)
    try createReader() // creates a new reader and calls reader.startReading()
  }

一旦视频完成其第一个循环(即一次offsetTime大于零),内存就会开始爆炸。注释掉创建具有更新时序的新缓冲区的代码可以修复内存泄漏(但这不可行,因为我需要更新的时序)。

大概CMSampleBufferCreateCopyWithNewTiming是为每个缓冲区分配新空间,并且该缓冲区永远不会被清理。我已经阅读了autoreleasepool一个潜在的选项,但是这个缓冲区被传递了很多(它通过一个自定义的发布者)。有没有办法消除内存泄漏?我可以为 1 个缓冲区分配空间并在每一帧上一次又一次地覆盖它 - 这可能吗?

4

0 回答 0