在很多工业现场中会使用到网络摄像头,它的优点是可以通过网络获取数据,没有距离的限制,只需要提供一个常用的普通网络接口就可以进行通讯,非常灵活。但是网络摄像头传输的数据都是通过了编码(现在常见格式有 H.264、MPEG-4、MJPEG)的,在接收到数据后,如果要在主板上进行处理和显示,就需要对数据进行解码。实现解码功能需要硬件的支持,依靠软件解码会消耗大量的CPU资源,而且性能也不理想,所以如果要连接网络摄像头,就需要使用到带有视频处理单元CPU,比如ESM6802和ESM8000系列主板。

        这类主板的CPU内部集成了一个拥有编解码功能的VPU(Video Processing Unit),通过VPU进行解码不会消耗过多的CPU资源,而VPU同时可以支持多种格式的解码,包括了常见的H.264、MPEG-4、MJPEG等格式。以ESM6802主板为例,能够支持的最大分辨率为1920x1088,同时最多可以处理4路摄像头的数据。下面我们就介绍如何使用在Qt界面中显示网络摄像头的数据。

        软件上同样还是使用gstreamer和Qt来实现,ESM6802的文件系统中已经集成了gstreamer和Qt5.8相关的库和工具,并且包含了处理网络摄像头通讯协议rtsp和操作VPU编解码的插件,而gstreamer能够支持多种图形系统包括Qt。所以利用gstreame能够快速方便的获取网络摄像头的数据并显示到Qt界面中,关于gstreamer的详细介绍,可以参考网站https://gstreamer.freedesktop.org/documentation/index.html?gi-language=c

        考虑到部分用户可能对gstreamer不熟悉,英创公司在gstreamer提供的API函数基础上重新封装了一层,通过库文件libemgst.so提供三个API函数和一个包含网络摄像头信息的结构体变量,利用这三个API函数,用户就可以将视频数据显示在Qt实现的界面上,而不需要使用代码对gstreamer单独进行操作,用户可以专注于Qt程序的开发。下面就详细介绍libemgst.so的三个API函数和结构体变量:

/**
 *	初始化gstreamer
*
*	本函数在C的main函数中调用,将会初始化gstreamer库
*
*	参数说明:
*	使用main函数的两个参数填入,gstramer初始化时需要读取main函数的两个参数
*
 *	返回值说明:
*	无返回值
*
 */
void init_emgst(int argc, char **argv);

/**
 *	包含网络摄像头设备信息的结构体
*	访问网络摄像头需要有ip地址,用户名和密码。如果是播放录像文件,还需要填入
*	录像文件的地址,如果是播放实时数据,路径填写为NULL即可。
*
*	参数说明:
*	ip:指向网络摄像头ip地址的指针
*	user:指向网络摄像头的用户名的指针
*	passwd:指向网络摄像头密码的指针
*	file:指向录像文件路径的指针,如果播放实时图像,赋值为NULL
*
 */
typedef struct _Rtsp_config {
    char *ip;
    char *user;
    char *passwd;
    char *file;
} Rtsp_config;

/**
 *	配置gstreamer,并显示到指定Qt窗口
*
*	函数会配置gstreamer,显示多个网络摄像头的数据到xwinid所指定的窗口,窗口的大
*  小通过参数width和height来确定,而网络摄像头的信息由指针rtsp_config传入,listsize
*	指定同时显示的通道数量。因为要动态建立网络摄像头的连接,函数不会退出,所以
*  需要在线程中调用。
*
*	参数说明:
*  xwinid:Qt窗口的id号,每一个窗口类中都包含这个成员变量。
*	width:显示视频数据所使用的窗口控件的宽度,单位pixel
*	height:显示视频数据所使用的窗口控件的高度,单位pixel
*  rtsp_config:指向设备信息结构体的指针,通常是一组包含设备信息的结构体指针。
*	详细解释可以查看对结构体Rtsp_config的说明
*  listsize:显示的视频通道数最多支持同时显示4路。支持1,2,4这三个值,函数会按
*	照固定模式对显示进行排列
*
 *	返回值说明:
*	0 表示正常退出gstreamer
*  -1 表示填入参数出现错误,启动gstreamer失败
*
 */
  int config_emrtsp(unsigned int xwinid, int width, int height, Rtsp_config *rtsp_config, int listsize);

/**
*	关闭gstreamer,释放资源
*
*	函数会关闭gstreamer输出,并释放相关资源
*
*	返回值说明:
*	无返回值
*
*/
void release_emrtsp(void);

        init_emgst函数进行gstreamer的初始化,在启动gstreamer功能之前需要调用一次,用户配合Qt使用时在main函数中调用即可:

