Blame view

src/interface/FFNvDecoderManager.cpp 10.5 KB
aac5773f   hucm   功能基本完成,接口待打磨
1
  #include "FFNvDecoderManager.h"
aac5773f   hucm   功能基本完成,接口待打磨
2
  
63e6f7bc   Hu Chunming   完成dvpp。但是nv和gb281...
3
  #ifdef USE_NVDEC
5686354a   Hu Chunming   初步编译成功cuvid部分的
4
  #include "../nvdec/FFNvDecoder.h"
63e6f7bc   Hu Chunming   完成dvpp。但是nv和gb281...
5
6
7
8
9
10
  #include "../gb28181/FFGB28181Decoder.h"
  #endif
  
  #ifdef USE_DVPP
  #include "./dvpp/DvppDecoderApi.h"
  #endif
3d2ab595   Hu Chunming   支持gb28181
11
12
13
  
  #include "logger.hpp"
  
aac5773f   hucm   功能基本完成,接口待打磨
14
15
  using namespace std;
  
48330793   Hu Chunming   修正解码线程自然结束时解码器内存没...
16
  
3d2ab595   Hu Chunming   支持gb28181
17
  AbstractDecoder* FFNvDecoderManager::createDecoder(MgrDecConfig config){
48330793   Hu Chunming   修正解码线程自然结束时解码器内存没...
18
19
20
  
      closeAllFinishedDecoder();
  
3d2ab595   Hu Chunming   支持gb28181
21
22
23
      if (config.cfg.post_decoded_cbk == nullptr || config.cfg.decode_finished_cbk== nullptr){
          return nullptr;
      }
3d2ab595   Hu Chunming   支持gb28181
24
  
3c7e3e11   Hu Chunming   1.修改日志
25
26
27
28
      std::lock_guard<std::mutex> l(m_mutex);
  
      auto it = decoderMap.find(config.name);
      if (it != decoderMap.end()){
372e629f   ming   gb28181支持TCP数据流
29
          LOG_ERROR("已存在name为{}的解码器", config.name);
aac5773f   hucm   功能基本完成,接口待打磨
30
31
          return nullptr;
      }
3d2ab595   Hu Chunming   支持gb28181
32
33
  
      AbstractDecoder* dec = nullptr;
63e6f7bc   Hu Chunming   完成dvpp。但是nv和gb281...
34
  #ifdef USE_NVDEC
3d2ab595   Hu Chunming   支持gb28181
35
36
      if(DECODER_TYPE_FFMPEG == config.dec_type){
          dec = new FFNvDecoder();
63e6f7bc   Hu Chunming   完成dvpp。但是nv和gb281...
37
38
39
      }
      
      if(DECODER_TYPE_GB28181 == config.dec_type){
3d2ab595   Hu Chunming   支持gb28181
40
41
          dec = new FFGB28181Decoder();
      }
63e6f7bc   Hu Chunming   完成dvpp。但是nv和gb281...
42
43
44
45
46
47
48
  #endif
  
  #ifdef USE_DVPP
      if(DECODER_TYPE_DVPP == config.dec_type){
          dec = new DvppDecoderApi();
      }
  #endif
aac5773f   hucm   功能基本完成,接口待打磨
49
      
3d2ab595   Hu Chunming   支持gb28181
50
51
      if (dec == nullptr){
          LOG_ERROR("没有指定解码器类型");
aac5773f   hucm   功能基本完成,接口待打磨
52
53
54
          return nullptr;
      }
      
d248da62   Hu Chunming   优化代码,修正dvpp的一些bug
55
      config.cfg.dec_name = config.name;
e96e6489   Hu Chunming   优化代码;添加isRunning函数
56
      bool bRet= dec->init(config.cfg);
aac5773f   hucm   功能基本完成,接口待打磨
57
58
      if (bRet)
      {
7319ea36   Hu Chunming   多显卡设置
59
          decoderMap[config.name] = dec;
3d2ab595   Hu Chunming   支持gb28181
60
61
  
          LOG_INFO("[{}][{}]- 解码器初始化成功",config.name, config.cfg.uri);
aac5773f   hucm   功能基本完成,接口待打磨
62
63
64
65
66
67
68
          return dec;
      }
      
      // 创建失败,关闭解码器
      dec->close();
      delete dec;
  
3d2ab595   Hu Chunming   支持gb28181
69
      LOG_ERROR("[{}][{}]- 解码器初始化失败!",config.name, config.cfg.uri);
aac5773f   hucm   功能基本完成,接口待打磨
70
71
72
      return nullptr;
  }
  
