音视频学习(二十二)——rtmp发流(tcp方式)

前言

本文主要介绍自研的RtmpStreamSender.dll,rtmp库提供接口接收裸流数据,支持将裸流数据封装为flv格式并通过rtmp协议发流。

关于rtmp协议基础介绍可查看:https://blog.csdn.net/www_dong/article/details/131026072

关于rtmp收流介绍可查看:https://blog.csdn.net/www_dong/article/details/135073488

设计

主要流程:

  1. rtmp服务端启动tcp(端口:1935)监听,创建rtmp server对象;
  2. rtmp客户端发起tcp连接,连接成功后等到收流;
  3. rtmp服务端以动态库的形式呈现,提供SendData接口,有上层应用调用时塞入数据;
  4. rtmp服务端接收数据,将数据封装为flv格式;
  5. rtmp服务端将flv格式的流发送给rtmp客户端;
  6. rtmp客户端将数据解复用、解码和播放;

流程

  • 创建tcp、启动端口监听;
c++ 复制代码
int RtmpServerThread(void* param)
{
	assert(param);
	CRtmpStreamSender* receiver = (CRtmpStreamSender*)param;
	receiver->RtmpServerWorker();
	return 0;
}

int CRtmpStreamSender::InitRtmpSession_()
{
	if (0 != ParseUrl_())
	{
		printf("parse url failed\n");
		return -1;
	}

	do
	{
		m_tcpServer = std::make_shared<ZDTcpServer>(nullptr, this);
		if (!m_tcpServer.get()
			|| 0 != m_tcpServer->TcpCreate()
			|| 0 != m_tcpServer->TcpBind(m_rtmpPort)
			|| 0 != m_tcpServer->TcpListen(5))
		{
			break;
		}
			
		m_command = std::make_shared<CRtmpServerCommand>(m_tcpServer);
		if (!m_command.get()
			|| 0 != m_command->Create())
		{
			break;
		}

		m_thread = std::thread(RtmpServerThread, this);
		return 0;
	} while (0);
	
	Stop();
	return 0;
}

// 在线程中等待连接
void CRtmpStreamSender::RtmpServerWorker()
{
	bool bAccept = false;
	while (m_running)
	{
		if (!bAccept)
		{
			if (0 == m_tcpServer->TcpAccept())
			{
				bAccept = true;
			}

			continue;
		}

		std::this_thread::sleep_for(std::chrono::seconds(5));
	}
}
  • 创建rtmp server
c++ 复制代码
// CRtmpServerCommand:librtmp封装类
m_command = std::make_shared<CRtmpServerCommand>(m_tcpServer);
if (!m_command.get()
    || 0 != m_command->Create())
{
    break;
}

// 创建rtmp server
int CRtmpServerCommand::Create()
{
    // 创建flv复用对象
	m_flvMuxer = flv_muxer_create(SendRtmpDataMutexFLV, this);
	if (!m_flvMuxer)
		return -1;

	struct rtmp_server_handler_t handler;
	memset(&handler, 0, sizeof(handler));
	handler.send = SendCallback;

    // 创建rtmp server
	m_rtmp = rtmp_server_create(this, &handler);
	if (!m_rtmp)
		return -1;

	return 0;
}
  • 数据封装
c++ 复制代码
// 上层通过SendData塞数据
int CRtmpStreamSender::SendData(void* data, int len, char* codec)
{
	if (!m_command.get())
		return -1;

	return m_command->InputData(data, len, codec);
}

// 数据做flv封装
int CRtmpServerCommand::InputData(void* data, int len, const std::string& codec)
{
	if (!m_flvMuxer || !data || len <= 0 || codec.empty())
		return -1;

	int ret = -1;
	if (0 == codec.compare("H264"))
	{
		ret = flv_muxer_avc(m_flvMuxer, data, len, 0, 0);
	}
	else if (0 == codec.compare("H265"))
	{
		ret = flv_muxer_hevc(m_flvMuxer, data, len, 0, 0);
	}
	else if (0 == codec.compare("AAC"))
	{
		ret = flv_muxer_aac(m_flvMuxer, data, len, 0, 0);
	}

	return 0;
}
  • 发送数据
c++ 复制代码
int CRtmpServerCommand::Package(int type, const void* data, size_t bytes, uint32_t timestamp)
{
	if (!m_rtmp || !data || bytes <= 0)
		return -1;

	int ret = -1;
	if (FLV_TYPE_AUDIO == type)
	{
		ret = rtmp_server_send_audio(m_rtmp, data, bytes, timestamp);
	}
	else if (FLV_TYPE_VIDEO == type)
	{
		ret = rtmp_server_send_video(m_rtmp, data, bytes, timestamp);
	}
	else if (FLV_TYPE_SCRIPT == type)
	{
		ret = rtmp_server_send_script(m_rtmp, data, bytes, timestamp);
	}

	if (0 != ret)
		return ret;

	return 0;
}
  • 资源销毁
c++ 复制代码
void CRtmpServerCommand::Destroy()
{
	if (m_rtmp)
	{
		rtmp_server_destroy(m_rtmp);
		m_rtmp = nullptr;
	}

	if (m_flvMuxer)
	{
		flv_muxer_destroy(m_flvMuxer);
		m_flvMuxer = nullptr;
	}
}
相关推荐
计算机安禾11 分钟前
【数据结构与算法】第1篇:为什么要学习数据结构与算法?专栏导学
c语言·开发语言·c++·学习·算法·visual studio code·visual studio
sensen_kiss12 分钟前
CPT304 SoftwareEngineeringII 软件工程 2 Pt.2 面向对象概念
学习·软件工程
李白你好12 分钟前
一个综合性的Web安全学习平台
学习·安全·web安全
EnglishJun15 分钟前
ARM嵌入式学习(九)--- C语言应用:点亮led
c语言·arm开发·学习
小陈phd19 分钟前
多模态大模型学习笔记(二十四)—— 核心技术篇① | 虚拟人形象生成:扩散模型与ControlNet原理
笔记·学习
01二进制代码漫游日记40 分钟前
通讯录(一)
c语言·数据结构·学习
江苏世纪龙科技44 分钟前
让汽修课堂“动”起来—哈弗M6汽车故障诊断与排除仿真教学软件
学习
芯跳加速44 分钟前
AI 视频自动化学习日记 · 第四天
人工智能·自动化·音视频
科技小E1 小时前
视频分析平台EasyGBS视频质量诊断技术解析与实战功能应用
音视频
一招定胜负1 小时前
【实战】Python + 讯飞语音识别 + 通义千问:课堂视频自动转结构化教学数据
python·音视频·语音识别