int main(int argc, char *argv[])
{
		//初始化英创主板gstreamer功能
		init_emgst(argc, argv);

		//Qt窗口初始化,由QtCreator自动生成
		QApplication a(argc, argv);
		MainWindow w;
		w.show();

		return a.exec();
}

H.264、MPEG-4、MJPEG编码格式都可以支持。

listsize指定同时显示的通道数,支持的值为1、2、4,显示的方式如下图:


jsmpeg实现多路播放监控视频 多路监控视频解决方案_jsmpeg实现多路播放监控视频

视频数据显示排列方式

         函数调用后gstreamer会将对应通道数的视频显示在指定的Qt窗口控件中,每一个区域所显示的网络摄像头数据是通过结构体指针rtsp_config指定的,下面是部分参考代码,创建了一个线程,通过一个按钮启动,把将graphicsView控件的id传给gstreamer,并显示设置的网络摄像头数据:

//派生一个Qt线程类,用获取config_emrtsp函数的参数,并在线程中启动
class Rtsp_pThread : public QThread
{
    Q_OBJECT

public:
    void run();

	//窗口控件的宽度
    int width;
    //窗口控件的高度
    int height;
    //显示的通道数量
    int num_of_chan;
    //窗口控件ID
    WId xwinid;
    //包含网络摄像头信息的结构体指针
    Rtsp_config *rtsp_config;
};
void MainWindow::on_confirm_released()
{
	//初始化结构体指针 
    rtsp_config = (Rtsp_config *)malloc(max_chan*sizeof(Rtsp_config));

    //填入每一路网络摄像头的ip,用户名和密码,因为不是播放录像文件,所以file一项填写的NULL。示例    是显示了四路网络摄像头的数据,因为都是访问的同一个摄像头,所以填写的信息是一致的
    for(i=0; i<max_chan; i++){
	    //网络摄像头的ip地址
    	(rtsp_config+i)->ip = (char*)malloc(20*sizeof(char));
    	sprintf((rtsp_config+i)->ip, "192.168.201.84");

	    //网络摄像头的用户名
    	(rtsp_config+i)->user = (char*)malloc(20*sizeof(char));
    	sprintf((rtsp_config+i)->user, "admin");

	    //网络摄像头的密码
    	(rtsp_config+i)->passwd = (char*)malloc(20*sizeof(char));
    	sprintf((rtsp_config+i)->passwd, "******");

    	(rtsp_config+i)->file = NULL;
    }

    //将显示区域的宽度,高度和等信息值赋给线程类的成员变量
    rtsp_thread.width = ui->graphicsView->width();
    rtsp_thread.height = ui->graphicsView->height();
    rtsp_thread.num_of_chan = max_chan;
    rtsp_thread.rtsp_config = rtsp_config;

    //启动线程
    rtsp_thread.start();
}

void Rtsp_pThread::run()
{
	//在线程中调用函数config_emrtsp,启动显示
    config_emrtsp(xwinid, width, height, rtsp_config, num_of_chan);
}

        release_emgst函数会停止gstreamer输出,并释放资源,在程序退出或者需要切换显示的时候调用,下面的代码是通过检测Qt按钮控件的点击事件,关闭显示:

void MainWindow::on_setting_released()
{
    int i;

    //停止显示,并释放相关资源
    release_emrtsp();
    //退出线程 
    rtsp_thread.quit();
}

        库文件libemgst.so已经预装在英创工控主板的文件系统中了,用户不需要针对主板再做任何设置。在开发程序的时候,将英创公司提供的emgst.h头文件和libemgst.so库文件添加到Qt工程中进去即可。头文件添加的方法十分简单,增加一个新的头文件就完成了。关于外部库的添加,Qt的开发工具QtCreator提供了非常简便的方法,用户只需要将libemgst.so拷贝到工程目录下,然后在QtCreator中的项目栏用鼠标右键单击工程,选择添加库(外部库),如下图:

jsmpeg实现多路播放监控视频 多路监控视频解决方案_jsmpeg实现多路播放监控视频_02

        选择外部库后,点击下一步,按照下图设置,在库文件一栏中选择到工程下的libemgst.so文件,如下图:

jsmpeg实现多路播放监控视频 多路监控视频解决方案_linux_03

        这样就完成了外部库的设置,用户可以在代码中调用libemgst.so提供的API函数了。英创公司提供了一个通过按钮切换显示视频通道数的例程,效果如下

jsmpeg实现多路播放监控视频 多路监控视频解决方案_jsmpeg实现多路播放监控视频_04

        通过这种方式,英创公司希望能够帮助用户缩短开发周期,让用户可以专注于应用程序的开发,而不用花时间去学习并在代码中设置gstreamer。对于例程感兴趣的用户可以和工程师联系,索取相关代码。