3d2ab595   Hu Chunming   支持gb28181
73
  bool FFNvDecoderManager::setPostDecArg(const string name, const void * userPtr)
aac5773f   hucm   功能基本完成,接口待打磨
74
75
76
  {
      if (name.empty())
      {
3d2ab595   Hu Chunming   支持gb28181
77
          LOG_ERROR("name 为空!");
0573bd98   Hu Chunming   优化代码;添加注释
78
          return false;
aac5773f   hucm   功能基本完成,接口待打磨
79
80
      }
  
3c7e3e11   Hu Chunming   1.修改日志
81
82
      std::lock_guard<std::mutex> l(m_mutex);
  
aac5773f   hucm   功能基本完成,接口待打磨
83
84
85
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
63e6f7bc   Hu Chunming   完成dvpp。但是nv和gb281...
86
          dec->second->setPostDecArg(userPtr);
0573bd98   Hu Chunming   优化代码;添加注释
87
          return true;
aac5773f   hucm   功能基本完成,接口待打磨
88
89
      }
  
3d2ab595   Hu Chunming   支持gb28181
90
      LOG_ERROR("没有找到name为{}的解码器",name);
0573bd98   Hu Chunming   优化代码;添加注释
91
      return false;
aac5773f   hucm   功能基本完成,接口待打磨
92
93
  }
  
3d2ab595   Hu Chunming   支持gb28181
94
  bool FFNvDecoderManager::setFinishedDecArg(const string name, const void * userPtr)
aac5773f   hucm   功能基本完成,接口待打磨
95
96
97
  {
      if (name.empty())
      {
3d2ab595   Hu Chunming   支持gb28181
98
99
100
101
102
103
104
105
106
          LOG_ERROR("name 为空!");
          return false;
      }
  
      std::lock_guard<std::mutex> l(m_mutex);
  
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
63e6f7bc   Hu Chunming   完成dvpp。但是nv和gb281...
107
          dec->second->setFinishedDecArg(userPtr);
3d2ab595   Hu Chunming   支持gb28181
108
109
110
111
112
113
114
115
116
117
118
119
          return true;
      }
  
      LOG_ERROR("没有找到name为{}的解码器",name);
      return false;
  }
  
  AbstractDecoder* FFNvDecoderManager::getDecoderByName(const string name)
  {
      if (name.empty())
      {
          LOG_ERROR("name 为空!");
aac5773f   hucm   功能基本完成,接口待打磨
120
121
122
          return nullptr;
      }
      
3c7e3e11   Hu Chunming   1.修改日志
123
124
      std::lock_guard<std::mutex> l(m_mutex);
  
aac5773f   hucm   功能基本完成,接口待打磨
125
126
127
128
129
130
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
          return dec->second;
      }
  
3d2ab595   Hu Chunming   支持gb28181
131
      LOG_ERROR("没有找到name为{}的解码器",name);
aac5773f   hucm   功能基本完成,接口待打磨
132
133
134
      return nullptr;
  }
  
92989af0   ming   更新解码器
135
  bool FFNvDecoderManager::startDecode(AbstractDecoder* dec){
aac5773f   hucm   功能基本完成,接口待打磨
136
137
      if (dec != nullptr && !dec->isRunning())
      {
92989af0   ming   更新解码器
138
          return dec->start();
aac5773f   hucm   功能基本完成,接口待打磨
139
      }
92989af0   ming   更新解码器
140
      return false;
aac5773f   hucm   功能基本完成,接口待打磨
141
142
  }
  
