打开APP
userphoto
未登录

开通VIP,畅享免费电子书等14项超值服

开通VIP
使用librtmp发布flash视频直播流(H264、AAC)的C++类代码
(2013-06-26 14:32:10)
[置顶] 使用librtmp发布flash视频直播流(H264、AAC)的C++类代码
分类: 音视频编码 2013-03-11 20:09 353人阅读 评论(5)  举报
rtmpflashc++h264aac
最近在研究使用开源librtmp库发布flash视频直播流,经过几天的折腾后,写了一个封装调用librtmp的类,时间紧迫没好好整理,仅供需要的朋友参考
头文件:RtmpStream.h
[cpp] view plaincopy
#pragma once
/// 对开源库rtmpdump的封装
#include
#include "common.h"
struct RTMP;
struct RTMPMetadata;
struct RTMPPacket;
class CRtmpStream
{
public:
CRtmpStream(void);
virtual ~CRtmpStream(void);
public:
/// 打开
BOOL Open(const char * apUrl);
/// 重新打开最近的连接
BOOL Reopen();
/// 关闭
void Close();
/// 是否打开
inline BOOL IsOpened() const
{
return m_pHandle != NULL;
}
/// 发送元数据
BOOL SendMetadata(const RTMPMetadata & astruMetadata);
/// 发送AAC数据
BOOL SendAACData(const void * apBuf, const unsigned int auBufLen, const unsigned int auTimestamp);
/// 发送H264数据
BOOL SendH264Data(const void * apBuf, const unsigned int auBufLen, const unsigned int auTimestamp,
const BOOL abIsKey, const int aiOffset);
private:
void CloseStream();
private:
RTMP *                    m_pHandle;
unsigned int            m_uStartAudioTime;
unsigned int            m_uStartVideoTime;
std::string                m_strLastUrl;
CCriticalSectionEx        m_oLock;
BOOL                    m_bIsLiving;
};
实现:RtmpStream.cpp
[cpp] view plaincopy
#include "stdafx.h"
#include "RtmpStream.h"
#include "rtmp/rtmp.h"
#include "rtmp/flvpush.h"
#include
#include
static class CInitSocket
{
public:
CInitSocket()
{
#ifdef WIN32
WSADATA wsa_data;
WORD w_version_requested = MAKEWORD(2, 2);
if ( WSAStartup(w_version_requested, &wsa_data) != 0 )
{
m_bResult = false;
return;
}
//else if ( LOBYTE(wsa_data.wVersion) != 2 || HIBYTE(wsa_data.wVersion) != 2 )
//{
//    WSACleanup();
//    return false;
//}
#endif
m_bResult = true;
}
~CInitSocket()
{
#ifdef WIN32
if ( m_bResult )
{
WSACleanup();
}
#endif
}
bool m_bResult;
} s_oInitSocket;
CRtmpStream::CRtmpStream(void)
: m_pHandle(NULL)
, m_bIsLiving(FALSE)
, m_uStartAudioTime(0)
, m_uStartVideoTime(0)
{
}
CRtmpStream::~CRtmpStream(void)
{
Close();
}
BOOL CRtmpStream::Open(const char * apUrl)
{
CCriticalSectionHelperEx oDoLock(m_oLock);
if ( m_bIsLiving )
{
return FALSE;
}
if ( IsOpened() )
{
return FALSE;
}
m_pHandle = RTMP_Alloc();
if ( m_pHandle == NULL )
{
assert(FALSE);
return FALSE;
}
RTMP_Init(m_pHandle);
if ( !RTMP_SetupURL(m_pHandle, apUrl) )
{
Close();
assert(FALSE);
return FALSE;
}
RTMP_EnableWrite(m_pHandle);
if ( !RTMP_Connect(m_pHandle, NULL) )
{
Close();
assert(FALSE);
return FALSE;
}
if ( !RTMP_ConnectStream(m_pHandle, 0) )
{
Close();
assert(FALSE);
return FALSE;
}
m_bIsLiving = TRUE;
m_uStartAudioTime = 0;
m_uStartVideoTime = 0;
if ( m_strLastUrl.c_str() != apUrl )
{
m_strLastUrl = apUrl;
}
return TRUE;
}
/// 重新打开最近的连接
BOOL CRtmpStream::Reopen()
{
return Open(m_strLastUrl.c_str());
}
void CRtmpStream::CloseStream()
{
if ( m_pHandle != NULL )
{
RTMP_Close(m_pHandle);
::Sleep(20);
RTMP_Free(m_pHandle);
m_pHandle = NULL;
}
}
void CRtmpStream::Close()
{
CCriticalSectionHelperEx oDoLock(m_oLock);
CloseStream();
m_bIsLiving = FALSE;
}
/// 发送元数据
BOOL CRtmpStream::SendMetadata(const RTMPMetadata & astruMetadata)
{
if ( !m_bIsLiving )
{
return FALSE;
}
CCriticalSectionHelperEx oDoLock(m_oLock);
if ( !IsOpened() )
{
assert(FALSE);
return FALSE;
}
if ( !RTMP_SendMetadata(m_pHandle, &astruMetadata) )
{
assert(FALSE);
return FALSE;
}
return TRUE;
}
/// 发送AAC数据
BOOL CRtmpStream::SendAACData(const void * apBuf, const unsigned int auBufLen, const unsigned int auTimestamp)
{
if ( !m_bIsLiving )
{
return FALSE;
}
if ( auBufLen == 0 )
{
return TRUE;
}
CCriticalSectionHelperEx oDoLock(m_oLock);
if ( !IsOpened() )
{
return TRUE;
}
unsigned int uCurTimestamp;
if ( m_uStartAudioTime )
{
// 有值
uCurTimestamp = auTimestamp - m_uStartAudioTime;
}
else
{
// is 0
uCurTimestamp = 1;
m_uStartAudioTime = auTimestamp;
}
if ( !RTMP_SendAACData(m_pHandle, apBuf, auBufLen, uCurTimestamp) )
{
Close();
Reopen();
assert(FALSE);
return FALSE;
}
return TRUE;
}
/// 发送H264数据
BOOL CRtmpStream::SendH264Data(const void * apBuf, const unsigned int auBufLen, const unsigned int auTimestamp,
const BOOL abIsKey, const int aiOffset)
{
if ( !m_bIsLiving )
{
return FALSE;
}
if ( auBufLen == 0 )
{
return TRUE;
}
CCriticalSectionHelperEx oDoLock(m_oLock);
if ( !IsOpened() )
{
return TRUE;
}
unsigned int uCurTimestamp;
if ( m_uStartVideoTime )
{
// 有值
uCurTimestamp = auTimestamp - m_uStartVideoTime;
}
else
{
// is 0
uCurTimestamp = 1;
m_uStartVideoTime = auTimestamp;
}
if ( !RTMP_SendH264Data(m_pHandle, apBuf, auBufLen, uCurTimestamp, abIsKey, aiOffset) )
{
Close();
Reopen();
assert(FALSE);
return FALSE;
}
return TRUE;
}
调用顺序:
1、打开Open
2、发送元数据SendMetadata
3、发送音频AAC数据SendAACData或视频H264数据SendH264Data
4、Close我的资源里有编译好的静态库和代码(使用mingw gcc编译,VC里也可直接使用)
本站仅提供存储服务,所有内容均由用户发布,如发现有害或侵权内容,请点击举报
打开APP,阅读全文并永久保存 查看更多类似文章
猜你喜欢
类似文章
【热】打开小程序,算一算2024你的财运
【基于libRTMP的流媒体直播之 AAC、H264 推送】
RTMP协议发送H.264编码及AAC编码的音视频,实现摄像头直播
RTMP协议发送H264 AAC音视频实例 C 实现的RTMP程序实例
[总结]RTMP流媒体技术零基础学习方法
最简单的基于librtmp的示例:接收(RTMP保存为FLV)
FFMpeg处理RTMP流的总结
更多类似文章 >>
生活服务
热点新闻
分享 收藏 导长图 关注 下载文章
绑定账号成功
后续可登录账号畅享VIP特权!
如果VIP功能使用有故障,
可点击这里联系客服!

联系客服