首页 数据库 mysql教程 使用 live555 直播来自 v4l2 的摄像头图像

使用 live555 直播来自 v4l2 的摄像头图像

Jun 07, 2016 pm 03:48 PM
c 使用 图像 摄像头 直播

结合前面的 采集 v4l2 视频, 使用 live555, 通过 rtsp 发布实时流. capture.h, capture.cpp, vcompress.h, vcompress.cpp 需要参考前面几片文章. 这里仅仅贴出 v4l2_x264_service.cpp [cpp] view plaincopy #includestdio.h #includestdlib.h #includeunistd

结合前面的 采集 v4l2 视频, 使用 live555, 通过 rtsp 发布实时流. capture.h, capture.cpp, vcompress.h, vcompress.cpp 需要参考前面几片文章. 这里仅仅贴出 v4l2_x264_service.cpp

[cpp] view plaincopy

  1. #include   
  2. #include   
  3. #include   
  4. #include   
  5.   
  6. #include   
  7. #include   
  8. #include   
  9.   
  10. #include   
  11. #include   
  12.   
  13. #include "capture.h"  
  14. #include "vcompress.h"  
  15.   
  16. static UsageEnvironment *_env = 0;  
  17.   
  18. #define SINK_PORT 3030  
  19.   
  20. #define VIDEO_WIDTH 320  
  21. #define VIDEO_HEIGHT 240  
  22. #define FRAME_PER_SEC 5.0  
  23.   
  24. pid_t gettid()  
  25. {  
  26.     return syscall(SYS_gettid);  
  27. }  
  28.   
  29.   
  30. // 使用 webcam + x264  
  31. class WebcamFrameSource : public FramedSource  
  32. {  
  33.     void *mp_capture, *mp_compress; // v4l2 + x264 encoder  
  34.     int m_started;  
  35.     void *mp_token;  
  36.   
  37. public:  
  38.     WebcamFrameSource (UsageEnvironment &env)  
  39.         : FramedSource(env)  
  40.     {  
  41.         fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);  
  42.         mp_capture = capture_open("/dev/video0", VIDEO_WIDTH, VIDEO_HEIGHT, PIX_FMT_YUV420P);  
  43.         if (!mp_capture) {  
  44.             fprintf(stderr, "%s: open /dev/video0 err\n", __func__);  
  45.             exit(-1);  
  46.         }  
  47.   
  48.         mp_compress = vc_open(VIDEO_WIDTH, VIDEO_HEIGHT, FRAME_PER_SEC);  
  49.         if (!mp_compress) {  
  50.             fprintf(stderr, "%s: open x264 err\n", __func__);  
  51.             exit(-1);  
  52.         }  
  53.   
  54.         m_started = 0;  
  55.         mp_token = 0;  
  56.     }  
  57.   
  58.     ~WebcamFrameSource ()  
  59.     {  
  60.         fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);  
  61.           
  62.         if (m_started) {  
  63.             envir().taskScheduler().unscheduleDelayedTask(mp_token);  
  64.         }  
  65.   
  66.         if (mp_compress)  
  67.             vc_close(mp_compress);  
  68.         if (mp_capture)  
  69.             capture_close(mp_capture);  
  70.     }  
  71.   
  72. protected:  
  73.     virtual void doGetNextFrame ()  
  74.     {  
  75.         if (m_started) return;  
  76.         m_started = 1;  
  77.   
  78.         // 根据 fps, 计算等待时间  
  79.         double delay = 1000.0 / FRAME_PER_SEC;  
  80.         int to_delay = delay * 1000;    // us  
  81.   
  82.         mp_token = envir().taskScheduler().scheduleDelayedTask(to_delay,  
  83.                 getNextFrame, this);  
  84.     }  

[cpp] view plaincopy

  1. virtual unsigned maxFrameSize() const        // 这个很重要, 如果不设置, 可能导致 getNextFrame() 出现 fMaxSize 小于实际编码帧的情况, 导致图像不完整  

[cpp] view plaincopy

  1. {    return 100*1024; }  