0573bd98   Hu Chunming   优化代码;添加注释
143
  bool FFNvDecoderManager::startDecodeByName(const string name){
aac5773f   hucm   功能基本完成,接口待打磨
144
145
       if (name.empty())
      {
3d2ab595   Hu Chunming   支持gb28181
146
          LOG_ERROR("name 为空!");
aac5773f   hucm   功能基本完成,接口待打磨
147
148
149
          return false;
      }
  
3c7e3e11   Hu Chunming   1.修改日志
150
151
      std::lock_guard<std::mutex> l(m_mutex);
  
aac5773f   hucm   功能基本完成,接口待打磨
152
153
154
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
3d2ab595   Hu Chunming   支持gb28181
155
          return dec->second->start();
aac5773f   hucm   功能基本完成,接口待打磨
156
157
      }
  
3d2ab595   Hu Chunming   支持gb28181
158
      LOG_ERROR("没有找到name为{}的解码器",name);
aac5773f   hucm   功能基本完成,接口待打磨
159
160
161
162
      return false;
  }
  
  void FFNvDecoderManager::startAllDecode(){
3c7e3e11   Hu Chunming   1.修改日志
163
164
165
  
      std::lock_guard<std::mutex> l(m_mutex);
  
aac5773f   hucm   功能基本完成,接口待打磨
166
      for(auto iter = decoderMap.begin(); iter != decoderMap.end(); iter++){
0573bd98   Hu Chunming   优化代码;添加注释
167
168
169
170
          if (!iter->second->isRunning())
          {
              iter->second->start();
          }
aac5773f   hucm   功能基本完成,接口待打磨
171
172
173
      }
  }
  
0573bd98   Hu Chunming   优化代码;添加注释
174
  bool FFNvDecoderManager::closeDecoderByName(const string name){
aac5773f   hucm   功能基本完成,接口待打磨
175
176
      if (name.empty())
      {
3d2ab595   Hu Chunming   支持gb28181
177
          LOG_ERROR("name 为空!");
aac5773f   hucm   功能基本完成,接口待打磨
178
179
180
          return false;
      }
  
3c7e3e11   Hu Chunming   1.修改日志
181
182
      std::lock_guard<std::mutex> l(m_mutex);
  
aac5773f   hucm   功能基本完成,接口待打磨
183
184
185
186
187
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
          dec->second->close();
          delete dec->second;
f40cc409   Hu Chunming   优化显存占用。当前在3080显卡上...
188
          dec->second = nullptr;
aac5773f   hucm   功能基本完成,接口待打磨
189
          decoderMap.erase(dec);
f40cc409   Hu Chunming   优化显存占用。当前在3080显卡上...
190
  
aac5773f   hucm   功能基本完成,接口待打磨
191
192
193
          return true;
      }
      
3d2ab595   Hu Chunming   支持gb28181
194
      LOG_ERROR("没有找到name为{}的解码器",name);
