This commit is contained in:
		
							parent
							
								
									6c9387fece
								
							
						
					
					
						commit
						6b928428a7
					
				|  | @ -0,0 +1,51 @@ | |||
| cmake_minimum_required(VERSION 3.18) | ||||
| set(PROJECT_NAME TestDecode) | ||||
| project(${PROJECT_NAME}) | ||||
| message(STATUS "project name : ${PROJECT_NAME}") | ||||
| 
 | ||||
| set(CMAKE_CXX_STANDARD 17) | ||||
| set(CMAKE_CXX_STANDARD_REQUIRED ON) | ||||
| 
 | ||||
| #set(CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/cmake" ${CMAKE_MODULE_PATH}) | ||||
| 
 | ||||
| # 设置项目生成目录 | ||||
| set(EXECUTABLE_OUTPUT_PATH  ${CMAKE_CURRENT_SOURCE_DIR}/app) | ||||
| 
 | ||||
| set(OpenCV_DIR "/usr/local/opencv4.9") | ||||
| find_package(OpenCV REQUIRED PATHS ${OpenCV_DIR}) | ||||
| message(STATUS ${OpenCV_VERSION}) | ||||
| 
 | ||||
| include_directories(${OpenCV_DIR}) | ||||
| message(STATUS ${OpenCV_INCLUDE_DIRS}) | ||||
| 
 | ||||
| set(RTSP_DECODER_ROOT ${CMAKE_CURRENT_SOURCE_DIR}/RTSPDocker) | ||||
| 
 | ||||
| # 添加子项目 | ||||
| add_subdirectory(RTSPDecoder) | ||||
| 
 | ||||
| # 设置库搜索路径 | ||||
| set(LIB_DIR "${CMAKE_CURRENT_SOURCE_DIR}/app/lib") | ||||
| #message(STATUS ${LIB_DIR}) | ||||
| #find_library(RTSPDecoder_LIB RTSPDecoder | ||||
| #        PATHS ${LIB_DIR} | ||||
| #        NO_DEFAULT_PATH) | ||||
| 
 | ||||
| add_executable(${PROJECT_NAME} main.cpp) | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| #message(STATUS  ${RTSPDecoder_LIB}) | ||||
| 
 | ||||
| # 链接动态库 | ||||
| target_link_libraries(${PROJECT_NAME} | ||||
|         PRIVATE | ||||
| #        ${RTSPDecoder_LIB} | ||||
|         RTSPDecoder | ||||
|         ${OpenCV_LIBS} | ||||
| ) | ||||
| 
 | ||||
| # 包含子项目头文件 | ||||
| target_include_directories(${PROJECT_NAME} | ||||
|         PRIVATE | ||||
|         ${RTSP_DECODER_ROOT} | ||||
| ) | ||||
							
								
								
									
										29
									
								
								README.md
								
								
								
								
							
							
						
						
									
										29
									
								
								README.md
								
								
								
								
							|  | @ -1,3 +1,30 @@ | |||
| # RtspDecoderByFFmpeg | ||||
|  基于FFmpeg制作的多路视频RTSP拉流解码库 | ||||
| 
 | ||||
| ## 解码工具库 | ||||
| ### 目录结构 | ||||
| - app 编译结果 (动态库+调用demo) | ||||
|     - lib 编译出来的动态库 | ||||
| - cmake | ||||
| - RTSPDecoder 解码库源码 | ||||
| 
 | ||||
| ### 编译 | ||||
| #### 依赖库版本 | ||||
| - cmake 3.10+ | ||||
| - cuda 12.1 | ||||
| - cudnn | ||||
| - FFmpeg 4.4.5 | ||||
| - Opencv 4.9 | ||||
| ### 编译方式 | ||||
| ``` | ||||
| mkdir build && cd build | ||||
| 
 | ||||
| cmake .. | ||||
| 
 | ||||
| make -j10 | ||||
| 
 | ||||
| # 可选 | ||||
| make install | ||||
| ``` | ||||
| 
 | ||||
| 
 | ||||
| # 基于FFmpeg制作的多路视频RTSP拉流解码库 | ||||
|  | @ -0,0 +1,60 @@ | |||
| cmake_minimum_required(VERSION 3.18) | ||||
| set(PROJECT_NAME RTSPDecoder) | ||||
| project(${PROJECT_NAME}) | ||||
| 
 | ||||
| set(CMAKE_CXX_STANDARD 17) | ||||
| set(CMAKE_CXX_STANDARD_REQUIRED ON) | ||||
| 
 | ||||
| if(POLICY CMP0146) | ||||
|     cmake_policy(SET CMP0146 OLD) | ||||
| endif() | ||||
| 
 | ||||
| # 设置项目生成目录 | ||||
| set(EXECUTABLE_OUTPUT_PATH  ${CMAKE_CURRENT_SOURCE_DIR}/app/lib) | ||||
| 
 | ||||
