使用 live555 直播来自 v4l2 的摄像头图像
结合前面的 采集 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
-
#include
-
#include
-
#include
-
#include
-
#include
-
#include
-
#include
-
#include
-
#include
- #include "capture.h"
- #include "vcompress.h"
- static UsageEnvironment *_env = 0;
- #define SINK_PORT 3030
- #define VIDEO_WIDTH 320
- #define VIDEO_HEIGHT 240
- #define FRAME_PER_SEC 5.0
- pid_t gettid()
- {
- return syscall(SYS_gettid);
- }
- // 使用 webcam + x264
- class WebcamFrameSource : public FramedSource
- {
- void *mp_capture, *mp_compress; // v4l2 + x264 encoder
- int m_started;
- void *mp_token;
- public:
- WebcamFrameSource (UsageEnvironment &env)
- : FramedSource(env)
- {
- fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);
- mp_capture = capture_open("/dev/video0", VIDEO_WIDTH, VIDEO_HEIGHT, PIX_FMT_YUV420P);
- if (!mp_capture) {
- fprintf(stderr, "%s: open /dev/video0 err\n", __func__);
- exit(-1);
- }
- mp_compress = vc_open(VIDEO_WIDTH, VIDEO_HEIGHT, FRAME_PER_SEC);
- if (!mp_compress) {
- fprintf(stderr, "%s: open x264 err\n", __func__);
- exit(-1);
- }
- m_started = 0;
- mp_token = 0;
- }
- ~WebcamFrameSource ()
- {
- fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);
- if (m_started) {
- envir().taskScheduler().unscheduleDelayedTask(mp_token);
- }
- if (mp_compress)
- vc_close(mp_compress);
- if (mp_capture)
- capture_close(mp_capture);
- }
- protected:
- virtual void doGetNextFrame ()
- {
- if (m_started) return;
- m_started = 1;
- // 根据 fps, 计算等待时间
- double delay = 1000.0 / FRAME_PER_SEC;
- int to_delay = delay * 1000; // us
- mp_token = envir().taskScheduler().scheduleDelayedTask(to_delay,
- getNextFrame, this);
- }
[cpp] view plaincopy
- virtual unsigned maxFrameSize() const // 这个很重要, 如果不设置, 可能导致 getNextFrame() 出现 fMaxSize 小于实际编码帧的情况, 导致图像不完整
[cpp] view plaincopy
- { return 100*1024; }
[cpp] view plaincopy
- private:
- static void getNextFrame (void *ptr)
- {
- ((WebcamFrameSource*)ptr)->getNextFrame1();
- }
- void getNextFrame1 ()
- {
- // capture:
- Picture pic;
- if (capture_get_picture(mp_capture, &pic)
- fprintf(stderr, "==== %s: capture_get_picture err\n", __func__);
- m_started = 0;
- return;
- }
- // compress
- const void *outbuf;
- int outlen;
- if (vc_compress(mp_compress, pic.data, pic.stride, &outbuf, &outlen)
- fprintf(stderr, "==== %s: vc_compress err\n", __func__);
- m_started = 0;
- return;
- }
- int64_t pts, dts;
- int key;
- vc_get_last_frame_info(mp_compress, &key, &pts, &dts);
- // save outbuf
- gettimeofday(&fPresentationTime, 0);
- fFrameSize = outlen;
- if (fFrameSize > fMaxSize) {
- fNumTruncatedBytes = fFrameSize - fMaxSize;
- fFrameSize = fMaxSize;
- }
- else {
- fNumTruncatedBytes = 0;
- }
- memmove(fTo, outbuf, fFrameSize);
- // notify
- afterGetting(this);
- m_started = 0;
- }
- };
- class WebcamOndemandMediaSubsession : public OnDemandServerMediaSubsession
- {
- public:
- static WebcamOndemandMediaSubsession *createNew (UsageEnvironment &env, FramedSource *source)
- {
- return new WebcamOndemandMediaSubsession(env, source);
- }
- protected:
- WebcamOndemandMediaSubsession (UsageEnvironment &env, FramedSource *source)
- : OnDemandServerMediaSubsession(env, True) // reuse the first source
- {
- fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);
- mp_source = source;
- mp_sdp_line = 0;
- }
- ~WebcamOndemandMediaSubsession ()
- {
- fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);
- if (mp_sdp_line) free(mp_sdp_line);
- }
- private:
- static void afterPlayingDummy (void *ptr)
- {
- fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);
- // ok
- WebcamOndemandMediaSubsession *This = (WebcamOndemandMediaSubsession*)ptr;
- This->m_done = 0xff;
- }
- static void chkForAuxSDPLine (void *ptr)
- {
- WebcamOndemandMediaSubsession *This = (WebcamOndemandMediaSubsession *)ptr;
- This->chkForAuxSDPLine1();
- }
- void chkForAuxSDPLine1 ()
- {
- fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);
- if (mp_dummy_rtpsink->auxSDPLine())
- m_done = 0xff;
- else {
- int delay = 100*1000; // 100ms
- nextTask() = envir().taskScheduler().scheduleDelayedTask(delay,
- chkForAuxSDPLine, this);
- }
- }
- protected:
- virtual const char *getAuxSDPLine (RTPSink *sink, FramedSource *source)
- {
- fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);
- if (mp_sdp_line) return mp_sdp_line;
- mp_dummy_rtpsink = sink;
- mp_dummy_rtpsink->startPlaying(*source, 0, 0);
- //mp_dummy_rtpsink->startPlaying(*source, afterPlayingDummy, this);
- chkForAuxSDPLine(this);
- m_done = 0;
- envir().taskScheduler().doEventLoop(&m_done);
- mp_sdp_line = strdup(mp_dummy_rtpsink->auxSDPLine());
- mp_dummy_rtpsink->stopPlaying();
- return mp_sdp_line;
- }
- virtual RTPSink *createNewRTPSink(Groupsock *rtpsock, unsigned char type, FramedSource *source)
- {
- fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);
- return H264VideoRTPSink::createNew(envir(), rtpsock, type);
- }
- virtual FramedSource *createNewStreamSource (unsigned sid, unsigned &bitrate)
- {
- fprintf(stderr, "[%d] %s .... calling\n", gettid(), __func__);
- bitrate = 500;
- return H264VideoStreamFramer::createNew(envir(), new WebcamFrameSource(envir()));
- }
- private:
- FramedSource *mp_source; // 对应 WebcamFrameSource
- char *mp_sdp_line;
- RTPSink *mp_dummy_rtpsink;
- char m_done;
- };
- static void test_task (void *ptr)
- {
- fprintf(stderr, "test: task ....\n");
- _env->taskScheduler().scheduleDelayedTask(100000, test_task, 0);
- }
- static void test (UsageEnvironment &env)
- {
- fprintf(stderr, "test: begin...\n");
- char done = 0;
- int delay = 100 * 1000;
- env.taskScheduler().scheduleDelayedTask(delay, test_task, 0);
- env.taskScheduler().doEventLoop(&done);
- fprintf(stderr, "test: end..\n");
- }
- int main (int argc, char **argv)
- {
- // env
- TaskScheduler *scheduler = BasicTaskScheduler::createNew();
- _env = BasicUsageEnvironment::createNew(*scheduler);
- // test
- //test(*_env);
- // rtsp server
- RTSPServer *rtspServer = RTSPServer::createNew(*_env, 8554);
- if (!rtspServer) {
- fprintf(stderr, "ERR: create RTSPServer err\n");
- ::exit(-1);
- }
- // add live stream
- do {
- WebcamFrameSource *webcam_source = 0;
- ServerMediaSession *sms = ServerMediaSession::createNew(*_env, "webcam", 0, "Session from /dev/video0");
- sms->addSubsession(WebcamOndemandMediaSubsession::createNew(*_env, webcam_source));
- rtspServer->addServerMediaSession(sms);
- char *url = rtspServer->rtspURL(sms);
- *_env "using url \"" "\"\n";
- delete [] url;
- } while (0);
- // run loop
- _env->taskScheduler().doEventLoop();
- return 1;
- }
需要 live555 + libavcodec + libswscale + libx264, client 使用 vlc, mplayer, quicktime, .....

핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

AI Hentai Generator
AI Hentai를 무료로 생성하십시오.

인기 기사

뜨거운 도구

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

SublimeText3 Mac 버전
신 수준의 코드 편집 소프트웨어(SublimeText3)

뜨거운 주제











CrystalDiskMark는 순차 및 무작위 읽기/쓰기 속도를 빠르게 측정하는 하드 드라이브용 소형 HDD 벤치마크 도구입니다. 다음으로 편집자님에게 CrystalDiskMark 소개와 crystaldiskmark 사용법을 소개하겠습니다~ 1. CrystalDiskMark 소개 CrystalDiskMark는 기계식 하드 드라이브와 솔리드 스테이트 드라이브(SSD)의 읽기 및 쓰기 속도와 성능을 평가하는 데 널리 사용되는 디스크 성능 테스트 도구입니다. ). 무작위 I/O 성능. 무료 Windows 응용 프로그램이며 사용자 친화적인 인터페이스와 다양한 테스트 모드를 제공하여 하드 드라이브 성능의 다양한 측면을 평가하고 하드웨어 검토에 널리 사용됩니다.

foobar2000은 언제든지 음악 리소스를 들을 수 있는 소프트웨어입니다. 모든 종류의 음악을 무손실 음질로 제공합니다. 음악 플레이어의 향상된 버전을 사용하면 더욱 포괄적이고 편안한 음악 경험을 얻을 수 있습니다. 컴퓨터에서 고급 오디오를 재생합니다. 이 장치는 보다 편리하고 효율적인 음악 재생 경험을 제공합니다. 인터페이스 디자인은 단순하고 명확하며 사용하기 쉽습니다. 또한 다양한 스킨과 테마를 지원하고, 자신의 선호도에 따라 설정을 개인화하며, 다양한 오디오 형식의 재생을 지원하는 전용 음악 플레이어를 생성합니다. 또한 볼륨을 조정하는 오디오 게인 기능도 지원합니다. 과도한 볼륨으로 인한 청력 손상을 방지하려면 자신의 청력 상태에 따라 조정하십시오. 다음엔 내가 도와줄게