aac5773f   hucm   功能基本完成,接口待打磨
195
196
197
198
199
      return false;
  }
  
  void FFNvDecoderManager::closeAllDecoder()
  {
3c7e3e11   Hu Chunming   1.修改日志
200
201
      std::lock_guard<std::mutex> l(m_mutex);
  
aac5773f   hucm   功能基本完成,接口待打磨
202
203
      for(auto iter = decoderMap.begin(); iter != decoderMap.end(); iter++){
          iter->second->close();
48330793   Hu Chunming   修正解码线程自然结束时解码器内存没...
204
          delete iter->second;
f40cc409   Hu Chunming   优化显存占用。当前在3080显卡上...
205
          iter->second = nullptr;
48330793   Hu Chunming   修正解码线程自然结束时解码器内存没...
206
207
      }
      decoderMap.clear();
48330793   Hu Chunming   修正解码线程自然结束时解码器内存没...
208
209
210
211
  }
  
  void FFNvDecoderManager::closeAllFinishedDecoder()
  {
3c7e3e11   Hu Chunming   1.修改日志
212
213
      std::lock_guard<std::mutex> l(m_mutex);
  
f40cc409   Hu Chunming   优化显存占用。当前在3080显卡上...
214
       for(auto iter = decoderMap.begin(); iter != decoderMap.end(); ){
48330793   Hu Chunming   修正解码线程自然结束时解码器内存没...
215
216
217
          if (iter->second->isFinished())
          {
              delete iter->second;
f40cc409   Hu Chunming   优化显存占用。当前在3080显卡上...
218
219
220
221
222
223
              iter->second = nullptr;
              iter = decoderMap.erase(iter);
          }
          else
          {
              iter++ ;
48330793   Hu Chunming   修正解码线程自然结束时解码器内存没...
224
          }
aac5773f   hucm   功能基本完成,接口待打磨
225
      }
aac5773f   hucm   功能基本完成,接口待打磨
226
227
228
229
  }
  
  int FFNvDecoderManager::count()
  {
48330793   Hu Chunming   修正解码线程自然结束时解码器内存没...
230
231
      closeAllFinishedDecoder();
  
3c7e3e11   Hu Chunming   1.修改日志
232
      std::lock_guard<std::mutex> l(m_mutex);
aac5773f   hucm   功能基本完成,接口待打磨
233
234
235
      return decoderMap.size();
  }
  
0573bd98   Hu Chunming   优化代码;添加注释
236
  bool FFNvDecoderManager::pauseDecoder(const string name)
aac5773f   hucm   功能基本完成,接口待打磨
237
238
239
  {
      if (name.empty())
      {
3d2ab595   Hu Chunming   支持gb28181
240
          LOG_ERROR("name 为空!");
aac5773f   hucm   功能基本完成,接口待打磨
241
242
243
          return false;
      }
  
3c7e3e11   Hu Chunming   1.修改日志
244
245
      std::lock_guard<std::mutex> l(m_mutex);
  
aac5773f   hucm   功能基本完成,接口待打磨
246
247
248
249
250
251
252
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
          dec->second->pause();
          return true;
      }
      
3d2ab595   Hu Chunming   支持gb28181
253
      LOG_ERROR("没有找到name为{}的解码器",name);
aac5773f   hucm   功能基本完成,接口待打磨
254
255
256
      return false;
  }
  
0573bd98   Hu Chunming   优化代码;添加注释
257
  bool FFNvDecoderManager::resumeDecoder(const string name)
aac5773f   hucm   功能基本完成,接口待打磨
258
259
260
  {
      if (name.empty())
      {
3d2ab595   Hu Chunming   支持gb28181
261
          LOG_ERROR("name 为空!");
aac5773f   hucm   功能基本完成,接口待打磨
262
263
264
          return false;
      }
  
3c7e3e11   Hu Chunming   1.修改日志
265
266
      std::lock_guard<std::mutex> l(m_mutex);
  
aac5773f   hucm   功能基本完成,接口待打磨
267
268
269
270
271
272
273
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
          dec->second->resume();
          return true;
      }
      
3d2ab595   Hu Chunming   支持gb28181
274
      LOG_ERROR("没有找到name为{}的解码器",name);
aac5773f   hucm   功能基本完成,接口待打磨
275
276
277
      return false;
  }
  
3d2ab595   Hu Chunming   支持gb28181
278
  bool FFNvDecoderManager::isSurport(MgrDecConfig& config)