[cpp] view plaincopy

  1. private:  
  2.     static void getNextFrame (void *ptr)  
  3.     {  
  4.         ((WebcamFrameSource*)ptr)->getNextFrame1();  
  5.     }  
  6.   
  7.     void getNextFrame1 ()  
  8.     {  
  9.         // capture:  
  10.         Picture pic;  
  11.         if (capture_get_picture(mp_capture, &pic) 
  12.             fprintf(stderr, "==== %s: capture_get_picture err\n", __func__);  
  13.             m_started = 0;  
  14.             return;  
  15.         }  
  16.   
  17.         // compress  
  18.         const void *outbuf;  
  19.         int outlen;  
  20.         if (vc_compress(mp_compress, pic.data, pic.stride, &outbuf, &outlen) 
  21.             fprintf(stderr, "==== %s: vc_compress err\n", __func__);  
  22.             m_started = 0;  
  23.             return;  
  24.         }  
  25.   
  26.         int64_t pts, dts;  
  27.         int key;  
  28.         vc_get_last_frame_info(mp_compress, &key, &pts, &dts);  
  29.   
  30.         // save outbuf  
  31.         gettimeofday(&fPresentationTime, 0);  
  32.         fFrameSize = outlen;  
  33.         if (fFrameSize > fMaxSize) {  
  34.             fNumTruncatedBytes = fFrameSize - fMaxSize;  
  35.             fFrameSize = fMaxSize;  
  36.         }  
  37.         else {  
  38.             fNumTruncatedBytes = 0;  
  39.         }  
  40.   
  41.         memmove(fTo, outbuf, fFrameSize);  
  42.   
  43.         // notify  
  44.         afterGetting(this);  
  45.   
  46.         m_started = 0;  
  47.     }  
  48. };  
  49.   
  50. class WebcamOndemandMediaSubsession : public OnDemandServerMediaSubsession  
  51. {  
  52. public:  
  53.     static WebcamOndemandMediaSubsession *createNew (UsageEnvironment &env, FramedSource *source)  
  54.     {  
  55.         return new WebcamOndemandMediaSubsession(env, source);  
  56.     }  
  57.   
  58. protected:  
  59.     WebcamOndemandMediaSubsession (UsageEnvironment &env, FramedSource *source)  
  60.         : OnDemandServerMediaSubsession(env, True) // reuse the first source  
  61.     {  
  62.         fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);  
  63.         mp_source = source;  
  64.         mp_sdp_line = 0;  
  65.     }  
  66.   
  67.     ~WebcamOndemandMediaSubsession ()  
  68.     {  
  69.         fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);  
  70.         if (mp_sdp_line) free(mp_sdp_line);  
  71.     }  
  72.   
  73. private:  
  74.     static void afterPlayingDummy (void *ptr)  
  75.     {  
  76.         fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);  
  77.         // ok  
  78.         WebcamOndemandMediaSubsession *This = (WebcamOndemandMediaSubsession*)ptr;  
  79.         This->m_done = 0xff;  
  80.     }  
  81.   
  82.     static void chkForAuxSDPLine (void *ptr)  
  83.     {  
  84.         WebcamOndemandMediaSubsession *This = (WebcamOndemandMediaSubsession *)ptr;  
  85.         This->chkForAuxSDPLine1();  
  86.     }  
  87.   
  88.     void chkForAuxSDPLine1 ()  
  89.     {  
  90.         fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);  
  91.         if (mp_dummy_rtpsink->auxSDPLine())  
  92.             m_done = 0xff;  
  93.         else {  
  94.             int delay = 100*1000;   // 100ms  
  95.             nextTask() = envir().taskScheduler().scheduleDelayedTask(delay,  
  96.                     chkForAuxSDPLine, this);  
  97.         }  
  98.     }  
  99.   
  100. protected:  
  101.     virtual const char *getAuxSDPLine (RTPSink *sink, FramedSource *source)  
  102.     {  
  103.         fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);  
  104.         if (mp_sdp_line) return mp_sdp_line;  
  105.   
  106.         mp_dummy_rtpsink = sink;  
  107.         mp_dummy_rtpsink->startPlaying(*source, 0, 0);  
  108.         //mp_dummy_rtpsink->startPlaying(*source, afterPlayingDummy, this);  
  109.         chkForAuxSDPLine(this);  
  110.         m_done = 0;  
  111.         envir().taskScheduler().doEventLoop(&m_done);  
  112.         mp_sdp_line = strdup(mp_dummy_rtpsink->auxSDPLine());  
  113.         mp_dummy_rtpsink->stopPlaying();  
  114.   
  115.         return mp_sdp_line;  
  116.     }  
  117.   
  118.     virtual RTPSink *createNewRTPSink(Groupsock *rtpsock, unsigned char type, FramedSource *source)  
  119.     {  
  120.         fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);  
  121.         return H264VideoRTPSink::createNew(envir(), rtpsock, type);  
  122.     }  
  123.   
  124.     virtual FramedSource *createNewStreamSource (unsigned sid, unsigned &bitrate)  
  125.     {  
  126.         fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);  
  127.         bitrate = 500;  
  128.         return H264VideoStreamFramer::createNew(envir(), new WebcamFrameSource(envir()));  
  129.     }  
  130.   
  131. private:  
  132.     FramedSource *mp_source;    // 对应 WebcamFrameSource  
  133.     char *mp_sdp_line;  
  134.     RTPSink *mp_dummy_rtpsink;  
  135.     char m_done;  
  136. };  
  137.   
  138. static void test_task (void *ptr)  
  139. {  
  140.     fprintf(stderr, "test: task ....\n");  
  141.     _env->taskScheduler().scheduleDelayedTask(100000, test_task, 0);  
  142. }  
  143.   
  144. static void test (UsageEnvironment &env)  
  145. {  
  146.     fprintf(stderr, "test: begin...\n");  
  147.   
  148.     char done = 0;  
  149.     int delay = 100 * 1000;  
  150.     env.taskScheduler().scheduleDelayedTask(delay, test_task, 0);  
  151.     env.taskScheduler().doEventLoop(&done);  
  152.   
  153.     fprintf(stderr, "test: end..\n");  
  154. }  
  155.   
  156. int main (int argc, char **argv)  
  157. {  
  158.     // env  
  159.     TaskScheduler *scheduler = BasicTaskScheduler::createNew();  
  160.     _env = BasicUsageEnvironment::createNew(*scheduler);  
  161.   
  162.     // test  
  163.     //test(*_env);  
  164.   
  165.     // rtsp server  
  166.     RTSPServer *rtspServer = RTSPServer::createNew(*_env, 8554);  
  167.     if (!rtspServer) {  
  168.         fprintf(stderr, "ERR: create RTSPServer err\n");  
  169.         ::exit(-1);  
  170.     }  
  171.   
  172.     // add live stream  
  173.     do {  
  174.         WebcamFrameSource *webcam_source = 0;  
  175.   
  176.         ServerMediaSession *sms = ServerMediaSession::createNew(*_env, "webcam", 0, "Session from /dev/video0");   
  177.         sms->addSubsession(WebcamOndemandMediaSubsession::createNew(*_env, webcam_source));  
  178.         rtspServer->addServerMediaSession(sms);  
  179.   
  180.         char *url = rtspServer->rtspURL(sms);  
  181.         *_env "using url \"" "\"\n";  
  182.         delete [] url;  
  183.     } while (0);  
  184.   
  185.     // run loop  
  186.     _env->taskScheduler().doEventLoop();  
  187.   
  188.     return 1;  
  189. }  

