1
0
mirror of https://gitee.com/drabel/LibQQt.git synced 2025-01-04 10:18:44 +08:00

update multimedia/qqtlogicvideomanager.cpp

This commit is contained in:
tianduanrui 2019-06-30 21:40:11 +08:00
parent 8d00ede1b3
commit 97352cdea5
3 changed files with 421 additions and 0 deletions

View File

@ -0,0 +1,342 @@
#include "qqtlogicvideomanager.h"
QQtLogicPreviewWidget::QQtLogicPreviewWidget ( QWidget* parent ) :
QWidget ( parent ),
ui ( new Ui::QQtLogicPreviewWidget )
{
ui->setupUi ( this );
memset ( &sinfo, 0, sizeof ( struct sensor_info ) );
pre_bpp = 16;
rate = 15; /* default to 15fps */
addr = 0;
phys = 0;
pre_size.w = 720;
pre_size.h = 480;
fd = 0;
memset ( & pre_memory, 0, sizeof ( struct camera_memory ) );
memset ( & pre_buf, 0, sizeof ( struct camera_buffer ) );
memset ( & pre_size, 0, sizeof ( struct frm_size ) );
tlb_base_phys = 0;
format = HAL_PIXEL_FORMAT_YCbCr_422_I;
bFullScreen = false;
m_parent = parent;
timer = new QTimer ( this );
timer->setSingleShot ( false );
connect ( timer, SIGNAL ( timeout() ), this, SLOT ( update() ) );
}
QQtLogicPreviewWidget::~QQtLogicPreviewWidget()
{
delete ui;
}
int QQtLogicPreviewWidget::play()
{
/*
*
*/
dmmu_init();
dmmu_get_page_table_base_phys ( &tlb_base_phys );
fd = ::open ( "/dev/cim", O_RDWR ); //av
if ( fd < 0 )
{
qDebug() << "Open device fail\n";
}
ioctl ( fd, CIMIO_SELECT_SENSOR, sinfo.sensor_id );
ioctl ( fd, CIMIO_GET_SENSORINFO, &sinfo );
ioctl ( fd, CIMIO_SET_TLB_BASE, tlb_base_phys ); //?????????
int i = 0;
if ( pre_buf.common && pre_buf.common->data )
{
dmmu_unmap_user_memory ( & ( pre_buf.dmmu_info ) );
free ( pre_buf.common->data );
}
pre_buf.fd = fd;
pre_buf.nr = 5;
pre_buf.size = pre_size.w * pre_size.h * 3;
pre_buf.common = &pre_memory;
pre_buf.common->size = pre_buf.size * pre_buf.nr;
pre_buf.common->data = memalign ( 4096, pre_buf.size * pre_buf.nr );
memset ( pre_buf.common->data, 0xa5, ( pre_buf.size * pre_buf.nr ) );
if ( pre_buf.common->data == NULL )
{
printf ( "==<%s L%d: null pointer!\n", __func__, __LINE__ );
return false;
}
pre_buf.paddr = 0;
pre_buf.dmmu_info.vaddr = pre_buf.common->data;
pre_buf.dmmu_info.size = pre_buf.common->size;
for ( i = 0; i < ( int ) ( pre_buf.common->size ); i += 0x1000 )
{
( ( uint8_t* ) ( pre_buf.common->data ) ) [i] = 0xff;
}
( ( uint8_t* ) ( pre_buf.common->data ) ) [pre_buf.common->size - 1] = 0xff;
dmmu_map_user_memory ( & ( pre_buf.dmmu_info ) );
for ( i = 0; i < pre_buf.nr; ++i )
{
pre_buf.yuvMeta[i].index = i;
pre_buf.yuvMeta[i].width = pre_size.w;
pre_buf.yuvMeta[i].height = pre_size.h;
pre_buf.yuvMeta[i].format = format;
pre_buf.yuvMeta[i].count = pre_buf.nr;
#ifdef __LINUX64__
pre_buf.yuvMeta[i].yAddr = ( int64_t ) pre_buf.common->data + ( pre_buf.size ) * i;
#else
pre_buf.yuvMeta[i].yAddr = ( int32_t ) pre_buf.common->data + ( pre_buf.size ) * i;
#endif
pre_buf.yuvMeta[i].yPhy = pre_buf.paddr + i * ( pre_buf.size );
if ( pre_buf.yuvMeta[i].format == HAL_PIXEL_FORMAT_YCbCr_422_I ) //yuv422 packed
{
pre_buf.yuvMeta[i].uAddr = pre_buf.yuvMeta[i].yAddr;
pre_buf.yuvMeta[i].vAddr = pre_buf.yuvMeta[i].uAddr;
pre_buf.yuvMeta[i].uPhy = pre_buf.yuvMeta[i].yPhy;
pre_buf.yuvMeta[i].vPhy = pre_buf.yuvMeta[i].uPhy;
pre_buf.yuvMeta[i].yStride = pre_buf.yuvMeta[i].width << 1;
pre_buf.yuvMeta[i].uStride = pre_buf.yuvMeta[i].yStride;
pre_buf.yuvMeta[i].vStride = pre_buf.yuvMeta[i].yStride;
}
else if ( pre_buf.yuvMeta[i].format == HAL_PIXEL_FORMAT_JZ_YUV_420_P ) //yuv420 planar
{
pre_buf.yuvMeta[i].uAddr = pre_buf.yuvMeta[i].yAddr + pre_size.w * pre_size.h;
pre_buf.yuvMeta[i].vAddr = pre_buf.yuvMeta[i].uAddr + pre_size.w * pre_size.h / 4;
pre_buf.yuvMeta[i].uPhy = pre_buf.yuvMeta[i].yPhy + pre_size.w * pre_size.h;
pre_buf.yuvMeta[i].vPhy = pre_buf.yuvMeta[i].uPhy + pre_size.w * pre_size.h / 4;
pre_buf.yuvMeta[i].yStride = pre_buf.yuvMeta[i].width << 1;
pre_buf.yuvMeta[i].uStride = pre_buf.yuvMeta[i].width / 2;
pre_buf.yuvMeta[i].vStride = pre_buf.yuvMeta[i].width / 2;
}
else if ( pre_buf.yuvMeta[i].format == HAL_PIXEL_FORMAT_RAW_SENSOR ) //raw bayer
{
pre_buf.yuvMeta[i].uAddr = pre_buf.yuvMeta[i].yAddr + pre_size.w * pre_size.h;
pre_buf.yuvMeta[i].vAddr = pre_buf.yuvMeta[i].uAddr;
pre_buf.yuvMeta[i].uPhy = pre_buf.yuvMeta[i].yPhy;
pre_buf.yuvMeta[i].vPhy = pre_buf.yuvMeta[i].uPhy;
pre_buf.yuvMeta[i].yStride = pre_buf.yuvMeta[i].width;
pre_buf.yuvMeta[i].uStride = pre_buf.yuvMeta[i].yStride;
pre_buf.yuvMeta[i].vStride = pre_buf.yuvMeta[i].yStride;
}
}
ioctl ( fd, CIMIO_SET_PREVIEW_FMT, format );
ioctl ( fd, CIMIO_SET_PREVIEW_SIZE, &pre_size );
ioctl ( fd, CIMIO_SET_PREVIEW_MEM, ( unsigned long ) ( pre_buf.yuvMeta ) );
ioctl ( fd, CIMIO_START_PREVIEW );
pp = ( unsigned char* ) malloc ( pre_size.w * pre_size.h * 3 * sizeof ( char ) );
frame = new QImage ( pp, pre_size.w, pre_size.h, QImage::Format_RGB888 );
timer->start ( 100 );
return fd;
}
int QQtLogicPreviewWidget::close()
{
bool ret = false;
if ( fd <= 0 )
printf ( "fd < 0\n" );
ret = ioctl ( fd, CIMIO_SHUTDOWN );
::close ( fd );
fd = 0;
dmmu_unmap_user_memory ( & ( pre_buf.dmmu_info ) );
dmmu_deinit();
memset ( pre_buf.yuvMeta, 0, pre_buf.nr * sizeof ( CameraYUVMeta ) );
pre_buf.size = 0;
pre_buf.nr = 0;
pre_buf.paddr = 0;
pre_buf.fd = -1;
if ( ( pre_buf.common != NULL ) && ( pre_buf.common->data != NULL ) )
{
free ( pre_buf.common->data );
pre_buf.common->data = NULL;
}
phys = 0;
timer->stop();
free ( pp );
delete frame;
return ret;
}
int QQtLogicPreviewWidget::convert_yuv_to_rgb_pixel ( int y, int u, int v )
{
unsigned int pixel32 = 0;
unsigned char* pixel = ( unsigned char* ) &pixel32;
int r, g, b;
b = y + ( ( 443 * ( u - 128 ) ) >> 8 );
b = ( b < 0 ) ? 0 : ( ( b > 255 ) ? 255 : b );
g = y - ( ( 179 * ( v - 128 ) + 86 * ( u - 128 ) ) >> 8 );
g = ( g < 0 ) ? 0 : ( ( g > 255 ) ? 255 : g );
r = y + ( ( 351 * ( v - 128 ) ) >> 8 );
r = ( r < 0 ) ? 0 : ( ( r > 255 ) ? 255 : r );
pixel[0] = r;
pixel[1] = g;
pixel[2] = b;
return pixel32;
}
int QQtLogicPreviewWidget::convert_yuv_to_rgb_buffer ( unsigned char* yuv, unsigned char* rgb, unsigned int width,
unsigned int height )
{
unsigned int in, out = 0;
unsigned int pixel_16;
unsigned char pixel_24[3];
unsigned int pixel32;
int y0, u, y1, v;
for ( in = 0; in < width * height * 2; in += 4 )
{
pixel_16 =
yuv[in + 3] << 24 |
yuv[in + 2] << 16 |
yuv[in + 1] << 8 |
yuv[in + 0];
y0 = ( pixel_16 & 0x000000ff );
u = ( pixel_16 & 0x0000ff00 ) >> 8;
y1 = ( pixel_16 & 0x00ff0000 ) >> 16;
v = ( pixel_16 & 0xff000000 ) >> 24;
pixel32 = convert_yuv_to_rgb_pixel ( y0, u, v );
pixel_24[0] = ( pixel32 & 0x000000ff );
pixel_24[1] = ( pixel32 & 0x0000ff00 ) >> 8;
pixel_24[2] = ( pixel32 & 0x00ff0000 ) >> 16;
rgb[out++] = pixel_24[0];
rgb[out++] = pixel_24[1];
rgb[out++] = pixel_24[2];
pixel32 = convert_yuv_to_rgb_pixel ( y1, u, v );
pixel_24[0] = ( pixel32 & 0x000000ff );
pixel_24[1] = ( pixel32 & 0x0000ff00 ) >> 8;
pixel_24[2] = ( pixel32 & 0x00ff0000 ) >> 16;
rgb[out++] = pixel_24[0];
rgb[out++] = pixel_24[1];
rgb[out++] = pixel_24[2];
}
return 0;
}
void QQtLogicPreviewWidget::paintEvent ( QPaintEvent* )
{
if ( fd <= 0 )
return;
QStylePainter painter ( this );
/*
* 线 log证明
*/
addr = ioctl ( fd, CIMIO_GET_FRAME );
p = ( uchar* ) addr;
/*
* yuv缺少alpha
*/
convert_yuv_to_rgb_buffer ( p, pp, pre_size.w, pre_size.h );
//frame->loadFromData((uchar *)pp, w * h * 3 * sizeof(char));
/*
*
*/
QRect srcRect ( 2, 6, pre_size.w, pre_size.h );
QRect dstRect = rect();
painter.scale ( 1.01, 1.02 );
/*
* OK
*/
painter.drawImage ( dstRect, *frame, srcRect );
//painter.drawPixmap(dstRect,QPixmap::fromImage(*frame,Qt::AutoColor),srcRect);;
/*
* OK
*/
//painter.drawItemPixmap(srcRect, Qt::AlignCenter, QPixmap::fromImage(*frame,Qt::AutoColor));
/*
* 30ms
*/
//update();
}
void QQtLogicPreviewWidget::mousePressEvent ( QMouseEvent* e )
{
static bool bGInit = false;
if ( !bGInit && !bFullScreen )
{
flags = windowFlags();
flags |= Qt::FramelessWindowHint;
geome = geometry();
bGInit = true;
}
#ifdef __EMBEDDED_LINUX__
//pline() << e->pos() << e->globalPos();
if ( e->pos().x() < 0 || e->pos().y() < 0 ||
e->pos().x() > geome.width() || e->pos().y() > geome.height() )
{
//在mips板上全屏返回的时候点击其他位置会多响应一次在此处过滤。
pline() << "warning!";
Q_UNUSED ( e );
return;
}
#endif
setAttribute ( Qt::WA_TranslucentBackground, true );
setAttribute ( Qt::WA_NoMousePropagation, true );
setAttribute ( Qt::WA_OpaquePaintEvent, true );
if ( bFullScreen )
{
flags ^= Qt::Window;
flags |= Qt::Widget;
setParent ( m_parent, flags );
setGeometry ( geome );
show();
bFullScreen = false;
}
else
{
int QQT_SCRN_WIDTH = QApplication::desktop()->availableGeometry().width();
int QQT_SCRN_HEIGHT = QApplication::desktop()->availableGeometry().height();
flags ^= Qt::Widget;
flags |= Qt::Window;
setParent ( 0, flags );
setGeometry ( 0, 0, QQT_SCRN_WIDTH, QQT_SCRN_HEIGHT );
show();
bFullScreen = true;
}
pline() << flags;
QWidget::mousePressEvent ( e );
}