aac5773f   hucm   功能基本完成,接口待打磨
279
  {
3d2ab595   Hu Chunming   支持gb28181
280
281
282
283
284
285
286
287
288
289
290
      {
          std::lock_guard<std::mutex> l(m_mutex);
  
          auto it = decoderMap.find(config.name);
          if (it != decoderMap.end()){
              LOG_ERROR("已存在name所标记的解码器");
              return false;
          }
      }
  
      AbstractDecoder* dec = nullptr;
63e6f7bc   Hu Chunming   完成dvpp。但是nv和gb281...
291
292
  #ifdef USE_NVDEC
      if(DECODER_TYPE_FFMPEG == config.dec_type){
3d2ab595   Hu Chunming   支持gb28181
293
          dec = new FFNvDecoder();
63e6f7bc   Hu Chunming   完成dvpp。但是nv和gb281...
294
295
296
      }
      
      if(DECODER_TYPE_GB28181 == config.dec_type){
3d2ab595   Hu Chunming   支持gb28181
297
298
          dec = new FFGB28181Decoder();
      }
63e6f7bc   Hu Chunming   完成dvpp。但是nv和gb281...
299
300
301
302
303
304
305
  #endif
  
  #ifdef USE_DVPP
      if(DECODER_TYPE_DVPP == config.dec_type){
          dec = new DvppDecoderApi();
      }
  #endif
3d2ab595   Hu Chunming   支持gb28181
306
307
308
309
310
311
312
313
314
315
316
      
      if (dec == nullptr){
          LOG_ERROR("没有指定解码器类型");
          return false;
      }
  
      bool bRet = dec->isSurport(config.cfg);
      delete dec;
      dec = nullptr;
  
      return bRet;
e96e6489   Hu Chunming   优化代码;添加isRunning函数
317
318
  }
  
0573bd98   Hu Chunming   优化代码;添加注释
319
  bool FFNvDecoderManager::isRunning(const string name){
bc52e542   Hu Chunming   添加关键帧解码功能
320
      if (name.empty())
e96e6489   Hu Chunming   优化代码;添加isRunning函数
321
      {
3d2ab595   Hu Chunming   支持gb28181
322
          LOG_ERROR("name 为空!");
e96e6489   Hu Chunming   优化代码;添加isRunning函数
323
324
325
          return false;
      }
  
3c7e3e11   Hu Chunming   1.修改日志
326
327
      std::lock_guard<std::mutex> l(m_mutex);
  
e96e6489   Hu Chunming   优化代码;添加isRunning函数
328
329
330
331
332
333
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
          return dec->second->isRunning();
      }
      
3d2ab595   Hu Chunming   支持gb28181
334
      LOG_ERROR("没有找到name为{}的解码器",name);
3c7e3e11   Hu Chunming   1.修改日志
335
336
337
338
339
340
      return false;
  }
  
  bool FFNvDecoderManager::isFinished(const string name){
      if (name.empty())
      {
3d2ab595   Hu Chunming   支持gb28181
341
          LOG_ERROR("name 为空!");
3c7e3e11   Hu Chunming   1.修改日志
342
343
344
345
346
347
348
349
350
351
352
          return false;
      }
  
      std::lock_guard<std::mutex> l(m_mutex);
  
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
          return dec->second->isFinished();
      }
      
3d2ab595   Hu Chunming   支持gb28181
353
      LOG_ERROR("没有找到name为{}的解码器",name);
3c7e3e11   Hu Chunming   1.修改日志
354
355
356
357
358
359
      return false;
  }
  
  bool FFNvDecoderManager::isPausing(const string name){
      if (name.empty())
      {
3d2ab595   Hu Chunming   支持gb28181
360
          LOG_ERROR("name 为空!");
3c7e3e11   Hu Chunming   1.修改日志
361
362
363
364
365
366
367
368
369
370
371
          return false;
      }
  
      std::lock_guard<std::mutex> l(m_mutex);
  
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
          return dec->second->isPausing();
      }
      
3d2ab595   Hu Chunming   支持gb28181
372
      LOG_ERROR("没有找到name为{}的解码器",name);
e96e6489   Hu Chunming   优化代码;添加isRunning函数
373
      return false;
bc52e542   Hu Chunming   添加关键帧解码功能
374
375
376
377
378
379
  }
  
  bool FFNvDecoderManager::setDecKeyframe(const string name, bool bKeyframe)
  {
      if (name.empty())
      {
3d2ab595   Hu Chunming   支持gb28181
380
          LOG_ERROR("name 为空!");
bc52e542   Hu Chunming   添加关键帧解码功能
381
382
383
          return false;
      }
  
3c7e3e11   Hu Chunming   1.修改日志
384
385
      std::lock_guard<std::mutex> l(m_mutex);
  
bc52e542   Hu Chunming   添加关键帧解码功能
386
387
388
389
390
391
392
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
          dec->second->setDecKeyframe(bKeyframe);
          return true;
      }
      
