mirror of
				https://github.com/hyperion-project/hyperion.ng.git
				synced 2025-03-01 10:33:28 +00:00 
			
		
		
		
	* - grabber auto off when not set as active prio - join aml and fb - on aml platform both grabbers are needed, so they joind in one module and share one prio. user don't the the nasty magic behind - aml: preparation for direct ge2d access * just save it, in the middle of ge2d impl * fix compile issues * now grabbing works basicly * add 3d support for ge2d * next step, we got some video from aml * switch back to rgba * remove unfinished ge2d stuff * commit missing changes * some urgent fixes, needs some beautifying, but it works now * fixes and refctoring
		
			
				
	
	
		
			172 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			172 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable File
		
	
	
	
	
| // STL includes
 | |
| #include <algorithm>
 | |
| #include <cassert>
 | |
| #include <iostream>
 | |
| #include <QFile>
 | |
| 
 | |
| // 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
 | |
| #include <utils/Logger.h>
 | |
| #include <grabber/AmlogicGrabber.h>
 | |
| #include "Amvideocap.h"
 | |
| 
 | |
| #define VIDEO_DEVICE   "/dev/amvideo"
 | |
| #define CAPTURE_DEVICE "/dev/amvideocap0"
 | |
| #include <iostream>
 | |
| 
 | |
| AmlogicGrabber::AmlogicGrabber(const unsigned width, const unsigned height)
 | |
| 	: Grabber("AMLOGICGRABBER", qMax(160u, width), qMax(160u, height)) // Minimum required width or height is 160
 | |
| 	, _captureDev(-1)
 | |
| 	, _videoDev(-1)
 | |
| 	, _lastError(0)
 | |
| 	, _fbGrabber("/dev/fb0",width,height)
 | |
| 	, _grabbingModeNotification(0)
 | |
| {
 | |
| 	Debug(_log, "constructed(%d x %d)",_width,_height);
 | |
| }
 | |
| 
 | |
| AmlogicGrabber::~AmlogicGrabber()
 | |
| {
 | |
| 	closeDev(_captureDev);
 | |
| 	closeDev(_videoDev);
 | |
| }
 | |
| 
 | |
| bool AmlogicGrabber::openDev(int &fd, const char* dev, int flags)
 | |
| {
 | |
| 	if (fd<0)
 | |
| 	{
 | |
| 		fd = open(dev, flags);
 | |
| 	}
 | |
| 	return fd >= 0;
 | |
| }
 | |
| 
 | |
| void AmlogicGrabber::closeDev(int &fd)
 | |
| {
 | |
| 	if (fd >= 0)
 | |
| 	{
 | |
| 		close(fd);
 | |
| 		fd = -1;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool AmlogicGrabber::isVideoPlaying()
 | |
| {
 | |
| 	if(!QFile::exists(VIDEO_DEVICE)) return false;
 | |
| 
 | |
| 	int videoDisabled = 1;
 | |
| 	if (!openDev(_videoDev, VIDEO_DEVICE, O_RDWR))
 | |
| 	{
 | |
| 		Error(_log, "Failed to open video device(%s): %d - %s", VIDEO_DEVICE, errno, strerror(errno));
 | |
| 		return false;
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		// 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;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return videoDisabled == 0;
 | |
| }
 | |
| 
 | |
| int AmlogicGrabber::grabFrame(Image<ColorRgb> & image)
 | |
| {
 | |
| 	if (!_enabled) return 0;
 | |
| 
 | |
| 	image.resize(_width,_height);
 | |
| 	// Make sure video is playing, else there is nothing to grab
 | |
| 	if (isVideoPlaying())
 | |
| 	{
 | |
| 		if (_grabbingModeNotification!=1)
 | |
| 		{
 | |
| 			Info(_log, "VPU mode");
 | |
| 			_grabbingModeNotification = 1;
 | |
| 			_lastError = 0;
 | |
| 		}
 | |
| 
 | |
| 		if (QFile::exists(CAPTURE_DEVICE))
 | |
| 		{
 | |
| 			grabFrame_amvideocap(image);
 | |
| 		}
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		if (_grabbingModeNotification!=2)
 | |
| 		{
 | |
| 			Info( _log, "FB mode");
 | |
| 			_grabbingModeNotification = 2;
 | |
| 			_lastError = 0;
 | |
| 		}
 | |
| 		_fbGrabber.grabFrame(image);
 | |
| 	}
 | |
| 
 | |
| 	closeDev(_videoDev);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int AmlogicGrabber::grabFrame_amvideocap(Image<ColorRgb> & image)
 | |
| {
 | |
| 	// If the device is not open, attempt to open it
 | |
| 	if (! openDev(_captureDev, CAPTURE_DEVICE, O_RDWR))
 | |
| 	{
 | |
| 		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)
 | |
| 	{
 | |
| 		ErrorIf(_lastError != 2,_log,"Failed to configure capture size (%d - %s)", errno, strerror(errno));
 | |
| 		closeDev(_captureDev);
 | |
| 		_lastError = 2;
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	// Read the snapshot into the memory
 | |
| 	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);
 | |
| 
 | |
| 	if (bytesRead < 0)
 | |
| 	{
 | |
| 		ErrorIf(_lastError != 3, _log,"Read of device failed: %d - %s", errno, strerror(errno));
 | |
| 		closeDev(_captureDev);
 | |
| 		_lastError = 3;
 | |
| 		return -1;
 | |
| 	}
 | |
| 	else if (bytesToRead != bytesRead)
 | |
| 	{
 | |
| 		// Read of snapshot failed
 | |
| 		ErrorIf(_lastError != 4, _log,"Capture failed to grab entire image [bytesToRead(%d) != bytesRead(%d)]", bytesToRead, bytesRead);
 | |
| 		closeDev(_captureDev);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	closeDev(_captureDev);
 | |
| 	
 | |
| 	_imageResampler.processImage((const uint8_t*)image_ptr, _width, _height, _width*3, PIXELFORMAT_BGR24, image);
 | |
| 	
 | |
| 	_lastError = 0;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 |