2015-08-07 14:37:41 +02:00
|
|
|
// STL includes
|
2015-08-20 09:51:44 +02:00
|
|
|
#include <algorithm>
|
2015-08-07 14:37:41 +02:00
|
|
|
#include <cassert>
|
|
|
|
#include <iostream>
|
2017-09-01 08:50:37 +02:00
|
|
|
#include <QFile>
|
2016-07-24 19:56:41 +02:00
|
|
|
|
2015-08-07 14:37:41 +02:00
|
|
|
// Linux includes
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
|
|
|
// Local includes
|
2016-07-24 19:56:41 +02:00
|
|
|
#include <utils/Logger.h>
|
2016-02-24 14:42:25 +01:00
|
|
|
#include <grabber/AmlogicGrabber.h>
|
2017-09-01 08:50:37 +02:00
|
|
|
#include "Amvideocap.h"
|
2015-08-07 14:37:41 +02:00
|
|
|
|
2017-09-01 08:50:37 +02:00
|
|
|
#define VIDEO_DEVICE "/dev/amvideo"
|
|
|
|
#define CAPTURE_DEVICE "/dev/amvideocap0"
|
2017-09-03 13:48:16 +02:00
|
|
|
#define GE2D_DEVICE "/dev/ge2d"
|
2015-08-07 14:37:41 +02:00
|
|
|
|
2016-07-24 19:56:41 +02:00
|
|
|
AmlogicGrabber::AmlogicGrabber(const unsigned width, const unsigned height)
|
2017-08-04 23:08:15 +02:00
|
|
|
: Grabber("AMLOGICGRABBER", qMax(160u, width), qMax(160u, height)) // Minimum required width or height is 160
|
2017-09-01 08:50:37 +02:00
|
|
|
, _captureDev(-1)
|
|
|
|
, _videoDev(-1)
|
2017-09-03 13:48:16 +02:00
|
|
|
, _ge2dDev(-1)
|
2017-09-01 08:50:37 +02:00
|
|
|
, _lastError(0)
|
|
|
|
, _fbGrabber("/dev/fb0",width,height)
|
|
|
|
, _grabbingModeNotification(0)
|
2017-09-03 13:48:16 +02:00
|
|
|
, _ge2dAvailable(true)
|
|
|
|
, _ge2dVideoBufferPtr(nullptr)
|
|
|
|
, _ge2dIonBuffer(nullptr)
|
2015-08-07 14:37:41 +02:00
|
|
|
{
|
2016-07-24 19:56:41 +02:00
|
|
|
Debug(_log, "constructed(%d x %d)",_width,_height);
|
2015-08-07 14:37:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
AmlogicGrabber::~AmlogicGrabber()
|
|
|
|
{
|
2017-09-01 08:50:37 +02:00
|
|
|
closeDev(_captureDev);
|
|
|
|
closeDev(_videoDev);
|
2017-09-03 13:48:16 +02:00
|
|
|
closeDev(_ge2dDev);
|
2017-09-01 08:50:37 +02:00
|
|
|
}
|
|
|
|
|
2017-09-03 13:48:16 +02:00
|
|
|
bool AmlogicGrabber::openDev(int &fd, const char* dev)
|
2017-09-01 08:50:37 +02:00
|
|
|
{
|
|
|
|
if (fd<0)
|
2015-08-07 14:37:41 +02:00
|
|
|
{
|
2017-09-03 13:48:16 +02:00
|
|
|
fd = open(dev, O_RDWR);
|
2015-08-07 14:37:41 +02:00
|
|
|
}
|
2017-09-01 08:50:37 +02:00
|
|
|
return fd >= 0;
|
2015-08-07 14:37:41 +02:00
|
|
|
}
|
|
|
|
|
2017-09-01 08:50:37 +02:00
|
|
|
void AmlogicGrabber::closeDev(int &fd)
|
|
|
|
{
|
|
|
|
if (fd >= 0)
|
|
|
|
{
|
|
|
|
close(fd);
|
|
|
|
fd = -1;
|
|
|
|
}
|
|
|
|
}
|
2015-08-07 14:37:41 +02:00
|
|
|
|
2015-08-20 09:51:44 +02:00
|
|
|
bool AmlogicGrabber::isVideoPlaying()
|
|
|
|
{
|
2017-09-01 08:50:37 +02:00
|
|
|
if(!QFile::exists(VIDEO_DEVICE)) return false;
|
2015-08-20 09:51:44 +02:00
|
|
|
|
2017-09-01 08:50:37 +02:00
|
|
|
int videoDisabled = 1;
|
2017-09-03 13:48:16 +02:00
|
|
|
if (!openDev(_videoDev, VIDEO_DEVICE))
|
2015-08-20 09:51:44 +02:00
|
|
|
{
|
2017-09-01 08:50:37 +02:00
|
|
|
Error(_log, "Failed to open video device(%s): %d - %s", VIDEO_DEVICE, errno, strerror(errno));
|
2015-08-20 09:51:44 +02:00
|
|
|
return false;
|
|
|
|
}
|
2017-09-01 08:50:37 +02:00
|
|
|
else
|
2015-08-20 09:51:44 +02:00
|
|
|
{
|
2017-09-01 08:50:37 +02:00
|
|
|
// Check the video disabled flag
|
|
|
|
if(ioctl(_videoDev, AMSTREAM_IOC_GET_VIDEO_DISABLE, &videoDisabled) < 0)
|
|
|
|
{
|
|
|
|
Error(_log, "Failed to retrieve video state from device: %d - %s", errno, strerror(errno));
|
|
|
|
closeDev(_videoDev);
|
|
|
|
return false;
|
|
|
|
}
|
2015-08-20 09:51:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return videoDisabled == 0;
|
|
|
|
}
|
|
|
|
|
2017-08-12 07:55:32 +02:00
|
|
|
int AmlogicGrabber::grabFrame(Image<ColorRgb> & image)
|
2015-08-07 14:37:41 +02:00
|
|
|
{
|
2017-09-01 08:50:37 +02:00
|
|
|
if (!_enabled) return 0;
|
2015-08-20 09:51:44 +02:00
|
|
|
|
2017-09-01 08:50:37 +02:00
|
|
|
image.resize(_width,_height);
|
2015-08-20 09:51:44 +02:00
|
|
|
// Make sure video is playing, else there is nothing to grab
|
2017-09-01 08:50:37 +02:00
|
|
|
if (isVideoPlaying())
|
2015-08-20 09:51:44 +02:00
|
|
|
{
|
2017-09-01 08:50:37 +02:00
|
|
|
if (_grabbingModeNotification!=1)
|
|
|
|
{
|
|
|
|
Info(_log, "VPU mode");
|
|
|
|
_grabbingModeNotification = 1;
|
|
|
|
_lastError = 0;
|
|
|
|
}
|
2015-08-20 09:51:44 +02:00
|
|
|
|
2017-09-03 13:48:16 +02:00
|
|
|
if (_ge2dAvailable)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
_ge2dAvailable = (QFile::exists(GE2D_DEVICE) && grabFrame_ge2d(image) == 0);
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
_ge2dAvailable = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!_ge2dAvailable)
|
|
|
|
{
|
|
|
|
closeDev(_videoDev);
|
|
|
|
closeDev(_ge2dDev);
|
|
|
|
Warning(_log, "GE2D capture interface not available! try Amvideocap instead");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (QFile::exists(CAPTURE_DEVICE))
|
2017-09-01 08:50:37 +02:00
|
|
|
{
|
|
|
|
grabFrame_amvideocap(image);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2015-08-08 08:13:59 +02:00
|
|
|
{
|
2017-09-01 08:50:37 +02:00
|
|
|
if (_grabbingModeNotification!=2)
|
2015-08-20 09:51:44 +02:00
|
|
|
{
|
2017-09-01 08:50:37 +02:00
|
|
|
Info( _log, "FB mode");
|
|
|
|
_grabbingModeNotification = 2;
|
|
|
|
_lastError = 0;
|
2015-08-20 09:51:44 +02:00
|
|
|
}
|
2017-09-01 08:50:37 +02:00
|
|
|
_fbGrabber.grabFrame(image);
|
2015-08-08 08:13:59 +02:00
|
|
|
}
|
2015-08-20 09:51:44 +02:00
|
|
|
|
2017-09-01 08:50:37 +02:00
|
|
|
closeDev(_videoDev);
|
2015-08-20 09:51:44 +02:00
|
|
|
|
2017-09-01 08:50:37 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int AmlogicGrabber::grabFrame_amvideocap(Image<ColorRgb> & image)
|
|
|
|
{
|
|
|
|
// If the device is not open, attempt to open it
|
2017-09-03 13:48:16 +02:00
|
|
|
if (! openDev(_captureDev, CAPTURE_DEVICE))
|
2017-09-01 08:50:37 +02:00
|
|
|
{
|
|
|
|
ErrorIf( _lastError != 1, _log,"Failed to open the AMLOGIC device (%d - %s):", errno, strerror(errno));
|
|
|
|
_lastError = 1;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
long r1 = ioctl(_captureDev, AMVIDEOCAP_IOW_SET_WANTFRAME_WIDTH, _width);
|
|
|
|
long r2 = ioctl(_captureDev, AMVIDEOCAP_IOW_SET_WANTFRAME_HEIGHT, _height);
|
|
|
|
|
|
|
|
if (r1<0 || r2<0 || _height==0 || _width==0)
|
2015-08-08 08:13:59 +02:00
|
|
|
{
|
2017-09-01 08:50:37 +02:00
|
|
|
ErrorIf(_lastError != 2,_log,"Failed to configure capture size (%d - %s)", errno, strerror(errno));
|
|
|
|
closeDev(_captureDev);
|
|
|
|
_lastError = 2;
|
2015-08-20 09:51:44 +02:00
|
|
|
return -1;
|
2015-08-08 08:13:59 +02:00
|
|
|
}
|
2015-08-20 09:51:44 +02:00
|
|
|
|
2015-08-07 14:37:41 +02:00
|
|
|
// Read the snapshot into the memory
|
2017-09-01 08:50:37 +02:00
|
|
|
image.resize(_width, _height);
|
|
|
|
_image_bgr.resize(_width, _height);
|
|
|
|
const ssize_t bytesToRead = _image_bgr.size();
|
|
|
|
void * image_ptr = _image_bgr.memptr();
|
|
|
|
const ssize_t bytesRead = pread(_captureDev, image_ptr, bytesToRead, 0);
|
2015-08-20 09:51:44 +02:00
|
|
|
|
2017-09-01 08:50:37 +02:00
|
|
|
if (bytesRead < 0)
|
2015-08-20 09:51:44 +02:00
|
|
|
{
|
2017-09-01 08:50:37 +02:00
|
|
|
ErrorIf(_lastError != 3, _log,"Read of device failed: %d - %s", errno, strerror(errno));
|
|
|
|
closeDev(_captureDev);
|
|
|
|
_lastError = 3;
|
2015-08-20 09:51:44 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
else if (bytesToRead != bytesRead)
|
2015-08-07 14:37:41 +02:00
|
|
|
{
|
|
|
|
// Read of snapshot failed
|
2017-09-01 08:50:37 +02:00
|
|
|
ErrorIf(_lastError != 4, _log,"Capture failed to grab entire image [bytesToRead(%d) != bytesRead(%d)]", bytesToRead, bytesRead);
|
|
|
|
closeDev(_captureDev);
|
2015-08-20 09:51:44 +02:00
|
|
|
return -1;
|
2015-08-07 14:37:41 +02:00
|
|
|
}
|
2015-08-20 09:51:44 +02:00
|
|
|
|
2017-09-01 08:50:37 +02:00
|
|
|
closeDev(_captureDev);
|
2017-09-03 13:48:16 +02:00
|
|
|
_useImageResampler = true;
|
2017-09-01 08:50:37 +02:00
|
|
|
_imageResampler.processImage((const uint8_t*)image_ptr, _width, _height, _width*3, PIXELFORMAT_BGR24, image);
|
|
|
|
_lastError = 0;
|
2017-08-12 07:55:32 +02:00
|
|
|
|
2015-08-20 09:51:44 +02:00
|
|
|
return 0;
|
2015-08-07 14:37:41 +02:00
|
|
|
}
|
2017-09-01 08:50:37 +02:00
|
|
|
|
2017-09-03 13:48:16 +02:00
|
|
|
|
|
|
|
int AmlogicGrabber::grabFrame_ge2d(Image<ColorRgb> & image)
|
|
|
|
{
|
|
|
|
if ( ! openDev(_ge2dDev, GE2D_DEVICE) || ! openDev(_videoDev, VIDEO_DEVICE))
|
|
|
|
{
|
|
|
|
Error(_log, "cannot open devices");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ion
|
|
|
|
if (_ge2dIonBuffer == nullptr)
|
|
|
|
{
|
|
|
|
_ge2dIonBuffer = new IonBuffer(_width * _height * 3); // BGR
|
|
|
|
_ge2dVideoBufferPtr = _ge2dIonBuffer->Map();
|
|
|
|
memset(_ge2dVideoBufferPtr, 0, _ge2dIonBuffer->BufferSize());
|
|
|
|
}
|
|
|
|
|
|
|
|
int canvas_index;
|
|
|
|
if (ioctl(_videoDev, AMVIDEO_EXT_GET_CURRENT_VIDEOFRAME, &canvas_index) < 0)
|
|
|
|
{
|
|
|
|
Error(_log, "AMSTREAM_EXT_GET_CURRENT_VIDEOFRAME failed.");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t canvas0addr;
|
|
|
|
|
|
|
|
if (ioctl(_videoDev, AMVIDEO_EXT_CURRENT_VIDEOFRAME_GET_CANVAS0ADDR, &canvas0addr) < 0)
|
|
|
|
{
|
|
|
|
Error(_log, "AMSTREAM_EXT_CURRENT_VIDEOFRAME_GET_CANVAS0ADDR failed.");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t ge2dformat;
|
|
|
|
if (ioctl(_videoDev, AMVIDEO_EXT_CURRENT_VIDEOFRAME_GET_GE2D_FORMAT, &ge2dformat) <0)
|
|
|
|
{
|
|
|
|
Error(_log, "AMSTREAM_EXT_CURRENT_VIDEOFRAME_GET_GE2D_FORMAT failed.");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t size;
|
|
|
|
if (ioctl(_videoDev, AMVIDEO_EXT_CURRENT_VIDEOFRAME_GET_SIZE, &size) < 0)
|
|
|
|
{
|
|
|
|
Error(_log, "AMSTREAM_EXT_CURRENT_VIDEOFRAME_GET_SIZE failed.");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned cropLeft = _cropLeft;
|
|
|
|
unsigned cropRight = _cropRight;
|
|
|
|
unsigned cropTop = _cropTop;
|
|
|
|
unsigned cropBottom = _cropBottom;
|
|
|
|
int videoWidth = (size >> 32) - cropLeft - cropRight;
|
|
|
|
int videoHeight = (size & 0xffffff) - cropTop - cropBottom;
|
|
|
|
|
|
|
|
// calculate final image dimensions and adjust top/left cropping in 3D modes
|
|
|
|
switch (_videoMode)
|
|
|
|
{
|
|
|
|
case VIDEO_3DSBS:
|
|
|
|
videoWidth /= 2;
|
|
|
|
cropLeft /= 2;
|
|
|
|
break;
|
|
|
|
case VIDEO_3DTAB:
|
|
|
|
videoHeight /= 2;
|
|
|
|
cropTop /= 2;
|
|
|
|
break;
|
|
|
|
case VIDEO_2D:
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct config_para_ex_s configex = { 0 };
|
|
|
|
configex.src_para.mem_type = CANVAS_TYPE_INVALID;
|
|
|
|
configex.src_para.canvas_index = canvas0addr;
|
|
|
|
configex.src_para.left = cropLeft;
|
|
|
|
configex.src_para.top = cropTop;
|
|
|
|
configex.src_para.width = videoWidth;
|
|
|
|
configex.src_para.height = videoHeight / 2;
|
|
|
|
configex.src_para.format = ge2dformat;
|
|
|
|
|
|
|
|
configex.dst_para.mem_type = CANVAS_ALLOC;
|
|
|
|
configex.dst_para.format = GE2D_FORMAT_S24_RGB;
|
|
|
|
configex.dst_para.left = 0;
|
|
|
|
configex.dst_para.top = 0;
|
|
|
|
configex.dst_para.width = _width;
|
|
|
|
configex.dst_para.height = _height;
|
|
|
|
|
|
|
|
configex.dst_planes[0].addr = (long unsigned int)_ge2dIonBuffer->PhysicalAddress();
|
|
|
|
configex.dst_planes[0].w = configex.dst_para.width;
|
|
|
|
configex.dst_planes[0].h = configex.dst_para.height;
|
|
|
|
|
|
|
|
if (ioctl(_ge2dDev, GE2D_CONFIG_EX, &configex) < 0)
|
|
|
|
{
|
|
|
|
Error(_log, "video GE2D_CONFIG_EX failed.");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ge2d_para_s blitRect = { 0 };
|
|
|
|
blitRect.src1_rect.x = 0;
|
|
|
|
blitRect.src1_rect.y = 0;
|
|
|
|
blitRect.src1_rect.w = configex.src_para.width;
|
|
|
|
blitRect.src1_rect.h = configex.src_para.height;
|
|
|
|
|
|
|
|
blitRect.dst_rect.x = 0;
|
|
|
|
blitRect.dst_rect.y = 0;
|
|
|
|
blitRect.dst_rect.w = configex.dst_para.width ;
|
|
|
|
blitRect.dst_rect.h = configex.dst_para.height;
|
|
|
|
|
|
|
|
// Blit to videoBuffer
|
|
|
|
if (ioctl(_ge2dDev, GE2D_STRETCHBLIT_NOALPHA, &blitRect) < 0)
|
|
|
|
{
|
|
|
|
Error(_log,"GE2D_STRETCHBLIT_NOALPHA failed.");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return video frame
|
|
|
|
if (ioctl(_videoDev, AMVIDEO_EXT_PUT_CURRENT_VIDEOFRAME) < 0)
|
|
|
|
{
|
|
|
|
Error(_log, "AMSTREAM_EXT_PUT_CURRENT_VIDEOFRAME failed.");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
_ge2dIonBuffer->Sync();
|
|
|
|
|
|
|
|
// Read the snapshot into the memory
|
|
|
|
_useImageResampler = false;
|
|
|
|
_imageResampler.processImage((const uint8_t*)_ge2dVideoBufferPtr, _width, _height, _width*3, PIXELFORMAT_BGR24, image);
|
|
|
|
|
|
|
|
closeDev(_videoDev);
|
|
|
|
closeDev(_ge2dDev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|