需要 live555 + libavcodec + libswscale + libx264, client 使用 vlc, mplayer, quicktime, .....

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
4 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳图形设置
4 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您听不到任何人,如何修复音频
4 周前 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解锁Myrise中的所有内容
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

crystaldiskmark是什么软件?-crystaldiskmark如何使用? crystaldiskmark是什么软件?-crystaldiskmark如何使用? Mar 18, 2024 pm 02:58 PM

CrystalDiskMark是一款适用于硬盘的小型HDD基准测试工具,可以快速测量顺序和随机读/写速度。接下来就让小编为大家介绍一下CrystalDiskMark,以及crystaldiskmark如何使用吧~一、CrystalDiskMark介绍CrystalDiskMark是一款广泛使用的磁盘性能测试工具,用于评估机械硬盘和固态硬盘(SSD)的读写速度和随机I/O性能。它是一款免费的Windows应用程序,并提供用户友好的界面和各种测试模式来评估硬盘驱动器性能的不同方面,并被广泛用于硬件评

foobar2000怎么下载?-foobar2000怎么使用 foobar2000怎么下载?-foobar2000怎么使用 Mar 18, 2024 am 10:58 AM

foobar2000是一款能随时收听音乐资源的软件,各种音乐无损音质带给你,增强版本的音乐播放器,让你得到更全更舒适的音乐体验,它的设计理念是将电脑端的高级音频播放器移植到手机上,提供更加便捷高效的音乐播放体验,界面设计简洁明了易于使用它采用了极简的设计风格,没有过多的装饰和繁琐的操作能够快速上手,同时还支持多种皮肤和主题,根据自己的喜好进行个性化设置,打造专属的音乐播放器支持多种音频格式的播放,它还支持音频增益功能根据自己的听力情况调整音量大小,避免过大的音量对听力造成损害。接下来就让小编为大

网易邮箱大师怎么用 网易邮箱大师怎么用 Mar 27, 2024 pm 05:32 PM

