blob: 5c10bec0476456e327a4b63e0186197698e4c5eb [file] [log] [blame]
zhipeng.he60bbc802021-09-13 16:12:40 +08001/*
2 * Copyright (c) 2020 Amlogic, Inc. All rights reserved.
3 *
4 * This source code is subject to the terms and conditions defined in the
5 * file 'LICENSE' which is part of this source code package.
6 *
7 * Description:
8 */
9#include <stdio.h>
10#include <string.h>
11#include <signal.h>
12#include <fstream>
13#include <unistd.h>
14#include <sys/time.h>
15#include <cstdlib>
16#include <time.h>
17#include <sys/types.h>
18#include <sys/stat.h>
19#include <fcntl.h>
20#include <thread>
21#include <atomic>
22#include <map>
23#include <mutex>
24#include <queue>
25#include <thread>
26#include <condition_variable>
27#include <chrono>
28#include <sys/time.h>
29#include <memory>
30#include <getopt.h>
31#include <chrono>
32#include <AmTsPlayer.h>
33#include <termios.h>
34
35#include <sys/select.h>
36#include <termios.h>
37#include <sys/ioctl.h>
38
39#include <iostream>
40
41using namespace std;
42
43#define ONLY_ONE_VIDEO 0
44
45#if ONLY_ONE_VIDEO
46const int changNum = 1;
47#else
48const int changNum = 2;
49#endif
50
51am_tsplayer_handle g_session[changNum] = {0};
52
53typedef struct {
54 char tsFilePatch[50];
55 am_tsplayer_video_codec videoFmt;
56 am_tsplayer_audio_codec audioFmt;
57 int vpid;
58 int apid;
59 am_tsplayer_input_source_type inPutTsType;
60 am_tsplayer_input_buffer_type drmmode;
61 int index;
62 bool run;
63 bool mute;
64 bool setAudio;
65 pthread_t mThread;
66} TsParams;
67
68void video_callback(void *user_data, am_tsplayer_event *event)
69{
Zhipeng Heee39f3d2022-08-31 13:30:45 +080070 if (!event || !user_data) {
71 return ;
72 }
zhipeng.he60bbc802021-09-13 16:12:40 +080073 TsParams* mTsParams = (TsParams*)user_data;
Zhipeng Heee39f3d2022-08-31 13:30:45 +080074 printf("video_callback type %d\n", event->type);
zhipeng.he60bbc802021-09-13 16:12:40 +080075 switch (event->type) {
76 case AM_TSPLAYER_EVENT_TYPE_VIDEO_CHANGED:
77 {
78 printf("index:%d [evt] AM_TSPLAYER_EVENT_TYPE_VIDEO_CHANGED: %d x %d @%d [%d]\n",
79 mTsParams->index,
80 event->event.video_format.frame_width,
81 event->event.video_format.frame_height,
82 event->event.video_format.frame_rate,
83 event->event.video_format.frame_aspectratio);
84 break;
85 }
86 case AM_TSPLAYER_EVENT_TYPE_USERDATA_AFD:
87 case AM_TSPLAYER_EVENT_TYPE_USERDATA_CC:
88 {
89 uint8_t* pbuf = event->event.mpeg_user_data.data;
90 uint32_t size = event->event.mpeg_user_data.len;
91 printf("index:%d [evt] USERDATA [%d] : %x-%x-%x-%x %x-%x-%x-%x ,size %d\n",
92 mTsParams->index,event->type, pbuf[0], pbuf[1], pbuf[2], pbuf[3],
93 pbuf[4], pbuf[5], pbuf[6], pbuf[7], size);
94 //UNUSED(pbuf);
95 //UNUSED(size);
96 break;
97 }
98 case AM_TSPLAYER_EVENT_TYPE_FIRST_FRAME:
99 {
100 printf("index:%d [evt] AM_TSPLAYER_EVENT_TYPE_FIRST_FRAME\n",mTsParams->index);
101 break;
102 }
103 case AM_TSPLAYER_EVENT_TYPE_DECODE_FIRST_FRAME_VIDEO:
104 {
105 printf("index:%d [evt] AM_TSPLAYER_EVENT_TYPE_DECODE_FIRST_FRAME_VIDEO\n",mTsParams->index);
106 break;
107 }
108 case AM_TSPLAYER_EVENT_TYPE_DECODE_FIRST_FRAME_AUDIO:
109 {
110 printf("index:%d [evt] AM_TSPLAYER_EVENT_TYPE_DECODE_FIRST_FRAME_AUDIO\n",mTsParams->index);
111 break;
112 }
113 case AM_TSPLAYER_EVENT_TYPE_AV_SYNC_DONE:
114 {
115 printf("index:%d [evt] AM_TSPLAYER_EVENT_TYPE_AV_SYNC_DONE\n",mTsParams->index);
116 break;
117 }
118 case AM_TSPLAYER_EVENT_TYPE_INPUT_VIDEO_BUFFER_DONE:
119 {
120 // printf("[evt] AM_TSPLAYER_EVENT_TYPE_INPUT_VIDEO_BUFFER_DONE,%p\n",event->event.ptr);
121 break;
122 }
123 case AM_TSPLAYER_EVENT_TYPE_INPUT_AUDIO_BUFFER_DONE:
124 {
125 // printf("[evt] AM_TSPLAYER_EVENT_TYPE_INPUT_AUDIO_BUFFER_DONE\n");
126 break;
127 }
128 default:
129 break;
130 }
131}
132
133void* TsPlayerThread(void *arg) {
134 TsParams* mTsParams = (TsParams*)arg;
135 mTsParams->run = true;
136 std::string inputTsName(mTsParams->tsFilePatch);
137 am_tsplayer_handle session;
138 am_tsplayer_avsync_mode avsyncMode= TS_SYNC_AMASTER;
139 am_tsplayer_video_trick_mode vTrickMode = AV_VIDEO_TRICK_MODE_NONE;
140 int kRwSize = 188*300;
141 int kRwTimeout = 30000;
142 char buf[kRwSize];
Zhipeng Hea485a7d2022-08-19 11:39:55 +0800143 long long fsize = 0;
zhipeng.he60bbc802021-09-13 16:12:40 +0800144 ifstream file(inputTsName.c_str(), ifstream::binary);
145 if (mTsParams->inPutTsType == TS_MEMORY) {
146 file.seekg(0, file.end);
147 fsize = file.tellg();
148 if (fsize <= 0) {
149 printf("file %s size %lld return\n", inputTsName.c_str(), fsize);
150 return 0;
151 }
152 file.seekg(0, file.beg);
153 }
154 printf("inde:%d file name = %s is_open %d, size %lld, inPutTsType %d drmmode:%d \n",
155 mTsParams->index,
156 inputTsName.c_str(),
157 file.is_open(),
158 fsize,
159 mTsParams->inPutTsType,
160 mTsParams->drmmode);
161 int x = 0, y = 0, w = 0, h = 0;
162 int tunnelid = 0;
163 switch (mTsParams->index) {
164 case 0:
165 tunnelid = 0;
166 x = 0;
167 y = 0;
168 w = 960;
169 h = 540;
170 break;
171 case 1:
172 tunnelid = 1;
173 x = 960;
174 y = 0;
175 w = 960;
176 h = 540;
177 break;
178 case 2:
179 tunnelid = 2;
180 x = 0;
181 y = 540;
182 w = 960;
183 h = 540;
184 break;
185 case 3:
186 tunnelid = 3;
187 x = 960;
188 y = 540;
189 w = 960;
190 h = 540;
191 break;
192 default:
193 break;
194 }
195
196
197 am_tsplayer_init_params parm = {mTsParams->inPutTsType, mTsParams->drmmode, mTsParams->index, 0};
198 AmTsPlayer_create(parm, &session);
199 g_session[mTsParams->index] = session;
200 AmTsPlayer_setSurface(session,(void*)&tunnelid);
201 AmTsPlayer_setVideoWindow(session, x, y, w, h);
202
203
204 uint32_t versionM, versionL;
205 AmTsPlayer_getVersion(&versionM, &versionL);
206 uint32_t instanceno;
207 AmTsPlayer_getInstansNo(session, &instanceno);
208 AmTsPlayer_setWorkMode(session, TS_PLAYER_MODE_NORMAL);
209 AmTsPlayer_registerCb(session, video_callback, (void*)mTsParams);
210 AmTsPlayer_setSyncMode(session, avsyncMode);
211
212 am_tsplayer_video_params vparm;
213 vparm.codectype = mTsParams->videoFmt;
214 vparm.pid = mTsParams->vpid;
215 AmTsPlayer_setVideoParams(session, &vparm);
216 AmTsPlayer_startVideoDecoding(session);
217
218 am_tsplayer_audio_params aparm;
219 aparm.codectype = mTsParams->audioFmt;
220 aparm.pid = mTsParams->apid;
221 AmTsPlayer_setAudioMute(session,false,mTsParams->mute);
222 AmTsPlayer_setAudioParams(session, &aparm);
223 AmTsPlayer_startAudioDecoding(session);
224
225 AmTsPlayer_showVideo(session);
226 AmTsPlayer_setTrickMode(session, vTrickMode);
227
228 am_tsplayer_input_buffer ibuf = {TS_INPUT_BUFFER_TYPE_NORMAL, (char*)buf, 0};
229 am_tsplayer_result res;
230 while (mTsParams->inPutTsType)
231 {
232 if (file.eof()) {
233 printf("file read eof index:%d\n",mTsParams->index);
234 break;
235 }
236
237 file.read(buf, (int)kRwSize);
238 ibuf.buf_size = kRwSize;
239
240 int retry = 100;
241 do {
242 if (!mTsParams->run) {
243 printf("break by quit******index:%d \n",mTsParams->index);
244 break;
245 }
246 res = AmTsPlayer_writeData(session, &ibuf, kRwTimeout);
247 if (res == AM_TSPLAYER_ERROR_RETRY ) {
248 usleep(300000);
249 } else {
250 break;
251 }
252 } while(res || retry-- > 0);
253
254 if (!mTsParams->run) {
255 break;
256 }
257 }
258 while (mTsParams->inPutTsType == TS_DEMOD &&
259 mTsParams->run) {
260 usleep(1000000);
261 }
262 mTsParams->run = false;
263
264 printf("AmTsPlayer_stopVideoDecoding index:%d\n",mTsParams->index);
265 AmTsPlayer_stopVideoDecoding(session);
266 AmTsPlayer_stopAudioDecoding(session);
267 printf("AmTsPlayer_release index:%d\n",mTsParams->index);
268 AmTsPlayer_release(session);
269 printf("AmTsPlayer_release ok index:%d\n",mTsParams->index);
270 return NULL;
271}
272
273void signHandler(int iSignNo) {
274 (void) iSignNo;
275 return;
276}
277
278int _kbhit() {
279 static const int STDIN = 0;
280 static bool initialized = false;
281
282 if (!initialized) {
283 // Use termios to turn off line buffering
284 termios term;
285 tcgetattr(STDIN, &term);
286 term.c_lflag &= ~ICANON;
287 tcsetattr(STDIN, TCSANOW, &term);
288 setbuf(stdin, NULL);
289 initialized = true;
290 }
291
292 int bytesWaiting;
293 ioctl(STDIN, FIONREAD, &bytesWaiting);
294 return bytesWaiting;
295}
296
297#if ONLY_ONE_VIDEO
298void changAudio(TsParams* mTsParams,int number) {
299 bool bSetMute = false;
300 bSetMute = (number == 0) ? false : true;
301 if (mTsParams[0].mute != bSetMute) {
302 mTsParams[0].mute = bSetMute;
303 mTsParams[0].setAudio = true;
304 AmTsPlayer_setAudioMute(g_session[0],false,mTsParams[0].mute);
305 printf("AmTsPlayer_setAudioMute : %d \n",mTsParams[0].mute);
306 }
307 return;
308}
309#else
310void changAudio(TsParams* mTsParams,int number) {
311 for (int i = 0 ; i < changNum;i++) {
312 if (mTsParams[i].mute == false) {
313 if (i == number) {
314 return;
315 } else {
316 mTsParams[i].mute = true;
317 mTsParams[i].setAudio = true;
318 AmTsPlayer_setAudioMute(g_session[i],false,mTsParams[i].mute);
319 printf("index:%d AmTsPlayer_setAudioMute : %d \n",i,mTsParams[i].mute);
320 break;
321 }
322 }
323 }
324 // usleep(200000);
325 mTsParams[number].mute = false;
326 mTsParams[number].setAudio = true;
327 AmTsPlayer_setAudioMute(g_session[number],false,mTsParams[number].mute);
328 printf("index:%d AmTsPlayer_setAudioMute : %d \n",number,mTsParams[number].mute);
329 return;
330}
331#endif
332
333void stopVideo(TsParams* mTsParams,int number) {
Zhipeng Hea485a7d2022-08-19 11:39:55 +0800334 printf("index:%d AmTsPlayer_stopVideoDecoding \n",number);
zhipeng.he60bbc802021-09-13 16:12:40 +0800335 AmTsPlayer_stopVideoDecoding(g_session[number]);
336 return;
337}
338
339void startVideo(TsParams* mTsParams,int number) {
340 am_tsplayer_video_params vparm;
341 vparm.codectype = mTsParams->videoFmt;
342 vparm.pid = mTsParams->vpid;
Zhipeng Hea485a7d2022-08-19 11:39:55 +0800343 printf("index:%d AmTsPlayer_startVideoDecoding \n",number);
zhipeng.he60bbc802021-09-13 16:12:40 +0800344 AmTsPlayer_setVideoParams(g_session[number], &vparm);
345 AmTsPlayer_startVideoDecoding(g_session[number]);
346
347 return;
348}
349
zhipeng.he92870d82022-03-15 18:06:20 +0800350void hideVideo(TsParams* mTsParams,int number) {
Zhipeng Hea485a7d2022-08-19 11:39:55 +0800351 printf("index:%d AmTsPlayer_hideVideo \n",number);
zhipeng.he92870d82022-03-15 18:06:20 +0800352 AmTsPlayer_hideVideo(g_session[number]);
353 return;
354}
355
356void showVideo(TsParams* mTsParams,int number) {
Zhipeng Hea485a7d2022-08-19 11:39:55 +0800357 printf("index:%d AmTsPlayer_showVideo \n",number);
zhipeng.he92870d82022-03-15 18:06:20 +0800358 AmTsPlayer_showVideo(g_session[number]);
359 return;
360}
361
362
zhipeng.he60bbc802021-09-13 16:12:40 +0800363
364void printfStatus(TsParams* mTsParams,char ch) {
365 int i = 0;
366 printf("\n");
367 printf("******************************\n");
368 printf("******* key input : %c ********\n",ch);
369 printf("******** q:quit ***********\n");
370 for (i = 0 ; i < changNum;i++) {
371 printf("*******%d:%d channel sound******\n",i,i);
372 }
373 for (i = 0 ; i < changNum;i++) {
374 printf("*** number:%d audioMute:%d ***\n",i,mTsParams[i].mute);
375 }
376 printf("******************************\n");
377
378}
379int main(int argc, char **argv)
380{
381 (void)argc;
382 (void)argv;
383 signal(SIGINT, signHandler);
384 TsParams mTsParams[changNum];
385 char const* file0 = "/data/1.ts";
386 int file0_len = strlen(file0);
387 memcpy(mTsParams[0].tsFilePatch,file0,file0_len);
388 mTsParams[0].tsFilePatch[file0_len] = '\0';
389 printf("file0:%s file0_len:%d\n",file0,file0_len);
390 mTsParams[0].videoFmt = AV_VIDEO_CODEC_H264;
391 mTsParams[0].audioFmt = AV_AUDIO_CODEC_AAC;
392 mTsParams[0].vpid = 0x100;
393 mTsParams[0].apid = 0x101;
394 mTsParams[0].inPutTsType = TS_MEMORY;
395 mTsParams[0].drmmode = TS_INPUT_BUFFER_TYPE_NORMAL;
396 mTsParams[0].index = 0;
397 mTsParams[0].mThread = 0;
398 mTsParams[0].run = true;
399 mTsParams[0].mute = false;
400 mTsParams[0].setAudio = false;
401
402
403#if !ONLY_ONE_VIDEO
404 char const* file1 = "/data/2.ts";
405 int file1_len = strlen(file1);
406 memcpy(mTsParams[1].tsFilePatch,file1,file1_len);
407 mTsParams[1].tsFilePatch[file1_len] = '\0';
408 mTsParams[1].videoFmt = AV_VIDEO_CODEC_H264;
409 mTsParams[1].audioFmt = AV_AUDIO_CODEC_MP3;
410 mTsParams[1].vpid = 0xc8;
411 mTsParams[1].apid = 0xc9;
412 mTsParams[1].inPutTsType = TS_MEMORY;
413 mTsParams[1].drmmode = TS_INPUT_BUFFER_TYPE_NORMAL;
414 mTsParams[1].index = 1;
415 mTsParams[1].mThread = 0;
416 mTsParams[1].run = true;
417 mTsParams[1].mute = true;
418 mTsParams[1].setAudio = false;
419#endif
420#if 0
421
422 char const* file2 = "/data/2.ts";
423 memcpy(mTsParams[2].tsFilePatch,file2,strlen(file2));
424 mTsParams[2].videoFmt = AV_VIDEO_CODEC_H264;
425 mTsParams[2].audioFmt = AV_AUDIO_CODEC_MP3;
426 mTsParams[2].vpid = 0xc8;
427 mTsParams[2].apid = 0xc9;
428 mTsParams[2].inPutTsType = TS_MEMORY;
429 mTsParams[2].drmmode = TS_INPUT_BUFFER_TYPE_NORMAL;
430 mTsParams[2].index = 2;
431 mTsParams[2].mThread = 0;
432 mTsParams[2].mute = true;
433 mTsParams[2].setAudio = false;
434
435 char const* file3 = "/data/2.ts";
436 memcpy(mTsParams[3].tsFilePatch,file3,strlen(file3));
437 mTsParams[3].videoFmt = AV_VIDEO_CODEC_H264;
438 mTsParams[3].audioFmt = AV_AUDIO_CODEC_MP3;
439 mTsParams[3].vpid = 0xc8;
440 mTsParams[3].apid = 0xc9;
441 mTsParams[3].inPutTsType = TS_MEMORY;
442 mTsParams[3].drmmode = TS_INPUT_BUFFER_TYPE_NORMAL;
443 mTsParams[3].index = 3;
444 mTsParams[3].mThread = 0;
445 mTsParams[3].run = true;
446 mTsParams[3].mute = true;
447 mTsParams[3].setAudio = false;
448#endif
449
450 int i = 0;
451 for (i = 0 ; i < changNum;i++) {
452 if (pthread_create(&(mTsParams[i].mThread), NULL, TsPlayerThread, &mTsParams[i]) == 0) {
453 printf("pthread_create %d ok %ld\n",i,mTsParams[i].mThread);
454 } else {
455 mTsParams[i].mThread = 0;
456 printf("pthread_create %d errno:%d (%s) \n",i,errno,strerror(errno));
457 }
458 usleep(5000);
459 }
460
461 char ch = 0;
462 bool isExit = false;
463 int run_false_index_count = 0;
464 while (1) {
465 if (_kbhit()) {
466 ch = getchar();
467 printfStatus(mTsParams,ch);
468 switch (ch) {
469 case 'q':
470 printf("----break\n");
471 for (i = 0 ; i < 4;i++) {
472 mTsParams[i].run = false;
473 }
474 isExit = true;
475 break;
476 case '0':
477 changAudio(mTsParams, 0);
478 break;
479 case '1':
480 changAudio(mTsParams, 1);
481 break;
482 case '2':
483 changAudio(mTsParams, 2);
484 break;
485 case '3':
486 changAudio(mTsParams, 3);
487 break;
488 case 'p':
489 //stop video 2 decoding,test
490 stopVideo(mTsParams,changNum-1);
491 break;
492 case 'r':
493 //start video 2 decoding,test
494 startVideo(mTsParams,changNum-1);
495 break;
zhipeng.he92870d82022-03-15 18:06:20 +0800496 case 'h':
497 printf("----hide the video 1\n");
498 hideVideo(mTsParams,changNum-1);
499 break;
500 case 's':
501 printf("----show the video 1\n");
502 showVideo(mTsParams,changNum-1);
503 break;
zhipeng.he60bbc802021-09-13 16:12:40 +0800504 default:
505 break;
506 }
507 ch = 0;
508 }
zhipeng.hef31a5c82021-11-17 20:50:31 +0800509 run_false_index_count = 0;
zhipeng.he60bbc802021-09-13 16:12:40 +0800510 for (i = 0 ; i < changNum;i++) {
511 if (mTsParams[i].run == false) {
512 run_false_index_count++;
513 }
514 }
515 if (run_false_index_count == changNum) {
516 isExit = true;
517 printf("---->all channel exit changNum:%d \n",changNum);
518 }
519 if (isExit) {
520 break;
521 }
522 usleep(200000);
523 }
524
525 for (i = 0 ; i < changNum;i++) {
526 printf("---->pthread_join %d \n",i);
527 mTsParams[i].run = false;
528 if (mTsParams[i].mThread != 0) {
529 pthread_join(mTsParams[i].mThread, NULL);
530 mTsParams[i].mThread = 0;
531 }
532 }
533
534 printf("---->pthread_join end\n");
535 return 0;
536}