NetEase Mailbox는 중국 네티즌들이 널리 사용하는 이메일 주소로, 안정적이고 효율적인 서비스로 항상 사용자들의 신뢰를 얻어 왔습니다. NetEase Mailbox Master는 휴대폰 사용자를 위해 특별히 제작된 이메일 소프트웨어로 이메일 보내기 및 받기 프로세스를 크게 단순화하고 이메일 처리를 더욱 편리하게 만듭니다. 따라서 NetEase Mailbox Master를 사용하는 방법과 그 기능이 무엇인지 아래에서 이 사이트의 편집자가 자세한 소개를 제공하여 도움을 드릴 것입니다! 먼저, 모바일 앱스토어에서 NetEase Mailbox Master 앱을 검색하여 다운로드하실 수 있습니다. App Store 또는 Baidu Mobile Assistant에서 "NetEase Mailbox Master"를 검색한 후 안내에 따라 설치하세요. 다운로드 및 설치가 완료되면 NetEase 이메일 계정을 열고 로그인합니다. 로그인 인터페이스는 아래와 같습니다.

오늘날 클라우드 스토리지는 우리의 일상 생활과 업무에 없어서는 안 될 부분이 되었습니다. 중국 최고의 클라우드 스토리지 서비스 중 하나인 Baidu Netdisk는 강력한 스토리지 기능, 효율적인 전송 속도 및 편리한 운영 경험으로 많은 사용자의 호감을 얻었습니다. 중요한 파일을 백업하고, 정보를 공유하고, 온라인으로 비디오를 시청하고, 음악을 듣고 싶은 경우 Baidu Cloud Disk는 귀하의 요구를 충족할 수 있습니다. 그러나 많은 사용자가 Baidu Netdisk 앱의 구체적인 사용 방법을 이해하지 못할 수 있으므로 이 튜토리얼에서는 Baidu Netdisk 앱 사용 방법을 자세히 소개합니다. Baidu 클라우드 네트워크 디스크 사용 방법: 1. 설치 먼저 Baidu Cloud 소프트웨어를 다운로드하고 설치할 때 사용자 정의 설치 옵션을 선택하십시오.

