1
0
mirror of https://github.com/Tha14/toxic.git synced 2024-06-29 13:57:45 +02:00
toxic/src/osx_video.m

342 lines
9.0 KiB
Mathematica
Raw Normal View History

2015-08-20 13:39:47 +02:00
/* osx_video.m
*
*
* Copyright (C) 2014 Toxic All Rights Reserved.
*
* This file is part of Toxic.
*
* Toxic is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Toxic is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Toxic. If not, see <http://www.gnu.org/licenses/>.
*
*/
#ifdef __OBJC__
#include "osx_video.h"
#import <Foundation/Foundation.h>
#import <AVFoundation/AVFoundation.h>
#include "line_info.h"
#include "settings.h"
#include <errno.h>
#include <stdbool.h>
#include <string.h>
#include <pthread.h>
#include <unistd.h>
#include <stdlib.h>
#include <assert.h>
/*
* Helper video format functions
*/
static uint8_t rgb_to_y(int r, int g, int b)
{
int y = ((9798 * r + 19235 * g + 3736 * b) >> 15);
return y > 255 ? 255 : y < 0 ? 0 : y;
2015-08-20 13:39:47 +02:00
}
static uint8_t rgb_to_u(int r, int g, int b)
{
int u = ((-5538 * r + -10846 * g + 16351 * b) >> 15) + 128;
return u > 255 ? 255 : u < 0 ? 0 : u;
2015-08-20 13:39:47 +02:00
}
static uint8_t rgb_to_v(int r, int g, int b)
{
int v = ((16351 * r + -13697 * g + -2664 * b) >> 15) + 128;
return v > 255 ? 255 : v < 0 ? 0 : v;
2015-08-20 13:39:47 +02:00
}
2015-08-20 21:23:01 +02:00
void bgrxtoyuv420(uint8_t *plane_y, uint8_t *plane_u, uint8_t *plane_v, uint8_t *rgb, uint16_t width, uint16_t height)
2015-08-20 13:39:47 +02:00
{
uint16_t x, y;
uint8_t *p;
uint8_t r, g, b;
for (y = 0; y != height; y += 2) {
2015-08-20 13:39:47 +02:00
p = rgb;
for (x = 0; x != width; x++) {
2015-08-20 13:39:47 +02:00
b = *rgb++;
g = *rgb++;
r = *rgb++;
2015-08-20 21:23:01 +02:00
rgb++;
2015-08-20 13:39:47 +02:00
*plane_y++ = rgb_to_y(r, g, b);
}
for (x = 0; x != width / 2; x++) {
2015-08-20 13:39:47 +02:00
b = *rgb++;
g = *rgb++;
r = *rgb++;
2015-08-20 21:23:01 +02:00
rgb++;
2015-08-20 13:39:47 +02:00
*plane_y++ = rgb_to_y(r, g, b);
b = *rgb++;
g = *rgb++;
r = *rgb++;
2015-08-20 21:23:01 +02:00
rgb++;
2015-08-20 13:39:47 +02:00
*plane_y++ = rgb_to_y(r, g, b);
b = ((int)b + (int) * (rgb - 8) + (int) * p + (int) * (p + 4) + 2) / 4;
p++;
g = ((int)g + (int) * (rgb - 7) + (int) * p + (int) * (p + 4) + 2) / 4;
p++;
r = ((int)r + (int) * (rgb - 6) + (int) * p + (int) * (p + 4) + 2) / 4;
p++;
2015-08-20 21:23:01 +02:00
p++;
2015-08-20 13:39:47 +02:00
*plane_u++ = rgb_to_u(r, g, b);
*plane_v++ = rgb_to_v(r, g, b);
2015-08-20 21:23:01 +02:00
p += 4;
2015-08-20 13:39:47 +02:00
}
}
}
/*
* End of helper video format functions
*/
/*
* Implementation for OSXVideo
*/
@implementation OSXVideo {
dispatch_queue_t _processingQueue;
AVCaptureSession *_session;
AVCaptureVideoDataOutput *_linkerVideo;
CVImageBufferRef _currentFrame;
pthread_mutex_t _frameLock;
2015-08-20 21:23:01 +02:00
BOOL _shouldMangleDimensions;
2015-08-20 13:39:47 +02:00
}
- (instancetype)initWithDeviceNames:
(char **)device_names AmtDevices:
(int *)size
{
2015-08-20 13:39:47 +02:00
_session = [[AVCaptureSession alloc] init];
NSArray *devices = [AVCaptureDevice devicesWithMediaType: AVMediaTypeVideo];
2015-08-20 21:23:01 +02:00
int i;
2015-08-20 21:23:01 +02:00
for (i = 0; i < [devices count]; ++i) {
AVCaptureDevice *device = [devices objectAtIndex: i];
2015-08-20 13:39:47 +02:00
char *video_input_name;
NSString *localizedName = [device localizedName];
video_input_name = (char *)malloc(strlen([localizedName cStringUsingEncoding: NSUTF8StringEncoding]) + 1);
strcpy(video_input_name, (char *)[localizedName cStringUsingEncoding: NSUTF8StringEncoding]);
2015-08-20 13:39:47 +02:00
device_names[i] = video_input_name;
}
2015-08-20 21:23:01 +02:00
2015-08-20 13:39:47 +02:00
if ( i <= 0 )
return nil;
2015-08-20 13:39:47 +02:00
*size = i;
return self;
}
- (void)dealloc
{
2015-08-20 21:23:01 +02:00
pthread_mutex_destroy(&_frameLock);
2015-08-20 13:39:47 +02:00
[_session release];
2015-08-20 21:23:01 +02:00
[_linkerVideo release];
dispatch_release(_processingQueue);
[super dealloc];
2015-08-20 13:39:47 +02:00
}
- (int)openVideoDeviceIndex:
(uint32_t)device_idx Width:
(uint16_t *)width Height:
(uint16_t *)height
{
2015-08-20 13:39:47 +02:00
pthread_mutex_init(&_frameLock, NULL);
2015-08-20 21:23:01 +02:00
pthread_mutex_lock(&_frameLock);
2015-08-20 13:39:47 +02:00
_processingQueue = dispatch_queue_create("Toxic processing queue", DISPATCH_QUEUE_SERIAL);
NSArray *devices = [AVCaptureDevice devicesWithMediaType: AVMediaTypeVideo];
AVCaptureDevice *device = [devices objectAtIndex: device_idx];
2015-08-20 21:23:01 +02:00
NSError *error = NULL;
AVCaptureInput *input = [[AVCaptureDeviceInput alloc] initWithDevice: device error: &error];
2015-08-20 21:23:01 +02:00
if ( error != NULL ) {
[input release];
return -1;
}
[_session beginConfiguration];
[_session addInput: input];
2015-08-20 21:23:01 +02:00
//_session.sessionPreset = AVCaptureSessionPreset640x480;
//*width = 640;
//*height = 480;
_shouldMangleDimensions = YES;
[_session commitConfiguration];
[input release];
[device release];
/* Obtain device resolution */
AVCaptureInputPort *port = [input.ports objectAtIndex: 0];
2015-08-20 21:23:01 +02:00
CMFormatDescriptionRef format_description = port.formatDescription;
2015-08-20 21:23:01 +02:00
if ( format_description ) {
CMVideoDimensions dimensions = CMVideoFormatDescriptionGetDimensions(format_description);
*width = dimensions.width;
*height = dimensions.height;
} else {
*width = 0;
*height = 0;
2015-08-20 13:39:47 +02:00
}
_linkerVideo = [[AVCaptureVideoDataOutput alloc] init];
[_linkerVideo setSampleBufferDelegate: self queue: _processingQueue];
2015-08-20 13:39:47 +02:00
// TODO possibly get a better pixel format
2015-08-20 21:23:01 +02:00
if (_shouldMangleDimensions) {
[_linkerVideo setVideoSettings: @ {
2017-06-30 00:26:06 +02:00
(id)kCVPixelBufferPixelFormatTypeKey: @(kCVPixelFormatType_32BGRA),
(id)kCVPixelBufferWidthKey: @640,
(id)kCVPixelBufferHeightKey: @480
}];
2015-08-20 21:23:01 +02:00
} else {
[_linkerVideo setVideoSettings: @ {(id)kCVPixelBufferPixelFormatTypeKey: @(kCVPixelFormatType_32BGRA)}];
2015-08-20 21:23:01 +02:00
}
[_session addOutput: _linkerVideo];
2015-08-20 13:39:47 +02:00
[_session startRunning];
2015-08-20 21:23:01 +02:00
pthread_mutex_unlock(&_frameLock);
2015-08-20 13:39:47 +02:00
return 0;
}
- (void)closeVideoDeviceIndex:
(uint32_t)device_idx
{
NSArray *devices = [AVCaptureDevice devicesWithMediaType: AVMediaTypeVideo];
AVCaptureDevice *device = [devices objectAtIndex: device_idx];
2015-08-20 21:23:01 +02:00
NSError *error = NULL;
AVCaptureInput *input = [[AVCaptureDeviceInput alloc] initWithDevice: device error: &error];
2015-08-20 13:39:47 +02:00
[_session stopRunning];
[_session removeOutput: _linkerVideo];
[_session removeInput: input];
2015-08-20 13:39:47 +02:00
[_linkerVideo release];
}
- (void)captureOutput:
(AVCaptureOutput *)captureOutput didOutputSampleBuffer:
(CMSampleBufferRef)sampleBuffer fromConnection:
(AVCaptureConnection *)connection
{
2015-08-20 21:23:01 +02:00
pthread_mutex_lock(&_frameLock);
CVImageBufferRef img = CMSampleBufferGetImageBuffer(sampleBuffer);
2015-08-20 21:23:01 +02:00
if (!img) {
NSLog(@"Toxic WARNING: Bad sampleBuffer from AVfoundation!");
} else {
CVPixelBufferUnlockBaseAddress(_currentFrame, kCVPixelBufferLock_ReadOnly);
RELEASE_CHK(CFRelease, _currentFrame);
_currentFrame = (CVImageBufferRef)CFRetain(img);
// we're not going to do anything to it, so it's safe to lock it always
CVPixelBufferLockBaseAddress(_currentFrame, kCVPixelBufferLock_ReadOnly);
}
2015-08-20 21:23:01 +02:00
pthread_mutex_unlock(&_frameLock);
}
- (int)getVideoFrameY:
(uint8_t *)y U:
(uint8_t *)u V:
(uint8_t *)v Width:
(uint16_t *)width Height:
(uint16_t *)height
{
2015-08-20 21:23:01 +02:00
if (!_currentFrame) {
return -1;
}
2015-08-20 13:39:47 +02:00
2015-08-20 21:23:01 +02:00
pthread_mutex_lock(&_frameLock);
CFRetain(_currentFrame);
2015-08-20 13:39:47 +02:00
2015-08-20 21:23:01 +02:00
CFTypeID imageType = CFGetTypeID(_currentFrame);
2015-08-20 13:39:47 +02:00
if (imageType == CVPixelBufferGetTypeID()) {
// TODO maybe handle other formats
2015-08-20 21:23:01 +02:00
bgrxtoyuv420(y, u, v, CVPixelBufferGetBaseAddress(_currentFrame), *width, *height);
2015-08-20 13:39:47 +02:00
} else if (imageType == CVOpenGLBufferGetTypeID()) {
// OpenGL pbuffer
} else if (imageType == CVOpenGLTextureGetTypeID()) {
// OpenGL Texture (Do we need to handle these?)
}
2015-08-20 21:23:01 +02:00
CVPixelBufferRelease(_currentFrame);
pthread_mutex_unlock(&_frameLock);
return 0;
2015-08-20 13:39:47 +02:00
}
@end
/*
* End of implementation for OSXVideo
*/
/*
* C-interface for OSXVideo
*/
static OSXVideo *_OSXVideo = nil;
2015-08-20 13:39:47 +02:00
int osx_video_init(char **device_names, int *size)
{
_OSXVideo = [[OSXVideo alloc] initWithDeviceNames: device_names AmtDevices: size];
2015-08-20 13:39:47 +02:00
if ( _OSXVideo == nil )
return -1;
return 0;
}
void osx_video_release()
{
[_OSXVideo release];
2015-08-20 21:23:01 +02:00
_OSXVideo = nil;
2015-08-20 13:39:47 +02:00
}
2015-08-20 21:23:01 +02:00
int osx_video_open_device(uint32_t selection, uint16_t *width, uint16_t *height)
2015-08-20 13:39:47 +02:00
{
2015-08-20 21:23:01 +02:00
if ( _OSXVideo == nil )
return -1;
return [_OSXVideo openVideoDeviceIndex: selection Width: width Height: height];
2015-08-20 13:39:47 +02:00
}
void osx_video_close_device(uint32_t device_idx)
{
[_OSXVideo closeVideoDeviceIndex: device_idx];
2015-08-20 13:39:47 +02:00
}
2015-08-20 21:23:01 +02:00
int osx_video_read_device(uint8_t *y, uint8_t *u, uint8_t *v, uint16_t *width, uint16_t *height)
2015-08-20 13:39:47 +02:00
{
2015-08-20 21:23:01 +02:00
if ( _OSXVideo == nil )
return -1;
return [_OSXVideo getVideoFrameY: y U: u V: v Width: width Height: height];
2015-08-20 13:39:47 +02:00
}
/*
* End of C-interface for OSXVideo
*/
#endif /* __OBJC__ */