// rtsp_manager.cpp #include "rtsp_manager.hpp" #include "logger.hpp" #include #include #include #include // 静态变量定义 GMainLoop *RTSPManager::loop = nullptr; GMainContext *RTSPManager::main_context = nullptr; GstRTSPServer *RTSPManager::server = nullptr; std::unordered_map RTSPManager::streaming_status; std::unordered_map RTSPManager::mounted_factories; std::mutex RTSPManager::mounted_factories_mutex; std::unordered_map> RTSPManager::media_map; std::mutex RTSPManager::media_map_mutex; void RTSPManager::init() { gst_init(nullptr, nullptr); LOG_INFO("[RTSP] GStreamer initialized."); } // 创建 media factory GstRTSPMediaFactory *RTSPManager::create_media_factory(const Camera &cam) { // 输出分辨率直接用相机原始分辨率 int out_width = cam.width; int out_height = cam.height; std::string launch_str = "( v4l2src device=" + cam.device + " ! video/x-raw,format=NV12,width=" + std::to_string(out_width) + ",height=" + std::to_string(out_height) + ",framerate=" + std::to_string(cam.fps) + "/1" " ! queue max-size-buffers=1 leaky=downstream" " ! mpph265enc rc-mode=cbr bps=" + std::to_string(cam.bitrate) + " gop=" + std::to_string(cam.fps) + " b-frames=0" " ! h265parse config-interval=1" " ! rtph265pay name=pay0 pt=96 config-interval=1 )"; GstRTSPMediaFactory *factory = gst_rtsp_media_factory_new(); gst_rtsp_media_factory_set_launch(factory, launch_str.c_str()); gst_rtsp_media_factory_set_shared(factory, TRUE); gst_rtsp_media_factory_set_suspend_mode(factory, GST_RTSP_SUSPEND_MODE_RESET); // 使用 media-configure 信号代替 media-created g_signal_connect_data(factory, "media-configure", G_CALLBACK(on_media_created), g_strdup(cam.name.c_str()), (GClosureNotify)g_free, static_cast(0)); return factory; } // 启动 RTSP server void RTSPManager::start(const std::vector &cameras) { server = gst_rtsp_server_new(); gst_rtsp_server_set_service(server, "8554"); loop = g_main_loop_new(nullptr, FALSE); main_context = g_main_loop_get_context(loop); gst_rtsp_server_attach(server, nullptr); LOG_INFO("[RTSP] Server running on rtsp://localhost:8554"); g_main_loop_run(loop); if (server) { g_object_unref(server); server = nullptr; } if (loop) { g_main_loop_unref(loop); loop = nullptr; } LOG_INFO("[RTSP] Server stopped."); } // media-configure 信号处理 void RTSPManager::on_media_created(GstRTSPMediaFactory *factory, GstRTSPMedia *media, gpointer user_data) { const char *cam_name = static_cast(user_data); g_object_ref(media); // 增加引用计数,防止被提前销毁 { std::lock_guard lock(media_map_mutex); media_map[cam_name].push_back(media); } // 连接 unprepared 信号,当 pipeline 被销毁时移除 g_signal_connect_data(media, "unprepared", G_CALLBACK(on_media_unprepared), g_strdup(cam_name), (GClosureNotify)g_free, static_cast(0)); } // unprepared 信号处理 void RTSPManager::on_media_unprepared(GstRTSPMedia *media, gpointer user_data) { const char *cam_name = static_cast(user_data); { std::lock_guard lock(media_map_mutex); auto it = media_map.find(cam_name); if (it != media_map.end()) { auto &media_list = it->second; media_list.erase(std::remove(media_list.begin(), media_list.end(), media), media_list.end()); if (media_list.empty()) { media_map.erase(it); } } } // === ★ 新增:减计数并在最后一个时唤醒等待线程 === { std::lock_guard lk(stop_mutex); if (--stopping_count[cam_name] == 0) stop_cv.notify_all(); } g_object_unref(media); // 释放引用 } // 挂载摄像头 gboolean RTSPManager::mount_camera_in_main(gpointer data) { Camera *cam = static_cast(data); if (!cam || !server) { delete cam; return G_SOURCE_REMOVE; } GstRTSPMountPoints *mounts = gst_rtsp_server_get_mount_points(server); if (!mounts) { delete cam; return G_SOURCE_REMOVE; } std::string mount_point = "/" + cam->name; GstRTSPMediaFactory *factory = create_media_factory(*cam); if (!factory) { g_object_unref(mounts); delete cam; return G_SOURCE_REMOVE; } gst_rtsp_mount_points_add_factory(mounts, mount_point.c_str(), factory); g_object_unref(mounts); { std::lock_guard lock(mounted_factories_mutex); mounted_factories[cam->name] = factory; streaming_status[cam->name] = true; } LOG_INFO("[RTSP] Camera '" + cam->name + "' mounted at rtsp://localhost:8554" + mount_point); delete cam; return G_SOURCE_REMOVE; } // 卸载摄像头 gboolean RTSPManager::unmount_camera_in_main(gpointer data) { Camera *cam = static_cast(data); if (!cam || !server) { delete cam; return G_SOURCE_REMOVE; } std::string cam_name = cam->name; std::string mount_point = "/" + cam_name; // 停止所有媒体 { std::lock_guard lock(media_map_mutex); auto it = media_map.find(cam_name); if (it != media_map.end()) { // === ★ 新增:记录需要等待的 media 数量 === { std::lock_guard lk(stop_mutex); stopping_count[cam_name] = it->second.size(); } for (GstRTSPMedia *media : it->second) { GstElement *pipeline = gst_rtsp_media_get_element(media); if (pipeline) { gst_element_set_state(pipeline, GST_STATE_NULL); gst_object_unref(pipeline); } // 这一步会触发 on_media_unprepared 信号 gst_rtsp_media_unprepare(media); } } } // 卸载 factory GstRTSPMountPoints *mounts = gst_rtsp_server_get_mount_points(server); if (mounts) { gst_rtsp_mount_points_remove_factory(mounts, mount_point.c_str()); g_object_unref(mounts); } { std::lock_guard lock(mounted_factories_mutex); auto it = mounted_factories.find(cam_name); if (it != mounted_factories.end()) { if (it->second && G_IS_OBJECT(it->second)) g_object_unref(it->second); mounted_factories.erase(it); } streaming_status[cam_name] = false; } LOG_INFO("[RTSP] Camera '" + cam_name + "' unmounted."); delete cam; return G_SOURCE_REMOVE; } // 公共挂载/卸载接口 void RTSPManager::mount_camera(const Camera &cam) { Camera *camCopy = new Camera(cam); g_main_context_invoke(main_context, [](gpointer data) -> gboolean { return RTSPManager::mount_camera_in_main(data); }, camCopy); // === ★ 新增:等待 on_media_unprepared 全部完成 === std::unique_lock lk(stop_mutex); stop_cv.wait(lk, [&] { auto it = stopping_count.find(cam.name); return it == stopping_count.end() || it->second == 0; }); } void RTSPManager::unmount_camera(const Camera &cam) { Camera *camCopy = new Camera(cam); g_main_context_invoke(main_context, [](gpointer data) -> gboolean { return RTSPManager::unmount_camera_in_main(data); }, camCopy); } // 是否正在流 bool RTSPManager::is_streaming(const std::string &cam_name) { std::lock_guard lock(mounted_factories_mutex); auto it = streaming_status.find(cam_name); return it != streaming_status.end() ? it->second : false; } // 停止 server void RTSPManager::stop() { // Ctrl-C 退出时,不再尝试逐路卸载摄像头, // 只负责让 g_main_loop_run() 退出,线程自然收尾。 if (loop) { g_main_context_invoke( main_context, [](gpointer data) -> gboolean { GMainLoop *lp = static_cast(data); if (lp) g_main_loop_quit(lp); return G_SOURCE_REMOVE; }, loop); } } // 新增接口:检查是否还有摄像头在流 bool RTSPManager::is_any_streaming() { std::lock_guard lock(mounted_factories_mutex); for (const auto &kv : streaming_status) if (kv.second) return true; return false; }