EasyRTMP内置进入摄像机中实现网络推流直播摄像机的功能
本文转自博客:http://blog.csdn.net/jinlong0603/article/details/57468084
在前面的《如何用传统摄像机实现直接对接平台,类似于海康萤石、大华乐橙的方案》中,我们描述了摄像机如何对接平台的功能,那么实际开发中,我们实现例如各种4G直播摄像机、直播解码器功能的摄像机,我们就需要将RTMP推送功能内置到摄像机,在摄像机内部调用自身SDK获取码流,再推送到nginx-rtmp、wowza、FMS等RTMP流媒体服务器实现直播;
EasyRTMP
EasyRTMP是EasyDarwin团队开发的一套支持Windows、Linux、arm、Android、iOS等各种平台的高效RTMP推送库。
EasyRTMP接收的是标准的H264流和AAC音频流。可以单独推送H264,也可以音视频混合推送。
内置进IPCamera
先初始化EasyRTMP和IPCamera的SDK,再调用IPCamera的SDK获取IPCamera的实时视频流。在实时流的回调中,调用EasyRTMP_SendPacket接口将音视频流实时传给EasyRTMP进行RTMP推送直播。值得注意的是如果需要推送音频的话,音频需要是AAC格式,如果没有AAC流,可以通过第三方工具转成AAC格式,如fdk-aac等。交叉编译后内置进IPCamera内部,就可以将普通的IPCamera升级成RTMP直播摄像机。GitHub上提供了hisiv100、hisiv200、hisiv300、hisiv400、linux i386、linux x64、Windows版本的EasyRTMP库, 其他版本arm平台的EasyRTMP库,可以提供编译工具链给EasyDarwin团队获取对应的库文件。
Demo代码
代码在GitHub上,见文末链接。可参考代码中EasyRTMP_SDK项目的代码,这个Demo就是调用了摄像机的SDK获取实时视频流传给EasyRTMP库进行直播推送。如下代码是SDK视频流回调的处理,其他具体去下载代码查看。
HI_S32 NETSDK_APICALL OnStreamCallback(HI_U32 u32Handle,
HI_U32 u32DataType,
HI_U8* pu8Buffer,
HI_U32 u32Length,
HI_VOID* pUserData
)
{
HI_S_AVFrame* pstruAV = HI_NULL;
HI_S_SysHeader* pstruSys = HI_NULL;
if (u32DataType == HI_NET_DEV_AV_DATA)
{
pstruAV = (HI_S_AVFrame*)pu8Buffer;
if (pstruAV->u32AVFrameFlag == HI_NET_DEV_VIDEO_FRAME_FLAG)
{
if(g_RtmpHandle == 0 ) return 0;
if(pstruAV->u32AVFrameLen > 0)
{
unsigned char* pbuf = (unsigned char*)(pu8Buffer+sizeof(HI_S_AVFrame));
EASY_AV_Frame avFrame;
memset(&avFrame, 0x00, sizeof(EASY_AV_Frame));
avFrame.u32AVFrameLen = pstruAV->u32AVFrameLen;
avFrame.pBuffer = (unsigned char*)pbuf;
avFrame.u32VFrameType = (pstruAV->u32VFrameType==HI_NET_DEV_VIDEO_FRAME_I)?EASY_SDK_VIDEO_FRAME_I:EASY_SDK_VIDEO_FRAME_P;
avFrame.u32AVFrameFlag = EASY_SDK_VIDEO_FRAME_FLAG;
avFrame.u32TimestampSec = pstruAV->u32AVFramePTS/1000;
avFrame.u32TimestampUsec = (pstruAV->u32AVFramePTS%1000)*1000;
EasyRTMP_SendPacket(g_RtmpHandle, &avFrame);
}
}
else
if (pstruAV->u32AVFrameFlag == HI_NET_DEV_AUDIO_FRAME_FLAG)
{
if(g_RtmpHandle == 0 ) return 0;
EASY_AV_Frame avFrame;
memset(&avFrame, 0x00, sizeof(EASY_AV_Frame));
avFrame.u32AVFrameFlag = EASY_SDK_AUDIO_FRAME_FLAG;
if(pstruAV->u32AVFrameLen <= 4) return 0;
if(EasyInitAACEncoder(EASY_SDK_AUDIO_CODEC_G711A) == 0)
{
memset(m_pAACEncBufer, 0, 64*1024);
unsigned int iAACBufferLen = 0;
unsigned char* pbuf = (unsigned char*)(pu8Buffer+sizeof(HI_S_AVFrame)) + 4;
if(Easy_AACEncoder_Encode(g_pAACEncoderHandle, (unsigned char*)pbuf, pstruAV->u32AVFrameLen-4, m_pAACEncBufer, &iAACBufferLen) > 0)
{
avFrame.pBuffer = (Easy_U8*)(m_pAACEncBufer);
avFrame.u32AVFrameLen = iAACBufferLen;
avFrame.u32TimestampSec = pstruAV->u32AVFramePTS/1000;
avFrame.u32TimestampUsec = (pstruAV->u32AVFramePTS%1000)*1000;
EasyRTMP_SendPacket(g_RtmpHandle, &avFrame);
}
}
}
}
else
if (u32DataType == HI_NET_DEV_SYS_DATA)
{
pstruSys = (HI_S_SysHeader*)pu8Buffer;
printf("Video W:%u H:%u Audio: %u \n", pstruSys->struVHeader.u32Width, pstruSys->struVHeader.u32Height, pstruSys->struAHeader.u32Format);
}
return HI_SUCCESS;
}
下载地址
Demo代码下载:https://github.com/EasyDarwin/EasyRTMP
Android EasyRTMP版本下载:https://fir.im/easyrtmp
获取更多信息
Copyright © EasyDarwin.org 2012-2017