3d2ab595   Hu Chunming   支持gb28181
393
      LOG_ERROR("没有找到name为{}的解码器",name);
3c7e3e11   Hu Chunming   1.修改日志
394
395
396
397
398
399
400
      return false;
  }
  
  bool FFNvDecoderManager::getResolution(const string name, int &width, int &height)
  {
      if (name.empty())
      {
3d2ab595   Hu Chunming   支持gb28181
401
          LOG_ERROR("name 为空!");
3c7e3e11   Hu Chunming   1.修改日志
402
403
404
405
406
407
408
409
410
411
412
413
          return false;
      }
  
      std::lock_guard<std::mutex> l(m_mutex);
  
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end())
      {
          dec->second->getResolution(width, height);
          return true;
      }
      
3d2ab595   Hu Chunming   支持gb28181
414
      LOG_ERROR("没有找到name为{}的解码器",name);
bc52e542   Hu Chunming   添加关键帧解码功能
415
      return false;
3c7e3e11   Hu Chunming   1.修改日志
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
  }
  
  vector<string> FFNvDecoderManager::getAllDecodeName(){
      
      closeAllFinishedDecoder();
  
      std::lock_guard<std::mutex> l(m_mutex);
  
      vector<string> decode_names;
      for(auto it = decoderMap.begin(); it != decoderMap.end(); ++it){
          decode_names.push_back(it->first);
      }
      return decode_names;
  }
  
  int FFNvDecoderManager::getCachedQueueLength(const string name){
      if (name.empty()){
3d2ab595   Hu Chunming   支持gb28181
433
          LOG_ERROR("name 为空!");
3c7e3e11   Hu Chunming   1.修改日志
434
435
436
437
438
439
440
441
442
443
          return -1;
      }
  
      std::lock_guard<std::mutex> l(m_mutex);
  
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end()){
          return dec->second->getCachedQueueLength();
      }
      
3d2ab595   Hu Chunming   支持gb28181
444
      LOG_ERROR("没有找到name为{}的解码器",name);
3c7e3e11   Hu Chunming   1.修改日志
445
446
447
      return -1;
  }
  
3c7e3e11   Hu Chunming   1.修改日志
448
  void FFNvDecoderManager::releaseFFImgInfo(FFImgInfo* info){
3d2ab595   Hu Chunming   支持gb28181
449
450
  	if(nullptr != info){
  		if(info->pData != nullptr){
63e6f7bc   Hu Chunming   完成dvpp。但是nv和gb281...
451
  			free(info->pData);
3d2ab595   Hu Chunming   支持gb28181
452
453
454
455
456
457
  			info->pData = nullptr;
  		}
  		delete info;
  		info = nullptr;
  	}
  }
92989af0   ming   更新解码器
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
  
  FFImgInfo* FFNvDecoderManager::snapshot_in_task(const string name){
      if (name.empty()){
          LOG_ERROR("name 为空!");
          return nullptr;
      }
  
      std::lock_guard<std::mutex> l(m_mutex);
  
      auto dec = decoderMap.find(name);
      if (dec != decoderMap.end()){
          return dec->second->snapshot();
      }
      
      LOG_ERROR("没有找到name为{}的解码器",name);
      return nullptr;
  }
  
  vector<FFImgInfo*> FFNvDecoderManager::timing_snapshot_all(){
  
      closeAllFinishedDecoder();
  
      std::lock_guard<std::mutex> l(m_mutex);
  
      vector<FFImgInfo*> vec;
      for(auto it = decoderMap.begin(); it != decoderMap.end(); ++it){
          if(it->second->isSnapTime()){
              FFImgInfo* imginfo = it->second->snapshot();
              if(imginfo != nullptr){
                  vec.push_back(imginfo);
              }
              it->second->updateLastSnapTime();
          }
      }
  
      return vec;
  }