int SrsPublishEdge::initialize(SrsSource* source, SrsRequest* req) { int ret = ERROR_SUCCESS; if ((ret = forwarder->initialize(source, this, req)) != ERROR_SUCCESS) { return ret; } return ret; } bool SrsPublishEdge::can_publish() { return state != SrsEdgeStatePublish; } int SrsPublishEdge::on_client_publish() { int ret = ERROR_SUCCESS; // error when not init state. if (state != SrsEdgeStateInit) { ret = ERROR_RTMP_EDGE_PUBLISH_STATE; srs_error("invalid state for client to publish stream on edge. " "state=%d, ret=%d", state, ret); return ret; } // @see https://github.com/ossrs/srs/issues/180 // to avoid multiple publish the same stream on the same edge, // directly enter the publish stage. if (true) { SrsEdgeState pstate = state; state = SrsEdgeStatePublish; srs_trace("edge change from %d to state %d (push).", pstate, state); } // start to forward stream to origin. ret = forwarder->start(); // @see https://github.com/ossrs/srs/issues/180 // when failed, revert to init if (ret != ERROR_SUCCESS) { SrsEdgeState pstate = state; state = SrsEdgeStateInit; srs_trace("edge revert from %d to state %d (push). ret=%d", pstate, state, ret); } return ret; }
SrsPublishEdge::on_client_publish这个函数在encoder 向 edge 推流是会调用这个函数,
用来启动farwarder,在 encoder 和 origin 之间转发 rtmp message.
这一篇分析 SrsRtmpConn::process_publish_message函数
对SrsSouce类的几个处理函数的代码
1. SrsSource::on_edge_proxy_publish
int SrsSource::on_edge_proxy_publish(SrsCommonMessage* msg) { return publish_edge->on_proxy_publish(msg); }
如果是edge,直接调用SrsPublishEdge类的on_proxy_publish方法,将msg转发到origin
int SrsPublishEdge::on_proxy_publish(SrsCommonMessage* msg) { return forwarder->proxy(msg); }
调用 SrsEdgeForwarder类的proxy函数
int SrsEdgeForwarder::proxy(SrsCommonMessage* msg) { int ret = ERROR_SUCCESS; if ((ret = send_error_code) != ERROR_SUCCESS) { srs_error("publish edge proxy thread send error, ret=%d", ret); return ret; } // the msg is auto free by source, // so we just ignore, or copy then send it. if (msg->size <= 0 || msg->header.is_set_chunk_size() || msg->header.is_window_ackledgement_size() || msg->header.is_ackledgement() ) { return ret; } SrsSharedPtrMessage copy; if ((ret = copy.create(msg)) != ERROR_SUCCESS) { srs_error("initialize the msg failed. ret=%d", ret); return ret; } srs_verbose("initialize shared ptr msg success."); copy.stream_id = stream_id; if ((ret = queue->enqueue(copy.copy())) != ERROR_SUCCESS) { srs_error("enqueue edge publish msg failed. ret=%d", ret); } return ret; }
int SrsEdgeForwarder::proxy(SrsCommonMessage* msg)函数
用来将从 encoder接收的 rtmp msg 保存在 queue中
int SrsEdgeForwarder::start() { int ret = ERROR_SUCCESS; send_error_code = ERROR_SUCCESS; std::string ep_server, ep_port; if ((ret = connect_server(ep_server, ep_port)) != ERROR_SUCCESS) { return ret; } srs_assert(client); client->set_recv_timeout(SRS_CONSTS_RTMP_RECV_TIMEOUT_US); client->set_send_timeout(SRS_CONSTS_RTMP_SEND_TIMEOUT_US); SrsRequest* req = _req; if ((ret = client->handshake()) != ERROR_SUCCESS) { srs_error("handshake with server failed. ret=%d", ret); return ret; } if ((ret = connect_app(ep_server, ep_port)) != ERROR_SUCCESS) { srs_error("connect with server failed. ret=%d", ret); return ret; } if ((ret = client->create_stream(stream_id)) != ERROR_SUCCESS) { srs_error("connect with server failed, stream_id=%d. ret=%d", stream_id, ret); return ret; } if ((ret = client->publish(req->stream, stream_id)) != ERROR_SUCCESS) { srs_error("publish failed, stream=%s, stream_id=%d. ret=%d", req->stream.c_str(), stream_id, ret); return ret; } return pthread->start(); }
int SrsEdgeForwarder::start()函数
在encoder 向 edge推流是调用
作用是用 SrsRtmpClient类连接到 origin
SrsEdgeForwarder包含
SrsReusableThread2类
#define SYS_MAX_EDGE_SEND_MSGS 128 int SrsEdgeForwarder::cycle() { int ret = ERROR_SUCCESS; client->set_recv_timeout(SRS_CONSTS_RTMP_PULSE_TIMEOUT_US); SrsPithyPrint* pprint = SrsPithyPrint::create_edge(); SrsAutoFree(SrsPithyPrint, pprint); SrsMessageArray msgs(SYS_MAX_EDGE_SEND_MSGS); while (!pthread->interrupted()) { if (send_error_code != ERROR_SUCCESS) { st_usleep(SRS_EDGE_FORWARDER_ERROR_US); continue; } // read from client. if (true) { SrsCommonMessage* msg = NULL; ret = client->recv_message(&msg); srs_verbose("edge loop recv message. ret=%d", ret); if (ret != ERROR_SUCCESS && ret != ERROR_SOCKET_TIMEOUT) { srs_error("edge push get server control message failed. ret=%d", ret); send_error_code = ret; continue; } //为什么没见到msg的处理函数直接就释放了
srs_freep(msg); } // forward all messages. // each msg in msgs.msgs must be free, for the SrsMessageArray never free them. int count = 0; if ((ret = queue->dump_packets(msgs.max, msgs.msgs, count)) != ERROR_SUCCESS) { srs_error("get message to push to origin failed. ret=%d", ret); return ret; } pprint->elapse(); // pithy print if (pprint->can_print()) { kbps->sample(); srs_trace("-> "SRS_CONSTS_LOG_EDGE_PUBLISH " time=%"PRId64", msgs=%d, okbps=%d,%d,%d, ikbps=%d,%d,%d", pprint->age(), count, kbps->get_send_kbps(), kbps->get_send_kbps_30s(), kbps->get_send_kbps_5m(), kbps->get_recv_kbps(), kbps->get_recv_kbps_30s(), kbps->get_recv_kbps_5m()); } // count 是 queue中消息的个数,也就是encoder发送到edge的rtmp msg得个数
// count<=0 没有包需要有 edge 转发到 origin
// ignore when no messages. if (count <= 0) { srs_verbose("no packets to push."); continue; } // sendout messages, all messages are freed by send_and_free_messages(). if ((ret = client->send_and_free_messages(msgs.msgs, count, stream_id)) != ERROR_SUCCESS) { srs_error("edge publish push message to server failed. ret=%d", ret); return ret; } } return ret; }
SrsEdgeForwarder::cycle的主要功能
有 SrsRtmpClient类来完成
1. 调用SrsRtmpClient::recv_message从 origin接收消息
2.
queue->dump_packets(msgs.max, msgs.msgs, count)将从encoder接收到的rtmp msg 保存在类型为
SrsMessageArray的变量msgs中,
client->send_and_free_messages(msgs.msgs, count, stream_id)将rtmp msg 转发到origin
client是 edge到 origin rtmp 连接,类型是SrsRtmpClient,
主要用于在encoder和origin转发 rtmp msg
edge 是的推流rtmp msg就分析完了,没做过多处理转发到origin
接下来分析origin的rtmp msg处理
int SrsRtmpConn::acquire_publish(SrsSource* source, bool is_edge) { int ret = ERROR_SUCCESS; if (!source->can_publish(is_edge)) { ret = ERROR_SYSTEM_STREAM_BUSY; srs_warn("stream %s is already publishing. ret=%d", req->get_stream_url().c_str(), ret); return ret; } // when edge, ignore the publish event, directly proxy it. if (is_edge) {
//在这里启动一个SrsEdgeForwarder转发encoder的rtmp msg
if ((ret = source->on_edge_start_publish()) != ERROR_SUCCESS) { srs_error("notice edge start publish stream failed. ret=%d", ret); return ret; } } else {
// origin 会运行到此处,从SrsSource的on_publish开始分析
if ((ret = source->on_publish()) != ERROR_SUCCESS) { srs_error("notify publish failed. ret=%d", ret); return ret; } } return ret; }
int SrsSource::on_publish() { int ret = ERROR_SUCCESS; // update the request object. srs_assert(_req); _can_publish = false; // whatever, the publish thread is the source or edge source, // save its id to srouce id.
// 更新_source_id,并通知所有的SrsConsumer(call SrsConsumer::update_source_id)
on_source_id_changed(_srs_context->get_id()); // reset the mix queue. mix_queue->clear(); // detect the monotonically again. is_monotonically_increase = true; last_packet_time = 0; // create forwarders
// 在vhost 里配置 forward 127.0.0.1:1936 127.0.0.1:1937;
// 以后再具体分析
if ((ret = create_forwarders()) != ERROR_SUCCESS) { srs_error("create forwarders failed. ret=%d", ret); return ret; } // 用 ffmpeg对流进行转码
// 配置 transcode --> engine --> output rtmp://127.0.0.1:[port]/[app]?vhost=[vhost]/[stream]_[engine];
// 以后具体分析
// TODO: FIXME: use initialize to set req. #ifdef SRS_AUTO_TRANSCODE if ((ret = encoder->on_publish(_req)) != ERROR_SUCCESS) { srs_error("start encoder failed. ret=%d", ret); return ret; } #endif // TODO: FIXME: use initialize to set req.
// hls相关,以后具体分析
#ifdef SRS_AUTO_HLS if ((ret = hls->on_publish(_req, false)) != ERROR_SUCCESS) { srs_error("start hls failed. ret=%d", ret); return ret; } #endif //DVR 相关,以后具体分析
// TODO: FIXME: use initialize to set req. #ifdef SRS_AUTO_DVR if ((ret = dvr->on_publish(_req)) != ERROR_SUCCESS) { srs_error("start dvr failed. ret=%d", ret); return ret; } #endif //HDS相关,以后具体分析
ifdef SRS_AUTO_HDS if ((ret = hds->on_publish(_req)) != ERROR_SUCCESS) { srs_error("start hds failed. ret=%d", ret); return ret; } #endif // notify the handler. srs_assert(handler); if ((ret = handler->on_publish(this, _req)) != ERROR_SUCCESS) { srs_error("handle on publish failed. ret=%d", ret); return ret; } SrsStatistic* stat = SrsStatistic::instance(); stat->on_stream_publish(_req, _source_id); return ret; }
主要分析
handler->on_publish(this, _req)
b srs_app_source.cpp:2203
推流后断下
bt command output above:
#0 SrsServer::on_publish (this=0x8d53e0, s=0x8fa0f0, r=0x8fcf00) at src/app/srs_app_server.cpp:1426 #1 0x000000000048e817 in SrsSource::on_publish (this=0x8fa0f0) at src/app/srs_app_source.cpp:2203 #2 0x0000000000481fb6 in SrsRtmpConn::acquire_publish (this=0x8f4ec0, source=0x8fa0f0, is_edge=false) at src/app/srs_app_rtmp_conn.cpp:980 #3 0x000000000048122e in SrsRtmpConn::publishing (this=0x8f4ec0, source=0x8fa0f0) at src/app/srs_app_rtmp_conn.cpp:847 #4 0x000000000047f759 in SrsRtmpConn::stream_service_cycle (this=0x8f4ec0) at src/app/srs_app_rtmp_conn.cpp:547 #5 0x000000000047e9d8 in SrsRtmpConn::service_cycle (this=0x8f4ec0) at src/app/srs_app_rtmp_conn.cpp:416 #6 0x000000000047d936 in SrsRtmpConn::do_cycle (this=0x8f4ec0) at src/app/srs_app_rtmp_conn.cpp:211 #7 0x000000000047bc6d in SrsConnection::cycle (this=0x8f4f48) at src/app/srs_app_conn.cpp:89 #8 0x00000000004ad6a3 in SrsOneCycleThread::cycle (this=0x8f4f90) at src/app/srs_app_thread.cpp:372 #9 0x00000000004ace3b in internal::SrsThread::thread_cycle (this=0x8f4fb0) at src/app/srs_app_thread.cpp:207 #10 0x00000000004ad049 in internal::SrsThread::thread_fun (arg=0x8f4fb0) at src/app/srs_app_thread.cpp:245 #11 0x0000000000535371 in _st_thread_main () at sched.c:327 #12 0x0000000000535ae1 in st_thread_create (start=0x7ffff7fdbc20, arg=0x1f7fdbb60, joinable=0, stk_size=5480568) at sched.c:591
SrsServer::on_publish代码如下
int SrsServer::on_publish(SrsSource* s, SrsRequest* r) { int ret = ERROR_SUCCESS; #ifdef SRS_AUTO_HTTP_SERVER if ((ret = http_server->http_mount(s, r)) != ERROR_SUCCESS) { return ret; } #endif return ret; }
SrsHttpServer::http_mount