网易邮箱,作为中国网民广泛使用的一种电子邮箱,一直以来以其稳定、高效的服务赢得了用户的信赖。而网易邮箱大师,则是专为手机用户打造的邮箱软件,它极大地简化了邮件的收发流程,让我们的邮件处理变得更加便捷。那么网易邮箱大师该如何使用,具体又有哪些功能呢,下文中本站小编将为大家带来详细的内容介绍,希望能帮助到大家!首先,您可以在手机应用商店搜索并下载网易邮箱大师应用。在应用宝或百度手机助手中搜索“网易邮箱大师”,然后按照提示进行安装即可。下载安装完成后,我们打开网易邮箱账号并进行登录,登录界面如下图所示

百度网盘app怎么用 百度网盘app怎么用 Mar 27, 2024 pm 06:46 PM

在如今云存储已经成为我们日常生活和工作中不可或缺的一部分。百度网盘作为国内领先的云存储服务之一,凭借其强大的存储功能、高效的传输速度以及便捷的操作体验,赢得了广大用户的青睐。而且无论你是想要备份重要文件、分享资料,还是在线观看视频、听取音乐,百度网盘都能满足你的需求。但是很多用户们可能对百度网盘app的具体使用方法还不了解,那么这篇教程就将为大家详细介绍百度网盘app如何使用,还有疑惑的用户们就快来跟着本文详细了解一下吧!百度云网盘怎么用:一、安装首先,下载并安装百度云软件时,请选择自定义安装选

小红书怎么开直播  小红书开启直播方法 小红书怎么开直播 小红书开启直播方法 Mar 28, 2024 pm 01:50 PM

  小红书都是一款你们非常熟悉的生活社区平台应用,这里的功能多多,真的都能够让大家任何的时候,都能看到各种各样的资讯内容,笔记十分多,图文并茂的,都能够让大家非常的满意,且有些时候都能够看到一些直播间的,所以大家也是产生了自己想开通直播的,能和大家一起聊聊天的,却不知道怎么开通直播的,下面小编也能给你们带来具体的操作方法,希望能够帮助到你们的。小红书开启直播方法:  1、首先打开小红书,点击首页底部的+。  2、然后切换至直播,点击开始直播入口。

BTCC教学:如何在BTCC交易所绑定使用MetaMask钱包? BTCC教学:如何在BTCC交易所绑定使用MetaMask钱包? Apr 26, 2024 am 09:40 AM

MetaMask(中文也叫小狐狸钱包)是一款免费的、广受好评的加密钱包软件。目前,BTCC已支持绑定MetaMask钱包,绑定后可使用MetaMask钱包进行快速登入,储值、买币等,且首次绑定还可获得20USDT体验金。在BTCCMetaMask钱包教学中,我们将详细介绍如何注册和使用MetaMask,以及如何在BTCC绑定并使用小狐狸钱包。MetaMask钱包是什么?MetaMask小狐狸钱包拥有超过3,000万用户,是当今最受欢迎的加密货币钱包之一。它可免费​​使用,可作为扩充功能安装在网络

PotPlayer怎么看直播?-PotPlayer看直播教程 PotPlayer怎么看直播?-PotPlayer看直播教程 Mar 19, 2024 pm 10:04 PM

小伙伴们知道PotPlayer怎么看直播吗?今天小编就来讲解PotPlayer看直播教程,感兴趣的快跟小编一起来看看吧,希望能够帮助到大家呢。首先我们打开PotPlayer,然后选择软件右下角的多功能侧边栏,然后我们进行点击;参考下图然后会弹出一个播放列表,我们可以选择“添加”选项,在这里可以对直播的设置进行调整和添加。这时候我们在弹出的下拉框中,我们选择添加链接,当然如果我们有直播源的文件的话,我们直接选择添加文件,然后将文件进行导入也是可以的。然后我们在弹出来的地址框中,我们输入自己想要观看

小米汽车app怎么用 小米汽车app怎么用 Apr 01, 2024 pm 09:19 PM

小米汽车软件提供远程车控功能,让用户可以通过手机或电脑远程控制车辆,例如开关车辆的门窗、启动引擎、控制车辆的空调和音响等,下文就是这个软件的使用及内容,一起了解下吧。小米汽车app功能及使用方法大全1、小米汽车app在3月25日上线苹果AppStore,现在安卓手机的应用商店中也可以下载了;购车:了解小米汽车核心亮点和技术参数,可预约试驾、配置订购您的小米汽车,支持在线处理提车待办事项。3、社区:了解小米汽车品牌资讯,交流用车体验,分享精彩车生活;4、车控:手机就是遥控器,远程控制,实时安防,轻

See all articles