0

0

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

php中文网

php中文网

发布时间:2016-06-07 15:48:35

|

1826人浏览过

|

来源于php中文网

原创

结合前面的 采集 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, .....

微动100多用户微信服务平台带分销系统
微动100多用户微信服务平台带分销系统

系统包含模块:1、卖场系统适用客户:实体卖场,可以分类管理,每个分类设置一个客服,客服可以使用手机管理分类商品2、万能表单用户可以自定义表单字段,收集各样信息,并可以导出Excel3、第三方接口方便用户自己开发,目前仅支持text格式4、留言板可以显示用户的头像和昵称5、场景二维码这是高级接口的使用,方便统计用户来源6、一键分享一个仿微信公众号详情界面,可以分享到朋友圈7、婚纱摄影一个相册+店面展

下载

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

43

2026.02.28

Golang 工程化架构设计:可维护与可演进系统构建
Golang 工程化架构设计:可维护与可演进系统构建

Go语言工程化架构设计专注于构建高可维护性、可演进的企业级系统。本专题深入探讨Go项目的目录结构设计、模块划分、依赖管理等核心架构原则,涵盖微服务架构、领域驱动设计(DDD)在Go中的实践应用。通过实战案例解析接口抽象、错误处理、配置管理、日志监控等关键工程化技术,帮助开发者掌握构建稳定、可扩展Go应用的最佳实践方法。

38

2026.02.28

Golang 性能分析与运行时机制:构建高性能程序
Golang 性能分析与运行时机制:构建高性能程序

Go语言以其高效的并发模型和优异的性能表现广泛应用于高并发、高性能场景。其运行时机制包括 Goroutine 调度、内存管理、垃圾回收等方面,深入理解这些机制有助于编写更高效稳定的程序。本专题将系统讲解 Golang 的性能分析工具使用、常见性能瓶颈定位及优化策略,并结合实际案例剖析 Go 程序的运行时行为,帮助开发者掌握构建高性能应用的关键技能。

35

2026.02.28

Golang 并发编程模型与工程实践:从语言特性到系统性能
Golang 并发编程模型与工程实践:从语言特性到系统性能

本专题系统讲解 Golang 并发编程模型,从语言级特性出发,深入理解 goroutine、channel 与调度机制。结合工程实践,分析并发设计模式、性能瓶颈与资源控制策略,帮助将并发能力有效转化为稳定、可扩展的系统性能优势。

20

2026.02.27

Golang 高级特性与最佳实践:提升代码艺术
Golang 高级特性与最佳实践:提升代码艺术

本专题深入剖析 Golang 的高级特性与工程级最佳实践,涵盖并发模型、内存管理、接口设计与错误处理策略。通过真实场景与代码对比,引导从“可运行”走向“高质量”,帮助构建高性能、可扩展、易维护的优雅 Go 代码体系。

18

2026.02.27

Golang 测试与调试专题:确保代码可靠性
Golang 测试与调试专题:确保代码可靠性

本专题聚焦 Golang 的测试与调试体系,系统讲解单元测试、表驱动测试、基准测试与覆盖率分析方法,并深入剖析调试工具与常见问题定位思路。通过实践示例,引导建立可验证、可回归的工程习惯,从而持续提升代码可靠性与可维护性。

3

2026.02.27

漫蛙app官网链接入口
漫蛙app官网链接入口

漫蛙App官网提供多条稳定入口,包括 https://manwa.me、https

235

2026.02.27

deepseek在线提问
deepseek在线提问

本合集汇总了DeepSeek在线提问技巧与免登录使用入口,助你快速上手AI对话、写作、分析等功能。阅读专题下面的文章了解更多详细内容。

11

2026.02.27

AO3官网直接进入
AO3官网直接进入

AO3官网最新入口合集,汇总2026年可用官方及镜像链接,助你快速稳定访问Archive of Our Own平台。阅读专题下面的文章了解更多详细内容。

382

2026.02.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Node.js 教程
Node.js 教程

共57课时 | 12.4万人学习

CSS3 教程
CSS3 教程

共18课时 | 6.4万人学习

Git 教程
Git 教程

共21课时 | 3.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号