1 /*--------------------------------------------------------------------------
2 Copyright (c) 2010-2011, 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 /*============================================================================
29                     V E N C _ T E S T. C P P
30 
31 DESCRIPTION
32 
33  This is the OMX test app .
34 
35 REFERENCES
36 
37 ============================================================================*/
38 
39 //usage
40 // FILE QVGA MP4 24 384000 100 enc_qvga.yuv QVGA_24.m4v
41 // FILE QCIF MP4 15 96000 0 foreman.qcif.yuv output_qcif.m4v
42 // FILE VGA MP4 24 1200000 218 enc_vga.yuv vga_output.m4v
43 #include <sys/types.h>
44 #include <sys/stat.h>
45 #include <string.h>
46 #include <unistd.h>
47 #include <stdlib.h>
48 #include <stdio.h>
49 #include <pthread.h>
50 #include <fcntl.h>
51 #include <sys/mman.h>
52 //#include <sys/time.h>
53 #include <time.h>
54 #include <sys/ioctl.h>
55 #include <limits.h>
56 #include <string.h>
57 //#include <sys/stat.h>
58 #include "OMX_QCOMExtns.h"
59 #include "OMX_Core.h"
60 
61 #define QCOM_EXT 1
62 
63 #include "OMX_Core.h"
64 #include "OMX_Video.h"
65 #include "OMX_Component.h"
66 #include "camera_test.h"
67 #include "fb_test.h"
68 #include "venc_util.h"
69 #include "extra_data_handler.h"
70 #ifdef USE_ION
71 #include <linux/msm_ion.h>
72 #endif
73 
74 //////////////////////////
75 // MACROS
76 //////////////////////////
77 
78 #define CHK(result) if (result != OMX_ErrorNone) { E("*************** error *************"); exit(0); }
79 #define TEST_LOG
80 #ifdef VENC_SYSLOG
81 #include "cutils/log.h"
82 /// Debug message macro
83 #define D(fmt, ...) LOGE("venc_test Debug %s::%d " fmt "\n",            \
84                          __FUNCTION__, __LINE__,                        \
85                          ## __VA_ARGS__)
86 
87 /// Error message macro
88 #define E(fmt, ...) LOGE("venc_test Error %s::%d " fmt "\n",          \
89                          __FUNCTION__, __LINE__,                      \
90                          ## __VA_ARGS__)
91 
92 #else
93      #ifdef TEST_LOG
94        #define D(fmt, ...) fprintf(stderr, "venc_test Debug %s::%d " fmt "\n", \
95                             __FUNCTION__, __LINE__,                     \
96                             ## __VA_ARGS__)
97 
98      /// Error message macro
99       #define E(fmt, ...) fprintf(stderr, "venc_test Error %s::%d " fmt "\n", \
100                             __FUNCTION__, __LINE__,                   \
101                             ## __VA_ARGS__)
102      #else
103       #define D(fmt, ...)
104       #define E(fmt, ...)
105          #endif
106 
107 #endif
108 
109 //////////////////////////
110 // CONSTANTS
111 //////////////////////////
112 static const int MAX_MSG = 100;
113 //#warning do not hardcode these use port definition
114 static const int PORT_INDEX_IN = 0;
115 static const int PORT_INDEX_OUT = 1;
116 
117 static const int NUM_IN_BUFFERS = 10;
118 static const int NUM_OUT_BUFFERS = 10;
119 
120 unsigned int num_in_buffers = 0;
121 unsigned int num_out_buffers = 0;
122 
123 //////////////////////////
124 /* MPEG4 profile and level table*/
125 static const unsigned int mpeg4_profile_level_table[][5]=
126 {
127     /*max mb per frame, max mb per sec, max bitrate, level, profile*/
128     {99,1485,64000,OMX_VIDEO_MPEG4Level0,OMX_VIDEO_MPEG4ProfileSimple},
129     {99,1485,64000,OMX_VIDEO_MPEG4Level1,OMX_VIDEO_MPEG4ProfileSimple},
130     {396,5940,128000,OMX_VIDEO_MPEG4Level2,OMX_VIDEO_MPEG4ProfileSimple},
131     {396,11880,384000,OMX_VIDEO_MPEG4Level3,OMX_VIDEO_MPEG4ProfileSimple},
132     {1200,36000,4000000,OMX_VIDEO_MPEG4Level4a,OMX_VIDEO_MPEG4ProfileSimple},
133     {1620,40500,8000000,OMX_VIDEO_MPEG4Level5,OMX_VIDEO_MPEG4ProfileSimple},
134     {3600,108000,12000000,OMX_VIDEO_MPEG4Level5,OMX_VIDEO_MPEG4ProfileSimple},
135     {0,0,0,0,0},
136 
137     {99,1485,128000,OMX_VIDEO_MPEG4Level0,OMX_VIDEO_MPEG4ProfileAdvancedSimple},
138     {99,1485,128000,OMX_VIDEO_MPEG4Level1,OMX_VIDEO_MPEG4ProfileAdvancedSimple},
139     {396,5940,384000,OMX_VIDEO_MPEG4Level2,OMX_VIDEO_MPEG4ProfileAdvancedSimple},
140     {396,11880,768000,OMX_VIDEO_MPEG4Level3,OMX_VIDEO_MPEG4ProfileAdvancedSimple},
141     {792,23760,3000000,OMX_VIDEO_MPEG4Level4,OMX_VIDEO_MPEG4ProfileAdvancedSimple},
142     {1620,48600,8000000,OMX_VIDEO_MPEG4Level5,OMX_VIDEO_MPEG4ProfileAdvancedSimple},
143     {0,0,0,0,0},
144 };
145 
146 /* H264 profile and level table*/
147 static const unsigned int h264_profile_level_table[][5]=
148 {
149      /*max mb per frame, max mb per sec, max bitrate, level, profile*/
150     {99,1485,64000,OMX_VIDEO_AVCLevel1,OMX_VIDEO_AVCProfileBaseline},
151     {99,1485,128000,OMX_VIDEO_AVCLevel1b,OMX_VIDEO_AVCProfileBaseline},
152     {396,3000,192000,OMX_VIDEO_AVCLevel11,OMX_VIDEO_AVCProfileBaseline},
153     {396,6000,384000,OMX_VIDEO_AVCLevel12,OMX_VIDEO_AVCProfileBaseline},
154     {396,11880,768000,OMX_VIDEO_AVCLevel13,OMX_VIDEO_AVCProfileBaseline},
155     {396,11880,2000000,OMX_VIDEO_AVCLevel2,OMX_VIDEO_AVCProfileBaseline},
156     {792,19800,4000000,OMX_VIDEO_AVCLevel21,OMX_VIDEO_AVCProfileBaseline},
157     {1620,20250,4000000,OMX_VIDEO_AVCLevel22,OMX_VIDEO_AVCProfileBaseline},
158     {1620,40500,10000000,OMX_VIDEO_AVCLevel3,OMX_VIDEO_AVCProfileBaseline},
159     {3600,108000,14000000,OMX_VIDEO_AVCLevel31,OMX_VIDEO_AVCProfileBaseline},
160     {5120,216000,20000000,OMX_VIDEO_AVCLevel32,OMX_VIDEO_AVCProfileBaseline},
161     {8192,245760,20000000,OMX_VIDEO_AVCLevel4,OMX_VIDEO_AVCProfileBaseline},
162     {0,0,0,0,0},
163 
164     {99,1485,64000,OMX_VIDEO_AVCLevel1,OMX_VIDEO_AVCProfileHigh},
165     {99,1485,160000,OMX_VIDEO_AVCLevel1b,OMX_VIDEO_AVCProfileHigh},
166     {396,3000,240000,OMX_VIDEO_AVCLevel11,OMX_VIDEO_AVCProfileHigh},
167     {396,6000,480000,OMX_VIDEO_AVCLevel12,OMX_VIDEO_AVCProfileHigh},
168     {396,11880,960000,OMX_VIDEO_AVCLevel13,OMX_VIDEO_AVCProfileHigh},
169     {396,11880,2500000,OMX_VIDEO_AVCLevel2,OMX_VIDEO_AVCProfileHigh},
170     {792,19800,5000000,OMX_VIDEO_AVCLevel21,OMX_VIDEO_AVCProfileHigh},
171     {1620,20250,5000000,OMX_VIDEO_AVCLevel22,OMX_VIDEO_AVCProfileHigh},
172     {1620,40500,12500000,OMX_VIDEO_AVCLevel3,OMX_VIDEO_AVCProfileHigh},
173     {3600,108000,17500000,OMX_VIDEO_AVCLevel31,OMX_VIDEO_AVCProfileHigh},
174     {5120,216000,25000000,OMX_VIDEO_AVCLevel32,OMX_VIDEO_AVCProfileHigh},
175     {8192,245760,25000000,OMX_VIDEO_AVCLevel4,OMX_VIDEO_AVCProfileHigh},
176     {0,0,0,0,0},
177 
178     {99,1485,64000,OMX_VIDEO_AVCLevel1,OMX_VIDEO_AVCProfileMain},
179     {99,1485,128000,OMX_VIDEO_AVCLevel1b,OMX_VIDEO_AVCProfileMain},
180     {396,3000,192000,OMX_VIDEO_AVCLevel11,OMX_VIDEO_AVCProfileMain},
181     {396,6000,384000,OMX_VIDEO_AVCLevel12,OMX_VIDEO_AVCProfileMain},
182     {396,11880,768000,OMX_VIDEO_AVCLevel13,OMX_VIDEO_AVCProfileMain},
183     {396,11880,2000000,OMX_VIDEO_AVCLevel2,OMX_VIDEO_AVCProfileMain},
184     {792,19800,4000000,OMX_VIDEO_AVCLevel21,OMX_VIDEO_AVCProfileMain},
185     {1620,20250,4000000,OMX_VIDEO_AVCLevel22,OMX_VIDEO_AVCProfileMain},
186     {1620,40500,10000000,OMX_VIDEO_AVCLevel3,OMX_VIDEO_AVCProfileMain},
187     {3600,108000,14000000,OMX_VIDEO_AVCLevel31,OMX_VIDEO_AVCProfileMain},
188     {5120,216000,20000000,OMX_VIDEO_AVCLevel32,OMX_VIDEO_AVCProfileMain},
189     {8192,245760,20000000,OMX_VIDEO_AVCLevel4,OMX_VIDEO_AVCProfileMain},
190     {0,0,0,0,0}
191 
192 };
193 
194 /* H263 profile and level table*/
195 static const unsigned int h263_profile_level_table[][5]=
196 {
197     /*max mb per frame, max mb per sec, max bitrate, level, profile*/
198     {99,1485,64000,OMX_VIDEO_H263Level10,OMX_VIDEO_H263ProfileBaseline},
199     {396,5940,128000,OMX_VIDEO_H263Level20,OMX_VIDEO_H263ProfileBaseline},
200     {396,11880,384000,OMX_VIDEO_H263Level30,OMX_VIDEO_H263ProfileBaseline},
201     {396,11880,2048000,OMX_VIDEO_H263Level40,OMX_VIDEO_H263ProfileBaseline},
202     {99,1485,128000,OMX_VIDEO_H263Level45,OMX_VIDEO_H263ProfileBaseline},
203     {396,19800,4096000,OMX_VIDEO_H263Level50,OMX_VIDEO_H263ProfileBaseline},
204     {810,40500,8192000,OMX_VIDEO_H263Level60,OMX_VIDEO_H263ProfileBaseline},
205     {1620,81000,16384000,OMX_VIDEO_H263Level70,OMX_VIDEO_H263ProfileBaseline},
206     {0,0,0,0,0}
207 };
208 
209 #define Log2(number, power)  { OMX_U32 temp = number; power = 0; while( (0 == (temp & 0x1)) &&  power < 16) { temp >>=0x1; power++; } }
210 #define FractionToQ16(q,num,den) { OMX_U32 power; Log2(den,power); q = num << (16 - power); }
211 
212 //////////////////////////
213 // TYPES
214 //////////////////////////
215 struct ProfileType
216 {
217    OMX_VIDEO_CODINGTYPE eCodec;
218    OMX_VIDEO_MPEG4LEVELTYPE eLevel;
219    OMX_VIDEO_CONTROLRATETYPE eControlRate;
220    OMX_VIDEO_AVCSLICEMODETYPE eSliceMode;
221    OMX_U32 nFrameWidth;
222    OMX_U32 nFrameHeight;
223    OMX_U32 nFrameBytes;
224 #ifdef BADGER
225    OMX_U32 nFramestride;
226    OMX_U32 nFrameScanlines;
227    OMX_U32 nFrameRead;
228 #endif
229    OMX_U32 nBitrate;
230    float nFramerate;
231    char* cInFileName;
232    char* cOutFileName;
233    OMX_U32 nUserProfile;
234 };
235 
236 enum MsgId
237 {
238    MSG_ID_OUTPUT_FRAME_DONE,
239    MSG_ID_INPUT_FRAME_DONE,
240    MSG_ID_MAX
241 };
242 union MsgData
243 {
244    struct
245    {
246       OMX_BUFFERHEADERTYPE* pBuffer;
247    } sBitstreamData;
248 };
249 struct Msg
250 {
251    MsgId id;
252    MsgData data;
253 };
254 struct MsgQ
255 {
256    Msg q[MAX_MSG];
257    int head;
258    int size;
259 };
260 
261 enum Mode
262 {
263    MODE_PREVIEW,
264    MODE_DISPLAY,
265    MODE_PROFILE,
266    MODE_FILE_ENCODE,
267    MODE_LIVE_ENCODE
268 };
269 
270 enum ResyncMarkerType
271 {
272    RESYNC_MARKER_NONE,     ///< No resync marker
273    RESYNC_MARKER_BYTE,     ///< BYTE Resync marker for MPEG4, H.264
274    RESYNC_MARKER_MB,       ///< MB resync marker for MPEG4, H.264
275    RESYNC_MARKER_GOB       ///< GOB resync marker for H.263
276 };
277 
278 union DynamicConfigData
279 {
280    OMX_VIDEO_CONFIG_BITRATETYPE bitrate;
281    OMX_CONFIG_FRAMERATETYPE framerate;
282    QOMX_VIDEO_INTRAPERIODTYPE intraperiod;
283    OMX_CONFIG_INTRAREFRESHVOPTYPE intravoprefresh;
284    OMX_CONFIG_ROTATIONTYPE rotation;
285    float f_framerate;
286 };
287 
288 struct DynamicConfig
289 {
290    bool pending;
291    unsigned frame_num;
292    OMX_INDEXTYPE config_param;
293    union DynamicConfigData config_data;
294 };
295 
296 #ifdef USE_ION
297 struct enc_ion
298 {
299    int ion_device_fd;
300    struct ion_allocation_data alloc_data;
301    struct ion_fd_data ion_alloc_fd;
302 };
303 #endif
304 
305 //////////////////////////
306 // MODULE VARS
307 //////////////////////////
308 static pthread_mutex_t m_mutex;
309 static pthread_cond_t m_signal;
310 static MsgQ m_sMsgQ;
311 
312 //#warning determine how many buffers we really have
313 OMX_STATETYPE m_eState = OMX_StateInvalid;
314 OMX_COMPONENTTYPE m_sComponent;
315 OMX_HANDLETYPE m_hHandle = NULL;
316 OMX_BUFFERHEADERTYPE* m_pOutBuffers[NUM_OUT_BUFFERS] = {NULL};
317 OMX_BUFFERHEADERTYPE* m_pInBuffers[NUM_IN_BUFFERS] = {NULL};
318 OMX_BOOL m_bInFrameFree[NUM_IN_BUFFERS];
319 
320 ProfileType m_sProfile;
321 
322 static int m_nFramePlay = 0;
323 static int m_eMode = MODE_PREVIEW;
324 static int m_nInFd = -1;
325 static int m_nOutFd = -1;
326 static int m_nTimeStamp = 0;
327 static int m_nFrameIn = 0; // frames pushed to encoder
328 static int m_nFrameOut = 0; // frames returned by encoder
329 static int m_nAVCSliceMode = 0;
330 static bool m_bWatchDogKicked = false;
331 FILE  *m_pDynConfFile = NULL;
332 static struct DynamicConfig dynamic_config;
333 
334 /* Statistics Logging */
335 static long long tot_bufsize = 0;
336 int ebd_cnt=0, fbd_cnt=0;
337 
338 #ifdef USE_ION
339 static const char* PMEM_DEVICE = "/dev/ion";
340 #elif MAX_RES_720P
341 static const char* PMEM_DEVICE = "/dev/pmem_adsp";
342 #elif MAX_RES_1080P_EBI
343 static const char* PMEM_DEVICE  = "/dev/pmem_adsp";
344 #elif MAX_RES_1080P
345 static const char* PMEM_DEVICE = "/dev/pmem_smipool";
346 #else
347 #error PMEM_DEVICE cannot be determined.
348 #endif
349 
350 #ifdef USE_ION
351 struct enc_ion ion_data;
352 #endif
353 //////////////////////////
354 // MODULE FUNCTIONS
355 //////////////////////////
356 
PmemMalloc(OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO * pMem,int nSize)357 void* PmemMalloc(OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO* pMem, int nSize)
358 {
359    void *pvirt = NULL;
360    int rc = 0;
361 
362    if (!pMem)
363       return NULL;
364 
365 #ifdef USE_ION
366   ion_data.ion_device_fd = open (PMEM_DEVICE,O_RDONLY);
367   if(ion_data.ion_device_fd < 0)
368   {
369       E("\nERROR: ION Device open() Failed");
370       return NULL;
371   }
372   nSize = (nSize + 4095) & (~4095);
373   ion_data.alloc_data.len = nSize;
374   ion_data.alloc_data.heap_id_mask = 0x1 << ION_CP_MM_HEAP_ID;
375   ion_data.alloc_data.align = 4096;
376   ion_data.alloc_data.flags = 0;
377 
378   rc = ioctl(ion_data.ion_device_fd,ION_IOC_ALLOC,&ion_data.alloc_data);
379   if(rc || !ion_data.alloc_data.handle) {
380          E("\n ION ALLOC memory failed ");
381          ion_data.alloc_data.handle=NULL;
382          return NULL;
383   }
384 
385   ion_data.ion_alloc_fd.handle = ion_data.alloc_data.handle;
386   rc = ioctl(ion_data.ion_device_fd,ION_IOC_MAP,&ion_data.ion_alloc_fd);
387   if(rc) {
388         E("\n ION MAP failed ");
389         ion_data.ion_alloc_fd.fd =-1;
390         ion_data.ion_alloc_fd.fd =-1;
391         return NULL;
392   }
393   pMem->pmem_fd = ion_data.ion_alloc_fd.fd;
394 #else
395    pMem->pmem_fd = open(PMEM_DEVICE, O_RDWR);
396    if ((int)(pMem->pmem_fd) < 0)
397       return NULL;
398    nSize = (nSize + 4095) & (~4095);
399 #endif
400    pMem->offset = 0;
401    pvirt = mmap(NULL, nSize,
402                 PROT_READ | PROT_WRITE,
403                 MAP_SHARED, pMem->pmem_fd, pMem->offset);
404    if (pvirt == (void*) MAP_FAILED)
405    {
406       close(pMem->pmem_fd);
407       pMem->pmem_fd = -1;
408 #ifdef USE_ION
409     if(ioctl(ion_data.ion_device_fd,ION_IOC_FREE,
410        &ion_data.alloc_data.handle)) {
411       E("ion recon buffer free failed");
412     }
413     ion_data.alloc_data.handle = NULL;
414     ion_data.ion_alloc_fd.fd =-1;
415     close(ion_data.ion_device_fd);
416     ion_data.ion_device_fd =-1;
417 #endif
418       return NULL;
419    }
420    D("allocated pMem->fd = %d pvirt=0x%x, pMem->phys=0x%x, size = %d", pMem->pmem_fd,
421        pvirt, pMem->offset, nSize);
422    return pvirt;
423 }
424 
PmemFree(OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO * pMem,void * pvirt,int nSize)425 int PmemFree(OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO* pMem, void* pvirt, int nSize)
426 {
427    if (!pMem || !pvirt)
428       return -1;
429 
430    nSize = (nSize + 4095) & (~4095);
431    munmap(pvirt, nSize);
432    close(pMem->pmem_fd);
433    pMem->pmem_fd = -1;
434 #ifdef USE_ION
435    if(ioctl(ion_data.ion_device_fd,ION_IOC_FREE,
436          &ion_data.alloc_data.handle)) {
437         E("ion recon buffer free failed");
438    }
439    ion_data.alloc_data.handle = NULL;
440    ion_data.ion_alloc_fd.fd =-1;
441    close(ion_data.ion_device_fd);
442    ion_data.ion_device_fd =-1;
443 #endif
444    return 0;
445 }
PrintFramePackArrangement(OMX_QCOM_FRAME_PACK_ARRANGEMENT framePackingArrangement)446 void PrintFramePackArrangement(OMX_QCOM_FRAME_PACK_ARRANGEMENT framePackingArrangement)
447 {
448     printf("id (%d)\n",
449            framePackingArrangement.id);
450     printf("cancel_flag (%d)\n",
451            framePackingArrangement.cancel_flag);
452     printf("type (%d)\n",
453            framePackingArrangement.type);
454     printf("quincunx_sampling_flag (%d)\n",
455            framePackingArrangement.quincunx_sampling_flag);
456    printf("content_interpretation_type (%d)\n",
457           framePackingArrangement.content_interpretation_type);
458    printf("spatial_flipping_flag (%d)\n",
459           framePackingArrangement.spatial_flipping_flag);
460    printf("frame0_flipped_flag (%d)\n",
461           framePackingArrangement.frame0_flipped_flag);
462    printf("field_views_flag (%d)\n",
463           framePackingArrangement.field_views_flag);
464    printf("current_frame_is_frame0_flag (%d)\n",
465           framePackingArrangement.current_frame_is_frame0_flag);
466    printf("frame0_self_contained_flag (%d)\n",
467           framePackingArrangement.frame0_self_contained_flag);
468    printf("frame1_self_contained_flag (%d)\n",
469           framePackingArrangement.frame1_self_contained_flag);
470    printf("frame0_grid_position_x (%d)\n",
471           framePackingArrangement.frame0_grid_position_x);
472    printf("frame0_grid_position_y (%d)\n",
473           framePackingArrangement.frame0_grid_position_y);
474    printf("frame1_grid_position_x (%d)\n",
475           framePackingArrangement.frame1_grid_position_x);
476    printf("frame1_grid_position_y (%d)\n",
477           framePackingArrangement.frame1_grid_position_y);
478    printf("reserved_byte (%d)\n",
479           framePackingArrangement.reserved_byte);
480    printf("repetition_period (%d)\n",
481           framePackingArrangement.repetition_period);
482    printf("extension_flag (%d)\n",
483           framePackingArrangement.extension_flag);
484 }
SetState(OMX_STATETYPE eState)485 void SetState(OMX_STATETYPE eState)
486 {
487 #define GOTO_STATE(eState)                      \
488    case eState:                                 \
489       {                                         \
490          D("Going to state " # eState"...");            \
491          OMX_SendCommand(m_hHandle,                     \
492                          OMX_CommandStateSet,           \
493                          (OMX_U32) eState,              \
494                          NULL);                         \
495          while (m_eState != eState)                     \
496          {                                              \
497             sleep(1);                               \
498          }                                              \
499          D("Now in state " # eState);                   \
500          break;                                         \
501       }
502 
503    switch (eState)
504    {
505       GOTO_STATE(OMX_StateLoaded);
506       GOTO_STATE(OMX_StateIdle);
507       GOTO_STATE(OMX_StateExecuting);
508       GOTO_STATE(OMX_StateInvalid);
509       GOTO_STATE(OMX_StateWaitForResources);
510       GOTO_STATE(OMX_StatePause);
511    }
512 }
513 ////////////////////////////////////////////////////////////////////////////////
ConfigureEncoder()514 OMX_ERRORTYPE ConfigureEncoder()
515 {
516    OMX_ERRORTYPE result = OMX_ErrorNone;
517    unsigned const int *profile_tbl = (unsigned int const *)mpeg4_profile_level_table;
518    OMX_U32 mb_per_sec, mb_per_frame;
519    bool profile_level_found = false;
520    OMX_U32 eProfile,eLevel;
521 
522    OMX_PARAM_PORTDEFINITIONTYPE portdef; // OMX_IndexParamPortDefinition
523 #ifdef QCOM_EXT
524       OMX_QCOM_PARAM_PORTDEFINITIONTYPE qPortDefnType;
525 #endif
526    portdef.nPortIndex = (OMX_U32) 0; // input
527    result = OMX_GetParameter(m_hHandle,
528                              OMX_IndexParamPortDefinition,
529                              &portdef);
530    E("\n OMX_IndexParamPortDefinition Get Paramter on input port");
531    CHK(result);
532    portdef.format.video.nFrameWidth = m_sProfile.nFrameWidth;
533    portdef.format.video.nFrameHeight = m_sProfile.nFrameHeight;
534 
535    E ("\n Height %d width %d bit rate %d",portdef.format.video.nFrameHeight
536       ,portdef.format.video.nFrameWidth,portdef.format.video.nBitrate);
537    result = OMX_SetParameter(m_hHandle,
538                              OMX_IndexParamPortDefinition,
539                              &portdef);
540    E("\n OMX_IndexParamPortDefinition Set Paramter on input port");
541    CHK(result);
542    // once more to get proper buffer size
543    result = OMX_GetParameter(m_hHandle,
544                              OMX_IndexParamPortDefinition,
545                              &portdef);
546    E("\n OMX_IndexParamPortDefinition Get Paramter on input port, 2nd pass");
547    CHK(result);
548    // update size accordingly
549    m_sProfile.nFrameBytes = portdef.nBufferSize;
550    portdef.nPortIndex = (OMX_U32) 1; // output
551    result = OMX_GetParameter(m_hHandle,
552                              OMX_IndexParamPortDefinition,
553                              &portdef);
554    E("\n OMX_IndexParamPortDefinition Get Paramter on output port");
555    CHK(result);
556    portdef.format.video.nFrameWidth = m_sProfile.nFrameWidth;
557    portdef.format.video.nFrameHeight = m_sProfile.nFrameHeight;
558    portdef.format.video.nBitrate = m_sProfile.nBitrate;
559    FractionToQ16(portdef.format.video.xFramerate,(int) (m_sProfile.nFramerate * 2),2);
560    result = OMX_SetParameter(m_hHandle,
561                              OMX_IndexParamPortDefinition,
562                              &portdef);
563    E("\n OMX_IndexParamPortDefinition Set Paramter on output port");
564    CHK(result);
565 
566 #ifdef QCOM_EXT
567 
568 qPortDefnType.nPortIndex = PORT_INDEX_IN;
569 qPortDefnType.nMemRegion = OMX_QCOM_MemRegionEBI1;
570 qPortDefnType.nSize = sizeof(OMX_QCOM_PARAM_PORTDEFINITIONTYPE);
571 
572 result = OMX_SetParameter(m_hHandle,
573                              (OMX_INDEXTYPE)OMX_QcomIndexPortDefn,
574                              &qPortDefnType);
575 
576 #endif
577    if (!m_sProfile.nUserProfile) // profile not set by user, go ahead with table calculation
578    {
579    //validate the ht,width,fps,bitrate and set the appropriate profile and level
580    if(m_sProfile.eCodec == OMX_VIDEO_CodingMPEG4)
581    {
582      profile_tbl = (unsigned int const *)mpeg4_profile_level_table;
583    }
584    else if(m_sProfile.eCodec == OMX_VIDEO_CodingAVC)
585    {
586      profile_tbl = (unsigned int const *)h264_profile_level_table;
587    }
588    else if(m_sProfile.eCodec == OMX_VIDEO_CodingH263)
589    {
590      profile_tbl = (unsigned int const *)h263_profile_level_table;
591    }
592 
593    mb_per_frame = ((m_sProfile.nFrameHeight+15)>>4)*
594                 ((m_sProfile.nFrameWidth+15)>>4);
595 
596    mb_per_sec = mb_per_frame*(m_sProfile.nFramerate);
597 
598    do{
599       if(mb_per_frame <= (int)profile_tbl[0])
600       {
601           if(mb_per_sec <= (int)profile_tbl[1])
602           {
603             if(m_sProfile.nBitrate <= (int)profile_tbl[2])
604             {
605               eLevel = (int)profile_tbl[3];
606               eProfile = (int)profile_tbl[4];
607               E("\n profile/level found: %d/%d\n",eProfile/eLevel);
608               profile_level_found = true;
609               break;
610             }
611           }
612       }
613       profile_tbl = profile_tbl + 5;
614    }while(profile_tbl[0] != 0);
615 
616    if ( profile_level_found != true )
617    {
618      E("\n Error: Unsupported profile/level\n");
619      return OMX_ErrorNone;
620    }
621    }
622    else // Profile set by user!
623    {
624       eProfile = m_sProfile.nUserProfile;
625       eLevel = 0;
626    }
627    if (m_sProfile.eCodec == OMX_VIDEO_CodingH263)
628    {
629       D("Configuring H263...");
630 
631       OMX_VIDEO_PARAM_H263TYPE h263;
632       result = OMX_GetParameter(m_hHandle,
633                                 OMX_IndexParamVideoH263,
634                                 &h263);
635       CHK(result);
636       h263.nPortIndex = (OMX_U32) PORT_INDEX_OUT;
637       h263.nPFrames = m_sProfile.nFramerate * 2 - 1; // intra period
638       h263.nBFrames = 0;
639       h263.eProfile = (OMX_VIDEO_H263PROFILETYPE)eProfile;
640       h263.eLevel = (OMX_VIDEO_H263LEVELTYPE)eLevel;
641       h263.bPLUSPTYPEAllowed = OMX_FALSE;
642       h263.nAllowedPictureTypes = 2;
643       h263.bForceRoundingTypeToZero = OMX_TRUE;
644       h263.nPictureHeaderRepetition = 0;
645       h263.nGOBHeaderInterval = 1;
646       result = OMX_SetParameter(m_hHandle,
647                                 OMX_IndexParamVideoH263,
648                                 &h263);
649    }
650    else
651    {
652       D("Configuring MP4/H264...");
653 
654       OMX_VIDEO_PARAM_PROFILELEVELTYPE profileLevel; // OMX_IndexParamVideoProfileLevelCurrent
655       profileLevel.nPortIndex = (OMX_U32) PORT_INDEX_OUT;
656       profileLevel.eProfile = eProfile;
657       profileLevel.eLevel =  eLevel;
658       result = OMX_SetParameter(m_hHandle,
659                                 OMX_IndexParamVideoProfileLevelCurrent,
660                                 &profileLevel);
661       E("\n OMX_IndexParamVideoProfileLevelCurrent Set Paramter port");
662       CHK(result);
663       //profileLevel.eLevel = (OMX_U32) m_sProfile.eLevel;
664       result = OMX_GetParameter(m_hHandle,
665                                 OMX_IndexParamVideoProfileLevelCurrent,
666                                 &profileLevel);
667       E("\n OMX_IndexParamVideoProfileLevelCurrent Get Paramter port");
668       D ("\n Profile = %d level = %d",profileLevel.eProfile,profileLevel.eLevel);
669       CHK(result);
670 
671         if (m_sProfile.eCodec == OMX_VIDEO_CodingMPEG4)
672         {
673         OMX_VIDEO_PARAM_MPEG4TYPE mp4; // OMX_IndexParamVideoMpeg4
674        result = OMX_GetParameter(m_hHandle,
675                                  OMX_IndexParamVideoMpeg4,
676                                  &mp4);
677        CHK(result);
678        mp4.nPortIndex = (OMX_U32) PORT_INDEX_OUT;
679        mp4.nTimeIncRes = 1000;
680        result = OMX_SetParameter(m_hHandle,
681                                  OMX_IndexParamVideoMpeg4,
682                                  &mp4);
683        CHK(result);
684          }
685    }
686    if (m_sProfile.eCodec == OMX_VIDEO_CodingAVC)
687    {
688 #if 1
689 /////////////C A B A C ///A N D/////D E B L O C K I N G /////////////////
690 
691       OMX_VIDEO_PARAM_AVCTYPE avcdata;
692       avcdata.nPortIndex = (OMX_U32)PORT_INDEX_OUT;
693       result = OMX_GetParameter(m_hHandle,
694                                 OMX_IndexParamVideoAvc,
695                                 &avcdata);
696       CHK(result);
697 // TEST VALUES (CHANGE FOR DIFF CONFIG's)
698     avcdata.eLoopFilterMode = OMX_VIDEO_AVCLoopFilterEnable;
699 //      avcdata.eLoopFilterMode = OMX_VIDEO_AVCLoopFilterDisable;
700 //    avcdata.eLoopFilterMode = OMX_VIDEO_AVCLoopFilterDisableSliceBoundary;
701    avcdata.bEntropyCodingCABAC = OMX_FALSE;
702 //   avcdata.bEntropyCodingCABAC = OMX_TRUE;
703    avcdata.nCabacInitIdc = 1;
704 ///////////////////////////////////////////////
705 
706       result = OMX_SetParameter(m_hHandle,
707                                 OMX_IndexParamVideoAvc,
708                                 &avcdata);
709       CHK(result);
710 
711 /////////////C A B A C ///A N D/////D E B L O C K I N G /////////////////
712 #endif
713    }
714 
715    OMX_VIDEO_PARAM_BITRATETYPE bitrate; // OMX_IndexParamVideoBitrate
716    bitrate.nPortIndex = (OMX_U32)PORT_INDEX_OUT;
717    result = OMX_GetParameter(m_hHandle,
718                              OMX_IndexParamVideoBitrate,
719                              &bitrate);
720    E("\n OMX_IndexParamVideoBitrate Get Paramter port");
721    CHK(result);
722    bitrate.eControlRate = m_sProfile.eControlRate;
723    bitrate.nTargetBitrate = m_sProfile.nBitrate;
724    result = OMX_SetParameter(m_hHandle,
725                              OMX_IndexParamVideoBitrate,
726                              &bitrate);
727    E("\n OMX_IndexParamVideoBitrate Set Paramter port");
728    CHK(result);
729 
730    OMX_VIDEO_PARAM_PORTFORMATTYPE framerate; // OMX_IndexParamVidePortFormat
731    framerate.nPortIndex = 0;
732    result = OMX_GetParameter(m_hHandle,
733                              OMX_IndexParamVideoPortFormat,
734                              &framerate);
735    E("\n OMX_IndexParamVideoPortFormat Get Paramter port");
736    CHK(result);
737    FractionToQ16(framerate.xFramerate,(int) (m_sProfile.nFramerate * 2),2);
738    result = OMX_SetParameter(m_hHandle,
739                              OMX_IndexParamVideoPortFormat,
740                              &framerate);
741    E("\n OMX_IndexParamVideoPortFormat Set Paramter port");
742    CHK(result);
743 
744 #if 1
745 ///////////////////I N T R A P E R I O D ///////////////////
746 
747       QOMX_VIDEO_INTRAPERIODTYPE intra;
748 
749       intra.nPortIndex = (OMX_U32) PORT_INDEX_OUT; // output
750       result = OMX_GetConfig(m_hHandle,
751                              (OMX_INDEXTYPE) QOMX_IndexConfigVideoIntraperiod,
752                              (OMX_PTR) &intra);
753 
754       if (result == OMX_ErrorNone)
755       {
756          intra.nPFrames = (OMX_U32) (2 * m_sProfile.nFramerate - 1); //setting I
757                                                                      //frame interval to
758                                                                      //2 x framerate
759          intra.nIDRPeriod = 1; //every I frame is an IDR
760          intra.nPortIndex = (OMX_U32) PORT_INDEX_OUT;
761          result = OMX_SetConfig(m_hHandle,
762                                 (OMX_INDEXTYPE) QOMX_IndexConfigVideoIntraperiod,
763                                 (OMX_PTR) &intra);
764       }
765       else
766       {
767          E("failed to get state", 0, 0, 0);
768       }
769 
770 
771 ///////////////////I N T R A P E R I O D ///////////////////
772 #endif
773 
774 #if 1
775 ///////////////////E R R O R C O R R E C T I O N ///////////////////
776 
777       ResyncMarkerType eResyncMarkerType = RESYNC_MARKER_NONE;
778       unsigned long int nResyncMarkerSpacing = 0;
779       OMX_BOOL enableHEC = OMX_FALSE;
780 
781 //For Testing ONLY
782    if (m_sProfile.eCodec == OMX_VIDEO_CodingMPEG4)
783    {
784 // MPEG4
785 //      eResyncMarkerType = RESYNC_MARKER_BYTE;
786 //      nResyncMarkerSpacing = 1920;
787       eResyncMarkerType = RESYNC_MARKER_MB;
788       nResyncMarkerSpacing = 50;
789       enableHEC = OMX_TRUE;
790    }
791    else if (m_sProfile.eCodec == OMX_VIDEO_CodingH263)
792    {
793 //H263
794       eResyncMarkerType = RESYNC_MARKER_GOB;
795       nResyncMarkerSpacing = 0;
796    }
797    else if (m_sProfile.eCodec == OMX_VIDEO_CodingAVC)
798    {
799 //H264
800 //      eResyncMarkerType = RESYNC_MARKER_BYTE;
801 //      nResyncMarkerSpacing = 1920;
802 
803       //nResyncMarkerSpacing sets the slice size in venc_set_multislice_cfg
804       //
805       //As of 9/24/10, it is known that the firmware has a bitstream
806       //corruption issue when RateControl and multislice are enabled for 720P
807       //So, disabling multislice for 720P when ratecontrol is enabled until
808       //the firmware issue is resolved.
809 
810       if ( ( (m_sProfile.nFrameWidth == 1280) && (m_sProfile.nFrameHeight = 720) ) &&
811            (m_sProfile.eControlRate  != OMX_Video_ControlRateDisable) )
812       {
813          eResyncMarkerType = RESYNC_MARKER_NONE;
814          nResyncMarkerSpacing = 0;
815       }
816       else
817       {
818          eResyncMarkerType = RESYNC_MARKER_MB;
819           nResyncMarkerSpacing = 50;
820       }
821    }
822 
823    OMX_VIDEO_PARAM_ERRORCORRECTIONTYPE errorCorrection; //OMX_IndexParamVideoErrorCorrection
824    errorCorrection.nPortIndex = (OMX_U32) PORT_INDEX_OUT; // output
825    result = OMX_GetParameter(m_hHandle,
826                              (OMX_INDEXTYPE) OMX_IndexParamVideoErrorCorrection,
827                              (OMX_PTR) &errorCorrection);
828 
829    errorCorrection.bEnableRVLC = OMX_FALSE;
830    errorCorrection.bEnableDataPartitioning = OMX_FALSE;
831 
832       if ((eResyncMarkerType == RESYNC_MARKER_BYTE) &&
833          (m_sProfile.eCodec == OMX_VIDEO_CodingMPEG4)){
834             errorCorrection.bEnableResync = OMX_TRUE;
835             errorCorrection.nResynchMarkerSpacing = nResyncMarkerSpacing;
836             errorCorrection.bEnableHEC = enableHEC;
837             }
838       else if ((eResyncMarkerType == RESYNC_MARKER_BYTE) &&
839                (m_sProfile.eCodec == OMX_VIDEO_CodingAVC)){
840          errorCorrection.bEnableResync = OMX_TRUE;
841          errorCorrection.nResynchMarkerSpacing = nResyncMarkerSpacing;
842          }
843       else if ((eResyncMarkerType == RESYNC_MARKER_GOB) &&
844                (m_sProfile.eCodec == OMX_VIDEO_CodingH263)){
845          errorCorrection.bEnableResync = OMX_FALSE;
846          errorCorrection.nResynchMarkerSpacing = nResyncMarkerSpacing;
847          errorCorrection.bEnableDataPartitioning = OMX_TRUE;
848          }
849 
850       result = OMX_SetParameter(m_hHandle,
851                             (OMX_INDEXTYPE) OMX_IndexParamVideoErrorCorrection,
852                             (OMX_PTR) &errorCorrection);
853    CHK(result);
854 
855       if (eResyncMarkerType == RESYNC_MARKER_MB){
856          if (m_sProfile.eCodec == OMX_VIDEO_CodingAVC){
857             OMX_VIDEO_PARAM_AVCTYPE avcdata;
858             avcdata.nPortIndex = (OMX_U32) PORT_INDEX_OUT; // output
859             result = OMX_GetParameter(m_hHandle,
860                                       OMX_IndexParamVideoAvc,
861                                       (OMX_PTR) &avcdata);
862             CHK(result);
863             if (result == OMX_ErrorNone)
864             {
865                avcdata.nSliceHeaderSpacing = nResyncMarkerSpacing;
866                result = OMX_SetParameter(m_hHandle,
867                                          OMX_IndexParamVideoAvc,
868                                          (OMX_PTR) &avcdata);
869                CHK(result);
870 
871             }
872          }
873          else if(m_sProfile.eCodec == OMX_VIDEO_CodingMPEG4){
874             OMX_VIDEO_PARAM_MPEG4TYPE mp4;
875             mp4.nPortIndex = (OMX_U32) PORT_INDEX_OUT; // output
876             result = OMX_GetParameter(m_hHandle,
877                                       OMX_IndexParamVideoMpeg4,
878                                       (OMX_PTR) &mp4);
879             CHK(result);
880 
881             if (result == OMX_ErrorNone)
882             {
883                mp4.nSliceHeaderSpacing = nResyncMarkerSpacing;
884                result = OMX_SetParameter(m_hHandle,
885                                          OMX_IndexParamVideoMpeg4,
886                                          (OMX_PTR) &mp4);
887                CHK(result);
888             }
889          }
890          }
891 
892 ///////////////////E R R O R C O R R E C T I O N ///////////////////
893 #endif
894 
895 #if 1
896 ///////////////////I N T R A R E F R E S H///////////////////
897       bool bEnableIntraRefresh = OMX_TRUE;
898 
899       if (result == OMX_ErrorNone)
900       {
901          OMX_VIDEO_PARAM_INTRAREFRESHTYPE ir; // OMX_IndexParamVideoIntraRefresh
902          ir.nPortIndex = (OMX_U32) PORT_INDEX_OUT; // output
903          result = OMX_GetParameter(m_hHandle,
904                                    OMX_IndexParamVideoIntraRefresh,
905                                    (OMX_PTR) &ir);
906          if (result == OMX_ErrorNone)
907          {
908             if (bEnableIntraRefresh)
909             {
910                ir.eRefreshMode = OMX_VIDEO_IntraRefreshCyclic;
911                ir.nCirMBs = 5;
912                result = OMX_SetParameter(m_hHandle,
913                                          OMX_IndexParamVideoIntraRefresh,
914                                          (OMX_PTR) &ir);
915                CHK(result);
916             }
917          }
918       }
919 #endif
920 #if 1
921 ///////////////////FRAMEPACKING DATA///////////////////
922       OMX_QCOM_FRAME_PACK_ARRANGEMENT framePackingArrangement;
923       FILE *m_pConfigFile;
924       char m_configFilename [128] = "/data/configFile.cfg";
925       memset(&framePackingArrangement, 0, sizeof(framePackingArrangement));
926       m_pConfigFile = fopen(m_configFilename, "r");
927       if (m_pConfigFile != NULL)
928       {
929          //read all frame packing data
930          framePackingArrangement.nPortIndex = (OMX_U32)PORT_INDEX_OUT;
931          int totalSizeToRead = FRAME_PACK_SIZE * sizeof(OMX_U32);
932          char *pFramePack = (char *) &(framePackingArrangement.id);
933          while ( ( (fscanf(m_pConfigFile, "%d", pFramePack)) != EOF ) &&
934                  (totalSizeToRead != 0) )
935          {
936             //printf("Addr = %p, Value read = %d, sizeToRead remaining=%d\n",
937             //       pFramePack, *pFramePack, totalSizeToRead);
938             pFramePack += sizeof(OMX_U32);
939             totalSizeToRead -= sizeof(OMX_U32);
940          }
941          //close the file.
942          fclose(m_pConfigFile);
943 
944          printf("Frame Packing data from config file:\n");
945          PrintFramePackArrangement(framePackingArrangement);
946       }
947       else
948       {
949          D("\n Config file does not exist or could not be opened.");
950          //set the default values
951          framePackingArrangement.nPortIndex = (OMX_U32)PORT_INDEX_OUT;
952          framePackingArrangement.id = 123;
953          framePackingArrangement.cancel_flag = false;
954          framePackingArrangement.type = 3;
955          framePackingArrangement.quincunx_sampling_flag = false;
956          framePackingArrangement.content_interpretation_type = 0;
957          framePackingArrangement.spatial_flipping_flag = true;
958          framePackingArrangement.frame0_flipped_flag = false;
959          framePackingArrangement.field_views_flag = false;
960          framePackingArrangement.current_frame_is_frame0_flag = false;
961          framePackingArrangement.frame0_self_contained_flag = true;
962          framePackingArrangement.frame1_self_contained_flag = false;
963          framePackingArrangement.frame0_grid_position_x = 3;
964          framePackingArrangement.frame0_grid_position_y = 15;
965          framePackingArrangement.frame1_grid_position_x = 11;
966          framePackingArrangement.frame1_grid_position_y = 7;
967          framePackingArrangement.reserved_byte = 0;
968          framePackingArrangement.repetition_period = 16381;
969          framePackingArrangement.extension_flag = false;
970 
971          printf("Frame Packing Defaults :\n");
972          PrintFramePackArrangement(framePackingArrangement);
973       }
974       result = OMX_SetConfig(m_hHandle,
975                 (OMX_INDEXTYPE)OMX_QcomIndexConfigVideoFramePackingArrangement,
976                 (OMX_PTR) &framePackingArrangement);
977       CHK(result);
978 
979 //////////////////////OMX_VIDEO_PARAM_INTRAREFRESHTYPE///////////////////
980 #endif
981 
982    OMX_CONFIG_FRAMERATETYPE enc_framerate; // OMX_IndexConfigVideoFramerate
983    enc_framerate.nPortIndex = (OMX_U32)PORT_INDEX_OUT;
984    result = OMX_GetConfig(m_hHandle,
985                           OMX_IndexConfigVideoFramerate,
986                           &enc_framerate);
987    CHK(result);
988    FractionToQ16(enc_framerate.xEncodeFramerate,(int) (m_sProfile.nFramerate * 2),2);
989    result = OMX_SetConfig(m_hHandle,
990                           OMX_IndexConfigVideoFramerate,
991                           &enc_framerate);
992    CHK(result);
993    return OMX_ErrorNone;
994 }
995 ////////////////////////////////////////////////////////////////////////////////
SendMessage(MsgId id,MsgData * data)996 void SendMessage(MsgId id, MsgData* data)
997 {
998    pthread_mutex_lock(&m_mutex);
999    if (m_sMsgQ.size >= MAX_MSG)
1000    {
1001       E("main msg m_sMsgQ is full");
1002       return;
1003    }
1004    m_sMsgQ.q[(m_sMsgQ.head + m_sMsgQ.size) % MAX_MSG].id = id;
1005    if (data)
1006       m_sMsgQ.q[(m_sMsgQ.head + m_sMsgQ.size) % MAX_MSG].data = *data;
1007    ++m_sMsgQ.size;
1008    pthread_cond_signal(&m_signal);
1009    pthread_mutex_unlock(&m_mutex);
1010 }
1011 ////////////////////////////////////////////////////////////////////////////////
PopMessage(Msg * msg)1012 void PopMessage(Msg* msg)
1013 {
1014    pthread_mutex_lock(&m_mutex);
1015    while (m_sMsgQ.size == 0)
1016    {
1017       pthread_cond_wait(&m_signal, &m_mutex);
1018    }
1019    *msg = m_sMsgQ.q[m_sMsgQ.head];
1020    --m_sMsgQ.size;
1021    m_sMsgQ.head = (m_sMsgQ.head + 1) % MAX_MSG;
1022    pthread_mutex_unlock(&m_mutex);
1023 }
1024 ////////////////////////////////////////////////////////////////////////////////
EVT_CB(OMX_IN OMX_HANDLETYPE hComponent,OMX_IN OMX_PTR pAppData,OMX_IN OMX_EVENTTYPE eEvent,OMX_IN OMX_U32 nData1,OMX_IN OMX_U32 nData2,OMX_IN OMX_PTR pEventData)1025 OMX_ERRORTYPE EVT_CB(OMX_IN OMX_HANDLETYPE hComponent,
1026                      OMX_IN OMX_PTR pAppData,
1027                      OMX_IN OMX_EVENTTYPE eEvent,
1028                      OMX_IN OMX_U32 nData1,
1029                      OMX_IN OMX_U32 nData2,
1030                      OMX_IN OMX_PTR pEventData)
1031 {
1032 #define SET_STATE(eState)                                   \
1033    case eState:                                             \
1034       {                                                     \
1035          D("" # eState " complete");                        \
1036          m_eState = eState;                                 \
1037          break;                                             \
1038       }
1039 
1040    if (eEvent == OMX_EventCmdComplete)
1041    {
1042       if ((OMX_COMMANDTYPE) nData1 == OMX_CommandStateSet)
1043       {
1044          switch ((OMX_STATETYPE) nData2)
1045          {
1046             SET_STATE(OMX_StateLoaded);
1047             SET_STATE(OMX_StateIdle);
1048             SET_STATE(OMX_StateExecuting);
1049             SET_STATE(OMX_StateInvalid);
1050             SET_STATE(OMX_StateWaitForResources);
1051             SET_STATE(OMX_StatePause);
1052          default:
1053             E("invalid state %d", (int) nData2);
1054           }
1055       }
1056    }
1057 
1058    else if (eEvent == OMX_EventError)
1059    {
1060       E("OMX_EventError");
1061    }
1062 
1063    else
1064    {
1065       E("unexpected event %d", (int) eEvent);
1066    }
1067    return OMX_ErrorNone;
1068 }
1069 ////////////////////////////////////////////////////////////////////////////////
EBD_CB(OMX_IN OMX_HANDLETYPE hComponent,OMX_IN OMX_PTR pAppData,OMX_IN OMX_BUFFERHEADERTYPE * pBuffer)1070 OMX_ERRORTYPE EBD_CB(OMX_IN OMX_HANDLETYPE hComponent,
1071                      OMX_IN OMX_PTR pAppData,
1072                      OMX_IN OMX_BUFFERHEADERTYPE* pBuffer)
1073 {
1074    D("Got EBD callback ts=%lld", pBuffer->nTimeStamp);
1075 
1076    for (int i = 0; i < num_in_buffers; i++)
1077    {
1078       // mark this buffer ready for use again
1079       if (m_pInBuffers[i] == pBuffer)
1080       {
1081 
1082          D("Marked input buffer idx %d as free, buf %p", i, pBuffer->pBuffer);
1083          m_bInFrameFree[i] = OMX_TRUE;
1084          break;
1085       }
1086    }
1087 
1088    if (m_eMode == MODE_LIVE_ENCODE)
1089    {
1090       CameraTest_ReleaseFrame(pBuffer->pBuffer,
1091                               ((OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO*)pBuffer->pAppPrivate));
1092    }
1093    else
1094    {
1095       // wake up main thread and tell it to send next frame
1096       MsgData data;
1097       data.sBitstreamData.pBuffer = pBuffer;
1098       SendMessage(MSG_ID_INPUT_FRAME_DONE,
1099                   &data);
1100 
1101    }
1102    return OMX_ErrorNone;
1103 }
1104 ////////////////////////////////////////////////////////////////////////////////
FBD_CB(OMX_OUT OMX_HANDLETYPE hComponent,OMX_OUT OMX_PTR pAppData,OMX_OUT OMX_BUFFERHEADERTYPE * pBuffer)1105 OMX_ERRORTYPE FBD_CB(OMX_OUT OMX_HANDLETYPE hComponent,
1106                      OMX_OUT OMX_PTR pAppData,
1107                      OMX_OUT OMX_BUFFERHEADERTYPE* pBuffer)
1108 {
1109    D("Got FBD callback ts=%lld", pBuffer->nTimeStamp);
1110 
1111    static long long prevTime = 0;
1112    long long currTime = GetTimeStamp();
1113 
1114    m_bWatchDogKicked = true;
1115 
1116    /* Empty Buffers should not be counted */
1117    if(pBuffer->nFilledLen !=0)
1118    {
1119       /* Counting Buffers supplied from OpneMax Encoder */
1120       fbd_cnt++;
1121       tot_bufsize += pBuffer->nFilledLen;
1122    }
1123    if (prevTime != 0)
1124    {
1125       long long currTime = GetTimeStamp();
1126       D("FBD_DELTA = %lld\n", currTime - prevTime);
1127    }
1128    prevTime = currTime;
1129 
1130    if (m_eMode == MODE_PROFILE)
1131    {
1132       // if we are profiling we are not doing file I/O
1133       // so just give back to encoder
1134       if (OMX_FillThisBuffer(m_hHandle, pBuffer) != OMX_ErrorNone)
1135       {
1136          E("empty buffer failed for profiling");
1137       }
1138    }
1139    else
1140    {
1141       // wake up main thread and tell it to write to file
1142       MsgData data;
1143       data.sBitstreamData.pBuffer = pBuffer;
1144       SendMessage(MSG_ID_OUTPUT_FRAME_DONE,
1145                   &data);
1146    }
1147    return OMX_ErrorNone;
1148 }
1149 ////////////////////////////////////////////////////////////////////////////////
VencTest_Initialize()1150 OMX_ERRORTYPE VencTest_Initialize()
1151 {
1152    OMX_ERRORTYPE result = OMX_ErrorNone;
1153    static OMX_CALLBACKTYPE sCallbacks = {EVT_CB, EBD_CB, FBD_CB};
1154    int i;
1155 
1156    for (i = 0; i < num_in_buffers; i++)
1157    {
1158       m_pInBuffers[i] = NULL;
1159    }
1160 
1161    result = OMX_Init();
1162    CHK(result);
1163 
1164    if (m_sProfile.eCodec == OMX_VIDEO_CodingMPEG4)
1165    {
1166         result = OMX_GetHandle(&m_hHandle,
1167                              "OMX.qcom.video.encoder.mpeg4",
1168                              NULL,
1169                              &sCallbacks);
1170      // CHK(result);
1171    }
1172    else if (m_sProfile.eCodec == OMX_VIDEO_CodingH263)
1173    {
1174       result = OMX_GetHandle(&m_hHandle,
1175                              "OMX.qcom.video.encoder.h263",
1176                              NULL,
1177                              &sCallbacks);
1178       CHK(result);
1179    }
1180    else
1181    {
1182       result = OMX_GetHandle(&m_hHandle,
1183                              "OMX.qcom.video.encoder.avc",
1184                              NULL,
1185                              &sCallbacks);
1186       CHK(result);
1187    }
1188 
1189 
1190    result = ConfigureEncoder();
1191    CHK(result);
1192 
1193    return result;
1194 }
1195 
1196 ////////////////////////////////////////////////////////////////////////////////
VencTest_RegisterYUVBuffer(OMX_BUFFERHEADERTYPE ** ppBufferHeader,OMX_U8 * pBuffer,OMX_PTR pAppPrivate)1197 OMX_ERRORTYPE VencTest_RegisterYUVBuffer(OMX_BUFFERHEADERTYPE** ppBufferHeader,
1198                                          OMX_U8 *pBuffer,
1199                                          OMX_PTR pAppPrivate)
1200 {
1201    OMX_ERRORTYPE result = OMX_ErrorNone;
1202 #if 0
1203    D("register buffer");
1204    if ((result = OMX_AllocateBuffer(m_hHandle,
1205                                ppBufferHeader,
1206                                (OMX_U32) PORT_INDEX_IN,
1207                                pAppPrivate,
1208                                m_sProfile.nFrameBytes
1209                                )) != OMX_ErrorNone)
1210    {
1211       E("use buffer failed");
1212    }
1213    else
1214    {
1215      E("Allocate Buffer Success %x", (*ppBufferHeader)->pBuffer);
1216    }
1217   #endif
1218    D("register buffer");
1219    D("Calling UseBuffer for Input port");
1220    if ((result = OMX_UseBuffer(m_hHandle,
1221                                ppBufferHeader,
1222                                (OMX_U32) PORT_INDEX_IN,
1223                                pAppPrivate,
1224                                m_sProfile.nFrameBytes,
1225                                pBuffer)) != OMX_ErrorNone)
1226    {
1227       E("use buffer failed");
1228    }
1229 
1230    return result;
1231 }
1232 ////////////////////////////////////////////////////////////////////////////////
VencTest_EncodeFrame(void * pYUVBuff,long long nTimeStamp)1233 OMX_ERRORTYPE VencTest_EncodeFrame(void* pYUVBuff,
1234                                    long long nTimeStamp)
1235 {
1236    OMX_ERRORTYPE result = OMX_ErrorUndefined;
1237    D("calling OMX empty this buffer");
1238    for (int i = 0; i < num_in_buffers; i++)
1239    {
1240       if (pYUVBuff == m_pInBuffers[i]->pBuffer)
1241       {
1242          m_pInBuffers[i]->nTimeStamp = nTimeStamp;
1243     D("Sending Buffer - %x", m_pInBuffers[i]->pBuffer);
1244          result = OMX_EmptyThisBuffer(m_hHandle,
1245                                       m_pInBuffers[i]);
1246          /* Counting Buffers supplied to OpenMax Encoder */
1247          if(OMX_ErrorNone == result)
1248             ebd_cnt++;
1249          CHK(result);
1250          break;
1251       }
1252    }
1253    return result;
1254 }
1255 ////////////////////////////////////////////////////////////////////////////////
VencTest_Exit(void)1256 OMX_ERRORTYPE VencTest_Exit(void)
1257 {
1258    int i;
1259    OMX_ERRORTYPE result = OMX_ErrorNone;
1260    D("trying to exit venc");
1261 
1262    D("going to idle state");
1263    SetState(OMX_StateIdle);
1264 
1265 
1266    D("going to loaded state");
1267    //SetState(OMX_StateLoaded);
1268       OMX_SendCommand(m_hHandle,
1269                       OMX_CommandStateSet,
1270                       (OMX_U32) OMX_StateLoaded,
1271                        NULL);
1272 
1273       for (i = 0; i < num_in_buffers; i++)
1274    {
1275       D("free buffer");
1276       if (m_pInBuffers[i]->pBuffer)
1277       {
1278         // free(m_pInBuffers[i]->pBuffer);
1279          result = OMX_FreeBuffer(m_hHandle,
1280                                  PORT_INDEX_IN,
1281                                  m_pInBuffers[i]);
1282          CHK(result);
1283       }
1284       else
1285       {
1286          E("buffer %d is null", i);
1287          result = OMX_ErrorUndefined;
1288          CHK(result);
1289       }
1290    }
1291    for (i = 0; i < num_out_buffers; i++)
1292    {
1293       D("free buffer");
1294       if (m_pOutBuffers[i]->pBuffer)
1295       {
1296          free(m_pOutBuffers[i]->pBuffer);
1297          result = OMX_FreeBuffer(m_hHandle,
1298                                  PORT_INDEX_OUT,
1299                                  m_pOutBuffers[i]);
1300          CHK(result);
1301 
1302       }
1303       else
1304       {
1305          E("buffer %d is null", i);
1306          result = OMX_ErrorUndefined;
1307          CHK(result);
1308       }
1309    }
1310 
1311      while (m_eState != OMX_StateLoaded)
1312      {
1313         sleep(1);
1314      }
1315    D("component_deinit...");
1316    result = OMX_Deinit();
1317    CHK(result);
1318 
1319    D("venc is exiting...");
1320    return result;
1321 }
1322 ////////////////////////////////////////////////////////////////////////////////
1323 
VencTest_ReadDynamicConfigMsg()1324 void VencTest_ReadDynamicConfigMsg()
1325 {
1326   char frame_n[8], config[16], param[8];
1327   char *dest = frame_n;
1328   bool end = false;
1329   int cntr, nparam = 0;
1330   memset(&dynamic_config, 0, sizeof(struct DynamicConfig));
1331   do
1332   {
1333     cntr = -1;
1334     do
1335     {
1336       dest[++cntr] = fgetc(m_pDynConfFile);
1337     } while(dest[cntr] != ' ' && dest[cntr] != '\t' && dest[cntr] != '\n' && dest[cntr] != '\r' && !feof(m_pDynConfFile));
1338     if (dest[cntr] == '\n' || dest[cntr] == '\r')
1339       end = true;
1340     dest[cntr] = NULL;
1341     if (dest == frame_n)
1342       dest = config;
1343     else if (dest == config)
1344       dest = param;
1345     else
1346       end = true;
1347     nparam++;
1348   } while (!end && !feof(m_pDynConfFile));
1349 
1350   if (nparam > 1)
1351   {
1352     dynamic_config.pending = true;
1353     dynamic_config.frame_num = atoi(frame_n);
1354     if (!strcmp(config, "bitrate"))
1355     {
1356       dynamic_config.config_param = OMX_IndexConfigVideoBitrate;
1357       dynamic_config.config_data.bitrate.nPortIndex = PORT_INDEX_OUT;
1358       dynamic_config.config_data.bitrate.nEncodeBitrate = strtoul(param, NULL, 10);
1359     }
1360     else if (!strcmp(config, "framerate"))
1361     {
1362       dynamic_config.config_param = OMX_IndexConfigVideoFramerate;
1363       dynamic_config.config_data.framerate.nPortIndex = PORT_INDEX_OUT;
1364       dynamic_config.config_data.f_framerate = atof(param);
1365     }
1366     else if (!strcmp(config, "iperiod"))
1367     {
1368       dynamic_config.config_param = (OMX_INDEXTYPE)QOMX_IndexConfigVideoIntraperiod;
1369       dynamic_config.config_data.intraperiod.nPortIndex = PORT_INDEX_OUT;
1370       dynamic_config.config_data.intraperiod.nPFrames = strtoul(param, NULL, 10) - 1;
1371       dynamic_config.config_data.intraperiod.nIDRPeriod = 1; // This value is ignored in OMX component
1372     }
1373     else if (!strcmp(config, "ivoprefresh"))
1374     {
1375       dynamic_config.config_param = OMX_IndexConfigVideoIntraVOPRefresh;
1376       dynamic_config.config_data.intravoprefresh.nPortIndex = PORT_INDEX_OUT;
1377       dynamic_config.config_data.intravoprefresh.IntraRefreshVOP = OMX_TRUE;
1378     }
1379     else if (!strcmp(config, "rotation"))
1380     {
1381       dynamic_config.config_param = OMX_IndexConfigCommonRotate;
1382       dynamic_config.config_data.rotation.nPortIndex = PORT_INDEX_OUT;
1383       dynamic_config.config_data.rotation.nRotation = strtoul(param, NULL, 10);
1384     }
1385     else
1386     {
1387       E("UNKNOWN CONFIG PARAMETER: %s!", config);
1388       dynamic_config.pending = false;
1389     }
1390   }
1391   else if (feof(m_pDynConfFile))
1392   {
1393     fclose(m_pDynConfFile);
1394     m_pDynConfFile = NULL;
1395   }
1396 }
1397 
VencTest_ProcessDynamicConfigurationFile()1398 void VencTest_ProcessDynamicConfigurationFile()
1399 {
1400   do
1401   {
1402     if (dynamic_config.pending)
1403     {
1404       if(m_nFrameIn == dynamic_config.frame_num)
1405       {
1406         if (dynamic_config.config_param == OMX_IndexConfigVideoFramerate)
1407         {
1408           m_sProfile.nFramerate = dynamic_config.config_data.f_framerate;
1409           FractionToQ16(dynamic_config.config_data.framerate.xEncodeFramerate,
1410                         (int)(m_sProfile.nFramerate * 2), 2);
1411         }
1412         if (OMX_SetConfig(m_hHandle, dynamic_config.config_param,
1413             &dynamic_config.config_data) != OMX_ErrorNone)
1414           E("ERROR: Setting dynamic config to OMX param[0x%x]", dynamic_config.config_param);
1415         dynamic_config.pending = false;
1416       }
1417       else if (m_nFrameIn > dynamic_config.frame_num)
1418       {
1419         E("WARNING: Config change requested in passed frame(%d)", dynamic_config.frame_num);
1420         dynamic_config.pending = false;
1421       }
1422     }
1423     if (!dynamic_config.pending)
1424       VencTest_ReadDynamicConfigMsg();
1425   } while (!dynamic_config.pending && m_pDynConfFile);
1426 }
1427 
1428 ////////////////////////////////////////////////////////////////////////////////
VencTest_ReadAndEmpty(OMX_BUFFERHEADERTYPE * pYUVBuffer)1429 OMX_ERRORTYPE VencTest_ReadAndEmpty(OMX_BUFFERHEADERTYPE* pYUVBuffer)
1430 {
1431    OMX_ERRORTYPE result = OMX_ErrorNone;
1432 #ifdef T_ARM
1433 #ifdef MAX_RES_720P
1434    if (read(m_nInFd,
1435             pYUVBuffer->pBuffer,
1436             m_sProfile.nFrameBytes) != m_sProfile.nFrameBytes)
1437    {
1438       return OMX_ErrorUndefined;
1439    }
1440 #elif BADGER
1441    int bytes;
1442    E("will read YUV now: %d bytes to buffer %p\n", m_sProfile.nFrameRead, pYUVBuffer->pBuffer);
1443    E("W: %d H: %d Str: %d scal: %d \n", m_sProfile.nFrameWidth, m_sProfile.nFrameHeight,
1444 		m_sProfile.nFramestride, m_sProfile.nFrameScanlines);
1445    bytes = read(m_nInFd, pYUVBuffer->pBuffer, m_sProfile.nFrameRead);
1446    if (bytes != m_sProfile.nFrameRead) {
1447 		E("read failed: %d != %d\n", read, m_sProfile.nFrameRead);
1448 		return OMX_ErrorUndefined;
1449    }
1450    E("\n\nRead %d bytes\n\n\n", m_sProfile.nFrameRead);
1451 #else
1452          OMX_U32 bytestoread = m_sProfile.nFrameWidth*m_sProfile.nFrameHeight;
1453          // read Y first
1454          if (read(m_nInFd,
1455               pYUVBuffer->pBuffer,
1456               bytestoread) != bytestoread)
1457             return OMX_ErrorUndefined;
1458 
1459          // check alignment for offset to C
1460          OMX_U32 offset_to_c = m_sProfile.nFrameWidth * m_sProfile.nFrameHeight;
1461 
1462          const OMX_U32 C_2K = (1024*2),
1463             MASK_2K = C_2K-1,
1464             IMASK_2K = ~MASK_2K;
1465 
1466          if (offset_to_c & MASK_2K)
1467          {
1468             // offset to C is not 2k aligned, adjustment is required
1469             offset_to_c = (offset_to_c & IMASK_2K) + C_2K;
1470          }
1471 
1472          bytestoread = m_sProfile.nFrameWidth*m_sProfile.nFrameHeight/2;
1473          // read C
1474          if (read(m_nInFd,
1475               pYUVBuffer->pBuffer + offset_to_c,
1476               bytestoread)!= bytestoread)
1477             return OMX_ErrorUndefined;
1478 #endif
1479 #else
1480    {
1481 	  char * pInputbuf = (char *)(pYUVBuffer->pBuffer) ;
1482 	      read(m_nInFd,pInputbuf,m_sProfile.nFrameBytes) ;
1483 
1484    }
1485 #endif
1486    if (m_pDynConfFile)
1487      VencTest_ProcessDynamicConfigurationFile();
1488    D("about to call VencTest_EncodeFrame...");
1489    pthread_mutex_lock(&m_mutex);
1490    ++m_nFrameIn;
1491 #ifdef BADGER
1492    pYUVBuffer->nFilledLen = m_sProfile.nFrameRead;
1493 #else
1494    pYUVBuffer->nFilledLen = m_sProfile.nFrameBytes;
1495 #endif
1496    D("Called Buffer with Data filled length %d",pYUVBuffer->nFilledLen);
1497 
1498       result = VencTest_EncodeFrame(pYUVBuffer->pBuffer,
1499                                  m_nTimeStamp);
1500 
1501    m_nTimeStamp += (1000000) / m_sProfile.nFramerate;
1502    CHK(result);
1503    pthread_mutex_unlock(&m_mutex);
1504    return result;
1505 }
1506 ////////////////////////////////////////////////////////////////////////////////
PreviewCallback(int nFD,int nOffset,void * pPhys,void * pVirt,long long nTimeStamp)1507 void PreviewCallback(int nFD,
1508                      int nOffset,
1509                      void* pPhys,
1510                      void* pVirt,
1511                      long long nTimeStamp)
1512 {
1513 
1514    D("================= preview frame %d, phys=0x%x, nTimeStamp(millis)=%lld",
1515      m_nFrameIn+1, pPhys, (nTimeStamp / 1000));
1516 
1517    if (m_nFrameIn == m_nFramePlay &&
1518        m_nFramePlay != 0)
1519    {
1520       // we will stop camera after last frame is encoded.
1521       // for now just ignore input frames
1522 
1523       CameraTest_ReleaseFrame(pPhys, pVirt);
1524       return;
1525    }
1526 
1527    // see if we should stop
1528    pthread_mutex_lock(&m_mutex);
1529    ++m_nFrameIn;
1530    pthread_mutex_unlock(&m_mutex);
1531 
1532 
1533    if (m_eMode == MODE_LIVE_ENCODE)
1534    {
1535 
1536       OMX_ERRORTYPE result;
1537 
1538       // register new camera buffers with encoder
1539       int i;
1540       for (i = 0; i < num_in_buffers; i++)
1541       {
1542          if (m_pInBuffers[i] != NULL &&
1543              m_pInBuffers[i]->pBuffer == pPhys)
1544          {
1545             break;
1546          }
1547          else if (m_pInBuffers[i] == NULL)
1548          {
1549             D("registering buffer...");
1550             result = VencTest_RegisterYUVBuffer(&m_pInBuffers[i],
1551                                                 (OMX_U8*) pPhys,
1552                                                 (OMX_PTR) pVirt); // store virt in app private field
1553             D("register done");
1554             CHK(result);
1555             break;
1556          }
1557       }
1558 
1559       if (i == num_in_buffers)
1560       {
1561          E("There are more camera buffers than we thought");
1562          CHK(1);
1563       }
1564 
1565       // encode the yuv frame
1566 
1567       D("StartEncodeTime=%lld", GetTimeStamp());
1568       result = VencTest_EncodeFrame(pPhys,
1569                                     nTimeStamp);
1570       CHK(result);
1571      // FBTest_DisplayImage(nFD, nOffset);
1572    }
1573    else
1574    {
1575      // FBTest_DisplayImage(nFD, nOffset);
1576       CameraTest_ReleaseFrame(pPhys, pVirt);
1577    }
1578 }
1579 ////////////////////////////////////////////////////////////////////////////////
usage(char * filename)1580 void usage(char* filename)
1581 {
1582    char* fname = strrchr(filename, (int) '/');
1583    fname = (fname == NULL) ? filename : fname;
1584 
1585    fprintf(stderr, "usage: %s LIVE <QCIF|QVGA> <MP4|H263> <FPS> <BITRATE> <NFRAMES> <OUTFILE>\n", fname);
1586    fprintf(stderr, "usage: %s FILE <QCIF|QVGA> <MP4|H263 <FPS> <BITRATE> <NFRAMES> <INFILE> <OUTFILE> ", fname);
1587    fprintf(stderr, "<Dynamic config file - opt> <Rate Control - opt> <AVC Slice Mode - opt>\n", fname);
1588    fprintf(stderr, "usage: %s PROFILE <QCIF|QVGA> <MP4|H263 <FPS> <BITRATE> <NFRAMES> <INFILE>\n", fname);
1589    fprintf(stderr, "usage: %s PREVIEW <QCIF|QVGA> <FPS> <NFRAMES>\n", fname);
1590    fprintf(stderr, "usage: %s DISPLAY <QCIF|QVGA> <FPS> <NFRAMES> <INFILE>\n", fname);
1591    fprintf(stderr, "\n       BITRATE - bitrate in kbps\n");
1592    fprintf(stderr, "       FPS - frames per second\n");
1593    fprintf(stderr, "       NFRAMES - number of frames to play, 0 for infinite\n");
1594    fprintf(stderr, "       RateControl (Values 0 - 4 for RC_OFF, RC_CBR_CFR, RC_CBR_VFR, RC_VBR_CFR, RC_VBR_VFR\n");
1595    exit(1);
1596 }
1597 
parseWxH(char * str,OMX_U32 * width,OMX_U32 * height)1598 bool parseWxH(char *str, OMX_U32 *width, OMX_U32 *height)
1599 {
1600    bool parseOK = false;
1601    const char delimiters[] = " x*,";
1602    char *token, *dupstr, *temp;
1603    OMX_U32 w, h;
1604 
1605    dupstr = strdup(str);
1606    token = strtok_r(dupstr, delimiters, &temp);
1607    if (token)
1608    {
1609        w = strtoul(token, NULL, 10);
1610        token = strtok_r(NULL, delimiters, &temp);
1611        if (token)
1612        {
1613            h = strtoul(token, NULL, 10);
1614            if (w != ULONG_MAX && h != ULONG_MAX)
1615            {
1616 #ifdef MAX_RES_720P
1617               if ((w * h >> 8) <= 3600)
1618               {
1619                  parseOK = true;
1620                  *width = w;
1621                  *height = h;
1622                  }
1623 #else
1624               if ((w * h >> 8) <= 8160)
1625               {
1626                  parseOK = true;
1627                  *width = w;
1628                  *height = h;
1629                  }
1630 #endif
1631               else
1632                  E("\nInvalid dimensions %dx%d",w,h);
1633               }
1634            }
1635        }
1636    free(dupstr);
1637    return parseOK;
1638 }
1639 
1640 ////////////////////////////////////////////////////////////////////////////////
parseArgs(int argc,char ** argv)1641 void parseArgs(int argc, char** argv)
1642 {
1643    int dyn_file_arg = argc;
1644    if (argc == 1)
1645    {
1646       usage(argv[0]);
1647    }
1648    else if (strcmp("PREVIEW", argv[1]) == 0 ||
1649             strcmp("preview", argv[1]) == 0)
1650    {
1651       m_eMode = MODE_PREVIEW;
1652       if (argc != 5)
1653       {
1654          usage(argv[0]);
1655       }
1656    }
1657    else if (strcmp("DISPLAY", argv[1]) == 0 ||
1658             strcmp("display", argv[1]) == 0)
1659    {
1660       m_eMode = MODE_DISPLAY;
1661       if (argc != 6)
1662       {
1663          usage(argv[0]);
1664       }
1665       m_sProfile.cInFileName = argv[5];
1666       m_sProfile.cOutFileName = NULL;
1667    }
1668    else if (strcmp("LIVE", argv[1]) == 0 ||
1669             strcmp("live", argv[1]) == 0)
1670    {//263
1671       m_eMode = MODE_LIVE_ENCODE;
1672       if (argc != 8)
1673       {
1674          usage(argv[0]);
1675       }
1676       m_sProfile.cInFileName = NULL;
1677       m_sProfile.cOutFileName = argv[7];
1678    }
1679    else if (strcmp("FILE", argv[1]) == 0 ||
1680             strcmp("file", argv[1]) == 0)
1681    {//263
1682       m_eMode = MODE_FILE_ENCODE;
1683 
1684       if(argc < 9 || argc > 13)
1685       {
1686           usage(argv[0]);
1687       }
1688       else
1689       {
1690          if (argc > 9)
1691             dyn_file_arg = 9;
1692 
1693          if (argc > 10)
1694          {
1695            m_sProfile.eControlRate = OMX_Video_ControlRateVariable;
1696             int RC = atoi(argv[10]);
1697 
1698             switch (RC)
1699             {
1700             case 0:
1701                m_sProfile.eControlRate  = OMX_Video_ControlRateDisable ;//VENC_RC_NONE
1702                break;
1703             case 1:
1704                m_sProfile.eControlRate  = OMX_Video_ControlRateConstant;//VENC_RC_CBR_CFR
1705                break;
1706 
1707             case 2:
1708                m_sProfile.eControlRate  = OMX_Video_ControlRateConstantSkipFrames;//VENC_RC_CBR_VFR
1709                break;
1710 
1711             case 3:
1712                m_sProfile.eControlRate  =OMX_Video_ControlRateVariable ;//VENC_RC_VBR_CFR
1713                break;
1714 
1715             case 4:
1716                m_sProfile.eControlRate  = OMX_Video_ControlRateVariableSkipFrames;//VENC_RC_VBR_VFR
1717                break;
1718 
1719            default:
1720                E("invalid rate control selection");
1721                m_sProfile.eControlRate = OMX_Video_ControlRateVariable; //VENC_RC_VBR_CFR
1722                break;
1723             }
1724          }
1725 
1726          if (argc > 11)
1727          {
1728             int profile_argi = 11;
1729             if(!strcmp(argv[3], "H264") || !strcmp(argv[3], "h264"))
1730             {
1731                profile_argi = 12;
1732                D("\nSetting AVCSliceMode ... ");
1733                int AVCSliceMode = atoi(argv[11]);
1734                switch(AVCSliceMode)
1735                {
1736                case 0:
1737                   m_sProfile.eSliceMode = OMX_VIDEO_SLICEMODE_AVCDefault;
1738                   break;
1739 
1740                case 1:
1741                   m_sProfile.eSliceMode = OMX_VIDEO_SLICEMODE_AVCMBSlice;
1742                   break;
1743 
1744                case 2:
1745                   m_sProfile.eSliceMode = OMX_VIDEO_SLICEMODE_AVCByteSlice;
1746                   break;
1747 
1748                default:
1749                   E("invalid Slice Mode");
1750                   m_sProfile.eSliceMode = OMX_VIDEO_SLICEMODE_AVCDefault;
1751                   break;
1752               }
1753             }
1754             if (profile_argi < argc)
1755             {
1756                if (!strncmp(argv[profile_argi], "0x", 2) || !strncmp(argv[profile_argi], "0x", 2))
1757                {
1758                   m_sProfile.nUserProfile = strtoul(argv[profile_argi], NULL, 16);
1759                }
1760                else
1761                {
1762                   m_sProfile.nUserProfile = strtoul(argv[profile_argi], NULL, 10);
1763                }
1764                if (!m_sProfile.nUserProfile || m_sProfile.nUserProfile == ULONG_MAX)
1765                {
1766                   E("invalid specified Profile %s, using default", argv[profile_argi]);
1767                   m_sProfile.nUserProfile = 0;
1768                }
1769             }
1770          }
1771       }
1772       m_sProfile.cInFileName = argv[7];
1773       m_sProfile.cOutFileName = argv[8];
1774    }
1775    else if (strcmp("PROFILE", argv[1]) == 0 ||
1776             strcmp("profile", argv[1]) == 0)
1777    {//263
1778       m_eMode = MODE_PROFILE;
1779       if (argc != 8)
1780       {
1781          usage(argv[0]);
1782       }
1783       m_sProfile.cInFileName = argv[7];
1784       m_sProfile.cOutFileName = NULL;
1785    }
1786    else
1787    {
1788       usage(argv[0]);
1789    }
1790 
1791 
1792    if (strcmp("QCIF", argv[2]) == 0 ||
1793        strcmp("qcif", argv[2]) == 0)
1794    {
1795       m_sProfile.nFrameWidth = 176;
1796       m_sProfile.nFrameHeight = 144;
1797       m_sProfile.nFrameBytes = 176*144*3/2;
1798       m_sProfile.eLevel = OMX_VIDEO_MPEG4Level0;
1799    }
1800    else if (strcmp("QVGA", argv[2]) == 0 ||
1801             strcmp("qvga", argv[2]) == 0)
1802    {
1803       m_sProfile.nFrameWidth = 320;
1804       m_sProfile.nFrameHeight = 240;
1805       m_sProfile.nFrameBytes = 320*240*3/2;
1806       m_sProfile.eLevel = OMX_VIDEO_MPEG4Level1;
1807    }
1808 
1809 
1810     else if (strcmp("VGA", argv[2]) == 0 ||
1811             strcmp("vga", argv[2]) == 0)
1812    {
1813       m_sProfile.nFrameWidth = 640;
1814       m_sProfile.nFrameHeight = 480;
1815       m_sProfile.nFrameBytes = 640*480*3/2;
1816       m_sProfile.eLevel = OMX_VIDEO_MPEG4Level1;
1817    }
1818 
1819     else if (strcmp("WVGA", argv[2]) == 0 ||
1820             strcmp("wvga", argv[2]) == 0)
1821    {
1822       m_sProfile.nFrameWidth = 800;
1823       m_sProfile.nFrameHeight = 480;
1824       m_sProfile.nFrameBytes = 800*480*3/2;
1825       m_sProfile.eLevel = OMX_VIDEO_MPEG4Level1;
1826    }
1827   else if (strcmp("CIF", argv[2]) == 0 ||
1828             strcmp("cif", argv[2]) == 0)
1829    {
1830       m_sProfile.nFrameWidth = 352;
1831       m_sProfile.nFrameHeight = 288;
1832       m_sProfile.nFrameBytes = 352*288*3/2;
1833       m_sProfile.eLevel = OMX_VIDEO_MPEG4Level1;
1834    }
1835    else if (strcmp("720", argv[2]) == 0)
1836    {
1837       m_sProfile.nFrameWidth = 1280;
1838       m_sProfile.nFrameHeight = 720;
1839       m_sProfile.nFrameBytes = 720*1280*3/2;
1840       m_sProfile.eLevel = OMX_VIDEO_MPEG4Level1;
1841    }
1842    else if (strcmp("1080", argv[2]) == 0)
1843    {
1844       m_sProfile.nFrameWidth = 1920;
1845       m_sProfile.nFrameHeight = 1080;
1846       m_sProfile.nFrameBytes = 1920*1080*3/2;
1847       m_sProfile.eLevel = OMX_VIDEO_MPEG4Level1;
1848    }
1849    else if (parseWxH(argv[2], &m_sProfile.nFrameWidth, &m_sProfile.nFrameHeight))
1850    {
1851       m_sProfile.nFrameBytes = m_sProfile.nFrameWidth*m_sProfile.nFrameHeight*3/2;
1852       m_sProfile.eLevel = OMX_VIDEO_MPEG4Level1;
1853    }
1854    else
1855    {
1856       usage(argv[0]);
1857    }
1858 
1859 #ifdef BADGER
1860    m_sProfile.nFramestride =  (m_sProfile.nFrameWidth + 31) & (~31);
1861    m_sProfile.nFrameScanlines = (m_sProfile.nFrameHeight + 31) & (~31);
1862    m_sProfile.nFrameBytes = ((m_sProfile.nFramestride * m_sProfile.nFrameScanlines * 3/2) + 4095) & (~4095);
1863    E("stride: %d, Scanlines: %d, Size: %d",
1864      m_sProfile.nFramestride, m_sProfile.nFrameScanlines, m_sProfile.nFrameBytes);
1865    m_sProfile.nFrameRead = m_sProfile.nFramestride * m_sProfile.nFrameScanlines * 3/2;
1866 #endif
1867    if (m_eMode == MODE_DISPLAY ||
1868        m_eMode == MODE_PREVIEW)
1869    {
1870       m_sProfile.nFramerate = atof(argv[3]);
1871       m_nFramePlay = atoi(argv[4]);
1872 
1873    }
1874    else if (m_eMode == MODE_LIVE_ENCODE ||
1875             m_eMode == MODE_FILE_ENCODE ||
1876             m_eMode == MODE_PROFILE)
1877    {
1878       if ((!strcmp(argv[3], "MP4")) || (!strcmp(argv[3], "mp4")))
1879       {
1880          m_sProfile.eCodec = OMX_VIDEO_CodingMPEG4;
1881       }
1882       else if ((!strcmp(argv[3], "H263")) || (!strcmp(argv[3], "h263")))
1883       {
1884          m_sProfile.eCodec = OMX_VIDEO_CodingH263;
1885       }
1886       else if ((!strcmp(argv[3], "H264")) || (!strcmp(argv[3], "h264")))
1887       {
1888          m_sProfile.eCodec = OMX_VIDEO_CodingAVC;
1889       }
1890       else
1891       {
1892          usage(argv[0]);
1893       }
1894 
1895       m_sProfile.nFramerate = atof(argv[4]);
1896       m_sProfile.nBitrate = atoi(argv[5]);
1897 //      m_sProfile.eControlRate = OMX_Video_ControlRateVariable;
1898       m_nFramePlay = atoi(argv[6]);
1899       if (dyn_file_arg < argc)
1900       {
1901         m_pDynConfFile = fopen(argv[dyn_file_arg], "r");
1902         if (!m_pDynConfFile)
1903           E("ERROR: Cannot open dynamic config file: %s", argv[dyn_file_arg]);
1904         else
1905         {
1906           memset(&dynamic_config, 0, sizeof(struct DynamicConfig));
1907         }
1908       }
1909    }
1910 }
1911 
Watchdog(void * data)1912 void* Watchdog(void* data)
1913 {
1914    while (1)
1915    {
1916       sleep(1000);
1917       if (m_bWatchDogKicked == true)
1918          m_bWatchDogKicked = false;
1919       else
1920          E("watchdog has not been kicked. we may have a deadlock");
1921    }
1922    return NULL;
1923 }
1924 
main(int argc,char ** argv)1925 int main(int argc, char** argv)
1926 {
1927    OMX_U8* pvirt = NULL;
1928    int result;
1929    float enc_time_sec=0.0,enc_time_usec=0.0;
1930 
1931    m_nInFd = -1;
1932    m_nOutFd = -1;
1933    m_nTimeStamp = 0;
1934    m_nFrameIn = 0;
1935    m_nFrameOut = 0;
1936 
1937    memset(&m_sMsgQ, 0, sizeof(MsgQ));
1938    parseArgs(argc, argv);
1939 
1940    D("fps=%d, bitrate=%d, width=%d, height=%d",
1941      m_sProfile.nFramerate,
1942      m_sProfile.nBitrate,
1943      m_sProfile.nFrameWidth,
1944      m_sProfile.nFrameHeight);
1945 
1946 
1947    //if (m_eMode != MODE_PREVIEW && m_eMode != MODE_DISPLAY)
1948    //{
1949      // pthread_t wd;
1950      // pthread_create(&wd, NULL, Watchdog, NULL);
1951    //}
1952 
1953    for (int x = 0; x < num_in_buffers; x++)
1954    {
1955       // mark all buffers as ready to use
1956       m_bInFrameFree[x] = OMX_TRUE;
1957    }
1958 
1959 
1960     if (m_eMode != MODE_PROFILE)
1961    {
1962       #if T_ARM
1963 	   m_nOutFd = open(m_sProfile.cOutFileName, O_WRONLY | O_CREAT | O_TRUNC, S_IRWXU | S_IRWXG | S_IRWXO);
1964       #else
1965 	  m_nOutFd = open(m_sProfile.cOutFileName,0);
1966       #endif
1967       if (m_nOutFd < 0)
1968       {
1969          E("could not open output file %s", m_sProfile.cOutFileName);
1970          CHK(1);
1971       }
1972    }
1973 
1974    pthread_mutex_init(&m_mutex, NULL);
1975    pthread_cond_init(&m_signal, NULL);
1976 
1977    if (m_eMode != MODE_PREVIEW)
1978    {
1979       VencTest_Initialize();
1980    }
1981 
1982    ////////////////////////////////////////
1983    // Camera + Encode
1984    ////////////////////////////////////////
1985    if (m_eMode == MODE_LIVE_ENCODE)
1986    {
1987      CameraTest_Initialize(m_sProfile.nFramerate,
1988                             m_sProfile.nFrameWidth,
1989                             m_sProfile.nFrameHeight,
1990                             PreviewCallback);
1991       CameraTest_Run();
1992    }
1993 
1994    if (m_eMode == MODE_FILE_ENCODE ||
1995        m_eMode == MODE_PROFILE)
1996    {
1997       int i;
1998       #if T_ARM
1999       m_nInFd = open(m_sProfile.cInFileName, O_RDONLY);
2000       #else
2001       m_nInFd = open(m_sProfile.cInFileName,1);
2002       #endif
2003 	  if (m_nInFd < 0)
2004       {
2005          E("could not open input file");
2006          CHK(1);
2007 
2008       }
2009       D("going to idle state");
2010       //SetState(OMX_StateIdle);
2011       OMX_SendCommand(m_hHandle,
2012                       OMX_CommandStateSet,
2013                       (OMX_U32) OMX_StateIdle,
2014                        NULL);
2015 
2016       OMX_PARAM_PORTDEFINITIONTYPE portDef;
2017 
2018       portDef.nPortIndex = 0;
2019       result = OMX_GetParameter(m_hHandle, OMX_IndexParamPortDefinition, &portDef);
2020       CHK(result);
2021 
2022       D("allocating Input buffers");
2023       num_in_buffers = portDef.nBufferCountActual;
2024       for (i = 0; i < portDef.nBufferCountActual; i++)
2025       {
2026          OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO* pMem = new OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO;
2027          pvirt = (OMX_U8*)PmemMalloc(pMem, m_sProfile.nFrameBytes);
2028 
2029          if(pvirt == NULL)
2030          {
2031             CHK(1);
2032          }
2033          result = VencTest_RegisterYUVBuffer(&m_pInBuffers[i],
2034                                              (OMX_U8*) pvirt,
2035                                              (OMX_PTR) pMem);
2036          CHK(result);
2037       }
2038    }
2039    else if (m_eMode == MODE_LIVE_ENCODE)
2040    {
2041        D("going to idle state");
2042        //SetState(OMX_StateIdle);
2043        OMX_SendCommand(m_hHandle,
2044                        OMX_CommandStateSet,
2045                        (OMX_U32) OMX_StateIdle,
2046                         NULL);
2047    }
2048 
2049    int i;
2050    OMX_PARAM_PORTDEFINITIONTYPE portDef;
2051 
2052    portDef.nPortIndex = 1;
2053    result = OMX_GetParameter(m_hHandle, OMX_IndexParamPortDefinition, &portDef);
2054    CHK(result);
2055 
2056    D("allocating output buffers");
2057    D("Calling UseBuffer for Output port");
2058    num_out_buffers = portDef.nBufferCountActual;
2059    for (i = 0; i < portDef.nBufferCountActual; i++)
2060    {
2061       void* pBuff;
2062 
2063       pBuff = malloc(portDef.nBufferSize);
2064      D("portDef.nBufferSize = %d ",portDef.nBufferSize);
2065       result = OMX_UseBuffer(m_hHandle,
2066                              &m_pOutBuffers[i],
2067                              (OMX_U32) PORT_INDEX_OUT,
2068                              NULL,
2069                              portDef.nBufferSize,
2070                              (OMX_U8*) pBuff);
2071       CHK(result);
2072    }
2073    D("allocate done");
2074 
2075         // D("Going to state " # eState"...");
2076 
2077          while (m_eState != OMX_StateIdle)
2078          {
2079             sleep(1);
2080          }
2081          //D("Now in state " # eState);
2082 
2083 
2084    D("going to executing state");
2085    SetState(OMX_StateExecuting);
2086    for (i = 0; i < num_out_buffers; i++)
2087    {
2088       D("filling buffer %d", i);
2089       result = OMX_FillThisBuffer(m_hHandle, m_pOutBuffers[i]);
2090       //sleep(1000);
2091       CHK(result);
2092    }
2093 
2094    if (m_eMode == MODE_FILE_ENCODE)
2095    {
2096       // encode the first frame to kick off the whole process
2097       VencTest_ReadAndEmpty(m_pInBuffers[0]);
2098     //  FBTest_DisplayImage(((PmemBuffer*) m_pInBuffers[0]->pAppPrivate)->fd,0);
2099    }
2100 
2101    if (m_eMode == MODE_PROFILE)
2102    {
2103       int i;
2104 
2105       // read several frames into memory
2106       D("reading frames into memory");
2107       for (i = 0; i < num_in_buffers; i++)
2108       {
2109         D("[%d] address 0x%x",i, m_pInBuffers[i]->pBuffer);
2110 #ifdef MAX_RES_720P
2111          read(m_nInFd,
2112               m_pInBuffers[i]->pBuffer,
2113               m_sProfile.nFrameBytes);
2114 #else
2115          // read Y first
2116          read(m_nInFd,
2117               m_pInBuffers[i]->pBuffer,
2118               m_sProfile.nFrameWidth*m_sProfile.nFrameHeight);
2119 
2120          // check alignment for offset to C
2121          OMX_U32 offset_to_c = m_sProfile.nFrameWidth * m_sProfile.nFrameHeight;
2122 
2123          const OMX_U32 C_2K = (1024*2),
2124             MASK_2K = C_2K-1,
2125             IMASK_2K = ~MASK_2K;
2126 
2127          if (offset_to_c & MASK_2K)
2128          {
2129             // offset to C is not 2k aligned, adjustment is required
2130             offset_to_c = (offset_to_c & IMASK_2K) + C_2K;
2131          }
2132 
2133          // read C
2134          read(m_nInFd,
2135               m_pInBuffers[i]->pBuffer + offset_to_c,
2136               m_sProfile.nFrameWidth*m_sProfile.nFrameHeight/2);
2137 #endif
2138 
2139       }
2140 
2141      // FBTest_Initialize(m_sProfile.nFrameWidth, m_sProfile.nFrameHeight);
2142 
2143       // loop over the mem-resident frames and encode them
2144       D("beging playing mem-resident frames...");
2145       for (i = 0; m_nFramePlay == 0 || i < m_nFramePlay; i++)
2146       {
2147          int idx = i % num_in_buffers;
2148          if (m_bInFrameFree[idx] == OMX_FALSE)
2149          {
2150             int j;
2151             E("the expected buffer is not free, but lets find another");
2152 
2153             idx = -1;
2154 
2155             // lets see if we can find another free buffer
2156             for (j = 0; j < num_in_buffers; j++)
2157             {
2158                if(m_bInFrameFree[j])
2159                {
2160                   idx = j;
2161                   break;
2162                }
2163             }
2164          }
2165 
2166          // if we have a free buffer let's encode it
2167          if (idx >= 0)
2168          {
2169             D("encode frame %d...m_pInBuffers[idx]->pBuffer=0x%x", i,m_pInBuffers[idx]->pBuffer);
2170             m_bInFrameFree[idx] = OMX_FALSE;
2171             VencTest_EncodeFrame(m_pInBuffers[idx]->pBuffer,
2172                                  m_nTimeStamp);
2173             D("display frame %d...", i);
2174         //    FBTest_DisplayImage(((PmemBuffer*) m_pInBuffers[idx]->pAppPrivate)->fd,0);
2175             m_nTimeStamp += 1000000 / m_sProfile.nFramerate;
2176          }
2177          else
2178          {
2179             E("wow, no buffers are free, performance "
2180               "is not so good. lets just sleep some more");
2181 
2182          }
2183          D("sleep for %d microsec", 1000000/m_sProfile.nFramerate);
2184          sleep (1000000 / m_sProfile.nFramerate);
2185       }
2186      // FBTest_Exit();
2187    }
2188 
2189    Msg msg;
2190    bool bQuit = false;
2191    while ((m_eMode == MODE_FILE_ENCODE || m_eMode == MODE_LIVE_ENCODE) &&
2192           !bQuit)
2193    {
2194       PopMessage(&msg);
2195       switch (msg.id)
2196       {
2197       //////////////////////////////////
2198       // FRAME IS ENCODED
2199       //////////////////////////////////
2200       case MSG_ID_INPUT_FRAME_DONE:
2201          /*pthread_mutex_lock(&m_mutex);
2202          ++m_nFrameOut;
2203          if (m_nFrameOut == m_nFramePlay && m_nFramePlay != 0)
2204          {
2205             bQuit = true;
2206          }
2207          pthread_mutex_unlock(&m_mutex);*/
2208 
2209          if (!bQuit && m_eMode == MODE_FILE_ENCODE)
2210          {
2211             D("pushing another frame down to encoder");
2212             if (VencTest_ReadAndEmpty(msg.data.sBitstreamData.pBuffer))
2213             {
2214                // we have read the last frame
2215                D("main is exiting...");
2216                bQuit = true;
2217             }
2218          }
2219        break;
2220       case MSG_ID_OUTPUT_FRAME_DONE:
2221          D("================ writing frame %d = %d bytes to output file",
2222            m_nFrameOut+1,
2223            msg.data.sBitstreamData.pBuffer->nFilledLen);
2224          D("StopEncodeTime=%lld", GetTimeStamp());
2225 
2226 
2227 		 write(m_nOutFd,
2228                msg.data.sBitstreamData.pBuffer->pBuffer,
2229                msg.data.sBitstreamData.pBuffer->nFilledLen);
2230 
2231 
2232          result = OMX_FillThisBuffer(m_hHandle,
2233                                      msg.data.sBitstreamData.pBuffer);
2234 
2235          if (result != OMX_ErrorNone)
2236          {
2237             CHK(result);
2238          }
2239 
2240          pthread_mutex_lock(&m_mutex);
2241          ++m_nFrameOut;
2242          if (m_nFrameOut == m_nFramePlay && m_nFramePlay != 0)
2243          {
2244             bQuit = true;
2245          }
2246          pthread_mutex_unlock(&m_mutex);
2247          break;
2248 
2249       default:
2250          E("invalid msg id %d", (int) msg.id);
2251       } // end switch (msg.id)
2252 
2253 /*  // TO UNCOMMENT FOR PAUSE TESTINGS
2254       if(m_nFrameOut == 10)
2255       {
2256          E("\nGoing to Pause state\n");
2257          SetState(OMX_StatePause);
2258          sleep(3);
2259 //REQUEST AN I FRAME AFTER PAUSE
2260          OMX_CONFIG_INTRAREFRESHVOPTYPE voprefresh;
2261          voprefresh.nPortIndex = (OMX_U32)PORT_INDEX_OUT;
2262          voprefresh.IntraRefreshVOP = OMX_TRUE;
2263          result = OMX_SetConfig(m_hHandle,
2264                                    OMX_IndexConfigVideoIntraVOPRefresh,
2265                                    &voprefresh);
2266          E("\n OMX_IndexConfigVideoIntraVOPRefresh Set Paramter port");
2267          CHK(result);
2268          E("\nGoing to executing state\n");
2269          SetState(OMX_StateExecuting);
2270       }
2271 */
2272    } // end while (!bQuit)
2273 
2274 
2275    if (m_eMode == MODE_LIVE_ENCODE)
2276    {
2277       CameraTest_Exit();
2278       close(m_nOutFd);
2279    }
2280    else if (m_eMode == MODE_FILE_ENCODE ||
2281             m_eMode == MODE_PROFILE)
2282    {
2283       // deallocate pmem buffers
2284       for (int i = 0; i < num_in_buffers; i++)
2285       {
2286          PmemFree((OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO*)m_pInBuffers[i]->pAppPrivate,
2287                   m_pInBuffers[i]->pBuffer,
2288                   m_sProfile.nFrameBytes);
2289          delete (OMX_QCOM_PLATFORM_PRIVATE_PMEM_INFO*) m_pInBuffers[i]->pAppPrivate;
2290       }
2291       close(m_nInFd);
2292 
2293       if (m_eMode == MODE_FILE_ENCODE)
2294       {
2295          close(m_nOutFd);
2296       }
2297       if (m_pDynConfFile)
2298       {
2299         fclose(m_pDynConfFile);
2300         m_pDynConfFile = NULL;
2301       }
2302    }
2303 
2304    if (m_eMode != MODE_PREVIEW)
2305    {
2306       D("exit encoder test");
2307       VencTest_Exit();
2308    }
2309 
2310    pthread_mutex_destroy(&m_mutex);
2311    pthread_cond_destroy(&m_signal);
2312 
2313    /* Time Statistics Logging */
2314    if(0 != m_sProfile.nFramerate)
2315    {
2316       enc_time_usec = m_nTimeStamp - (1000000 / m_sProfile.nFramerate);
2317       enc_time_sec =enc_time_usec/1000000;
2318       if(0 != enc_time_sec)
2319       {
2320          printf("Total Frame Rate: %f",ebd_cnt/enc_time_sec);
2321          printf("\nEncoder Bitrate :%lf Kbps",(tot_bufsize*8)/(enc_time_sec*1000));
2322       }
2323    }
2324    else
2325    {
2326       printf("\n\n Encode Time is zero");
2327    }
2328    printf("\nTotal Number of Frames :%d",ebd_cnt);
2329    printf("\nNumber of dropped frames during encoding:%d\n",ebd_cnt-fbd_cnt);
2330    /* End of Time Statistics Logging */
2331 
2332    D("main has exited");
2333    return 0;
2334 }
2335