Xiaohongshu는 여러분에게 매우 친숙한 생활 커뮤니티 플랫폼 애플리케이션으로, 많은 기능을 갖추고 있으며, 언제든지 다양한 정보 콘텐츠를 볼 수 있습니다. 때로는 일부 라이브 방송실을 볼 수 있으므로 모든 사람들이 라이브 방송을 시작하고 모든 사람과 채팅하고 싶지만 라이브 방송을 시작하는 방법을 모릅니다. 아래 편집자가 구체적인 작업 방법을 알려 드릴 수도 있습니다. 그것이 당신을 도울 수 있기를 바랍니다. Xiaohongshu에서 라이브 스트리밍을 시작하는 방법: 1. 먼저 Xiaohongshu를 열고 홈페이지 하단의 +를 클릭합니다. 2. 그런 다음 라이브 방송으로 전환하고 라이브 방송 시작 입구를 클릭합니다.

MetaMask(중국어로 Little Fox Wallet이라고도 함)는 무료이며 호평을 받는 암호화 지갑 소프트웨어입니다. 현재 BTCC는 MetaMask 지갑에 대한 바인딩을 지원합니다. 바인딩 후 MetaMask 지갑을 사용하여 빠르게 로그인하고 가치를 저장하고 코인을 구매할 수 있으며 첫 바인딩에는 20 USDT 평가판 보너스도 받을 수 있습니다. BTCCMetaMask 지갑 튜토리얼에서는 MetaMask 등록 및 사용 방법, BTCC에서 Little Fox 지갑을 바인딩하고 사용하는 방법을 자세히 소개합니다. MetaMask 지갑이란 무엇입니까? 3천만 명 이상의 사용자를 보유한 MetaMask Little Fox Wallet은 오늘날 가장 인기 있는 암호화폐 지갑 중 하나입니다. 무료로 사용할 수 있으며 확장으로 네트워크에 설치할 수 있습니다.

여러분, PotPlayer로 라이브 방송을 시청하는 방법을 알고 계시나요? 오늘은 PotPlayer로 라이브 방송을 시청하는 방법을 설명하겠습니다. 관심이 있으시면 저와 함께 시청해 보세요. 먼저 PotPlayer를 열고 소프트웨어의 오른쪽 하단에 있는 다기능 사이드바를 선택한 다음 아래 그림을 참조하면 재생 목록이 표시됩니다. 여기서 "추가" 옵션을 선택할 수 있습니다. 라이브 방송 설정을 조정하고 추가하세요. 이때 나타나는 드롭다운 상자에서 링크를 추가하도록 선택합니다. 물론 생방송 소스 파일이 있는 경우 파일을 직접 추가한 다음 파일을 가져올 수 있습니다. 그러다 뜨는 주소창에 보고 싶은 방송을 입력합니다

Xiaomi 자동차 소프트웨어는 원격 자동차 제어 기능을 제공하여 사용자가 휴대폰이나 컴퓨터를 통해 차량의 문과 창문을 열고 닫고, 엔진 시동을 걸고, 차량의 에어컨과 오디오를 제어하는 등 차량을 원격으로 제어할 수 있습니다. 다음은 이 소프트웨어의 사용법과 내용에 대해 함께 알아봅시다. Xiaomi Auto 앱 기능 및 사용 방법의 전체 목록 1. Xiaomi Auto 앱은 3월 25일 Apple AppStore에 출시되었으며 이제 Android 휴대폰의 앱 스토어에서 다운로드할 수 있습니다. 자동차 구매: Xiaomi Auto의 핵심 특징과 기술 매개 변수를 이해합니다. , 시승 예약을 하고 Xiaomi 차량을 구성 및 주문하고 차량 픽업 작업 항목의 온라인 처리를 지원하세요. 3. 커뮤니티: Xiaomi Auto 브랜드 정보를 이해하고, 자동차 경험을 교환하고, 멋진 자동차 생활을 공유하세요. 4. 자동차 제어: 휴대폰은 원격 제어, 원격 제어, 실시간 보안, 간편함을 제공합니다.
