yituo_video/src/rtsp_manager.cpp

339 lines
9.5 KiB
C++
Raw Normal View History

2025-09-09 13:28:05 +08:00
// rtsp_manager.cpp
2025-09-08 14:55:07 +08:00
#include "rtsp_manager.hpp"
#include "logger.hpp"
2025-11-21 15:32:56 +08:00
2025-09-08 14:55:07 +08:00
#include <iostream>
2025-09-09 13:38:23 +08:00
#include <algorithm>
2025-09-10 13:13:51 +08:00
#include <thread>
#include <chrono>
2025-11-24 16:37:53 +08:00
#include <fcntl.h>
#include <sys/ioctl.h>
#include <linux/videodev2.h>
2025-09-08 14:55:07 +08:00
2025-09-09 09:57:24 +08:00
// 静态变量定义
2025-09-08 14:55:07 +08:00
GMainLoop *RTSPManager::loop = nullptr;
2025-09-09 11:02:41 +08:00
GMainContext *RTSPManager::main_context = nullptr;
2025-09-08 14:55:07 +08:00
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;
std::unordered_map<std::string, GstRTSPMediaFactory *> RTSPManager::mounted_factories;
std::mutex RTSPManager::mounted_factories_mutex;
2025-11-21 15:32:56 +08:00
2025-09-09 13:28:05 +08:00
std::unordered_map<std::string, std::vector<GstRTSPMedia *>> RTSPManager::media_map;
std::mutex RTSPManager::media_map_mutex;
2025-09-09 10:30:50 +08:00
2025-11-24 16:37:53 +08:00
bool set_v4l2_format(const std::string &dev, int width, int height)
{
int fd = open(dev.c_str(), O_RDWR);
if (fd < 0)
{
LOG_ERROR("Failed to open " + dev);
return false;
}
struct v4l2_format fmt;
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;
}
close(fd);
return true;
}
2025-09-08 14:55:07 +08:00
void RTSPManager::init()
{
gst_init(nullptr, nullptr);
LOG_INFO("[RTSP] GStreamer initialized.");
}
GstRTSPMediaFactory *RTSPManager::create_media_factory(const Camera &cam)
{
2025-11-24 17:36:32 +08:00
// 程序启动时设置一次 v4l2 格式
2025-11-24 16:37:53 +08:00
if (!set_v4l2_format(cam.device, cam.width, cam.height))
{
LOG_ERROR("[RTSP] Failed to set V4L2 format for " + cam.name);
}
2025-09-16 18:19:42 +08:00
int out_width = cam.width;
int out_height = cam.height;
2025-11-24 17:36:32 +08:00
// 先拼 caps方便看
std::string caps =
"video/x-raw,format=NV12,"
"width=" +
2025-11-24 15:57:49 +08:00
std::to_string(out_width) +
2025-09-22 09:10:17 +08:00
",height=" + std::to_string(out_height) +
2025-11-24 17:36:32 +08:00
",framerate=" + std::to_string(cam.fps) + "/1";
// 关键改动:
// 1) v4l2src 加 is-live / do-timestamp避免 preroll 卡死
// 2) queue 放宽限制,避免首帧被丢导致 pipeline 出错
// 3) suspend_mode 改为 NONE避免客户端断开时 RESET pipeline 导致反复 s_stream(0/1)
std::string launch_str =
"( v4l2src device=" + cam.device +
" io-mode=2 is-live=true do-timestamp=true"
" ! " +
caps +
" ! queue max-size-buffers=0 max-size-bytes=0 max-size-time=0"
" ! mpph264enc rc-mode=cbr"
" bps=" +
std::to_string(cam.bitrate) +
2025-11-21 16:23:56 +08:00
" gop=" + std::to_string(cam.fps) +
2025-11-21 16:28:35 +08:00
" header-mode=1"
" ! h264parse config-interval=1"
2025-11-21 16:23:56 +08:00
" ! rtph264pay name=pay0 pt=96 )";
2025-09-08 14:55:07 +08:00
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);
2025-11-24 17:36:32 +08:00
// 不再在客户端断开时 RESET / 销毁 pipeline避免频繁触发 MIPI / maxim4c 重新配置
gst_rtsp_media_factory_set_suspend_mode(factory, GST_RTSP_SUSPEND_MODE_NONE);
2025-11-21 16:08:26 +08:00
2025-11-21 16:23:56 +08:00
g_signal_connect_data(factory,
"media-configure",
2025-11-21 16:16:43 +08:00
G_CALLBACK(on_media_created),
2025-11-21 15:32:56 +08:00
g_strdup(cam.name.c_str()),
2025-11-21 16:23:56 +08:00
(GClosureNotify)g_free,
(GConnectFlags)0);
2025-09-09 13:28:05 +08:00
2025-09-08 14:55:07 +08:00
return factory;
}
2025-11-24 15:01:56 +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");
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-11-24 15:01:56 +08:00
// 先获取 mountpoints
GstRTSPMountPoints *mounts = gst_rtsp_server_get_mount_points(server);
// 在这里统一挂载所有 enabled 摄像头
for (const auto &cam : cams)
{
if (!cam.enabled)
continue;
GstRTSPMediaFactory *factory = create_media_factory(cam);
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;
}
LOG_INFO("[RTSP] Camera '" + cam.name +
"' mounted at rtsp://0.0.0.0:8554" + mount_point);
}
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-11-21 15:32:56 +08:00
void RTSPManager::on_media_created(GstRTSPMediaFactory *, GstRTSPMedia *media, gpointer user_data)
2025-09-09 13:28:05 +08:00
{
const char *cam_name = static_cast<const char *>(user_data);
2025-11-21 15:32:56 +08:00
LOG_INFO(std::string("[RTSP] media-configure for camera: ") + cam_name);
g_object_ref(media);
2025-09-09 13:38:23 +08:00
{
std::lock_guard<std::mutex> lock(media_map_mutex);
media_map[cam_name].push_back(media);
}
2025-09-09 13:33:37 +08:00
g_signal_connect_data(media, "unprepared", G_CALLBACK(on_media_unprepared),
2025-11-21 15:32:56 +08:00
g_strdup(cam_name),
(GClosureNotify)g_free,
(GConnectFlags)0);
2025-09-09 13:28:05 +08:00
}
void RTSPManager::on_media_unprepared(GstRTSPMedia *media, gpointer user_data)
{
const char *cam_name = static_cast<const char *>(user_data);
2025-11-21 15:32:56 +08:00
LOG_INFO(std::string("[RTSP] media-unprepared: ") + cam_name);
2025-09-09 13:28:05 +08:00
{
2025-09-09 13:38:23 +08:00
std::lock_guard<std::mutex> lock(media_map_mutex);
auto it = media_map.find(cam_name);
if (it != media_map.end())
2025-09-09 13:28:05 +08:00
{
2025-11-21 15:32:56 +08:00
auto &vec = it->second;
vec.erase(std::remove(vec.begin(), vec.end(), media), vec.end());
if (vec.empty())
2025-09-09 13:38:23 +08:00
media_map.erase(it);
2025-09-09 13:28:05 +08:00
}
}
2025-11-21 14:52:53 +08:00
2025-11-21 15:32:56 +08:00
g_object_unref(media);
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-09-09 10:30:50 +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-09-09 11:02:41 +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-09-09 11:02:41 +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
}
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-09-09 10:30:50 +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-11-21 15:32:56 +08:00
// 停掉 media
2025-09-09 13:38:23 +08:00
{
std::lock_guard<std::mutex> lock(media_map_mutex);
auto it = media_map.find(cam_name);
if (it != media_map.end())
{
for (GstRTSPMedia *media : it->second)
{
2025-11-20 09:10:39 +08:00
GstElement *pipeline = gst_rtsp_media_get_element(media);
if (pipeline)
{
2025-11-20 09:19:41 +08:00
gst_element_set_state(pipeline, GST_STATE_NULL);
2025-11-20 09:10:39 +08:00
gst_object_unref(pipeline);
}
2025-11-20 08:47:51 +08:00
gst_rtsp_media_unprepare(media);
2025-09-09 13:38:23 +08:00
}
2025-11-21 15:32:56 +08:00
it->second.clear();
media_map.erase(it);
2025-09-09 13:38:23 +08:00
}
}
2025-11-21 15:32:56 +08:00
// 移除 factory
2025-09-09 11:02:41 +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-11-21 15:32:56 +08:00
if (it->second)
2025-09-09 13:33:37 +08:00
g_object_unref(it->second);
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-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
void RTSPManager::mount_camera(const Camera &cam)
{
Camera *camCopy = new Camera(cam);
g_main_context_invoke(main_context, [](gpointer data) -> gboolean
2025-09-09 11:02:41 +08:00
{ return RTSPManager::mount_camera_in_main(data); }, camCopy);
2025-09-09 11:00:57 +08:00
}
void RTSPManager::unmount_camera(const Camera &cam)
{
Camera *camCopy = new Camera(cam);
g_main_context_invoke(main_context, [](gpointer data) -> gboolean
2025-09-09 11:02:41 +08:00
{ return RTSPManager::unmount_camera_in_main(data); }, camCopy);
2025-09-09 11:00:57 +08:00
}
bool RTSPManager::is_streaming(const std::string &cam_name)
{
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);
for (auto &kv : streaming_status)
if (kv.second)
return true;
return false;
}
2025-09-09 11:00:57 +08:00
void RTSPManager::stop()
{
if (loop)
{
2025-11-20 11:06:02 +08:00
g_main_context_invoke(
main_context,
[](gpointer data) -> gboolean
{
2025-11-21 15:32:56 +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
}
}