#define PY_SSIZE_T_CLEAN
#include <Python.h>
const double bt709_coeff_r = 0.2126;
const double bt709_coeff_g = 0.7152;
const double bt709_coeff_b = 0.0722;
const double bt709_coeff_ri = 1.0-bt709_coeff_r;
const double bt709_coeff_bi = 1.0-bt709_coeff_b;
const double bt709_coeff_bg = bt709_coeff_b / bt709_coeff_g;
const double bt709_coeff_rg = bt709_coeff_r / bt709_coeff_g;
const int y_offset = 16 << 8;
const int y_range = 219;
const int cr_offset = 128 << 8;
const int cr_range = 224;
const int cr_middle = 224/2;
const double bt709_ri_range = bt709_coeff_ri / cr_middle;
const double bt709_bi_range = bt709_coeff_bi / cr_middle;
static PyObject *method_atem_to_rgb(PyObject *self, PyObject *args) {
Py_buffer input_buffer;
Py_ssize_t data_length;
unsigned int width, height;
PyObject *res;
/* Parse arguments */
if(!PyArg_ParseTuple(args, "y*II", &input_buffer, &width, &height)) {
return NULL;
}
data_length = input_buffer.len;
char *buffer;
char *resbuffer = (char *) malloc(data_length);
char *outbuffer = resbuffer;
int pixel_size = 8;
buffer = input_buffer.buf;
for(int i=0; i<data_length; i += pixel_size ){
buffer += pixel_size;
// Convert 10-bit BT.709 Y'CbCrA 4:2:2 to RGB
// Unpack bytes to 2xY 2xA and a B and R pair
unsigned short a1 = (buffer[0] << 4) + ((buffer[1] & 0xf0) >> 4);
unsigned short a2 = (buffer[4] << 4) + ((buffer[5] & 0xf0) >> 4);
unsigned short cb = ((buffer[1] & 0x0f) << 6) + ((buffer[2] & 0xfc) >> 2);
unsigned short cr = ((buffer[5] & 0x0f) << 6) + ((buffer[6] & 0xfc) >> 2);
unsigned short y1 = ((buffer[2] & 0x03) << 8) + (buffer[3] & 0xff);
unsigned short y2 = ((buffer[6] & 0x03) << 8) + (buffer[7] & 0xff);
float cbf = bt709_bi_range * ((cb<<6) - cr_offset);
float crf = bt709_ri_range * ((cr<<6) - cr_offset);
float y1f = ((double)(y1 << 6)-y_offset) / y_range;
float y2f = ((double)(y2 << 6)-y_offset) / y_range;
float r1f = fmin(255, y1f + crf);
float g1f = fmin(255, y1f - cbf * bt709_coeff_bg - crf * bt709_coeff_rg);
float b1f = fmin(255, y1f + cbf);
float r2f = fmin(255, y2f + crf);
float g2f = fmin(255, y2f - cbf * bt709_coeff_bg - crf * bt709_coeff_rg);
float b2f = fmin(255, y2f + cbf);
outbuffer[0] = (unsigned char)r1f;
outbuffer[1] = (unsigned char)g1f;
outbuffer[2] = (unsigned char)b1f;
outbuffer[3] = (unsigned char)(((double)(a1 -16)) / 3.6);
outbuffer[4] = (unsigned char)r2f;
outbuffer[5] = (unsigned char)g2f;
outbuffer[6] = (unsigned char)b2f;
outbuffer[7] = (unsigned char)(((double)(a2 -16)) / 3.6);
outbuffer += pixel_size;
}
res = Py_BuildValue("y#", resbuffer, data_length);
free(resbuffer);
return res;
}
static PyMethodDef MediaConvertMethods[] = {
{"atem_to_rgb", method_atem_to_rgb, METH_VARARGS, "Convert an Atem YCbCrA frame to RGB8888"},
{NULL, NULL, 0, NULL},
};
static struct PyModuleDef mediaconvertmodule = {
PyModuleDef_HEAD_INIT,
"pyatem.mediaconvert",
"Native code for converting frames",
-1,
MediaConvertMethods,
};
PyMODINIT_FUNC PyInit_mediaconvert(void){
return PyModule_Create(&mediaconvertmodule);
}