2025-09-09 13:28:05 +08:00
|
|
|
|
// rtsp_manager.cpp
|
2025-09-08 14:55:07 +08:00
|
|
|
|
#include "rtsp_manager.hpp"
|
2025-11-21 15:32:56 +08:00
|
|
|
|
|
2025-11-24 16:37:53 +08:00
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
|
#include <linux/videodev2.h>
|
2025-12-17 14:13:52 +08:00
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
#include <chrono>
|
|
|
|
|
|
#include <iostream>
|
|
|
|
|
|
#include <thread>
|
2025-12-17 15:50:48 +08:00
|
|
|
|
#include <unordered_map>
|
2025-12-17 14:13:52 +08:00
|
|
|
|
|
|
|
|
|
|
#include "logger.hpp"
|
2025-09-08 14:55:07 +08:00
|
|
|
|
|
2025-09-09 09:57:24 +08:00
|
|
|
|
// 静态变量定义
|
2025-12-17 14:13:52 +08:00
|
|
|
|
GMainLoop* RTSPManager::loop = nullptr;
|
|
|
|
|
|
GMainContext* RTSPManager::main_context = nullptr;
|
|
|
|
|
|
GstRTSPServer* RTSPManager::server = nullptr;
|
2025-11-21 15:32:56 +08:00
|
|
|
|
|
2025-09-09 11:02:41 +08:00
|
|
|
|
std::unordered_map<std::string, bool> RTSPManager::streaming_status;
|
2025-12-17 14:13:52 +08:00
|
|
|
|
std::unordered_map<std::string, GstRTSPMediaFactory*> RTSPManager::mounted_factories;
|
2025-09-09 11:02:41 +08:00
|
|
|
|
std::mutex RTSPManager::mounted_factories_mutex;
|
2025-11-21 15:32:56 +08:00
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
// ==============================
|
|
|
|
|
|
// ✅ 不再保存 GstRTSPMedia* 裸指针
|
|
|
|
|
|
// 改为:每路 camera 的“活跃 session 计数”
|
|
|
|
|
|
// ==============================
|
|
|
|
|
|
static std::unordered_map<std::string, int> g_client_count;
|
|
|
|
|
|
static std::mutex g_client_count_mutex;
|
2025-09-09 10:30:50 +08:00
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
// 日志降噪:media-configure 过于频繁时只提示
|
2025-12-17 15:40:48 +08:00
|
|
|
|
static std::unordered_map<std::string, std::chrono::steady_clock::time_point> last_media_ts;
|
|
|
|
|
|
static std::mutex last_media_ts_mutex;
|
|
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
bool set_v4l2_format(const std::string& dev, int width, int height)
|
2025-11-24 16:37:53 +08:00
|
|
|
|
{
|
|
|
|
|
|
int fd = open(dev.c_str(), O_RDWR);
|
|
|
|
|
|
if (fd < 0)
|
|
|
|
|
|
{
|
|
|
|
|
|
LOG_ERROR("Failed to open " + dev);
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
struct v4l2_format fmt;
|
2025-11-25 08:54:26 +08:00
|
|
|
|
memset(&fmt, 0, sizeof(fmt));
|
|
|
|
|
|
fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
|
|
|
|
|
|
|
2025-11-25 09:21:08 +08:00
|
|
|
|
// 读格式,若相同则略过
|
2025-11-25 08:54:26 +08:00
|
|
|
|
if (ioctl(fd, VIDIOC_G_FMT, &fmt) == 0)
|
|
|
|
|
|
{
|
|
|
|
|
|
bool match = true;
|
2025-12-17 14:13:52 +08:00
|
|
|
|
if (fmt.fmt.pix_mp.width != (unsigned int)width || fmt.fmt.pix_mp.height != (unsigned int)height ||
|
2025-11-25 08:54:26 +08:00
|
|
|
|
fmt.fmt.pix_mp.pixelformat != V4L2_PIX_FMT_NV12)
|
|
|
|
|
|
{
|
|
|
|
|
|
match = false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (match)
|
|
|
|
|
|
{
|
|
|
|
|
|
close(fd);
|
2025-12-17 14:13:52 +08:00
|
|
|
|
LOG_INFO("[RTSP] V4L2 format already NV12 " + std::to_string(width) + "x" + std::to_string(height) +
|
|
|
|
|
|
" for " + dev);
|
2025-11-25 08:54:26 +08:00
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-11-24 16:37:53 +08:00
|
|
|
|
memset(&fmt, 0, sizeof(fmt));
|
|
|
|
|
|
fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
|
|
|
|
|
|
fmt.fmt.pix_mp.width = width;
|
|
|
|
|
|
fmt.fmt.pix_mp.height = height;
|
|
|
|
|
|
fmt.fmt.pix_mp.pixelformat = V4L2_PIX_FMT_NV12;
|
|
|
|
|
|
fmt.fmt.pix_mp.num_planes = 1;
|
|
|
|
|
|
|
|
|
|
|
|
if (ioctl(fd, VIDIOC_S_FMT, &fmt) < 0)
|
|
|
|
|
|
{
|
|
|
|
|
|
LOG_ERROR("VIDIOC_S_FMT failed for " + dev);
|
|
|
|
|
|
close(fd);
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-11-25 08:54:26 +08:00
|
|
|
|
LOG_INFO("[RTSP] Set V4L2 format to NV12 " + std::to_string(width) + "x" + std::to_string(height) + " for " + dev);
|
2025-11-24 16:37:53 +08:00
|
|
|
|
close(fd);
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-08 14:55:07 +08:00
|
|
|
|
void RTSPManager::init()
|
|
|
|
|
|
{
|
|
|
|
|
|
gst_init(nullptr, nullptr);
|
|
|
|
|
|
LOG_INFO("[RTSP] GStreamer initialized.");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
GstRTSPMediaFactory* RTSPManager::create_media_factory(const Camera& cam)
|
2025-09-08 14:55:07 +08:00
|
|
|
|
{
|
2025-11-25 09:13:52 +08:00
|
|
|
|
// 启动前把 v4l2 格式设成我们想要的
|
2025-11-25 09:02:06 +08:00
|
|
|
|
set_v4l2_format(cam.device, cam.width, cam.height);
|
2025-11-24 16:37:53 +08:00
|
|
|
|
|
2025-11-25 09:02:06 +08:00
|
|
|
|
int w = cam.width;
|
|
|
|
|
|
int h = cam.height;
|
2025-09-16 18:19:42 +08:00
|
|
|
|
|
2025-11-24 17:36:32 +08:00
|
|
|
|
std::string caps =
|
|
|
|
|
|
"video/x-raw,format=NV12,"
|
|
|
|
|
|
"width=" +
|
2025-12-17 14:13:52 +08:00
|
|
|
|
std::to_string(w) + ",height=" + std::to_string(h) + ",framerate=" + std::to_string(cam.fps) + "/1";
|
2025-11-24 17:36:32 +08:00
|
|
|
|
|
2025-11-25 09:27:20 +08:00
|
|
|
|
// 关键:tee 一路给 RTSP,一路给 fakesink,保持 v4l2src / mpph264enc 永远在跑
|
2025-12-17 14:13:52 +08:00
|
|
|
|
std::string launch_str = "( v4l2src device=" + cam.device +
|
|
|
|
|
|
" io-mode=2 is-live=true do-timestamp=true"
|
|
|
|
|
|
" ! " +
|
|
|
|
|
|
caps +
|
|
|
|
|
|
" ! tee name=t "
|
|
|
|
|
|
" ! queue leaky=downstream max-size-time=0 max-size-bytes=0 max-size-buffers=0"
|
|
|
|
|
|
" ! mpph264enc name=enc rc-mode=cbr bps=" +
|
|
|
|
|
|
std::to_string(cam.bitrate) + " gop=" + std::to_string(cam.fps) +
|
|
|
|
|
|
" header-mode=1"
|
|
|
|
|
|
" ! h264parse"
|
|
|
|
|
|
" ! rtph264pay name=pay0 pt=96 config-interval=1 "
|
|
|
|
|
|
" t. ! queue leaky=downstream max-size-buffers=3 max-size-bytes=0 max-size-time=0"
|
|
|
|
|
|
" ! fakesink sync=false async=false )";
|
2025-11-25 08:54:26 +08:00
|
|
|
|
|
2025-11-25 09:13:52 +08:00
|
|
|
|
LOG_INFO("[RTSP] Launch for " + cam.name + ": " + launch_str);
|
2025-09-08 14:55:07 +08:00
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
GstRTSPMediaFactory* factory = gst_rtsp_media_factory_new();
|
2025-09-08 14:55:07 +08:00
|
|
|
|
gst_rtsp_media_factory_set_launch(factory, launch_str.c_str());
|
2025-11-25 08:54:26 +08:00
|
|
|
|
|
2025-12-17 15:40:48 +08:00
|
|
|
|
// 所有客户端共享同一个 pipeline,避免频繁拉起 v4l2 / encoder
|
2025-11-25 09:35:09 +08:00
|
|
|
|
gst_rtsp_media_factory_set_shared(factory, TRUE);
|
2025-11-24 17:36:32 +08:00
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
// 客户端断开时不要乱 suspend/reset
|
2025-11-24 17:36:32 +08:00
|
|
|
|
gst_rtsp_media_factory_set_suspend_mode(factory, GST_RTSP_SUSPEND_MODE_NONE);
|
2025-11-21 16:08:26 +08:00
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
g_signal_connect_data(factory, "media-configure", G_CALLBACK(on_media_created), g_strdup(cam.name.c_str()),
|
|
|
|
|
|
(GClosureNotify)g_free, (GConnectFlags)0);
|
2025-09-09 13:28:05 +08:00
|
|
|
|
|
2025-09-08 14:55:07 +08:00
|
|
|
|
return factory;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
void RTSPManager::start(const std::vector<Camera>& cams)
|
2025-09-08 14:55:07 +08:00
|
|
|
|
{
|
|
|
|
|
|
server = gst_rtsp_server_new();
|
|
|
|
|
|
gst_rtsp_server_set_service(server, "8554");
|
|
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
// ✅ attach 之前设置 backlog
|
2025-12-17 15:40:48 +08:00
|
|
|
|
gst_rtsp_server_set_backlog(server, 32);
|
|
|
|
|
|
|
2025-09-08 14:55:07 +08:00
|
|
|
|
loop = g_main_loop_new(nullptr, FALSE);
|
2025-09-09 11:02:41 +08:00
|
|
|
|
main_context = g_main_loop_get_context(loop);
|
2025-09-09 11:00:57 +08:00
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
GstRTSPMountPoints* mounts = gst_rtsp_server_get_mount_points(server);
|
2025-11-24 15:01:56 +08:00
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
for (const auto& cam : cams)
|
2025-11-24 15:01:56 +08:00
|
|
|
|
{
|
2025-12-17 14:13:52 +08:00
|
|
|
|
if (!cam.enabled) continue;
|
2025-11-24 15:01:56 +08:00
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
GstRTSPMediaFactory* factory = create_media_factory(cam);
|
2025-11-24 15:01:56 +08:00
|
|
|
|
std::string mount_point = "/" + cam.name;
|
|
|
|
|
|
|
|
|
|
|
|
gst_rtsp_mount_points_add_factory(mounts, mount_point.c_str(), factory);
|
|
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(mounted_factories_mutex);
|
|
|
|
|
|
mounted_factories[cam.name] = factory;
|
|
|
|
|
|
streaming_status[cam.name] = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
// 初始化计数
|
|
|
|
|
|
{
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(g_client_count_mutex);
|
|
|
|
|
|
g_client_count.emplace(cam.name, 0);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
LOG_INFO("[RTSP] Camera '" + cam.name + "' mounted at rtsp://0.0.0.0:8554" + mount_point);
|
2025-11-24 15:01:56 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
g_object_unref(mounts);
|
|
|
|
|
|
|
2025-09-08 14:55:07 +08:00
|
|
|
|
gst_rtsp_server_attach(server, nullptr);
|
|
|
|
|
|
|
2025-11-24 15:01:56 +08:00
|
|
|
|
LOG_INFO("[RTSP] Server running on rtsp://0.0.0.0:8554");
|
2025-09-08 14:55:07 +08:00
|
|
|
|
g_main_loop_run(loop);
|
2025-09-08 15:51:10 +08:00
|
|
|
|
|
|
|
|
|
|
if (server)
|
|
|
|
|
|
{
|
|
|
|
|
|
g_object_unref(server);
|
|
|
|
|
|
server = nullptr;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (loop)
|
|
|
|
|
|
{
|
|
|
|
|
|
g_main_loop_unref(loop);
|
|
|
|
|
|
loop = nullptr;
|
|
|
|
|
|
}
|
2025-09-09 13:38:23 +08:00
|
|
|
|
|
2025-09-08 15:51:10 +08:00
|
|
|
|
LOG_INFO("[RTSP] Server stopped.");
|
2025-09-08 14:55:07 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
void RTSPManager::on_media_created(GstRTSPMediaFactory*, GstRTSPMedia* media, gpointer user_data)
|
2025-09-09 13:28:05 +08:00
|
|
|
|
{
|
2025-12-17 14:13:52 +08:00
|
|
|
|
const char* cam_name = static_cast<const char*>(user_data);
|
2025-11-21 15:32:56 +08:00
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
// ✅ 只做日志降噪:别让“抑制”影响生命周期处理
|
|
|
|
|
|
bool noisy = false;
|
2025-12-17 15:40:48 +08:00
|
|
|
|
auto now = std::chrono::steady_clock::now();
|
|
|
|
|
|
{
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(last_media_ts_mutex);
|
|
|
|
|
|
auto& last = last_media_ts[cam_name];
|
2025-12-17 15:50:48 +08:00
|
|
|
|
if (last.time_since_epoch().count() != 0 && now - last < std::chrono::seconds(2)) noisy = true;
|
2025-12-17 15:40:48 +08:00
|
|
|
|
last = now;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
int count_after = 0;
|
2025-12-17 15:40:48 +08:00
|
|
|
|
{
|
2025-12-17 15:50:48 +08:00
|
|
|
|
std::lock_guard<std::mutex> lock(g_client_count_mutex);
|
|
|
|
|
|
int& c = g_client_count[cam_name]; // 若不存在会创建;这里无所谓
|
|
|
|
|
|
c++;
|
|
|
|
|
|
count_after = c;
|
2025-12-17 15:40:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
if (noisy)
|
2025-09-09 13:38:23 +08:00
|
|
|
|
{
|
2025-12-17 15:50:48 +08:00
|
|
|
|
LOG_WARN(std::string("[RTSP] media-configure frequent: ") + cam_name +
|
|
|
|
|
|
" (active_sessions=" + std::to_string(count_after) + ")");
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
LOG_INFO(std::string("[RTSP] media-configure for camera: ") + cam_name +
|
|
|
|
|
|
" (active_sessions=" + std::to_string(count_after) + ")");
|
2025-09-09 13:38:23 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
// ✅ 生命周期绑定:unprepared 一定会对应减计数
|
2025-12-17 14:13:52 +08:00
|
|
|
|
g_signal_connect_data(media, "unprepared", G_CALLBACK(on_media_unprepared), g_strdup(cam_name),
|
|
|
|
|
|
(GClosureNotify)g_free, (GConnectFlags)0);
|
2025-09-09 13:28:05 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
void RTSPManager::on_media_unprepared(GstRTSPMedia* /*media*/, gpointer user_data)
|
2025-09-09 13:28:05 +08:00
|
|
|
|
{
|
2025-12-17 14:13:52 +08:00
|
|
|
|
const char* cam_name = static_cast<const char*>(user_data);
|
2025-11-21 15:32:56 +08:00
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
int count_after = 0;
|
2025-12-17 14:13:52 +08:00
|
|
|
|
{
|
2025-12-17 15:50:48 +08:00
|
|
|
|
std::lock_guard<std::mutex> lock(g_client_count_mutex);
|
|
|
|
|
|
auto it = g_client_count.find(cam_name);
|
|
|
|
|
|
if (it == g_client_count.end())
|
|
|
|
|
|
{
|
|
|
|
|
|
// 可能发生在 unmount 后 teardown 仍在进行的边界情况
|
|
|
|
|
|
LOG_WARN(std::string("[RTSP] media-unprepared but cam not in client_count: ") + cam_name);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (it->second > 0) it->second--;
|
|
|
|
|
|
count_after = it->second;
|
2025-09-09 13:28:05 +08:00
|
|
|
|
}
|
2025-11-21 14:52:53 +08:00
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
LOG_INFO(std::string("[RTSP] media-unprepared: ") + cam_name + " (active_sessions=" + std::to_string(count_after) +
|
|
|
|
|
|
")");
|
|
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
// ✅ 不要 g_object_unref(media) —— gst-rtsp-server 会处理
|
2025-09-09 13:28:05 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-09 11:02:41 +08:00
|
|
|
|
gboolean RTSPManager::mount_camera_in_main(gpointer data)
|
2025-09-09 09:57:24 +08:00
|
|
|
|
{
|
2025-12-17 14:13:52 +08:00
|
|
|
|
Camera* cam = static_cast<Camera*>(data);
|
2025-09-09 11:02:41 +08:00
|
|
|
|
if (!cam || !server)
|
2025-09-09 10:30:50 +08:00
|
|
|
|
{
|
|
|
|
|
|
delete cam;
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
GstRTSPMountPoints* mounts = gst_rtsp_server_get_mount_points(server);
|
2025-09-09 10:30:50 +08:00
|
|
|
|
|
|
|
|
|
|
std::string mount_point = "/" + cam->name;
|
2025-12-17 14:13:52 +08:00
|
|
|
|
GstRTSPMediaFactory* factory = create_media_factory(*cam);
|
2025-09-09 09:57:24 +08:00
|
|
|
|
|
|
|
|
|
|
gst_rtsp_mount_points_add_factory(mounts, mount_point.c_str(), factory);
|
|
|
|
|
|
g_object_unref(mounts);
|
|
|
|
|
|
|
2025-09-09 10:30:50 +08:00
|
|
|
|
{
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(mounted_factories_mutex);
|
|
|
|
|
|
mounted_factories[cam->name] = factory;
|
2025-09-09 11:02:41 +08:00
|
|
|
|
streaming_status[cam->name] = true;
|
2025-09-09 10:30:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
{
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(g_client_count_mutex);
|
|
|
|
|
|
g_client_count[cam->name] = 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-09 10:30:50 +08:00
|
|
|
|
LOG_INFO("[RTSP] Camera '" + cam->name + "' mounted at rtsp://localhost:8554" + mount_point);
|
2025-11-21 15:32:56 +08:00
|
|
|
|
|
2025-09-09 10:30:50 +08:00
|
|
|
|
delete cam;
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
2025-09-09 09:57:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-09 11:02:41 +08:00
|
|
|
|
gboolean RTSPManager::unmount_camera_in_main(gpointer data)
|
2025-09-09 09:57:24 +08:00
|
|
|
|
{
|
2025-12-17 14:13:52 +08:00
|
|
|
|
Camera* cam = static_cast<Camera*>(data);
|
2025-09-09 11:02:41 +08:00
|
|
|
|
if (!cam || !server)
|
2025-09-09 10:30:50 +08:00
|
|
|
|
{
|
|
|
|
|
|
delete cam;
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
|
|
}
|
2025-11-21 15:32:56 +08:00
|
|
|
|
|
2025-09-09 13:28:05 +08:00
|
|
|
|
std::string cam_name = cam->name;
|
|
|
|
|
|
std::string mount_point = "/" + cam_name;
|
|
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
// ✅ 仅移除 mount;media teardown 交给 gst-rtsp-server
|
2025-12-17 14:13:52 +08:00
|
|
|
|
GstRTSPMountPoints* mounts = gst_rtsp_server_get_mount_points(server);
|
2025-09-09 11:34:17 +08:00
|
|
|
|
if (mounts)
|
2025-09-09 10:30:50 +08:00
|
|
|
|
{
|
2025-09-09 11:34:17 +08:00
|
|
|
|
gst_rtsp_mount_points_remove_factory(mounts, mount_point.c_str());
|
|
|
|
|
|
g_object_unref(mounts);
|
2025-09-09 10:30:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(mounted_factories_mutex);
|
2025-09-09 13:28:05 +08:00
|
|
|
|
auto it = mounted_factories.find(cam_name);
|
2025-09-09 10:30:50 +08:00
|
|
|
|
if (it != mounted_factories.end())
|
|
|
|
|
|
{
|
2025-12-17 15:50:48 +08:00
|
|
|
|
// 为了避免 double-unref:这里不手动 unref factory
|
2025-09-09 11:34:17 +08:00
|
|
|
|
mounted_factories.erase(it);
|
2025-09-09 10:30:50 +08:00
|
|
|
|
}
|
2025-09-09 13:28:05 +08:00
|
|
|
|
streaming_status[cam_name] = false;
|
2025-09-09 10:30:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 15:50:48 +08:00
|
|
|
|
{
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(g_client_count_mutex);
|
|
|
|
|
|
g_client_count.erase(cam_name);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-09 13:28:05 +08:00
|
|
|
|
LOG_INFO("[RTSP] Camera '" + cam_name + "' unmounted.");
|
2025-09-09 10:30:50 +08:00
|
|
|
|
delete cam;
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
|
|
}
|
2025-09-09 11:00:57 +08:00
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
void RTSPManager::mount_camera(const Camera& cam)
|
2025-09-09 11:00:57 +08:00
|
|
|
|
{
|
2025-12-17 14:13:52 +08:00
|
|
|
|
Camera* camCopy = new Camera(cam);
|
|
|
|
|
|
g_main_context_invoke(
|
|
|
|
|
|
main_context, [](gpointer data) -> gboolean { return RTSPManager::mount_camera_in_main(data); }, camCopy);
|
2025-09-09 11:00:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
void RTSPManager::unmount_camera(const Camera& cam)
|
2025-09-09 11:00:57 +08:00
|
|
|
|
{
|
2025-12-17 14:13:52 +08:00
|
|
|
|
Camera* camCopy = new Camera(cam);
|
|
|
|
|
|
g_main_context_invoke(
|
|
|
|
|
|
main_context, [](gpointer data) -> gboolean { return RTSPManager::unmount_camera_in_main(data); }, camCopy);
|
2025-09-09 11:00:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-17 14:13:52 +08:00
|
|
|
|
bool RTSPManager::is_streaming(const std::string& cam_name)
|
2025-09-09 11:00:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(mounted_factories_mutex);
|
|
|
|
|
|
auto it = streaming_status.find(cam_name);
|
|
|
|
|
|
return it != streaming_status.end() ? it->second : false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-11-21 15:32:56 +08:00
|
|
|
|
bool RTSPManager::is_any_streaming()
|
|
|
|
|
|
{
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(mounted_factories_mutex);
|
2025-12-17 14:13:52 +08:00
|
|
|
|
for (auto& kv : streaming_status)
|
|
|
|
|
|
if (kv.second) return true;
|
2025-11-21 15:32:56 +08:00
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-09 11:00:57 +08:00
|
|
|
|
void RTSPManager::stop()
|
|
|
|
|
|
{
|
2025-12-17 14:49:52 +08:00
|
|
|
|
if (!loop) return;
|
|
|
|
|
|
|
|
|
|
|
|
if (server) gst_rtsp_server_set_backlog(server, 0); // optional
|
|
|
|
|
|
|
|
|
|
|
|
if (main_context)
|
2025-09-09 11:00:57 +08:00
|
|
|
|
{
|
2025-11-20 11:06:02 +08:00
|
|
|
|
g_main_context_invoke(
|
|
|
|
|
|
main_context,
|
|
|
|
|
|
[](gpointer data) -> gboolean
|
|
|
|
|
|
{
|
2025-12-17 14:13:52 +08:00
|
|
|
|
g_main_loop_quit(static_cast<GMainLoop*>(data));
|
2025-11-20 11:06:02 +08:00
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
|
|
},
|
|
|
|
|
|
loop);
|
2025-09-09 11:00:57 +08:00
|
|
|
|
}
|
2025-12-17 14:49:52 +08:00
|
|
|
|
|
|
|
|
|
|
g_main_loop_quit(loop);
|
2025-09-09 11:00:57 +08:00
|
|
|
|
}
|