View File

@ -0,0 +1,75 @@

#ifndef QQTLOGICVIDEOMANAGER_H
#define QQTLOGICVIDEOMANAGER_H
#include "qqt-qt.h"
#include "qqtlinux.h"
#include "qqtcore.h"
#include "qqt-local.h"
#ifdef __cplusplus
extern "C" {
#endif /* __cplusplus */
#include "graphics.h"
#include "dmmu.h"
#include "hal.h"
#include "jz_cim.h"
#ifdef __cplusplus
}
#endif /* __cplusplus */
/**
* @brief The QQtLogicPreviewWidget class
* QQt模拟摄像头预览控件
* dmmu的Qt Wrapper类
*/
class QQTSHARED_EXPORT QQtLogicPreviewWidget : public QWidget
{
Q_OBJECT
public:
explicit QQtLogicPreviewWidget ( QWidget* parent = 0 );
~QQtLogicPreviewWidget();
int play();
int close();
private:
Ui::QQtLogicPreviewWidget* ui;
struct sensor_info sinfo;
int pre_bpp;
int rate; /* default to 15fps */
unsigned int addr;
unsigned int phys;
int fd;
int format;
struct camera_memory pre_memory;
struct camera_buffer pre_buf;
struct frm_size pre_size;
unsigned int tlb_base_phys;
uchar* pp;
uchar* p;
QImage* frame;
QTimer* timer;
bool bFullScreen;
QWidget* m_parent;
QRect geome;
Qt::WindowFlags flags;
int convert_yuv_to_rgb_pixel ( int y, int u, int v );
int convert_yuv_to_rgb_buffer ( unsigned char* yuv, unsigned char* rgb, unsigned int width, unsigned int height );
// QWidget interface
protected:
void paintEvent ( QPaintEvent* );
void mousePressEvent ( QMouseEvent* e );
};
#endif // QQTLOGICVIDEOMANAGER_H

View File

@ -122,6 +122,10 @@ contains (DEFINES, __MULTIMEDIA__) {
$$PWD/multimedia/dmmu/jz_cim.h \
$$PWD/multimedia/dmmu/graphics.h \
$$PWD/multimedia/dmmu/hal.h
#SOURCES += $$PWD/multimedia/qqtlogicvideomanager.cpp
#HEADERS += $$PWD/multimedia/qqtlogicvideomanager.h
SOURCES += $$PWD/multimedia/qqtlogicpreviewwidget.cpp
HEADERS += $$PWD/multimedia/qqtlogicpreviewwidget.h
FORMS += $$PWD/multimedia/qqtlogicpreviewwidget.ui