iOS - AVassetWriter outputting file of 0 size and not giving errors

505 Views Asked by At

I'm still very new to swift (and programming) and I'm trying to output the CVPixelbuffer I get from ARFrame to a video in realtime (without the AR stuff on top).

I've set up the AVAssetWriter and Input and on each frame I try to append the CVPixelbuffer (converted into a CMSampleBuffer).

The code creates a file of 0 bytes and doesn't give any errors. I can't figure out what's wrong, and the documentation is rather cryptic for my level.

I'm pretty sure the timeStamp for each frame is incorrect. But I can't figure out if each frame simple needs a time, or the total time since the start of the video.

Then there is the size and the format of the cvPixelBuffer. I also can't figure out if I need to tell the AVAssetwriter what the size & format of the data is that is coming or if it figures that out by itself.

Thank you for your time.

class VideoHandler
{
    var videoWriter: AVAssetWriter?
    var videoWriterInput: AVAssetWriterInput?

    var videoURL = URL(fileURLWithPath: "")
    let videoSize = CGSize(width: 1280, height: 720)

    func start()
    {
        videoURL = getDocumentsDirectory().appendingPathComponent("Temp.mov")

        do {
            try FileManager.default.removeItem(at: videoURL)
        } catch {}

        do {
            try videoWriter = AVAssetWriter(outputURL: videoURL, fileType: AVFileType.mov)
        } catch let error as NSError {
            print(error)
            videoWriter = nil
        }

        let videoSettings: [String : AnyObject] = [
            AVVideoCodecKey  : AVVideoCodecType.h264 as AnyObject,
            AVVideoWidthKey  : videoSize.width as AnyObject,
            AVVideoHeightKey : videoSize.height as AnyObject,
            //        AVVideoCompressionPropertiesKey : [
            //          AVVideoAverageBitRateKey : NSInteger(1000000),
            //          AVVideoMaxKeyFrameIntervalKey : NSInteger(16),
            //          AVVideoProfileLevelKey : AVVideoProfileLevelH264BaselineAutoLevel
            //        ]
            ]

        videoWriterInput = AVAssetWriterInput(mediaType: AVMediaType.video, outputSettings: videoSettings)
        videoWriterInput?.expectsMediaDataInRealTime = true

        //TODO: Force unwrapping... so dangerous but don't know any other way.

        if videoWriter!.canAdd(videoWriterInput!)
        {
            videoWriter!.add(videoWriterInput!)
        }
        else
        {
            print("Can't add videoWriterInput")
        }

        if videoWriter?.startWriting() ?? false
        {
            videoWriter?.startSession(atSourceTime: CMTime.zero)
        }

        print("Video writing started.")
    }


    func addFrame(buffer: CVPixelBuffer, timeStamp: Double)
    {
        if videoWriterInput?.isReadyForMoreMediaData ?? false
        {
            videoWriterInput?.append(getCMSampleBuffer(buffer: buffer, timeStamp: timeStamp))
        }
    }

    func getCMSampleBuffer(buffer: CVPixelBuffer, timeStamp: Double) -> CMSampleBuffer
    {
        // Don't know yet how to set the time and timescale stuff properly.

        var info = CMSampleTimingInfo()
//        info.presentationTimeStamp = CMTime.zero
        info.presentationTimeStamp = CMTime(seconds: timeStamp, preferredTimescale: 1)
//        info.duration = CMTime.invalid
//        info.decodeTimeStamp = CMTime.invalid

        var formatDesc: CMFormatDescription? = nil

        CMVideoFormatDescriptionCreateForImageBuffer(
            allocator: kCFAllocatorDefault,
            imageBuffer: buffer,
            formatDescriptionOut: &formatDesc)

        var sampleBuffer: CMSampleBuffer? = nil

        CMSampleBufferCreateReadyWithImageBuffer(allocator: kCFAllocatorDefault,
                                                 imageBuffer: buffer,
                                                 formatDescription: formatDesc!,
                                                 sampleTiming: &info,
                                                 sampleBufferOut: &sampleBuffer);

        return sampleBuffer!
    }


    func stop()
    {
        videoWriter?.finishWriting
        {
            print("Video writing finished.")
        }
        videoWriter = nil
    }

    func getDocumentsDirectory() -> URL
    {
        let paths = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask)
        return paths[0]
    }

}
0

There are 0 best solutions below