28 #import <AVFoundation/AVFoundation.h>
74 #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1080
143 - (
void) captureOutput:(AVCaptureOutput *)captureOutput
144 didOutputSampleBuffer:(CMSampleBufferRef)videoFrame
145 fromConnection:(AVCaptureConnection *)connection;
153 if (
self = [super
init]) {
159 - (
void) captureOutput:(AVCaptureOutput *)captureOutput
160 didOutputSampleBuffer:(CMSampleBufferRef)videoFrame
161 fromConnection:(AVCaptureConnection *)connection
189 - (
void) captureOutput:(AVCaptureOutput *)captureOutput
190 didOutputSampleBuffer:(CMSampleBufferRef)audioFrame
191 fromConnection:(AVCaptureConnection *)connection;
199 if (
self = [super
init]) {
205 - (
void) captureOutput:(AVCaptureOutput *)captureOutput
206 didOutputSampleBuffer:(CMSampleBufferRef)audioFrame
207 fromConnection:(AVCaptureConnection *)connection
228 [ctx->capture_session stopRunning];
230 [ctx->capture_session release];
231 [ctx->video_output release];
232 [ctx->audio_output release];
233 [ctx->avf_delegate release];
234 [ctx->avf_audio_delegate release];
269 NSError *error = nil;
270 AVCaptureInput* capture_input = nil;
273 capture_input = (AVCaptureInput*) [[[AVCaptureDeviceInput alloc] initWithDevice:video_device error:&error] autorelease];
275 capture_input = (AVCaptureInput*) video_device;
278 if (!capture_input) {
280 [[error localizedDescription] UTF8String]);
285 [ctx->capture_session addInput:capture_input];
292 ctx->
video_output = [[AVCaptureVideoDataOutput alloc] init];
305 pxl_fmt_spec = avf_pixel_formats[i];
312 av_log(s,
AV_LOG_ERROR,
"Selected pixel format (%s) is not supported by AVFoundation.\n",
318 if ([[ctx->
video_output availableVideoCVPixelFormatTypes] indexOfObject:[NSNumber numberWithInt:pxl_fmt_spec.
avf_id]] == NSNotFound) {
319 av_log(s,
AV_LOG_ERROR,
"Selected pixel format (%s) is not supported by the input device.\n",
325 for (NSNumber *pxl_fmt
in [ctx->
video_output availableVideoCVPixelFormatTypes]) {
329 if ([pxl_fmt intValue] == avf_pixel_formats[i].
avf_id) {
330 pxl_fmt_dummy = avf_pixel_formats[i];
340 pxl_fmt_spec = pxl_fmt_dummy;
355 NSNumber *pixel_format = [NSNumber numberWithUnsignedInt:pxl_fmt_spec.avf_id];
356 NSDictionary *capture_dict = [NSDictionary dictionaryWithObject:pixel_format
357 forKey:(id)kCVPixelBufferPixelFormatTypeKey];
359 [ctx->video_output setVideoSettings:capture_dict];
360 [ctx->video_output setAlwaysDiscardsLateVideoFrames:YES];
364 dispatch_queue_t queue = dispatch_queue_create(
"avf_queue", NULL);
365 [ctx->video_output setSampleBufferDelegate:ctx->avf_delegate queue:queue];
366 dispatch_release(queue);
369 [ctx->capture_session addOutput:ctx->video_output];
381 NSError *error = nil;
382 AVCaptureDeviceInput* audio_dev_input = [[[AVCaptureDeviceInput alloc] initWithDevice:audio_device error:&error] autorelease];
384 if (!audio_dev_input) {
386 [[error localizedDescription] UTF8String]);
391 [ctx->capture_session addInput:audio_dev_input];
398 ctx->
audio_output = [[AVCaptureAudioDataOutput alloc] init];
407 dispatch_queue_t queue = dispatch_queue_create(
"avf_audio_queue", NULL);
408 [ctx->audio_output setSampleBufferDelegate:ctx->avf_audio_delegate queue:queue];
409 dispatch_release(queue);
412 [ctx->capture_session addOutput:ctx->audio_output];
427 CFRunLoopRunInMode(kCFRunLoopDefaultMode, 0.1, YES);
442 CVImageBufferRef image_buffer = CMSampleBufferGetImageBuffer(ctx->
current_frame);
443 CGSize image_buffer_size = CVImageBufferGetEncodedSize(image_buffer);
447 stream->
codec->
width = (int)image_buffer_size.width;
448 stream->
codec->
height = (
int)image_buffer_size.height;
465 CFRunLoopRunInMode(kCFRunLoopDefaultMode, 0.1, YES);
480 CMFormatDescriptionRef format_desc = CMSampleBufferGetFormatDescription(ctx->
current_audio_frame);
481 const AudioStreamBasicDescription *basic_desc = CMAudioFormatDescriptionGetStreamBasicDescription(format_desc);
495 ctx->
audio_float = basic_desc->mFormatFlags & kAudioFormatFlagIsFloat;
496 ctx->
audio_be = basic_desc->mFormatFlags & kAudioFormatFlagIsBigEndian;
498 ctx->
audio_packed = basic_desc->mFormatFlags & kAudioFormatFlagIsPacked;
501 if (basic_desc->mFormatID == kAudioFormatLinearPCM &&
530 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
534 uint32_t num_screens = 0;
539 #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1070
540 CGGetActiveDisplayList(0, NULL, &num_screens);
546 NSArray *devices = [AVCaptureDevice devicesWithMediaType:AVMediaTypeVideo];
548 for (AVCaptureDevice *device
in devices) {
549 const char *
name = [[device localizedName] UTF8String];
550 index = [devices indexOfObject:device];
554 #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1070
555 if (num_screens > 0) {
556 CGDirectDisplayID screens[num_screens];
557 CGGetActiveDisplayList(num_screens, screens, &num_screens);
558 for (
int i = 0; i < num_screens; i++) {
565 devices = [AVCaptureDevice devicesWithMediaType:AVMediaTypeAudio];
566 for (AVCaptureDevice *device
in devices) {
567 const char *
name = [[device localizedName] UTF8String];
568 int index = [devices indexOfObject:device];
575 AVCaptureDevice *video_device = nil;
576 AVCaptureDevice *audio_device = nil;
578 NSArray *video_devices = [AVCaptureDevice devicesWithMediaType:AVMediaTypeVideo];
594 video_device = [video_devices objectAtIndex:ctx->video_device_index];
596 #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1070
597 CGDirectDisplayID screens[num_screens];
598 CGGetActiveDisplayList(num_screens, screens, &num_screens);
599 AVCaptureScreenInput* capture_screen_input = [[[AVCaptureScreenInput alloc] initWithDisplayID:screens[ctx->video_device_index - ctx->
num_video_devices]] autorelease];
600 video_device = (AVCaptureDevice*) capture_screen_input;
609 video_device = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];
612 for (AVCaptureDevice *device
in video_devices) {
614 video_device = device;
619 #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1070
623 if(sscanf(ctx->
video_filename,
"Capture screen %d", &idx) && idx < num_screens) {
624 CGDirectDisplayID screens[num_screens];
625 CGGetActiveDisplayList(num_screens, screens, &num_screens);
626 AVCaptureScreenInput* capture_screen_input = [[[AVCaptureScreenInput alloc] initWithDisplayID:screens[idx]] autorelease];
627 video_device = (AVCaptureDevice*) capture_screen_input;
642 NSArray *devices = [AVCaptureDevice devicesWithMediaType:AVMediaTypeAudio];
649 audio_device = [devices objectAtIndex:ctx->audio_device_index];
653 audio_device = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeAudio];
655 NSArray *devices = [AVCaptureDevice devicesWithMediaType:AVMediaTypeAudio];
657 for (AVCaptureDevice *device
in devices) {
659 audio_device = device;
672 if (!video_device && !audio_device) {
685 av_log(s,
AV_LOG_DEBUG,
"audio device '%s' opened\n", [[audio_device localizedName] UTF8String]);
697 [ctx->capture_session startRunning];
724 CVImageBufferRef image_buffer = CMSampleBufferGetImageBuffer(ctx->
current_frame);
727 if (
av_new_packet(pkt, (
int)CVPixelBufferGetDataSize(image_buffer)) < 0) {
737 CVPixelBufferLockBaseAddress(image_buffer, 0);
739 void*
data = CVPixelBufferGetBaseAddress(image_buffer);
742 CVPixelBufferUnlockBaseAddress(image_buffer, 0);
747 int block_buffer_size = CMBlockBufferGetDataLength(block_buffer);
749 if (!block_buffer || !block_buffer_size) {
772 if (ret != kCMBlockBufferNoErr) {
779 #define INTERLEAVE_OUTPUT(bps) \
781 int##bps##_t **src; \
782 int##bps##_t *dest; \
783 src = av_malloc(ctx->audio_channels * sizeof(int##bps##_t*)); \
784 if (!src) return AVERROR(EIO); \
785 for (c = 0; c < ctx->audio_channels; c++) { \
786 src[c] = ((int##bps##_t*)ctx->audio_buffer) + c * num_samples; \
788 dest = (int##bps##_t*)pkt->data; \
789 shift = bps - ctx->audio_bits_per_sample; \
790 for (sample = 0; sample < num_samples; sample++) \
791 for (c = 0; c < ctx->audio_channels; c++) \
792 *dest++ = src[c][sample] << shift; \
802 OSStatus
ret = CMBlockBufferCopyDataBytes(block_buffer, 0, pkt->
size, pkt->
data);
803 if (ret != kCMBlockBufferNoErr) {
829 {
"list_devices",
"list available devices", offsetof(
AVFContext, list_devices),
AV_OPT_TYPE_INT, {.i64=0}, 0, 1,
AV_OPT_FLAG_DECODING_PARAM,
"list_devices" },
847 .
name =
"avfoundation",
854 .priv_class = &avf_class,