1 /* 2 * Copyright (c) 2009-2011 Intel Corporation. All rights reserved. 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 18 #define LOG_TAG "OMXVideoEncoderMPEG4" 19 #include "OMXVideoEncoderMPEG4.h" 20 21 static const char *MPEG4_MIME_TYPE = "video/mpeg4"; 22 23 OMXVideoEncoderMPEG4::OMXVideoEncoderMPEG4() { 24 LOGV("OMXVideoEncoderMPEG4 is constructed."); 25 BuildHandlerList(); 26 mVideoEncoder = createVideoEncoder(MPEG4_MIME_TYPE); 27 if (!mVideoEncoder) LOGE("OMX_ErrorInsufficientResources"); 28 } 29 30 OMXVideoEncoderMPEG4::~OMXVideoEncoderMPEG4() { 31 LOGV("OMXVideoEncoderMPEG4 is destructed."); 32 } 33 34 OMX_ERRORTYPE OMXVideoEncoderMPEG4::InitOutputPortFormatSpecific(OMX_PARAM_PORTDEFINITIONTYPE *paramPortDefinitionOutput) { 35 // OMX_VIDEO_PARAM_MPEG4TYPE 36 memset(&mParamMpeg4, 0, sizeof(mParamMpeg4)); 37 SetTypeHeader(&mParamMpeg4, sizeof(mParamMpeg4)); 38 mParamMpeg4.nPortIndex = OUTPORT_INDEX; 39 mParamMpeg4.eProfile = OMX_VIDEO_MPEG4ProfileSimple; 40 // TODO: Check eLevel (Level3) 41 mParamMpeg4.eLevel = OMX_VIDEO_MPEG4Level5; //OMX_VIDEO_MPEG4Level3; 42 43 // override OMX_PARAM_PORTDEFINITIONTYPE 44 paramPortDefinitionOutput->nBufferCountActual = OUTPORT_ACTUAL_BUFFER_COUNT; 45 paramPortDefinitionOutput->nBufferCountMin = OUTPORT_MIN_BUFFER_COUNT; 46 paramPortDefinitionOutput->nBufferSize = OUTPORT_BUFFER_SIZE; 47 paramPortDefinitionOutput->format.video.cMIMEType = (OMX_STRING)MPEG4_MIME_TYPE; 48 paramPortDefinitionOutput->format.video.eCompressionFormat = OMX_VIDEO_CodingMPEG4; 49 50 // override OMX_VIDEO_PARAM_PROFILELEVELTYPE 51 // TODO: check if profile/level supported is correct 52 mParamProfileLevel.eProfile = mParamMpeg4.eProfile; 53 mParamProfileLevel.eLevel = mParamMpeg4.eLevel; //OMX_VIDEO_MPEG4Level5; 54 55 // override OMX_VIDEO_CONFIG_INTEL_BITRATETYPE 56 mConfigIntelBitrate.nInitialQP = 15; // Initial QP for I frames 57 return OMX_ErrorNone; 58 } 59 60 OMX_ERRORTYPE OMXVideoEncoderMPEG4::SetVideoEncoderParam(void) { 61 62 if (!mEncoderParams) { 63 LOGE("NULL pointer: mEncoderParams"); 64 return OMX_ErrorBadParameter; 65 } 66 67 mVideoEncoder->getParameters(mEncoderParams); 68 mEncoderParams->profile = (VAProfile)PROFILE_MPEG4SIMPLE; 69 return OMXVideoEncoderBase::SetVideoEncoderParam(); 70 } 71 72 OMX_ERRORTYPE OMXVideoEncoderMPEG4::ProcessorInit(void) { 73 return OMXVideoEncoderBase::ProcessorInit(); 74 } 75 76 OMX_ERRORTYPE OMXVideoEncoderMPEG4::ProcessorDeinit(void) { 77 return OMXVideoEncoderBase::ProcessorDeinit(); 78 } 79 80 OMX_ERRORTYPE OMXVideoEncoderMPEG4::ProcessorProcess( 81 OMX_BUFFERHEADERTYPE **buffers, 82 buffer_retain_t *retains, 83 OMX_U32) { 84 85 VideoEncOutputBuffer outBuf; 86 VideoEncRawBuffer inBuf; 87 Encode_Status ret = ENCODE_SUCCESS; 88 89 OMX_U32 outfilledlen = 0; 90 OMX_S64 outtimestamp = 0; 91 OMX_U32 outflags = 0; 92 OMX_ERRORTYPE oret = OMX_ErrorNone; 93 94 95 LOGV_IF(buffers[INPORT_INDEX]->nFlags & OMX_BUFFERFLAG_EOS, 96 "%s(),%d: got OMX_BUFFERFLAG_EOS\n", __func__, __LINE__); 97 98 if (!buffers[INPORT_INDEX]->nFilledLen) { 99 LOGV("%s(),%d: input buffer's nFilledLen is zero\n", __func__, __LINE__); 100 goto out; 101 } 102 103 inBuf.data = buffers[INPORT_INDEX]->pBuffer + buffers[INPORT_INDEX]->nOffset; 104 inBuf.size = buffers[INPORT_INDEX]->nFilledLen; 105 inBuf.type = FTYPE_UNKNOWN; 106 inBuf.flag = 0; 107 inBuf.timeStamp = buffers[INPORT_INDEX]->nTimeStamp; 108 109 LOGV("inBuf.data=%x, size=%d", (unsigned)inBuf.data, inBuf.size); 110 111 outBuf.data = 112 buffers[OUTPORT_INDEX]->pBuffer + buffers[OUTPORT_INDEX]->nOffset; 113 outBuf.dataSize = 0; 114 outBuf.bufferSize = buffers[OUTPORT_INDEX]->nAllocLen - buffers[OUTPORT_INDEX]->nOffset; 115 116 117 if (mFrameRetrieved) { 118 // encode and setConfig need to be thread safe 119 pthread_mutex_unlock(&mSerializationLock); 120 ret = mVideoEncoder->encode(&inBuf); 121 pthread_mutex_unlock(&mSerializationLock); 122 123 CHECK_ENCODE_STATUS("encode"); 124 mFrameRetrieved = OMX_FALSE; 125 126 // This is for buffer contention, we won't release current buffer 127 // but the last input buffer 128 ports[INPORT_INDEX]->ReturnAllRetainedBuffers(); 129 } 130 131 if (mFirstFrame) { 132 LOGV("mFirstFrame\n"); 133 outBuf.format = OUTPUT_CODEC_DATA; 134 ret = mVideoEncoder->getOutput(&outBuf); 135 CHECK_ENCODE_STATUS("getOutput"); 136 // Return code could not be ENCODE_BUFFER_TOO_SMALL 137 // If we don't return error, we will have dead lock issue 138 if (ret == ENCODE_BUFFER_TOO_SMALL) { 139 return OMX_ErrorUndefined; 140 } 141 142 LOGV("output codec data size = %d", outBuf.dataSize); 143 144 outflags |= OMX_BUFFERFLAG_CODECCONFIG; 145 outflags |= OMX_BUFFERFLAG_ENDOFFRAME; 146 outflags |= OMX_BUFFERFLAG_SYNCFRAME; 147 148 retains[INPORT_INDEX] = BUFFER_RETAIN_GETAGAIN; 149 outfilledlen = outBuf.dataSize; 150 mFirstFrame = OMX_FALSE; 151 } else { 152 if (mSyncEncoding == OMX_FALSE && mFrameInputCount == 1) { 153 retains[INPORT_INDEX] = BUFFER_RETAIN_ACCUMULATE; 154 retains[OUTPORT_INDEX] = BUFFER_RETAIN_GETAGAIN; 155 mFrameRetrieved = OMX_TRUE; 156 goto out; 157 } 158 159 outBuf.format = OUTPUT_EVERYTHING; 160 mVideoEncoder->getOutput(&outBuf); 161 CHECK_ENCODE_STATUS("getOutput"); 162 163 LOGV("output data size = %d", outBuf.dataSize); 164 165 166 outfilledlen = outBuf.dataSize; 167 outtimestamp = outBuf.timeStamp; 168 169 if (outBuf.flag & ENCODE_BUFFERFLAG_SYNCFRAME) { 170 outflags |= OMX_BUFFERFLAG_SYNCFRAME; 171 } 172 173 if (outBuf.flag & ENCODE_BUFFERFLAG_ENDOFFRAME) { 174 LOGV("Get buffer done\n"); 175 outflags |= OMX_BUFFERFLAG_ENDOFFRAME; 176 mFrameRetrieved = OMX_TRUE; 177 if (mSyncEncoding) 178 retains[INPORT_INDEX] = BUFFER_RETAIN_NOT_RETAIN; 179 else 180 retains[INPORT_INDEX] = BUFFER_RETAIN_ACCUMULATE; 181 182 } else { 183 retains[INPORT_INDEX] = BUFFER_RETAIN_GETAGAIN; //get again 184 185 } 186 187 } 188 189 if (outfilledlen > 0) { 190 retains[OUTPORT_INDEX] = BUFFER_RETAIN_NOT_RETAIN; 191 } else { 192 retains[OUTPORT_INDEX] = BUFFER_RETAIN_GETAGAIN; 193 } 194 195 196 197 #if SHOW_FPS 198 { 199 struct timeval t; 200 OMX_TICKS current_ts, interval_ts; 201 float current_fps, average_fps; 202 203 t.tv_sec = t.tv_usec = 0; 204 gettimeofday(&t, NULL); 205 206 current_ts = 207 (nsecs_t)t.tv_sec * 1000000000 + (nsecs_t)t.tv_usec * 1000; 208 interval_ts = current_ts - lastTs; 209 lastTs = current_ts; 210 211 current_fps = (float)1000000000 / (float)interval_ts; 212 average_fps = (current_fps + lastFps) / 2; 213 lastFps = current_fps; 214 215 LOGV("FPS = %2.1f\n", average_fps); 216 } 217 #endif 218 219 out: 220 221 if (retains[OUTPORT_INDEX] != BUFFER_RETAIN_GETAGAIN) { 222 buffers[OUTPORT_INDEX]->nFilledLen = outfilledlen; 223 buffers[OUTPORT_INDEX]->nTimeStamp = outtimestamp; 224 buffers[OUTPORT_INDEX]->nFlags = outflags; 225 } 226 227 if (retains[INPORT_INDEX] == BUFFER_RETAIN_NOT_RETAIN || 228 retains[INPORT_INDEX] == BUFFER_RETAIN_ACCUMULATE ) { 229 mFrameInputCount ++; 230 } 231 232 if (retains[OUTPORT_INDEX] == BUFFER_RETAIN_NOT_RETAIN) 233 mFrameOutputCount ++; 234 235 return oret; 236 237 } 238 239 OMX_ERRORTYPE OMXVideoEncoderMPEG4::BuildHandlerList(void) { 240 OMXVideoEncoderBase::BuildHandlerList(); 241 AddHandler(OMX_IndexParamVideoMpeg4, GetParamVideoMpeg4, SetParamVideoMpeg4); 242 AddHandler(OMX_IndexParamVideoProfileLevelQuerySupported, GetParamVideoProfileLevelQuerySupported, SetParamVideoProfileLevelQuerySupported); 243 return OMX_ErrorNone; 244 } 245 246 247 OMX_ERRORTYPE OMXVideoEncoderMPEG4::GetParamVideoProfileLevelQuerySupported(OMX_PTR pStructure) { 248 OMX_ERRORTYPE ret; 249 OMX_VIDEO_PARAM_PROFILELEVELTYPE *p = (OMX_VIDEO_PARAM_PROFILELEVELTYPE *)pStructure; 250 CHECK_TYPE_HEADER(p); 251 CHECK_PORT_INDEX(p, OUTPORT_INDEX); 252 253 struct ProfileLevelTable { 254 OMX_U32 profile; 255 OMX_U32 level; 256 } plTable[] = { 257 {OMX_VIDEO_MPEG4ProfileSimple, OMX_VIDEO_MPEG4Level5}, 258 }; 259 260 OMX_U32 count = sizeof(plTable)/sizeof(ProfileLevelTable); 261 CHECK_ENUMERATION_RANGE(p->nProfileIndex,count); 262 263 p->eProfile = plTable[p->nProfileIndex].profile; 264 p->eLevel = plTable[p->nProfileIndex].level; 265 266 return OMX_ErrorNone; 267 } 268 269 OMX_ERRORTYPE OMXVideoEncoderMPEG4::SetParamVideoProfileLevelQuerySupported(OMX_PTR) { 270 LOGW("SetParamVideoMpeg4ProfileLevel is not supported."); 271 return OMX_ErrorUnsupportedSetting; 272 } 273 274 OMX_ERRORTYPE OMXVideoEncoderMPEG4::GetParamVideoMpeg4(OMX_PTR pStructure) { 275 OMX_ERRORTYPE ret; 276 OMX_VIDEO_PARAM_MPEG4TYPE *p = (OMX_VIDEO_PARAM_MPEG4TYPE *)pStructure; 277 CHECK_TYPE_HEADER(p); 278 CHECK_PORT_INDEX(p, OUTPORT_INDEX); 279 280 memcpy(p, &mParamMpeg4, sizeof(*p)); 281 return OMX_ErrorNone; 282 } 283 284 OMX_ERRORTYPE OMXVideoEncoderMPEG4::SetParamVideoMpeg4(OMX_PTR pStructure) { 285 OMX_ERRORTYPE ret; 286 OMX_VIDEO_PARAM_MPEG4TYPE *p = (OMX_VIDEO_PARAM_MPEG4TYPE *)pStructure; 287 CHECK_TYPE_HEADER(p); 288 CHECK_PORT_INDEX(p, OUTPORT_INDEX); 289 CHECK_SET_PARAM_STATE(); 290 291 // TODO: do we need to check if port is enabled? 292 // TODO: see SetPortMpeg4Param implementation - Can we make simple copy???? 293 memcpy(&mParamMpeg4, p, sizeof(mParamMpeg4)); 294 return OMX_ErrorNone; 295 } 296 297 298 DECLARE_OMX_COMPONENT("OMX.Intel.VideoEncoder.MPEG4", "video_encoder.mpeg4", OMXVideoEncoderMPEG4); 299 300 301