mirror of
				https://github.com/kunkundi/crossdesk.git
				synced 2025-10-26 20:25:34 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			117 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			117 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "nvidia_video_decoder.h"
 | |
| 
 | |
| #include "log.h"
 | |
| 
 | |
| // #define SAVE_DECODED_NV12_STREAM
 | |
| // #define SAVE_RECEIVED_H264_STREAM
 | |
| 
 | |
| NvidiaVideoDecoder::NvidiaVideoDecoder(std::shared_ptr<SystemClock> clock)
 | |
|     : clock_(clock) {}
 | |
| NvidiaVideoDecoder::~NvidiaVideoDecoder() {
 | |
| #ifdef SAVE_DECODED_NV12_STREAM
 | |
|   if (file_nv12_) {
 | |
|     fflush(file_nv12_);
 | |
|     fclose(file_nv12_);
 | |
|     file_nv12_ = nullptr;
 | |
|   }
 | |
| #endif
 | |
| 
 | |
| #ifdef SAVE_RECEIVED_H264_STREAM
 | |
|   if (file_h264_) {
 | |
|     fflush(file_h264_);
 | |
|     fclose(file_h264_);
 | |
|     file_h264_ = nullptr;
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   if (cuda_context_) {
 | |
|     cuCtxDestroy(cuda_context_);
 | |
|     cuda_context_ = nullptr;
 | |
|   }
 | |
| }
 | |
| 
 | |
| int NvidiaVideoDecoder::Init() {
 | |
|   ck(cuInit(0));
 | |
|   int nGpu = 0;
 | |
|   int iGpu = 0;
 | |
| 
 | |
|   ck(cuDeviceGetCount(&nGpu));
 | |
|   if (nGpu < 1) {
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   cuDeviceGet(&cuda_device_, iGpu);
 | |
| 
 | |
|   cuCtxCreate(&cuda_context_, 0, cuda_device_);
 | |
|   if (!cuda_context_) {
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   decoder =
 | |
|       new NvDecoder(cuda_context_, false, cudaVideoCodec_H264, true, false,
 | |
|                     nullptr, nullptr, false, 4096, 2160, 1000, false);
 | |
| 
 | |
| #ifdef SAVE_DECODED_NV12_STREAM
 | |
|   file_nv12_ = fopen("decoded_nv12_stream.yuv", "w+b");
 | |
|   if (!file_nv12_) {
 | |
|     LOG_WARN("Fail to open decoded_nv12_stream.yuv");
 | |
|   }
 | |
| #endif
 | |
| 
 | |
| #ifdef SAVE_RECEIVED_H264_STREAM
 | |
|   file_h264_ = fopen("received_h264_stream.h264", "w+b");
 | |
|   if (!file_h264_) {
 | |
|     LOG_WARN("Fail to open received_h264_stream.h264");
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| int NvidiaVideoDecoder::Decode(
 | |
|     const ReceivedFrame &received_frame,
 | |
|     std::function<void(const DecodedFrame &)> on_receive_decoded_frame) {
 | |
|   if (!decoder) {
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   const uint8_t *data = received_frame.Buffer();
 | |
|   size_t size = received_frame.Size();
 | |
| 
 | |
| #ifdef SAVE_RECEIVED_H264_STREAM
 | |
|   fwrite((unsigned char *)data, 1, size, file_h264_);
 | |
| #endif
 | |
| 
 | |
|   if ((*(data + 4) & 0x1f) == 0x07) {
 | |
|     LOG_INFO("Receive key frame");
 | |
|   }
 | |
| 
 | |
|   int num_frame_returned = decoder->Decode(data, (int)size);
 | |
|   for (size_t i = 0; i < num_frame_returned; ++i) {
 | |
|     cudaVideoSurfaceFormat format = decoder->GetOutputFormat();
 | |
|     if (format == cudaVideoSurfaceFormat_NV12) {
 | |
|       uint8_t *decoded_frame_buffer = nullptr;
 | |
|       decoded_frame_buffer = decoder->GetFrame();
 | |
|       if (decoded_frame_buffer) {
 | |
|         if (on_receive_decoded_frame) {
 | |
|           DecodedFrame decoded_frame(
 | |
|               decoded_frame_buffer,
 | |
|               decoder->GetWidth() * decoder->GetHeight() * 3 / 2,
 | |
|               decoder->GetWidth(), decoder->GetHeight());
 | |
|           decoded_frame.SetReceivedTimestamp(
 | |
|               received_frame.ReceivedTimestamp());
 | |
|           decoded_frame.SetCapturedTimestamp(
 | |
|               received_frame.CapturedTimestamp());
 | |
|           decoded_frame.SetDecodedTimestamp(clock_->CurrentTime());
 | |
|           on_receive_decoded_frame(decoded_frame);
 | |
| #ifdef SAVE_DECODED_NV12_STREAM
 | |
|           fwrite((unsigned char *)decoded_frame.Buffer(), 1,
 | |
|                  decoded_frame.Size(), file_nv12_);
 | |
| #endif
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| } |