1 /*--------------------------------------------------------------------------
2 Copyright (c) 2010, Code Aurora Forum. All rights reserved.
3
4 Redistribution and use in source and binary forms, with or without
5 modification, are permitted provided that the following conditions are met:
6 * Redistributions of source code must retain the above copyright
7 notice, this list of conditions and the following disclaimer.
8 * Redistributions in binary form must reproduce the above copyright
9 notice, this list of conditions and the following disclaimer in the
10 documentation and/or other materials provided with the distribution.
11 * Neither the name of Code Aurora nor
12 the names of its contributors may be used to endorse or promote
13 products derived from this software without specific prior written
14 permission.
15
16 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
17 AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19 NON-INFRINGEMENT ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
20 CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
21 EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
22 PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
23 OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
24 WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
25 OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
26 ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 --------------------------------------------------------------------------*/
28 #include "video_encoder_test.h"
29
30 #define DEBUG_PRINT printf
31 /************************************************************************/
32 /* #DEFINES */
33 /************************************************************************/
34
35 #define VOP_START_CODE 0x000001B6
36 #define SHORT_HEADER_START_CODE 0x00008000
37 #define H264_START_CODE 0x00000001
38
39 /************************************************************************/
40 /* STATIC VARIABLES */
41 /************************************************************************/
42
43 static int Code_type;
44 static int total_frames = 0;
45 static unsigned int header_code = 0;
46 static pthread_mutex_t read_lock;
47
48 static unsigned int read_frame ( unsigned char *dataptr,unsigned int length,
49 FILE * inputBufferFile
50 );
clp2(unsigned x)51 static unsigned clp2(unsigned x)
52 {
53 x = x - 1;
54 x = x | (x >> 1);
55 x = x | (x >> 2);
56 x = x | (x >> 4);
57 x = x | (x >> 8);
58 x = x | (x >>16);
59 return x + 1;
60 }
61
62
63 static void* video_thread (void *);
64 static void* async_thread (void *);
65
66
67
main(int argc,char ** argv)68 int main (int argc, char **argv)
69 {
70 struct video_encoder_context *encoder_context = NULL;
71 char *file_name = NULL;
72 FILE *file_ptr = NULL;
73 int temp1 =0,temp2 =0;
74 int error = 1;
75 unsigned int i = 0;
76
77 file_name = argv [1];
78 file_ptr = fopen (file_name,"rb");
79
80 if (file_ptr == NULL)
81 {
82 DEBUG_PRINT("\n File is not located ");
83 return -1;
84 }
85
86
87 encoder_context = (struct video_encoder_context *) \
88 calloc (sizeof (struct video_encoder_context),1);
89
90 encoder_context->outputBufferFile = NULL;
91 encoder_context->inputBufferFile = NULL;
92 encoder_context->video_driver_fd = -1;
93
94 if (encoder_context == NULL)
95 {
96 return -1;
97 }
98
99 encoder_context->inputBufferFile = file_ptr;
100 encoder_context->input_width = 176;
101 encoder_context->input_height = 144;
102 encoder_context->codectype = VEN_CODEC_MPEG4;
103 encoder_context->fps_num = 60;
104 encoder_context->fps_den = 2;
105 encoder_context->inputformat = VEN_INPUTFMT_NV12;
106 encoder_context->targetbitrate = 128000;
107
108 file_ptr = fopen ("/data/output.m4v","wb");
109 if (file_ptr == NULL)
110 {
111 DEBUG_PRINT("\n File can't be created");
112 return -1;
113 }
114 encoder_context->outputBufferFile = file_ptr;
115
116 switch (atoi(argv[2]))
117 {
118 case 0:
119 DEBUG_PRINT("\n MPEG4 codec selected");
120 encoder_context->codectype = VEN_CODEC_MPEG4;
121 Code_type = 0;
122 break;
123 case 1:
124 DEBUG_PRINT("\n H.263");
125 encoder_context->codectype = VEN_CODEC_H263;
126 Code_type = 0;
127 break;
128 case 2:
129 DEBUG_PRINT("\n H.264");
130 encoder_context->codectype = VEN_CODEC_H264;
131 Code_type = 1;
132 break;
133 default:
134 DEBUG_PRINT("\n Wrong codec type");
135 error = -1;
136 break;
137 }
138
139 if (error != -1)
140 {
141 temp1 = atoi(argv[3]);
142 temp2 = atoi(argv[4]);
143
144 if (((temp1%16) != 0) || ((temp2%16) != 0))
145 {
146 error = -1;
147 }
148 else
149 {
150 encoder_context->input_width = temp1;
151 encoder_context->input_height = temp2;
152 }
153 }
154
155 switch (atoi(argv[5]))
156 {
157 case 0:
158 DEBUG_PRINT("\n No Sink");
159 encoder_context->outputBufferFile = NULL;
160 break;
161 }
162
163 if (error != -1)
164 {
165 encoder_context->targetbitrate = atoi (argv[6]);
166 }
167
168 if ( error != -1 && (init_encoder (encoder_context) == -1 ))
169 {
170 DEBUG_PRINT("\n Init decoder fails ");
171 error = -1;
172 }
173 DEBUG_PRINT("\n Decoder open successfull");
174
175
176 /*Allocate input and output buffers*/
177 if (error != -1 && (allocate_buffer (0,encoder_context)== -1))
178 {
179 DEBUG_PRINT("\n Error in input Buffer allocation");
180 error = -1;
181 }
182
183 if (error != -1 && (allocate_buffer (1,encoder_context)== -1))
184 {
185 DEBUG_PRINT("\n Error in output Buffer allocation");
186 error = -1;
187 }
188
189
190 if (error != -1 && (start_encoding (encoder_context) == -1))
191 {
192 DEBUG_PRINT("\n Error in start decoding call");
193 error = -1;
194 }
195
196 if (error != -1 && (stop_encoding (encoder_context) == -1))
197 {
198 DEBUG_PRINT("\n Error in stop decoding call");
199 error = -1;
200 }
201
202 DEBUG_PRINT("\n De-init the decoder");
203 if ((deinit_encoder (encoder_context) == -1))
204 {
205 error = -1;
206 }
207
208
209 (void)free_buffer (INPUT_BUFFER,encoder_context);
210 (void)free_buffer (OUTPUT_BUFFER,encoder_context);
211
212 if (encoder_context->inputBufferFile != NULL)
213 {
214 fclose (encoder_context->inputBufferFile);
215 }
216 if (encoder_context->outputBufferFile != NULL)
217 {
218 fclose (encoder_context->outputBufferFile);
219 }
220 DEBUG_PRINT ("\n Total Number of frames decoded %d",total_frames);
221 DEBUG_PRINT("\n closing the driver");
222 free (encoder_context);
223
224 return error;
225 }
226
init_encoder(struct video_encoder_context * init_decode)227 int init_encoder ( struct video_encoder_context *init_decode )
228 {
229 struct venc_ioctl_msg ioctl_msg = {NULL,NULL};
230 struct venc_basecfg basecfg;
231 struct video_queue_context *queue_ptr = NULL;
232 struct venc_ratectrlcfg ratecrl;
233 pthread_mutexattr_t init_values;
234 struct venc_profile profile;
235 struct ven_profilelevel profilelevel;
236
237 DEBUG_PRINT("\n Before calling the open");
238
239 init_decode->video_driver_fd = open ("/dev/msm_vidc_enc", \
240 O_RDWR | O_NONBLOCK);
241
242
243
244 if (init_decode->video_driver_fd < 0)
245 {
246 DEBUG_PRINT("\n Open failed");
247 return -1;
248 }
249
250 basecfg.codectype = init_decode->codectype;
251 basecfg.dvs_height = 0;
252 basecfg.dvs_width = 0;
253 basecfg.fps_den = init_decode->fps_den;
254 basecfg.fps_num = init_decode->fps_num;
255 basecfg.input_height = init_decode->input_height;
256 basecfg.input_width = init_decode->input_width;
257 basecfg.inputformat = init_decode->inputformat;
258 basecfg.targetbitrate = init_decode->targetbitrate;
259
260 /*Initialize Decoder with codec type and resolution*/
261 ioctl_msg.inputparam = &basecfg;
262 ioctl_msg.outputparam = NULL;
263
264 if (ioctl (init_decode->video_driver_fd,VEN_IOCTL_SET_BASE_CFG,
265 (void*)&ioctl_msg) < 0)
266 {
267 DEBUG_PRINT("\n Set base config type failed");
268 return -1;
269 }
270
271 /*Initialize Decoder with codec type and resolution*/
272 DEBUG_PRINT ("\n Switch off rate control");
273 ioctl_msg.inputparam = &ratecrl;
274 ioctl_msg.outputparam = NULL;
275 ratecrl.rcmode = VEN_RC_OFF;
276 if (ioctl (init_decode->video_driver_fd,VEN_IOCTL_SET_RATE_CTRL_CFG,
277 (void*)&ioctl_msg) < 0)
278 {
279 DEBUG_PRINT("\n Set rate control failed");
280 return -1;
281 }
282
283 if (basecfg.codectype == VEN_CODEC_H264)
284 {
285 DEBUG_PRINT ("\n Set the VEN_IOCTL_SET_CODEC_PROFILE High");
286 ioctl_msg.inputparam = &profile;
287 ioctl_msg.outputparam = NULL;
288 profile.profile = VEN_PROFILE_H264_BASELINE;
289 if (ioctl (init_decode->video_driver_fd,VEN_IOCTL_SET_CODEC_PROFILE,
290 (void*)&ioctl_msg) < 0)
291 {
292 DEBUG_PRINT("\n Set VEN_IOCTL_SET_CODEC_PROFILE failed");
293 return -1;
294 }
295
296 DEBUG_PRINT ("\n Set the VEN_IOCTL_SET_CODEC_PROFILE High");
297 ioctl_msg.inputparam = &profilelevel;
298 ioctl_msg.outputparam = NULL;
299 profilelevel.level = VEN_LEVEL_H264_1p1;
300 if (ioctl (init_decode->video_driver_fd,VEN_IOCTL_SET_PROFILE_LEVEL,
301 (void*)&ioctl_msg) < 0)
302 {
303 DEBUG_PRINT("\n Set VEN_IOCTL_SET_CODEC_PROFILE failed");
304 return -1;
305 }
306
307 if (basecfg.input_width > 720)
308 {
309 DEBUG_PRINT ("\n Set the VEN_IOCTL_SET_CODEC_PROFILE High");
310 ioctl_msg.inputparam = &profile;
311 ioctl_msg.outputparam = NULL;
312 profile.profile = VEN_PROFILE_H264_HIGH;
313 if (ioctl (init_decode->video_driver_fd,VEN_IOCTL_SET_CODEC_PROFILE,
314 (void*)&ioctl_msg) < 0)
315 {
316 DEBUG_PRINT("\n Set VEN_IOCTL_SET_CODEC_PROFILE failed");
317 return -1;
318 }
319
320 DEBUG_PRINT ("\n Set the VEN_IOCTL_SET_CODEC_PROFILE High");
321 ioctl_msg.inputparam = &profilelevel;
322 ioctl_msg.outputparam = NULL;
323 profilelevel.level = VEN_LEVEL_H264_3p1;
324 if (ioctl (init_decode->video_driver_fd,VEN_IOCTL_SET_PROFILE_LEVEL,
325 (void*)&ioctl_msg) < 0)
326 {
327 DEBUG_PRINT("\n Set VEN_IOCTL_SET_CODEC_PROFILE failed");
328 return -1;
329 }
330 }
331 }
332
333 DEBUG_PRINT("\n Query Input bufffer requirements");
334 /*Get the Buffer requirements for input and output ports*/
335
336
337
338 ioctl_msg.inputparam = NULL;
339 ioctl_msg.outputparam = &init_decode->input_buffer;
340
341 if (ioctl (init_decode->video_driver_fd,VEN_IOCTL_GET_INPUT_BUFFER_REQ,
342 (void*)&ioctl_msg) < 0)
343 {
344 DEBUG_PRINT("\n Requesting for input buffer requirements failed");
345 return -1;
346 }
347
348 DEBUG_PRINT("\n input Size=%d min count =%d actual count = %d", \
349 (int)init_decode->input_buffer.datasize,\
350 (int)init_decode->input_buffer.mincount,\
351 (int)init_decode->input_buffer.actualcount);
352
353
354 ioctl_msg.inputparam = &init_decode->input_buffer;
355 ioctl_msg.outputparam = NULL;
356 init_decode->input_buffer.actualcount = init_decode->input_buffer.mincount + 2;
357
358 if (ioctl (init_decode->video_driver_fd,VEN_IOCTL_SET_INPUT_BUFFER_REQ,
359 (void*)&ioctl_msg) < 0)
360 {
361 DEBUG_PRINT("\n Set Buffer Requirements Failed");
362 return -1;
363 }
364
365
366 DEBUG_PRINT("\n Query output bufffer requirements");
367 ioctl_msg.inputparam = NULL;
368 ioctl_msg.outputparam = &init_decode->output_buffer;
369
370 if (ioctl (init_decode->video_driver_fd,VEN_IOCTL_GET_OUTPUT_BUFFER_REQ,
371 (void*)&ioctl_msg) < 0)
372 {
373 DEBUG_PRINT("\n Requesting for output buffer requirements failed");
374 return -1;
375 }
376
377 DEBUG_PRINT("\n output Size=%d min count =%d actual count = %d", \
378 (int)init_decode->output_buffer.datasize,\
379 (int)init_decode->output_buffer.mincount,\
380 (int)init_decode->output_buffer.actualcount);
381
382 /*Create Queue related data structures*/
383 queue_ptr = &init_decode->queue_context;
384 queue_ptr->commandq_size = 50;
385 queue_ptr->dataq_size = 50;
386
387 sem_init(&queue_ptr->sem_message,0, 0);
388 sem_init(&init_decode->sem_synchronize,0, 0);
389
390 pthread_mutexattr_init (&init_values);
391 pthread_mutex_init (&queue_ptr->mutex,&init_values);
392 pthread_mutex_init (&read_lock,&init_values);
393 DEBUG_PRINT("\n create Queues");
394 queue_ptr->ptr_cmdq = (struct video_msgq*) \
395 calloc (sizeof (struct video_msgq),
396 queue_ptr->commandq_size);
397 queue_ptr->ptr_dataq = (struct video_msgq*) \
398 calloc (sizeof (struct video_msgq),
399 queue_ptr->dataq_size
400 );
401
402 if ( queue_ptr->ptr_cmdq == NULL ||
403 queue_ptr->ptr_dataq == NULL
404 )
405 {
406 return -1;
407 }
408 DEBUG_PRINT("\n create Threads");
409 /*Create two threads*/
410 if ( (pthread_create (&init_decode->videothread_id,NULL,video_thread,
411 init_decode) < 0) ||
412 (pthread_create (&init_decode->asyncthread_id,NULL,async_thread,
413 init_decode) < 0)
414 )
415 {
416 return -1;
417 }
418
419 return 1;
420 }
421
422
423
free_buffer(unsigned int buffer_dir,struct video_encoder_context * encoder_context)424 int free_buffer ( unsigned int buffer_dir,
425 struct video_encoder_context *encoder_context
426 )
427 {
428 unsigned int buffercount = 0,i=0;
429 struct venc_bufferpayload **ptemp = NULL;
430
431 if (encoder_context == NULL)
432 {
433 return -1;
434 }
435
436 if (buffer_dir == INPUT_BUFFER && encoder_context->ptr_inputbuffer)
437 {
438 buffercount = encoder_context->input_buffer.actualcount;
439 ptemp = encoder_context->ptr_inputbuffer;
440
441 for (i=0;i<buffercount;i++)
442 {
443 if (ptemp [i])
444 {
445 if (ptemp [i]->fd != -1)
446 {
447 munmap ( ptemp [i]->pbuffer,ptemp [i]->maped_size);
448 ptemp [i]->pbuffer = NULL;
449 close (ptemp [i]->fd);
450 }
451 free (ptemp [i]);
452 ptemp [i] = NULL;
453 }
454 }
455 free (encoder_context->ptr_inputbuffer);
456 encoder_context->ptr_inputbuffer = NULL;
457 }
458 else if ( buffer_dir == OUTPUT_BUFFER && encoder_context->ptr_outputbuffer )
459 {
460 buffercount = encoder_context->output_buffer.actualcount;
461 ptemp = encoder_context->ptr_outputbuffer;
462
463 if (ptemp)
464 {
465 for (i=0;i<buffercount;i++)
466 {
467 if (ptemp [i])
468 {
469 if (ptemp [i]->fd != -1)
470 {
471 munmap ( ptemp [i]->pbuffer,ptemp [i]->maped_size);
472 ptemp [i]->pbuffer = NULL;
473 close (ptemp [i]->fd);
474 }
475 free (ptemp [i]);
476 ptemp [i] = NULL;
477 }
478 }
479 free (ptemp);
480 encoder_context->ptr_outputbuffer = NULL;
481 }
482 }
483
484 return 1;
485 }
486
allocate_buffer(unsigned int buffer_dir,struct video_encoder_context * encoder_context)487 int allocate_buffer ( unsigned int buffer_dir,
488 struct video_encoder_context *encoder_context
489 )
490 {
491 struct venc_bufferpayload **ptemp = NULL;
492 struct venc_ioctl_msg ioctl_msg = {NULL,NULL};
493 unsigned int buffercount = 0,i=0,alignedsize=0;
494 unsigned int buffersize = 0;
495
496 if ( encoder_context == NULL)
497 {
498 DEBUG_PRINT ("\nallocate_buffer: context is NULL");
499 return -1;
500 }
501
502 if ( buffer_dir == INPUT_BUFFER )
503 {
504 /*Check if buffers are allocated*/
505 if (encoder_context->ptr_inputbuffer != NULL)
506 {
507 DEBUG_PRINT ("\nallocate_buffer: encoder_context->ptr_inputbuffer is set");
508 return -1;
509 }
510
511 buffercount = encoder_context->input_buffer.actualcount;
512 alignedsize = encoder_context->input_buffer.alignment;
513 buffersize = encoder_context->input_buffer.datasize;
514 buffersize = (buffersize + alignedsize) & (~alignedsize);
515 }
516 else if (buffer_dir == OUTPUT_BUFFER)
517 {
518 /*Check if buffers are allocated*/
519 if (encoder_context->ptr_outputbuffer != NULL)
520 {
521 DEBUG_PRINT ("\nallocate_buffer: Double allcoate output");
522 return -1;
523 }
524
525 buffercount = encoder_context->output_buffer.actualcount;
526 alignedsize = encoder_context->output_buffer.alignment;
527 buffersize = encoder_context->output_buffer.datasize;
528 buffersize = (buffersize + alignedsize) & (~alignedsize);
529
530 }
531 else
532 {
533 DEBUG_PRINT ("\nallocate_buffer: Wrong buffer directions");
534 return -1;
535 }
536
537 ptemp = (struct venc_bufferpayload **)\
538 calloc (sizeof (struct venc_bufferpayload *),buffercount);
539
540 if (ptemp == NULL)
541 {
542 DEBUG_PRINT ("\nallocate_buffer: venc_bufferpayload failure");
543 return -1;
544 }
545
546
547 if (buffer_dir == OUTPUT_BUFFER)
548 {
549 DEBUG_PRINT ("\nallocate_buffer: OUT");
550 encoder_context->ptr_outputbuffer = ptemp;
551 }
552 else
553 {
554 DEBUG_PRINT ("\nallocate_buffer: IN");
555 encoder_context->ptr_inputbuffer = ptemp;
556 }
557
558 /*Allocate buffer headers*/
559 for (i=0; i< buffercount; i++)
560 {
561 ptemp [i] = (struct venc_bufferpayload*)\
562 calloc (sizeof (struct venc_bufferpayload),1);
563
564 if (ptemp [i] == NULL)
565 {
566 DEBUG_PRINT ("\nallocate_buffer: ptemp [i] calloc failure");
567 return -1;
568 }
569 ptemp [i]->fd = -1;
570 }
571
572 for (i=0; i< buffercount; i++)
573 {
574 ptemp [i]->fd = open ("/dev/pmem_adsp", O_RDWR | O_SYNC);
575
576 if (ptemp [i]->fd < 0)
577 {
578 DEBUG_PRINT ("\nallocate_buffer: open pmem failed");
579 return -1;
580 }
581
582 ptemp [i]->pbuffer = mmap(NULL,clp2(buffersize),PROT_READ|PROT_WRITE,
583 MAP_SHARED,ptemp [i]->fd,0);
584 DEBUG_PRINT ("\n pmem fd = %d virt addr = %p",ptemp [i]->fd,\
585 ptemp [i]->pbuffer);
586 if (ptemp [i]->pbuffer == MAP_FAILED)
587 {
588 ptemp [i]->pbuffer = NULL;
589 DEBUG_PRINT ("\nallocate_buffer: MMAP failed");
590 return -1;
591 }
592 ptemp [i]->nsize = buffersize;
593 ptemp [i]->maped_size = clp2 (buffersize);
594
595 ioctl_msg.inputparam = ptemp [i];
596 ioctl_msg.outputparam = NULL;
597
598 if (buffer_dir == OUTPUT_BUFFER)
599 {
600 if (ioctl (encoder_context->video_driver_fd,VEN_IOCTL_SET_OUTPUT_BUFFER,
601 &ioctl_msg) < 0)
602 {
603 DEBUG_PRINT ("\nallocate_buffer: Set Output Buffer IOCTL failed");
604 return -1;
605 }
606 }
607 else
608 {
609 if (ioctl (encoder_context->video_driver_fd,VEN_IOCTL_SET_INPUT_BUFFER,
610 &ioctl_msg) < 0)
611 {
612 DEBUG_PRINT ("\nallocate_buffer: Set input Buffer IOCTL failed");
613 return -1;
614 }
615 }
616
617 }
618 DEBUG_PRINT ("\nallocate_buffer: Success");
619 return 1;
620 }
621
622
623
start_encoding(struct video_encoder_context * encoder_context)624 int start_encoding (struct video_encoder_context *encoder_context)
625 {
626 struct venc_ioctl_msg ioctl_msg = {NULL,NULL};
627 struct venc_buffer enc_buffer;
628 unsigned int i = 0;
629 unsigned int data_len =0;
630
631
632 if (encoder_context == NULL)
633 {
634 return -1;
635 }
636
637 if (ioctl (encoder_context->video_driver_fd,VEN_IOCTL_CMD_START,
638 NULL) < 0)
639 {
640 DEBUG_PRINT("\n Start failed");
641 return -1;
642 }
643
644 DEBUG_PRINT("\n Start Issued successfully waiting for Start Done");
645 /*Wait for Start command response*/
646 sem_wait (&encoder_context->sem_synchronize);
647
648 /*Push output Buffers*/
649 i = 0;
650 while (i < encoder_context->output_buffer.actualcount)
651 {
652 enc_buffer.clientdata = (void *)encoder_context->ptr_outputbuffer [i];
653 enc_buffer.flags = 0;
654 enc_buffer.size = encoder_context->ptr_outputbuffer [i]->nsize;
655 enc_buffer.len = 0;
656 enc_buffer.ptrbuffer = encoder_context->ptr_outputbuffer [i]->pbuffer;
657 enc_buffer.offset = 0;
658 enc_buffer.timestamp = 0;
659
660 DEBUG_PRINT ("\n Client Data on output = %p",(void *)enc_buffer.clientdata);
661 ioctl_msg.inputparam = &enc_buffer;
662 ioctl_msg.outputparam = NULL;
663
664 if (ioctl (encoder_context->video_driver_fd,
665 VEN_IOCTL_CMD_FILL_OUTPUT_BUFFER,&ioctl_msg) < 0)
666 {
667 DEBUG_PRINT("\n fill output frame failed");
668 return -1;
669 }
670 i++;
671 }
672
673
674 /*push input buffers*/
675 i = 0;
676 while (i < encoder_context->input_buffer.actualcount)
677 {
678 DEBUG_PRINT("\n Read Frame from File");
679
680 enc_buffer.clientdata = (void *)encoder_context->ptr_inputbuffer [i];
681 enc_buffer.flags = 0;
682 enc_buffer.size = encoder_context->ptr_inputbuffer [i]->nsize;
683 enc_buffer.len = 0;
684 enc_buffer.ptrbuffer = encoder_context->ptr_inputbuffer [i]->pbuffer;
685 enc_buffer.offset = 0;
686 enc_buffer.timestamp = total_frames *
687 ((encoder_context->fps_den * 1000000)/encoder_context->fps_num);
688 enc_buffer.len = (encoder_context->input_height *
689 encoder_context->input_width *3)/2;
690 data_len = read_frame ( enc_buffer.ptrbuffer,
691 enc_buffer.len,
692 encoder_context->inputBufferFile);
693 if (data_len == 0)
694 {
695 DEBUG_PRINT("\n Length is zero error");
696 return -1;
697 }
698 enc_buffer.len = data_len;
699 DEBUG_PRINT("\n Read Frame from File szie = %d",(int)data_len);
700
701 DEBUG_PRINT ("\n Client Data on output = %p",(void *)enc_buffer.clientdata);
702 ioctl_msg.inputparam = &enc_buffer;
703 ioctl_msg.outputparam = NULL;
704
705 if (ioctl (encoder_context->video_driver_fd,
706 VEN_IOCTL_CMD_ENCODE_FRAME,&ioctl_msg) < 0)
707 {
708 DEBUG_PRINT("\n Encode input frame failed");
709 return -1;
710 }
711 total_frames++;
712 i++;
713 }
714 DEBUG_PRINT ("\n Wait for EOS");
715 /*Wait for EOS or Error condition*/
716 sem_wait (&encoder_context->sem_synchronize);
717 DEBUG_PRINT ("\n Reached EOS");
718
719 return 1;
720 }
721
stop_encoding(struct video_encoder_context * encoder_context)722 int stop_encoding (struct video_encoder_context *encoder_context)
723 {
724 struct venc_ioctl_msg ioctl_msg = {NULL,NULL};
725 struct venc_bufferflush buffer_flush;
726
727 if (encoder_context == NULL)
728 {
729 return -1;
730 }
731 buffer_flush.flush_mode = VEN_FLUSH_INPUT;
732 ioctl_msg.inputparam = &buffer_flush;
733 ioctl_msg.outputparam = NULL;
734
735 if (ioctl(encoder_context->video_driver_fd,VEN_IOCTL_CMD_FLUSH,
736 &ioctl_msg) < 0)
737 {
738 DEBUG_PRINT("\n Flush input failed");
739 }
740 else
741 {
742 sem_wait (&encoder_context->sem_synchronize);
743 }
744
745 buffer_flush.flush_mode = VEN_FLUSH_OUTPUT;
746 ioctl_msg.inputparam = &buffer_flush;
747 ioctl_msg.outputparam = NULL;
748
749 if (ioctl(encoder_context->video_driver_fd,VEN_IOCTL_CMD_FLUSH,
750 &ioctl_msg) < 0)
751 {
752 DEBUG_PRINT("\n Flush output failed");
753 }
754 else
755 {
756 sem_wait (&encoder_context->sem_synchronize);
757 }
758
759 DEBUG_PRINT("\n Stop VEN_IOCTL_CMD_STOP");
760 if (ioctl(encoder_context->video_driver_fd,VEN_IOCTL_CMD_STOP,NULL) < 0)
761 {
762 DEBUG_PRINT("\n Stop failed");
763 }
764 else
765 {
766 sem_wait (&encoder_context->sem_synchronize);
767 }
768 return 1;
769 }
770
deinit_encoder(struct video_encoder_context * init_decode)771 int deinit_encoder (struct video_encoder_context *init_decode)
772 {
773 if (init_decode == NULL)
774 {
775 return -1;
776 }
777
778 /*Close the driver*/
779 if (init_decode->video_driver_fd != -1)
780 {
781 close (init_decode->video_driver_fd);
782 }
783
784 if (init_decode->queue_context.ptr_cmdq)
785 {
786 free (init_decode->queue_context.ptr_cmdq);
787 init_decode->queue_context.ptr_cmdq = NULL;
788 }
789
790 if (init_decode->queue_context.ptr_dataq)
791 {
792 free (init_decode->queue_context.ptr_dataq);
793 init_decode->queue_context.ptr_dataq = NULL;
794 }
795
796 sem_destroy (&init_decode->queue_context.sem_message);
797 sem_destroy (&init_decode->sem_synchronize);
798
799 pthread_mutex_destroy(&init_decode->queue_context.mutex);
800 pthread_mutex_destroy (&read_lock);
801
802 return 1;
803 }
804
video_thread(void * context)805 static void* video_thread (void *context)
806 {
807 struct video_encoder_context *encoder_context = NULL;
808 struct video_msgq *queueitem = NULL;
809 struct venc_ioctl_msg ioctl_msg = {NULL,NULL};
810 struct venc_bufferpayload *tempbuffer = NULL;
811 struct venc_buffer enc_buffer;
812 unsigned int data_len =0;
813
814
815 if (context == NULL)
816 {
817 DEBUG_PRINT("\n video thread recieved NULL context");
818 return NULL;
819 }
820 encoder_context = (struct video_encoder_context *) context;
821
822 /* Thread function which will accept commands from async thread
823 * or main thread
824 */
825 while (1)
826 {
827 queueitem = queue_get_cmd (&encoder_context ->queue_context);
828 if (queueitem != NULL)
829 {
830 switch (queueitem->cmd)
831 {
832 case VEN_MSG_START:
833 DEBUG_PRINT("\n recived start done command");
834 sem_post (&encoder_context->sem_synchronize);
835 break;
836
837 case VEN_MSG_STOP:
838 DEBUG_PRINT("\n recieved stop done");
839 sem_post (&encoder_context->sem_synchronize);
840 break;
841
842 case VEN_MSG_INPUT_BUFFER_DONE:
843
844 tempbuffer = (struct venc_bufferpayload *)queueitem->clientdata;
845 if (tempbuffer == NULL)
846 {
847 DEBUG_PRINT("\n FATAL ERROR input buffer address is bad");
848 sem_post (&encoder_context->sem_synchronize);
849 break;
850 }
851 tempbuffer->filled_len = (encoder_context->input_height *
852 encoder_context->input_width *3)/2;
853
854 data_len = read_frame ( tempbuffer->pbuffer,
855 tempbuffer->filled_len,
856 encoder_context->inputBufferFile);
857
858 if (data_len == 0)
859 {
860 DEBUG_PRINT ("\n End of stream reached");
861 sem_post (&encoder_context->sem_synchronize);
862 break;
863 }
864 enc_buffer.clientdata = (void *)tempbuffer;
865 enc_buffer.flags = 0;
866 enc_buffer.ptrbuffer = tempbuffer->pbuffer;
867 enc_buffer.size = tempbuffer->nsize;
868 enc_buffer.len = tempbuffer->filled_len;
869 enc_buffer.offset = 0;
870 enc_buffer.timestamp = total_frames *
871 ((encoder_context->fps_den * 1000000)/encoder_context->fps_num);
872
873 /*TODO: Time stamp needs to be updated*/
874 ioctl_msg.inputparam = &enc_buffer;
875 ioctl_msg.outputparam = NULL;
876 total_frames++;
877 if (ioctl(encoder_context->video_driver_fd,VEN_IOCTL_CMD_ENCODE_FRAME,
878 &ioctl_msg) < 0)
879 {
880 DEBUG_PRINT("\n Decoder frame failed");
881 sem_post (&encoder_context->sem_synchronize);
882 }
883 DEBUG_PRINT("\n Input buffer done send next buffer current value = %d",\
884 total_frames);
885 break;
886
887 case VEN_MSG_OUTPUT_BUFFER_DONE:
888
889 tempbuffer = (struct venc_bufferpayload *)queueitem->clientdata;
890 if (tempbuffer == NULL)
891 {
892 DEBUG_PRINT("\n FATAL ERROR input buffer address is bad");
893 sem_post (&encoder_context->sem_synchronize);
894 break;
895 }
896
897 if (encoder_context->outputBufferFile != NULL)
898 {
899 fwrite (tempbuffer->pbuffer,1,tempbuffer->filled_len,
900 encoder_context->outputBufferFile);
901 }
902
903
904 DEBUG_PRINT("\n recieved output buffer consume outbuffer");
905 DEBUG_PRINT("\nValues outputbuffer->bufferaddr = %p",\
906 tempbuffer->pbuffer);
907 enc_buffer.clientdata = (void *)tempbuffer;
908 enc_buffer.flags = 0;
909 enc_buffer.size = tempbuffer->nsize;
910 enc_buffer.len = 0;
911 enc_buffer.ptrbuffer = tempbuffer->pbuffer;
912 enc_buffer.offset = 0;
913 enc_buffer.timestamp = 0;
914
915 ioctl_msg.inputparam = &enc_buffer;
916 ioctl_msg.outputparam = NULL;
917
918 if (ioctl (encoder_context->video_driver_fd,
919 VEN_IOCTL_CMD_FILL_OUTPUT_BUFFER,&ioctl_msg) < 0)
920 {
921 DEBUG_PRINT("\n Decoder frame failed");
922 return NULL;
923 }
924
925 break;
926
927 case VEN_MSG_FLUSH_INPUT_DONE:
928 DEBUG_PRINT("\n Flush input complete");
929 sem_post (&encoder_context->sem_synchronize);
930 break;
931
932 case VEN_MSG_FLUSH_OUPUT_DONE:
933 DEBUG_PRINT("\n Flush output complete");
934 sem_post (&encoder_context->sem_synchronize);
935 break;
936 }
937 }
938 else
939 {
940 DEBUG_PRINT("\n Error condition recieved NULL from Queue");
941 }
942
943 if (queueitem->cmd == VEN_MSG_STOP)
944 {
945 DEBUG_PRINT("\n Playback has ended thread will exit");
946 return NULL;
947 }
948 }
949 }
950
async_thread(void * context)951 static void* async_thread (void *context)
952 {
953 struct video_encoder_context *encoder_context = NULL;
954 struct video_msgq queueitem ;
955 struct venc_msg venc_msg;
956 struct venc_bufferpayload *tempbuffer = NULL;
957 struct venc_ioctl_msg ioctl_msg = {NULL,NULL};
958 int result = -1;
959
960 if (context == NULL)
961 {
962 DEBUG_PRINT("\n aynsc thread recieved NULL context");
963 return NULL;
964 }
965 encoder_context = (struct video_encoder_context *) context;
966 DEBUG_PRINT("\n Entering the async thread");
967
968 while (1)
969 {
970 ioctl_msg.inputparam = NULL;
971 ioctl_msg.outputparam = (void*)&venc_msg;
972 DEBUG_PRINT ("\n Sizeof venc_msginfo = %d ",sizeof (venc_msg));
973 DEBUG_PRINT("\n Address of Venc msg in async thread %p",\
974 ioctl_msg.outputparam);
975 if (ioctl (encoder_context->video_driver_fd,VEN_IOCTL_CMD_READ_NEXT_MSG,\
976 (void*)&ioctl_msg) < 0)
977 {
978 DEBUG_PRINT("\n Error in ioctl read next msg");
979 }
980 else
981 {
982 switch (venc_msg.msgcode)
983 {
984 case VEN_MSG_START:
985 case VEN_MSG_STOP:
986 case VEN_MSG_INDICATION:
987 DEBUG_PRINT("\nSTOP/START Indiacation");
988 queueitem.cmd = venc_msg.msgcode;
989 queueitem.status = venc_msg.statuscode;
990 queueitem.clientdata = NULL;
991 break;
992
993 case VEN_MSG_INPUT_BUFFER_DONE:
994 DEBUG_PRINT("\nINPUT buffer done Indiacation");
995 queueitem.cmd = venc_msg.msgcode;
996 queueitem.status = venc_msg.statuscode;
997 queueitem.clientdata = (void *)venc_msg.buf.clientdata;
998 DEBUG_PRINT("\nInput Client data pointer is %p",queueitem.clientdata);
999 tempbuffer = (struct venc_bufferpayload *) queueitem.clientdata;
1000 DEBUG_PRINT ("\n Input Address of tempbuffer %p",tempbuffer);
1001 tempbuffer->filled_len = venc_msg.buf.len;
1002 DEBUG_PRINT ("\n Input value of tempbuffer tempbuffer->filled_len %d",(int)tempbuffer->filled_len);
1003 break;
1004 case VEN_MSG_OUTPUT_BUFFER_DONE:
1005 DEBUG_PRINT("\nOUPUT buffer done Indiacation");
1006 queueitem.cmd = venc_msg.msgcode;
1007 queueitem.status = venc_msg.statuscode;
1008 queueitem.clientdata = (void *)venc_msg.buf.clientdata;
1009 DEBUG_PRINT("\nOutput Client data pointer is %p",queueitem.clientdata);
1010 tempbuffer = (struct venc_bufferpayload *) queueitem.clientdata;
1011 DEBUG_PRINT ("\n Output Address of tempbuffer %p",tempbuffer);
1012 tempbuffer->filled_len = venc_msg.buf.len;
1013 DEBUG_PRINT ("\n Output value of tempbuffer tempbuffer->filled_len %d",(int)tempbuffer->filled_len);
1014 break;
1015
1016 default:
1017 DEBUG_PRINT("\nIn Default of get next message %d",(int)venc_msg.msgcode);
1018 queueitem.cmd = venc_msg.msgcode;
1019 queueitem.status = venc_msg.statuscode;
1020 queueitem.clientdata = NULL;
1021 break;
1022 }
1023 result = queue_post_cmdq (&encoder_context->queue_context,&queueitem);
1024 while (result == 0)
1025 {
1026 result = queue_post_cmdq (&encoder_context->queue_context,&queueitem);
1027 }
1028
1029 if (result == -1)
1030 {
1031 DEBUG_PRINT("\n FATAL ERROR WITH Queue");
1032 }
1033 }
1034 if (venc_msg.msgcode == VEN_MSG_STOP)
1035 {
1036 /*Thread can exit at this point*/
1037 return NULL;
1038 }
1039 }
1040 }
1041
1042
read_frame(unsigned char * dataptr,unsigned int length,FILE * inputBufferFile)1043 static unsigned int read_frame (unsigned char *dataptr, unsigned int length,
1044 FILE * inputBufferFile)
1045 {
1046
1047 unsigned int readOffset = 0;
1048 int bytes_read = 0;
1049 unsigned int code = 0;
1050 int found = 0;
1051
1052 DEBUG_PRINT ("\n Inside the readframe");
1053
1054 if (dataptr == NULL && length == 0)
1055 {
1056 DEBUG_PRINT ("\n dataptr = %p length = %d",dataptr,(int)length);
1057 return 0;
1058 }
1059
1060 pthread_mutex_lock(&read_lock);
1061 bytes_read = fread(&dataptr[readOffset],1,length,inputBufferFile);
1062 pthread_mutex_unlock(&read_lock);
1063
1064 return bytes_read;
1065 }
1066