| set(CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/cmake" ${CMAKE_MODULE_PATH}) | ||||
| 
 | ||||
| # 如果使用CUDA | ||||
| option(USE_CUDA "Enable CUDA support" ON) | ||||
| if(USE_CUDA) | ||||
|     find_package(CUDA REQUIRED) | ||||
| endif() | ||||
| 
 | ||||
| # 查找依赖库 | ||||
| find_package(OpenCV REQUIRED) | ||||
| find_package(FFmpeg REQUIRED) | ||||
| 
 | ||||
| 
 | ||||
| # 添加动态库 | ||||
| add_library(${PROJECT_NAME} SHARED | ||||
|         RTSPDecoder.cpp | ||||
|         RTSPDecoder.h | ||||
| ) | ||||
| 
 | ||||
| target_include_directories(${PROJECT_NAME} PRIVATE | ||||
|         ${OpenCV_INCLUDE_DIRS} | ||||
|         ${FFmpeg_INCLUDE_DIRS} | ||||
| ) | ||||
| 
 | ||||
| target_link_libraries(${PROJECT_NAME} PRIVATE | ||||
|         ${OpenCV_LIBS} | ||||
| #        ${FFmpeg_LIBRARIES} | ||||
|         avutil avcodec avformat avdevice avfilter swscale swresample | ||||
| ) | ||||
| 
 | ||||
| if(USE_CUDA AND CUDA_FOUND) | ||||
|     target_include_directories(${PROJECT_NAME} PRIVATE ${CUDA_INCLUDE_DIRS}) | ||||
|     target_link_libraries(${PROJECT_NAME} PRIVATE ${CUDA_LIBRARIES}) | ||||
|     target_compile_definitions(${PROJECT_NAME} PRIVATE USE_CUDA_ACCEL) | ||||
| endif() | ||||
| 
 | ||||
| # 安装规则 | ||||
| install(TARGETS ${PROJECT_NAME} | ||||
|         LIBRARY DESTINATION lib | ||||
|         ARCHIVE DESTINATION lib | ||||
| ) | ||||
| 
 | ||||
| install(FILES RTSPDecoder.h | ||||
|         DESTINATION include | ||||
| ) | ||||
| 
 | ||||
|  | @ -0,0 +1,538 @@ | |||
| #include "RTSPDecoder.h" | ||||
| #include <thread> | ||||
| #include <queue> | ||||
| #include <condition_variable> | ||||
| #include <stdexcept> | ||||
| #include <cuda_runtime_api.h> | ||||
| 
 | ||||
| #define MAX_RETRY_COUNT 5 | ||||
| #define RETRY_DELAY_MS 1000 | ||||
| 
 | ||||
| RTSPDecoder::RTSPDecoder() { | ||||
|     avformat_network_init(); | ||||
| } | ||||
| 
 | ||||
| RTSPDecoder::~RTSPDecoder() { | ||||
|     // 停止所有解码线程
 | ||||
|     for (auto& stream : streams_) { | ||||
|         if (stream && stream->running) { | ||||
|             stream->running = false; | ||||
|             if (stream->decode_thread.joinable()) { | ||||
|                 stream->decode_thread.join(); | ||||
|             } | ||||
|         } | ||||
|         cleanupStream(stream.get()); | ||||
|     } | ||||
| 
 | ||||
|     avformat_network_deinit(); | ||||
| } | ||||
| 
 | ||||
| bool RTSPDecoder::init(const DecoderConfig& config) { | ||||
|     if (initialized_) return true; | ||||
| 
 | ||||
|     config_ = config; | ||||
| 
 | ||||
|     // 验证配置
 | ||||
|     if (config_.max_streams < 1 || config_.max_streams > 60) { | ||||
|         throw std::invalid_argument("Max streams must be between 1 and 60"); | ||||
|     } | ||||
| 
 | ||||
|     // 初始化GPU分配
 | ||||
|     if (config_.use_hw_accel) { | ||||
|         int gpu_count = getGPUCount(); | ||||
|         if (gpu_count == 0) { | ||||
|             config_.use_hw_accel = false; | ||||
|         } else if (config_.gpu_id >= 0 && config_.gpu_id < gpu_count) { | ||||
|             next_gpu_id_ = config_.gpu_id; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     initialized_ = true; | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| int RTSPDecoder::addStream(const std::string& rtsp_url) { | ||||
|     if (!initialized_) { | ||||
|         throw std::runtime_error("Decoder not initialized"); | ||||
|     } | ||||
| 
 | ||||
|     std::lock_guard<std::mutex> lock(streams_mutex_); | ||||
| 
 | ||||
|     if (streams_.size() >= config_.max_streams) { | ||||
|         throw std::runtime_error("Maximum number of streams reached"); | ||||
|     } | ||||
| 
 | ||||
|     auto ctx = std::make_unique<StreamContext>(); | ||||
|     int stream_id = static_cast<int>(streams_.size()); | ||||
| 
 | ||||
|     // 分配GPU
 | ||||
|     int gpu_id = config_.use_hw_accel ? allocateGPU() : -1; | ||||
| 
 | ||||
|     // 初始化格式上下文
 | ||||
|     ctx->format_ctx = avformat_alloc_context(); | ||||
|     if (!ctx->format_ctx) { | ||||
|         throw std::runtime_error("Failed to allocate format context"); | ||||
|     } | ||||
| 
 | ||||
|     // 设置RTSP参数
 | ||||
|     AVDictionary* options = nullptr; | ||||
|     av_dict_set(&options, "rtsp_transport", "tcp", 0); | ||||
|     av_dict_set(&options, "stimeout", "5000000", 0); // 5秒超时
 | ||||
| 
 | ||||
|     // 打开输入流
 | ||||
|     int retry_count = 0; | ||||
|     while (retry_count < MAX_RETRY_COUNT) { | ||||
|         int ret = avformat_open_input(&ctx->format_ctx, rtsp_url.c_str(), nullptr, &options); | ||||
|         if (ret == 0) break; | ||||
| 
 | ||||
|         retry_count++; | ||||
|         if (retry_count < MAX_RETRY_COUNT) { | ||||
|             std::this_thread::sleep_for(std::chrono::milliseconds(RETRY_DELAY_MS)); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     av_dict_free(&options); | ||||
| 
 | ||||
|     if (retry_count == MAX_RETRY_COUNT) { | ||||
|         cleanupStream(ctx.get()); | ||||
|         throw std::runtime_error("Failed to open RTSP stream after retries"); | ||||
|     } | ||||
| 
 | ||||
|     // 查找流信息
 | ||||
|     if (avformat_find_stream_info(ctx->format_ctx, nullptr) < 0) { | ||||
|         cleanupStream(ctx.get()); | ||||
|         throw std::runtime_error("Failed to find stream information"); | ||||
|     } | ||||
| 
 | ||||
|     // 查找视频流
 | ||||
|     for (unsigned int i = 0; i < ctx->format_ctx->nb_streams; i++) { | ||||
|         if (ctx->format_ctx->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) { | ||||
|             ctx->video_stream_idx = i; | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     if (ctx->video_stream_idx == -1) { | ||||
|         cleanupStream(ctx.get()); | ||||
|         throw std::runtime_error("No video stream found"); | ||||
|     } | ||||
| 
 | ||||
|     // 获取解码器参数
 | ||||
|     AVCodecParameters* codecpar = ctx->format_ctx->streams[ctx->video_stream_idx]->codecpar; | ||||
| 
 | ||||
|     // 查找解码器
 | ||||
|     const AVCodec* codec = nullptr; | ||||
|     if (config_.use_hw_accel && gpu_id >= 0) { | ||||
|         codec = avcodec_find_decoder(codecpar->codec_id); | ||||
|     } else { | ||||
|         codec = avcodec_find_decoder(codecpar->codec_id); | ||||
|     } | ||||
| 
 | ||||
|     if (!codec) { | ||||
|         cleanupStream(ctx.get()); | ||||
|         throw std::runtime_error("Unsupported codec"); | ||||
|     } | ||||
| 
 | ||||
|     // 创建解码器上下文
 | ||||
|     ctx->codec_ctx = avcodec_alloc_context3(codec); | ||||
|     if (!ctx->codec_ctx) { | ||||
|         cleanupStream(ctx.get()); | ||||
|         throw std::runtime_error("Failed to allocate codec context"); | ||||
|     } | ||||
| 
 | ||||
|     // 复制参数到解码器上下文
 | ||||
|     if (avcodec_parameters_to_context(ctx->codec_ctx, codecpar) < 0) { | ||||
|         cleanupStream(ctx.get()); | ||||
|         throw std::runtime_error("Failed to copy codec parameters"); | ||||
|     } | ||||
| 
 | ||||
|     // 初始化硬件加速
 | ||||
|     if (config_.use_hw_accel && gpu_id >= 0) { | ||||
|         if (!initHWAccel(*ctx, gpu_id)) { | ||||
|             cleanupStream(ctx.get()); | ||||
|             throw std::runtime_error("Failed to initialize hardware acceleration"); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // 打开解码器
 | ||||
|     if (avcodec_open2(ctx->codec_ctx, codec, nullptr) < 0) { | ||||
|         cleanupStream(ctx.get()); | ||||
|         throw std::runtime_error("Failed to open codec"); | ||||
|     } | ||||
| 
 | ||||
|     // 设置线程数 (根据CPU核心数自动调整)
 | ||||
|     ctx->codec_ctx->thread_count = std::thread::hardware_concurrency(); | ||||
|     if (ctx->codec_ctx->thread_count == 0) { | ||||
|         ctx->codec_ctx->thread_count = 4; // 默认值
 | ||||
|     } | ||||
| 
 | ||||
|     // 启动解码线程
 | ||||
|     ctx->gpu_id = gpu_id; | ||||
|     ctx->running = true; | ||||
|     ctx->decode_thread = std::thread(&RTSPDecoder::decodeThreadFunc, this, stream_id, ctx.get()); | ||||
| 
 | ||||
|     streams_.push_back(std::move(ctx)); | ||||
|     return stream_id; | ||||
| } | ||||
| 
 | ||||
| bool RTSPDecoder::removeStream(int stream_id) { | ||||
|     std::lock_guard<std::mutex> lock(streams_mutex_); | ||||
| 
 | ||||
|     if (stream_id < 0 || stream_id >= streams_.size() || !streams_[stream_id]) { | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     auto& ctx = streams_[stream_id]; | ||||
| 
 | ||||
|     // 停止解码线程
 | ||||
|     ctx->running = false; | ||||
|     if (ctx->decode_thread.joinable()) { | ||||
|         ctx->decode_thread.join(); | ||||
|     } | ||||
| 
 | ||||
|     // 清理资源
 | ||||
|     cleanupStream(ctx.get()); | ||||
| 
 | ||||
|     // 从列表中移除
 | ||||
|     streams_[stream_id].reset(); | ||||
| 
 | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| bool RTSPDecoder::getFrame(int stream_id, cv::Mat& frame, int timeout_ms) { | ||||
|     if (stream_id < 0 || stream_id >= streams_.size() || !streams_[stream_id]) { | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     auto& ctx = streams_[stream_id]; | ||||
|     std::unique_lock<std::mutex> lock(ctx->mutex); | ||||
| 
 | ||||
|     if (ctx->frame_queue.empty()) { | ||||
|         if (timeout_ms <= 0) { | ||||
|             return false; | ||||
|         } | ||||
| 
 | ||||
|         if (ctx->cond.wait_for(lock, std::chrono::milliseconds(timeout_ms)) == std::cv_status::timeout) { | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     if (ctx->frame_queue.empty()) { | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     frame = ctx->frame_queue.front(); | ||||
|     ctx->frame_queue.pop(); | ||||
| 
 | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| int RTSPDecoder::getActiveStreamCount() const { | ||||
|     int count = 0; | ||||
|     for (const auto& stream : streams_) { | ||||
|         if (stream && stream->running) { | ||||
|             count++; | ||||
|         } | ||||
|     } | ||||
|     return count; | ||||
| } | ||||
| 
 | ||||
| int RTSPDecoder::getGPUCount() { | ||||
|     int count = 0; | ||||
|     cudaError_t err = cudaGetDeviceCount(&count); | ||||
|     if (err != cudaSuccess) { | ||||
|         return 0; | ||||
|     } | ||||
|     return count; | ||||
| } | ||||
| 
 | ||||
| bool RTSPDecoder::initHWAccel(StreamContext& ctx, int gpu_id) { | ||||
|     AVBufferRef* hw_device_ctx = nullptr; | ||||
| 
 | ||||
|     // 创建CUDA设备上下文
 | ||||
|     char device_name[32]; | ||||
|     snprintf(device_name, sizeof(device_name), "%d", gpu_id); | ||||
| 
 | ||||
|     if (av_hwdevice_ctx_create(&hw_device_ctx, AV_HWDEVICE_TYPE_CUDA, device_name, nullptr, 0) < 0) { | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     ctx.hw_device_ctx = av_buffer_ref(hw_device_ctx); | ||||
|     ctx.codec_ctx->hw_device_ctx = av_buffer_ref(hw_device_ctx); | ||||
|     av_buffer_unref(&hw_device_ctx); | ||||
| 
 | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| void RTSPDecoder::decodeThreadFunc(int stream_id, StreamContext* ctx) { | ||||
|     AVFrame* frame = av_frame_alloc(); | ||||
|     AVFrame* sw_frame = nullptr; | ||||
|     AVPacket* pkt = av_packet_alloc(); | ||||
| 
 | ||||
|     if (!frame || !pkt) { | ||||
|         ctx->running = false; | ||||
|         if (frame) av_frame_free(&frame); | ||||
|         if (pkt) av_packet_free(&pkt); | ||||
|         return; | ||||
|     } | ||||
| 
 | ||||
|     // 如果使用硬件加速,准备软件帧
 | ||||
|     if (ctx->hw_device_ctx) { | ||||
|         sw_frame = av_frame_alloc(); | ||||
|         if (!sw_frame) { | ||||
|             ctx->running = false; | ||||
|             av_frame_free(&frame); | ||||
|             av_packet_free(&pkt); | ||||
|             return; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // 解码循环
 | ||||
|     while (ctx->running) { | ||||
|         // 读取数据包
 | ||||
|         int ret = av_read_frame(ctx->format_ctx, pkt); | ||||
|         if (ret < 0) { | ||||
|             if (ret == AVERROR(EAGAIN)) { | ||||
|                 continue; | ||||
|             } | ||||
| 
 | ||||
|             // 处理错误或EOF
 | ||||
|             if (ret != AVERROR_EOF) { | ||||
|                 // 重试逻辑
 | ||||
|                 int retry_count = 0; | ||||
|                 while (retry_count < MAX_RETRY_COUNT && ctx->running) { | ||||
|                     avformat_close_input(&ctx->format_ctx); | ||||
|                     AVDictionary* options = nullptr; | ||||
|                     av_dict_set(&options, "rtsp_transport", "tcp", 0); | ||||
|                     av_dict_set(&options, "stimeout", "5000000", 0); | ||||
| 
 | ||||
|                     ret = avformat_open_input(&ctx->format_ctx, ctx->format_ctx->url, nullptr, &options); | ||||
|                     av_dict_free(&options); | ||||
| 
 | ||||
|                     if (ret == 0) { | ||||
|                         avformat_find_stream_info(ctx->format_ctx, nullptr); | ||||
|                         break; | ||||
|                     } | ||||
| 
 | ||||
|                     retry_count++; | ||||
|                     std::this_thread::sleep_for(std::chrono::milliseconds(RETRY_DELAY_MS)); | ||||
|                 } | ||||
| 
 | ||||
|                 if (retry_count == MAX_RETRY_COUNT) { | ||||
|                     ctx->running = false; | ||||
|                     break; | ||||
|                 } | ||||
|             } else { | ||||
|                 // EOF, 重新开始
 | ||||
|                 av_seek_frame(ctx->format_ctx, ctx->video_stream_idx, 0, AVSEEK_FLAG_BACKWARD); | ||||
|             } | ||||
| 
 | ||||
|             av_packet_unref(pkt); | ||||
|             continue; | ||||
|         } | ||||
| 
 | ||||
|         // 检查是否是视频流
 | ||||
|         if (pkt->stream_index != ctx->video_stream_idx) { | ||||
|             av_packet_unref(pkt); | ||||
|             continue; | ||||
|         } | ||||
| 
 | ||||
|         // 发送数据包到解码器
 | ||||
|         ret = avcodec_send_packet(ctx->codec_ctx, pkt); | ||||
|         av_packet_unref(pkt); | ||||
| 
 | ||||
|         if (ret < 0 && ret != AVERROR(EAGAIN)) { | ||||
|             continue; | ||||
|         } | ||||
| 
 | ||||
|         // 接收解码后的帧
 | ||||
|         while (ctx->running) { | ||||
|             ret = avcodec_receive_frame(ctx->codec_ctx, frame); | ||||
|             if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF) { | ||||
|                 break; | ||||
|             } | ||||
| 
 | ||||
|             if (ret < 0) { | ||||
|                 break; | ||||
|             } | ||||
| 
 | ||||
|             AVFrame* output_frame = frame; | ||||
| 
 | ||||
|             // 如果使用硬件加速,需要将帧从GPU内存复制到CPU内存
 | ||||
|             if (ctx->hw_device_ctx) { | ||||
|                 if (av_hwframe_transfer_data(sw_frame, frame, 0) < 0) { | ||||
|                     av_frame_unref(frame); | ||||
|                     break; | ||||
|                 } | ||||
| 
 | ||||
|                 output_frame = sw_frame; | ||||
|             } | ||||
| 
 | ||||
|             // 转换为OpenCV Mat
 | ||||
|             int width = config_.width > 0 ? config_.width : output_frame->width; | ||||
|             int height = config_.height > 0 ? config_.height : output_frame->height; | ||||
| 
 | ||||
|             // 检查是否需要缩放
 | ||||
|             if (width != output_frame->width || height != output_frame->height || | ||||
|                 output_frame->format != AV_PIX_FMT_BGR24) { | ||||
|                 if (!ctx->sws_ctx) { | ||||
|                     ctx->sws_ctx = sws_getContext( | ||||
|                         output_frame->width, output_frame->height, | ||||
|                         static_cast<AVPixelFormat>(output_frame->format), | ||||
|                         width, height, AV_PIX_FMT_BGR24, | ||||
|                         SWS_BILINEAR, nullptr, nullptr, nullptr); | ||||
| 
 | ||||
|                     if (!ctx->sws_ctx) { | ||||
|                         av_frame_unref(output_frame); | ||||
|                         break; | ||||
|                     } | ||||
|                 } | ||||
| 
 | ||||
|                 // 创建目标帧
 | ||||
|                 AVFrame* bgr_frame = av_frame_alloc(); | ||||
|                 if (!bgr_frame) { | ||||
|                     av_frame_unref(output_frame); | ||||
|                     break; | ||||
|                 } | ||||
| 
 | ||||
|                 bgr_frame->format = AV_PIX_FMT_BGR24; | ||||
|                 bgr_frame->width = width; | ||||
|                 bgr_frame->height = height; | ||||
| 
 | ||||
|                 if (av_frame_get_buffer(bgr_frame, 0) < 0) { | ||||
|                     av_frame_free(&bgr_frame); | ||||
|                     av_frame_unref(output_frame); | ||||
|                     break; | ||||
|                 } | ||||
| 
 | ||||
|                 // 执行转换
 | ||||
|                 sws_scale(ctx->sws_ctx, | ||||
|                          output_frame->data, output_frame->linesize, 0, output_frame->height, | ||||
|                          bgr_frame->data, bgr_frame->linesize); | ||||
| 
 | ||||
|                 // 创建OpenCV Mat
 | ||||
|                 cv::Mat mat(height, width, CV_8UC3, bgr_frame->data[0], bgr_frame->linesize[0]); | ||||
| 
 | ||||
|                 // 复制数据 (因为bgr_frame会被释放)
 | ||||
|                 cv::Mat mat_copy = mat.clone(); | ||||
| 
 | ||||
|                 av_frame_free(&bgr_frame); | ||||
| 
 | ||||
|                                 // 将帧添加到队列
 | ||||
|                 { | ||||
|                     std::lock_guard<std::mutex> lock(ctx->mutex); | ||||
|                     if (ctx->frame_queue.size() >= config_.buffer_size) { | ||||
|                         ctx->frame_queue.pop(); | ||||
|                     } | ||||
|                     ctx->frame_queue.push(mat_copy); | ||||
|                     ctx->cond.notify_one(); | ||||
|                 } | ||||
|             } else { | ||||
|                 // 直接转换格式
 | ||||
|                 cv::Mat mat(output_frame->height, output_frame->width, CV_8UC3); | ||||
| 
 | ||||
|                 // 根据像素格式处理
 | ||||
|                 if (output_frame->format == AV_PIX_FMT_NV12) { | ||||
|                     // NV12转BGR
 | ||||
|                     cv::Mat y_plane(output_frame->height, output_frame->width, CV_8UC1, output_frame->data[0]); | ||||
|                     cv::Mat uv_plane(output_frame->height / 2, output_frame->width / 2, CV_8UC2, output_frame->data[1]); | ||||
|                     cv::cvtColorTwoPlane(y_plane, uv_plane, mat, cv::COLOR_YUV2BGR_NV12); | ||||
|                 } else if (output_frame->format == AV_PIX_FMT_YUV420P) { | ||||
|                     // YUV420P转BGR
 | ||||
|                     cv::Mat yuv_mat(output_frame->height * 3 / 2, output_frame->width, CV_8UC1, output_frame->data[0]); | ||||
|                     cv::cvtColor(yuv_mat, mat, cv::COLOR_YUV2BGR_I420); | ||||
|                 } else if (output_frame->format == AV_PIX_FMT_BGR24) { | ||||
|                     // 直接复制
 | ||||
|                     cv::Mat(output_frame->height, output_frame->width, CV_8UC3, output_frame->data[0]).copyTo(mat); | ||||
|                 } else { | ||||
|                     // 其他格式使用SWS转换
 | ||||
|                     if (!ctx->sws_ctx) { | ||||
|                         ctx->sws_ctx = sws_getContext( | ||||
|                             output_frame->width, output_frame->height, | ||||
|                             static_cast<AVPixelFormat>(output_frame->format), | ||||
|                             width, height, AV_PIX_FMT_BGR24, | ||||
|                             SWS_BILINEAR, nullptr, nullptr, nullptr); | ||||
|                     } | ||||
| 
 | ||||
|                     if (ctx->sws_ctx) { | ||||
|                         AVFrame* bgr_frame = av_frame_alloc(); | ||||
|                         bgr_frame->format = AV_PIX_FMT_BGR24; | ||||
|                         bgr_frame->width = width; | ||||
|                         bgr_frame->height = height; | ||||
| 
 | ||||
|                         if (av_frame_get_buffer(bgr_frame, 0) == 0) { | ||||
|                             sws_scale(ctx->sws_ctx, | ||||
|                                      output_frame->data, output_frame->linesize, 0, output_frame->height, | ||||
|                                      bgr_frame->data, bgr_frame->linesize); | ||||
| 
 | ||||
|                             mat = cv::Mat(height, width, CV_8UC3, bgr_frame->data[0]); | ||||
|                         } | ||||
| 
 | ||||
|                         av_frame_free(&bgr_frame); | ||||
|                     } | ||||
|                 } | ||||
| 
 | ||||
|                 // 将帧添加到队列
 | ||||
|                 if (!mat.empty()) { | ||||
|                     std::lock_guard<std::mutex> lock(ctx->mutex); | ||||
|                     if (ctx->frame_queue.size() >= config_.buffer_size) { | ||||
|                         ctx->frame_queue.pop(); | ||||
|                     } | ||||
|                     ctx->frame_queue.push(mat); | ||||
|                     ctx->cond.notify_one(); | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             av_frame_unref(output_frame); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // 清理资源
 | ||||
|     av_frame_free(&frame); | ||||
|     av_frame_free(&sw_frame); | ||||
|     av_packet_free(&pkt); | ||||
| 
 | ||||
|     if (ctx->sws_ctx) { | ||||
|         sws_freeContext(ctx->sws_ctx); | ||||
|         ctx->sws_ctx = nullptr; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void RTSPDecoder::cleanupStream(StreamContext* ctx) { | ||||
|     if (!ctx) return; | ||||
| 
 | ||||
|     if (ctx->sws_ctx) { | ||||
|         sws_freeContext(ctx->sws_ctx); | ||||
|         ctx->sws_ctx = nullptr; | ||||
|     } | ||||
| 
 | ||||
|     if (ctx->codec_ctx) { | ||||
|         avcodec_free_context(&ctx->codec_ctx); | ||||
|     } | ||||
| 
 | ||||
|     if (ctx->format_ctx) { | ||||
|         avformat_close_input(&ctx->format_ctx); | ||||
|     } | ||||
| 
 | ||||
|     if (ctx->hw_device_ctx) { | ||||
|         av_buffer_unref(&ctx->hw_device_ctx); | ||||
|     } | ||||
| 
 | ||||
|     // 清空帧队列
 | ||||
|     std::lock_guard<std::mutex> lock(ctx->mutex); | ||||
|     while (!ctx->frame_queue.empty()) { | ||||
|         ctx->frame_queue.pop(); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| int RTSPDecoder::allocateGPU() { | ||||
|     if (!config_.use_hw_accel) return -1; | ||||
| 
 | ||||
|     int gpu_count = getGPUCount(); | ||||
|     if (gpu_count == 0) return -1; | ||||
| 
 | ||||
|     // 简单的轮询分配策略
 | ||||
|     int gpu_id = next_gpu_id_++ % gpu_count; | ||||
|     return gpu_id; | ||||
| } | ||||
| 
 | ||||
|  | @ -0,0 +1,91 @@ | |||
| #ifndef RTSP_DECODER_H | ||||
| #define RTSP_DECODER_H | ||||
| 
 | ||||
| #include <vector> | ||||
| #include <string> | ||||
| #include <memory> | ||||
| #include <mutex> | ||||
| #include <atomic> | ||||
| #include <condition_variable> | ||||
| #include <thread> | ||||
| #include <opencv2/opencv.hpp> | ||||
| 
 | ||||
| extern "C" { | ||||
| #include <libavcodec/avcodec.h> | ||||
| #include <libavformat/avformat.h> | ||||
| #include <libavutil/hwcontext.h> | ||||
| // #include <libavutil/hwcontext_cuda.h>
 | ||||
| #include <libswscale/swscale.h> | ||||
| } | ||||
| 
 | ||||
| class RTSPDecoder { | ||||
| public: | ||||
|     struct DecoderConfig { | ||||
|         int max_streams = 60;          // 最大支持流数
 | ||||
|         int gpu_id = -1;               // 指定GPU ID (-1表示自动分配)
 | ||||
|         int width = 0;                 // 输出宽度 (0表示保持原始)
 | ||||
|         int height = 0;                // 输出高度 (0表示保持原始)
 | ||||
|         int fps = 0;                   // 目标帧率 (0表示保持原始)
 | ||||
|         bool use_hw_accel = true;      // 是否使用硬件加速
 | ||||
|         int buffer_size = 1024;          // 帧缓冲队列大小
 | ||||
|     }; | ||||
| 
 | ||||
|     // 构造函数
 | ||||
|     RTSPDecoder(); | ||||
| 
 | ||||
|     // 析构函数
 | ||||
|     ~RTSPDecoder(); | ||||
| 
 | ||||
|     // 初始化解码器
 | ||||
|     bool init(const DecoderConfig& config); | ||||
| 
 | ||||
|     // 添加RTSP流
 | ||||
|     int addStream(const std::string& rtsp_url); | ||||
| 
 | ||||
|     // 移除RTSP流
 | ||||
|     bool removeStream(int stream_id); | ||||
| 
 | ||||
|     // 获取解码后的帧
 | ||||
|     bool getFrame(int stream_id, cv::Mat& frame, int timeout_ms = 1000); | ||||
| 
 | ||||
|     // 获取当前活跃的流数量
 | ||||
|     int getActiveStreamCount() const; | ||||
| 
 | ||||
|     // 获取GPU数量
 | ||||
|     static int getGPUCount(); | ||||
| 
 | ||||
| private: | ||||
|     struct StreamContext { | ||||
|         AVFormatContext* format_ctx = nullptr; | ||||
|         AVCodecContext* codec_ctx = nullptr; | ||||
|         AVBufferRef* hw_device_ctx = nullptr; | ||||
|         SwsContext* sws_ctx = nullptr; | ||||
|         int video_stream_idx = -1; | ||||
|         int gpu_id = 0; | ||||
|         std::atomic<bool> running{false}; | ||||
|         std::mutex mutex; | ||||
|         std::condition_variable cond; | ||||
|         std::queue<cv::Mat> frame_queue; | ||||
|         std::thread decode_thread; | ||||
|     }; | ||||
| 
 | ||||
|     DecoderConfig config_; | ||||
|     std::vector<std::unique_ptr<StreamContext>> streams_; | ||||
|     std::mutex streams_mutex_; | ||||
|     std::atomic<int> next_gpu_id_{0}; | ||||
|     std::atomic<bool> initialized_{false}; | ||||
| 
 | ||||
|     // 初始化硬件加速
 | ||||
|     bool initHWAccel(StreamContext& ctx, int gpu_id); | ||||
| 
 | ||||
|     // 解码线程函数
 | ||||
|     void decodeThreadFunc(int stream_id, StreamContext* ctx); | ||||
| 
 | ||||
|     // 清理流上下文
 | ||||
|     void cleanupStream(StreamContext* ctx); | ||||
| 
 | ||||
|     // 分配GPU
 | ||||
|     int allocateGPU(); | ||||
| }; | ||||
| 
 | ||||
| #endif // RTSP_DECODER_H
 | ||||
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							|  | @ -0,0 +1,146 @@ | |||
| # vim: ts=2 sw=2 | ||||
| # - Try to find the required ffmpeg components(default: AVFORMAT, AVUTIL, AVCODEC) | ||||
| # | ||||
| # Once done this will define | ||||
| #  FFMPEG_FOUND         - System has the all required components. | ||||
| #  FFMPEG_INCLUDE_DIRS  - Include directory necessary for using the required components headers. | ||||
| #  FFMPEG_LIBRARIES     - Link these to use the required ffmpeg components. | ||||
| #  FFMPEG_DEFINITIONS   - Compiler switches required for using the required ffmpeg components. | ||||
| # | ||||
| # For each of the components it will additionaly set. | ||||
| #   - AVCODEC | ||||
| #   - AVDEVICE | ||||
| #   - AVFORMAT | ||||
| #   - AVUTIL | ||||
| #   - POSTPROCESS | ||||
| #   - SWSCALE | ||||
| # the following variables will be defined | ||||
| #  <component>_FOUND        - System has <component> | ||||
| #  <component>_INCLUDE_DIRS - Include directory necessary for using the <component> headers | ||||
| #  <component>_LIBRARIES    - Link these to use <component> | ||||
| #  <component>_DEFINITIONS  - Compiler switches required for using <component> | ||||
| #  <component>_VERSION      - The components version | ||||
| # | ||||
| # Copyright (c) 2006, Matthias Kretz, <kretz@kde.org> | ||||
| # Copyright (c) 2008, Alexander Neundorf, <neundorf@kde.org> | ||||
| # Copyright (c) 2011, Michael Jansen, <kde@michael-jansen.biz> | ||||
| # | ||||
| # Redistribution and use is allowed according to the terms of the BSD license. | ||||
| # For details see the accompanying COPYING-CMAKE-SCRIPTS file. | ||||
| 
 | ||||
| include(FindPackageHandleStandardArgs) | ||||
| 
 | ||||
| # The default components were taken from a survey over other FindFFMPEG.cmake files | ||||
| if (NOT FFmpeg_FIND_COMPONENTS) | ||||
|     set(FFmpeg_FIND_COMPONENTS AVCODEC AVFORMAT AVUTIL) | ||||
| endif () | ||||
| 
 | ||||
| # | ||||
| ### Macro: set_component_found | ||||
| # | ||||
| # Marks the given component as found if both *_LIBRARIES AND *_INCLUDE_DIRS is present. | ||||
| # | ||||
| macro(set_component_found _component ) | ||||
|     if (${_component}_LIBRARIES AND ${_component}_INCLUDE_DIRS) | ||||
|         # message(STATUS "  - ${_component} found.") | ||||
|         set(${_component}_FOUND TRUE) | ||||
|     else () | ||||
|         # message(STATUS "  - ${_component} not found.") | ||||
|     endif () | ||||
| endmacro() | ||||
| 
 | ||||
| # | ||||
| ### Macro: find_component | ||||
| # | ||||
| # Checks for the given component by invoking pkgconfig and then looking up the libraries and | ||||
| # include directories. | ||||
| # | ||||
| macro(find_component _component _pkgconfig _library _header) | ||||
| 
 | ||||
|     if (NOT WIN32) | ||||
|         # use pkg-config to get the directories and then use these values | ||||
|         # in the FIND_PATH() and FIND_LIBRARY() calls | ||||
|         find_package(PkgConfig) | ||||
|         if (PKG_CONFIG_FOUND) | ||||
|             pkg_check_modules(PC_${_component} ${_pkgconfig}) | ||||
|         endif () | ||||
|     endif (NOT WIN32) | ||||
| 
 | ||||
|     find_path(${_component}_INCLUDE_DIRS ${_header} | ||||
|             HINTS | ||||
|             ${PC_LIB${_component}_INCLUDEDIR} | ||||
|             ${PC_LIB${_component}_INCLUDE_DIRS} | ||||
|             ) | ||||
| 
 | ||||
|     find_library(${_component}_LIBRARIES NAMES ${_library} | ||||
|             HINTS | ||||
|             ${PC_LIB${_component}_LIBDIR} | ||||
|             ${PC_LIB${_component}_LIBRARY_DIRS} | ||||
|             ) | ||||
| 
 | ||||
|     set(${_component}_DEFINITIONS  ${PC_${_component}_CFLAGS_OTHER} CACHE STRING "The ${_component} CFLAGS.") | ||||
|     set(${_component}_VERSION      ${PC_${_component}_VERSION}      CACHE STRING "The ${_component} version number.") | ||||
| 
 | ||||
|     set_component_found(${_component}) | ||||
| 
 | ||||
|     mark_as_advanced( | ||||
|             ${_component}_INCLUDE_DIRS | ||||
|             ${_component}_LIBRARIES | ||||
|             ${_component}_DEFINITIONS | ||||
|             ${_component}_VERSION) | ||||
| 
 | ||||
| endmacro() | ||||
| 
 | ||||
| 
 | ||||
| # Check for cached results. If there are skip the costly part. | ||||
| if (NOT FFMPEG_LIBRARIES) | ||||
| 
 | ||||
|     # Check for all possible component. | ||||
|     find_component(AVCODEC  libavcodec  avcodec  libavcodec/avcodec.h) | ||||
|     find_component(AVFORMAT libavformat avformat libavformat/avformat.h) | ||||
|     find_component(AVDEVICE libavdevice avdevice libavdevice/avdevice.h) | ||||
|     find_component(AVUTIL   libavutil   avutil   libavutil/avutil.h) | ||||
|     find_component(SWSCALE  libswscale  swscale  libswscale/swscale.h) | ||||
|     find_component(POSTPROC libpostproc postproc libpostproc/postprocess.h) | ||||
| 
 | ||||
|     # Check if the required components were found and add their stuff to the FFMPEG_* vars. | ||||
|     foreach (_component ${FFmpeg_FIND_COMPONENTS}) | ||||
|         if (${_component}_FOUND) | ||||
|             # message(STATUS "Required component ${_component} present.") | ||||
|             set(FFMPEG_LIBRARIES   ${FFMPEG_LIBRARIES}   ${${_component}_LIBRARIES}) | ||||
|             set(FFMPEG_DEFINITIONS ${FFMPEG_DEFINITIONS} ${${_component}_DEFINITIONS}) | ||||
|             list(APPEND FFMPEG_INCLUDE_DIRS ${${_component}_INCLUDE_DIRS}) | ||||
|         else () | ||||
|             # message(STATUS "Required component ${_component} missing.") | ||||
|         endif () | ||||
|     endforeach () | ||||
| 
 | ||||
|     # Build the include path with duplicates removed. | ||||
|     if (FFMPEG_INCLUDE_DIRS) | ||||
|         list(REMOVE_DUPLICATES FFMPEG_INCLUDE_DIRS) | ||||
|     endif () | ||||
| 
 | ||||
|     # cache the vars. | ||||
|     set(FFMPEG_INCLUDE_DIRS ${FFMPEG_INCLUDE_DIRS} CACHE STRING "The FFmpeg include directories." FORCE) | ||||
|     set(FFMPEG_LIBRARIES    ${FFMPEG_LIBRARIES}    CACHE STRING "The FFmpeg libraries." FORCE) | ||||
|     set(FFMPEG_DEFINITIONS  ${FFMPEG_DEFINITIONS}  CACHE STRING "The FFmpeg cflags." FORCE) | ||||
| 
 | ||||
|     mark_as_advanced(FFMPEG_INCLUDE_DIRS | ||||
|             FFMPEG_LIBRARIES | ||||
|             FFMPEG_DEFINITIONS) | ||||
| 
 | ||||
| endif () | ||||
| 
 | ||||
| # Now set the noncached _FOUND vars for the components. | ||||
| foreach (_component AVCODEC AVDEVICE AVFORMAT AVUTIL POSTPROCESS SWSCALE) | ||||
|     set_component_found(${_component}) | ||||
| endforeach () | ||||
| 
 | ||||
| # Compile the list of required vars | ||||
| set(_FFmpeg_REQUIRED_VARS FFMPEG_LIBRARIES FFMPEG_INCLUDE_DIRS) | ||||
| foreach (_component ${FFmpeg_FIND_COMPONENTS}) | ||||
|     list(APPEND _FFmpeg_REQUIRED_VARS ${_component}_LIBRARIES ${_component}_INCLUDE_DIRS) | ||||
| endforeach () | ||||
| 
 | ||||
| # Give a nice error message if some of the required vars are missing. | ||||
| find_package_handle_standard_args(FFmpeg DEFAULT_MSG ${_FFmpeg_REQUIRED_VARS}) | ||||
|  | @ -0,0 +1,90 @@ | |||
| #include "RTSPDecoder/RTSPDecoder.h" | ||||
| #include <iostream> | ||||
| #include <sys/time.h> | ||||
| #include <opencv2/opencv.hpp> | ||||
| 
 | ||||
| std::string getDateTime_usec() | ||||
| { | ||||
|     time_t timep = time(NULL); | ||||
|     struct tm *p = localtime(&timep); | ||||
| 
 | ||||
|     struct timeval tv; | ||||
|     gettimeofday(&tv, NULL); | ||||
| 
 | ||||
|     int msec = tv.tv_usec / 1000; | ||||
| 
 | ||||
|     char tmp[30] = {0}; | ||||
|     sprintf(tmp, "%04d-%02d-%02d %02d:%02d:%02d.%03d", 1900 + p->tm_year, 1 + p->tm_mon, p->tm_mday, p->tm_hour, p->tm_min, p->tm_sec, msec); | ||||
| 
 | ||||
|     return std::string(tmp); | ||||
| } | ||||
| 
 | ||||
| int main() { | ||||
|     RTSPDecoder decoder; | ||||
|     RTSPDecoder::DecoderConfig config; | ||||
|     config.max_streams = 10;  // 最大10路流
 | ||||
|     config.use_hw_accel = true; // 使用硬件加速
 | ||||
|     config.width = 1920;     // 输出宽度
 | ||||
|     config.height = 1080;    // 输出高度
 | ||||
|     config.buffer_size = 5;   // 每路流缓冲5帧
 | ||||
| 
 | ||||
|     if (!decoder.init(config)) { | ||||
|         std::cerr << "Failed to initialize decoder" << std::endl; | ||||
|         return 1; | ||||
|     } | ||||
| 
 | ||||
|     // 添加RTSP流
 | ||||
|     std::vector<std::string> rtsp_urls = { | ||||
|         "./1.mp4", | ||||
|         "./1.mp4", | ||||
|         "./1.mp4", | ||||
|         "./1.mp4", | ||||
|         "./1.mp4", | ||||
|         "./1.mp4", | ||||
|         "./1.mp4", | ||||
|         "./1.mp4", | ||||
|         "./1.mp4", | ||||
|         "./1.mp4", | ||||
|         // 添加更多流...
 | ||||
|     }; | ||||
| 
 | ||||
|     std::vector<int> stream_ids; | ||||
|     for (const auto& url : rtsp_urls) { | ||||
|         try { | ||||
|             int stream_id = decoder.addStream(url); | ||||
|             stream_ids.push_back(stream_id); | ||||
|             std::cout << "Added stream " << stream_id << ": " << url << std::endl; | ||||
|         } catch (const std::exception& e) { | ||||
|             std::cerr << "Failed to add stream: " << e.what() << std::endl; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     int iID = 0; | ||||
|     // 主循环
 | ||||
|     while (true) { | ||||
|         for (int stream_id : stream_ids) { | ||||
|             ++iID; | ||||
|             cv::Mat frame; | ||||
|             if (decoder.getFrame(stream_id, frame, 1)) { | ||||
| 
 | ||||
|                 // 处理帧...
 | ||||
|                 if (!cv::imwrite("./jpg/Stream_" + std::to_string(stream_id) + "_" + std::to_string(iID) + ".jpg", frame)) | ||||
|                 { | ||||
|                     std::cerr << "Save Failed ./jpg/Stream_" + std::to_string(stream_id) + "_" + std::to_string(iID) + ".jpg" << "  size:" << frame.size() << std::endl; | ||||
|                 } | ||||
|                 std::cout << getDateTime_usec() << " Stream " << stream_id << " -- " << std::to_string(iID) << std::endl; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         if (cv::waitKey(1) == 27) { // ESC键退出
 | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // 清理
 | ||||
|     for (int stream_id : stream_ids) { | ||||
|         decoder.removeStream(stream_id); | ||||
|     } | ||||
| 
 | ||||
|     return 0; | ||||
| } | ||||
		Loading…
	
		Reference in New Issue