RTMP 推流代码

初始化部分
av_log_set_level(1);
m_pRtmpContex = (RtmpContex*)malloc(sizeof(RtmpContex));
memset(m_Url, 0, 128);
memcpy(m_Url, RTMPUrl, strlen(RTMPUrl));

InitNet();
m_pRtmpContex->pRtmp = RTMP_Alloc();
RTMP_Init(m_pRtmpContex->pRtmp);
m_pRtmpContex->pRtmp->Link.timeout = 5;
RTMP_SetupURL(m_pRtmpContex->pRtmp, RTMPUrl);
RTMP_EnableWrite(m_pRtmpContex->pRtmp);

if (!RTMP_Connect(m_pRtmpContex->pRtmp, NULL))
{
	RTMP_Close(m_pRtmpContex->pRtmp);
	OutputDebugString("RTMP ::RTMP_Connect fail\n");
	return;
}

if (!RTMP_ConnectStream(m_pRtmpContex->pRtmp, 0))
{
RTMP_Close(m_pRtmpContex->pRtmp);
OutputDebugString(“RTMP ::RTMP_ConnectStream fail\n”);
return;
}

RTMPPacket_Reset(&m_packet);
RTMPPacket_Alloc(&m_packet, DUMMY_SINK_RECEIVE_BUFFER);
OutputDebugString(“RTMP ::Init sucessful\n”);

发包

int i = 0;
int ret = 0;
long timeoffset = 0;
RTMPPacket_Reset(&m_packet);
unsigned char body = (unsigned char)m_packet.m_body;
if (!m_isfirst){ return 0; }

if (!m_firframe) //第一个包
{
memcpy(body + i, “\x17\x00\x00\x00\x00\x01”, 6); i += 6;
memcpy(body + i, m_pRtmpContex->spsBuf + 1, 3); i += 3;

body[i++] = 0xff;
body[i++] = 0xe1;
body[i++] = (m_pRtmpContex->spslen >> 8) & 0xff;
body[i++] = m_pRtmpContex->spslen & 0xff;
memcpy(body + i, m_pRtmpContex->spsBuf, m_pRtmpContex->spslen); i += m_pRtmpContex->spslen;

body[i++] = 0x01;
	body[i++] = (m_pRtmpContex->ppsLen >> 8) & 0xff;
	body[i++] = m_pRtmpContex->ppsLen & 0xff;

memcpy(body + i, m_pRtmpContex->ppsBuf, m_pRtmpContex->ppsLen); i += m_pRtmpContex->ppsLen;

timeoffset = timeGetTime();
m_pRtmpContex->StartTime = (uint32_t)timeoffset;
m_packet.m_packetType = RTMP_PACKET_TYPE_VIDEO;
m_packet.m_nBodySize = i;
m_packet.m_nChannel = 0x04;
m_packet.m_nTimeStamp = (uint32_t)timeoffset;
m_packet.m_hasAbsTimestamp = 0;
m_packet.m_headerType = RTMP_PACKET_SIZE_LARGE;
m_packet.m_nInfoField2 = m_pRtmpContex->pRtmp->m_stream_id;
ret = RTMP_SendPacket(m_pRtmpContex->pRtmp, &m_packet, true);
m_firframe = true;
return 0;
}

//第一个包过后,后面的头有变化
这里说一下SPS PPS 如果包中涵盖该信息是不用单独发的,如果包中不含有,每个GOP是需要带的
switch (m_type)
{
case AV_MEDIU_TYPE_PFRAME:
memcpy(body + i, “\x27\x01\x00\x00\0x00”, 5); i += 5;
body[i++] = (len >> 24) & 0xff;
body[i++] = (len >> 16) & 0xff;
body[i++] = (len >> 8) & 0xff;
body[i++] = len & 0xff;
memcpy(&body[i], data, len); i = i + len;
//时间戳最好是重新打,如果不重新打,最好保证是升序
timeoffset = timeGetTime();
m_packet.m_hasAbsTimestamp = 0;
m_packet.m_packetType = RTMP_PACKET_TYPE_VIDEO;
m_packet.m_nInfoField2 = m_pRtmpContex->pRtmp->m_stream_id;
m_packet.m_nChannel = 0x04;
m_packet.m_headerType = RTMP_PACKET_SIZE_LARGE;
m_packet.m_nTimeStamp = (uint32_t)timeoffset;
m_packet.m_nBodySize = i;
ret = RTMP_SendPacket(m_pRtmpContex->pRtmp, &m_packet, true);
if (!ret){ OutputDebugString(“RTMP ::RTMP_SendPacket fail\n”); }
return 0;
break;
case AV_MEDIU_TYPE_PPS:
if (!m_firframe) return 0;
timeoffset = timeGetTime();
memcpy(body + i, “\x17\x01\x00\x00\0x00”, 5); i += 5;
body[i++] = (m_pRtmpContex->spslen >> 24) & 0xff;
body[i++] = (m_pRtmpContex->spslen >> 16) & 0xff;
body[i++] = (m_pRtmpContex->spslen >> 8) & 0xff;
body[i++] = m_pRtmpContex->spslen & 0xff;
memcpy(body + i, m_pRtmpContex->spsBuf, m_pRtmpContex->spslen); i += m_pRtmpContex->spslen;

		body[i++] = (m_pRtmpContex->ppsLen >> 24) & 0xff;
		body[i++] = (m_pRtmpContex->ppsLen >> 16) & 0xff;
		body[i++] = (m_pRtmpContex->ppsLen >> 8) & 0xff;
		body[i++] = m_pRtmpContex->ppsLen & 0xff;
		memcpy(body + i, m_pRtmpContex->ppsBuf, m_pRtmpContex->ppsLen); i += m_pRtmpContex->ppsLen;
		m_packet.m_hasAbsTimestamp = 0;
		m_packet.m_packetType      = RTMP_PACKET_TYPE_VIDEO;
		m_packet.m_nInfoField2     = m_pRtmpContex->pRtmp->m_stream_id;
		m_packet.m_nChannel        = 0x04;
		m_packet.m_headerType      = RTMP_PACKET_SIZE_LARGE;
		m_packet.m_nTimeStamp      = (uint32_t)timeoffset;
		m_packet.m_nBodySize       = i;
		ret = RTMP_SendPacket(m_pRtmpContex->pRtmp, &m_packet, true);
		return 0;
		break;
case AV_MEDIU_TYPE_IFRAME:
		memcpy(body+i, "\x17\x01\x00\x00\0x00", 5); i += 5;
		body[i++] = (len >> 24) & 0xff;
		body[i++] = (len >> 16) & 0xff;
		body[i++] = (len >> 8) & 0xff;
		body[i++] = len & 0xff;
		memcpy(body + i, data, len); i = i + len;
		timeoffset = timeGetTime();
		
		m_packet.m_hasAbsTimestamp = 0;
		m_packet.m_packetType      = RTMP_PACKET_TYPE_VIDEO;
		m_packet.m_nInfoField2     = m_pRtmpContex->pRtmp->m_stream_id;
		m_packet.m_nChannel        = 0x04;
		m_packet.m_headerType      = RTMP_PACKET_SIZE_LARGE;
		m_packet.m_nTimeStamp      = (uint32_t)timeoffset;
		m_packet.m_nBodySize       = i;
		ret = RTMP_SendPacket(m_pRtmpContex->pRtmp, &m_packet, true);
		if (!ret){ OutputDebugString("RTMP ::RTMP_SendPacket  fail\n"); }
		return 0;
		break;
				default:
		break;
}

return 0;

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值