我正在尝试根据用户的输入来裁剪和缩放CMSampleBufferRef,在ratio上,以下代码采用CMSampleBufferRef,将其转换为CVImageBufferRef并使用CVPixelBuffer根据其字节来裁剪内部图像。此过程的目标是使裁剪后的CVPixelBufferRef缩放到视频

- (CVPixelBufferRef)modifyImage:(CMSampleBufferRef) sampleBuffer {
    @synchronized (self) {
        CVImageBufferRef imageBuffer = CMSampleBufferGetImageBuffer(sampleBuffer);
        // Lock the image buffer
        CVPixelBufferLockBaseAddress(imageBuffer,0);

        // Get information about the image
        uint8_t *baseAddress = (uint8_t *)CVPixelBufferGetBaseAddress(imageBuffer);
        size_t bytesPerRow = CVPixelBufferGetBytesPerRow(imageBuffer);
        size_t width = CVPixelBufferGetWidth(imageBuffer);
        size_t height = CVPixelBufferGetHeight(imageBuffer);

        CVPixelBufferRef pxbuffer;
        NSDictionary *options = [NSDictionary dictionaryWithObjectsAndKeys:
                                 [NSNumber numberWithBool:YES], kCVPixelBufferCGImageCompatibilityKey,
                                 [NSNumber numberWithBool:YES], kCVPixelBufferCGBitmapContextCompatibilityKey,
                                 [NSNumber numberWithInt:720], kCVPixelBufferWidthKey,
                                 [NSNumber numberWithInt:1280], kCVPixelBufferHeightKey,
                                 nil];
        NSInteger tempWidth = (NSInteger) (width / ratio);
        NSInteger tempHeight = (NSInteger) (height / ratio);

        NSInteger baseAddressStart = 100 + 100 * bytesPerRow;
        CVReturn status = CVPixelBufferCreateWithBytes(kCFAllocatorDefault, tempWidth, tempHeight, kCVPixelFormatType_32BGRA, &baseAddress[baseAddressStart], bytesPerRow, MyPixelBufferReleaseCallback, NULL, (CFDictionaryRef)options, &pxbuffer);

        if (status != 0) {
            CKLog(@"%d", status);
            return NULL;
        }

        CVPixelBufferUnlockBaseAddress(imageBuffer,0);

        return pxbuffer;
    }
}

一切正常,除了当我尝试使用此方法将其写入视频输出时,它会不断收到内存警告。如果我保持相同的比例就可以了
- (void)writeBufferFrame:(CMSampleBufferRef)sampleBuffer pixelBuffer:(CVPixelBufferRef)pixelBuffer {
    CMTime lastSampleTime = CMSampleBufferGetPresentationTimeStamp(sampleBuffer);

    if(self.videoWriter.status != AVAssetWriterStatusWriting)
    {
        CKLog(@"%d", self.videoWriter.status);
        [self.videoWriter startWriting];
        [self.videoWriter startSessionAtSourceTime:lastSampleTime];
    }

    CVPixelBufferRef pxbuffer = [self modifyImage:sampleBuffer];
    BOOL success = [self.avAdaptor appendPixelBuffer:pxbuffer withPresentationTime:lastSampleTime];
    if (!success)
        NSLog(@"Warning:  Unable to write buffer to video");
}

我还尝试了使用CMSampleBufferRefCGContext的不同方法。如果您可以在这里为任何方法提供解决方案,我可以给您满分

最佳答案

尝试在对-CVPixelBufferLockBaseAddress和-CVPixelBufferUnlockBaseAddress的两个调用中都使用kCVPixelBufferLock_ReadOnly标志。
有时可以通过复制像素缓冲区来解决此问题。执行一次分配:

unsigned char *data = (unsigned char*)malloc(ySize * sizeof(unsigned char));
之后,将数据从pixelBuffer复制到数据

size_t大小=高度* bytesPerRow;
memcpy(data,baseAddress,size);

之后,使用数据。希望这会有所帮助。

10-06 09:00