summaryrefslogtreecommitdiff
path: root/libavcodec/omx.c (plain)
blob: 19b4f33836b68fabe7be921ca959c619dd28e535
1/*
2 * OMX Video encoder
3 * Copyright (C) 2011 Martin Storsjo
4 *
5 * This file is part of FFmpeg.
6 *
7 * FFmpeg is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
11 *
12 * FFmpeg is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
16 *
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with FFmpeg; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
20 */
21
22#include "config.h"
23
24#if CONFIG_OMX_RPI
25#define OMX_SKIP64BIT
26#endif
27
28#include <dlfcn.h>
29#include <OMX_Core.h>
30#include <OMX_Component.h>
31#include <pthread.h>
32#include <stdio.h>
33#include <stdlib.h>
34#include <sys/time.h>
35
36#include "libavutil/avstring.h"
37#include "libavutil/avutil.h"
38#include "libavutil/common.h"
39#include "libavutil/imgutils.h"
40#include "libavutil/log.h"
41#include "libavutil/opt.h"
42
43#include "avcodec.h"
44#include "h264.h"
45#include "internal.h"
46
47#ifdef OMX_SKIP64BIT
48static OMX_TICKS to_omx_ticks(int64_t value)
49{
50 OMX_TICKS s;
51 s.nLowPart = value & 0xffffffff;
52 s.nHighPart = value >> 32;
53 return s;
54}
55static int64_t from_omx_ticks(OMX_TICKS value)
56{
57 return (((int64_t)value.nHighPart) << 32) | value.nLowPart;
58}
59#else
60#define to_omx_ticks(x) (x)
61#define from_omx_ticks(x) (x)
62#endif
63
64#define INIT_STRUCT(x) do { \
65 x.nSize = sizeof(x); \
66 x.nVersion = s->version; \
67 } while (0)
68#define CHECK(x) do { \
69 if (x != OMX_ErrorNone) { \
70 av_log(avctx, AV_LOG_ERROR, \
71 "err %x (%d) on line %d\n", x, x, __LINE__); \
72 return AVERROR_UNKNOWN; \
73 } \
74 } while (0)
75
76typedef struct OMXContext {
77 void *lib;
78 void *lib2;
79 OMX_ERRORTYPE (*ptr_Init)(void);
80 OMX_ERRORTYPE (*ptr_Deinit)(void);
81 OMX_ERRORTYPE (*ptr_ComponentNameEnum)(OMX_STRING, OMX_U32, OMX_U32);
82 OMX_ERRORTYPE (*ptr_GetHandle)(OMX_HANDLETYPE*, OMX_STRING, OMX_PTR, OMX_CALLBACKTYPE*);
83 OMX_ERRORTYPE (*ptr_FreeHandle)(OMX_HANDLETYPE);
84 OMX_ERRORTYPE (*ptr_GetComponentsOfRole)(OMX_STRING, OMX_U32*, OMX_U8**);
85 OMX_ERRORTYPE (*ptr_GetRolesOfComponent)(OMX_STRING, OMX_U32*, OMX_U8**);
86 void (*host_init)(void);
87} OMXContext;
88
89static av_cold void *dlsym_prefixed(void *handle, const char *symbol, const char *prefix)
90{
91 char buf[50];
92 snprintf(buf, sizeof(buf), "%s%s", prefix ? prefix : "", symbol);
93 return dlsym(handle, buf);
94}
95
96static av_cold int omx_try_load(OMXContext *s, void *logctx,
97 const char *libname, const char *prefix,
98 const char *libname2)
99{
100 if (libname2) {
101 s->lib2 = dlopen(libname2, RTLD_NOW | RTLD_GLOBAL);
102 if (!s->lib2) {
103 av_log(logctx, AV_LOG_WARNING, "%s not found\n", libname);
104 return AVERROR_ENCODER_NOT_FOUND;
105 }
106 s->host_init = dlsym(s->lib2, "bcm_host_init");
107 if (!s->host_init) {
108 av_log(logctx, AV_LOG_WARNING, "bcm_host_init not found\n");
109 dlclose(s->lib2);
110 s->lib2 = NULL;
111 return AVERROR_ENCODER_NOT_FOUND;
112 }
113 }
114 s->lib = dlopen(libname, RTLD_NOW | RTLD_GLOBAL);
115 if (!s->lib) {
116 av_log(logctx, AV_LOG_WARNING, "%s not found\n", libname);
117 return AVERROR_ENCODER_NOT_FOUND;
118 }
119 s->ptr_Init = dlsym_prefixed(s->lib, "OMX_Init", prefix);
120 s->ptr_Deinit = dlsym_prefixed(s->lib, "OMX_Deinit", prefix);
121 s->ptr_ComponentNameEnum = dlsym_prefixed(s->lib, "OMX_ComponentNameEnum", prefix);
122 s->ptr_GetHandle = dlsym_prefixed(s->lib, "OMX_GetHandle", prefix);
123 s->ptr_FreeHandle = dlsym_prefixed(s->lib, "OMX_FreeHandle", prefix);
124 s->ptr_GetComponentsOfRole = dlsym_prefixed(s->lib, "OMX_GetComponentsOfRole", prefix);
125 s->ptr_GetRolesOfComponent = dlsym_prefixed(s->lib, "OMX_GetRolesOfComponent", prefix);
126 if (!s->ptr_Init || !s->ptr_Deinit || !s->ptr_ComponentNameEnum ||
127 !s->ptr_GetHandle || !s->ptr_FreeHandle ||
128 !s->ptr_GetComponentsOfRole || !s->ptr_GetRolesOfComponent) {
129 av_log(logctx, AV_LOG_WARNING, "Not all functions found in %s\n", libname);
130 dlclose(s->lib);
131 s->lib = NULL;
132 if (s->lib2)
133 dlclose(s->lib2);
134 s->lib2 = NULL;
135 return AVERROR_ENCODER_NOT_FOUND;
136 }
137 return 0;
138}
139
140static av_cold OMXContext *omx_init(void *logctx, const char *libname, const char *prefix)
141{
142 static const char * const libnames[] = {
143#if CONFIG_OMX_RPI
144 "/opt/vc/lib/libopenmaxil.so", "/opt/vc/lib/libbcm_host.so",
145#else
146 "libOMX_Core.so", NULL,
147 "libOmxCore.so", NULL,
148#endif
149 NULL
150 };
151 const char* const* nameptr;
152 int ret = AVERROR_ENCODER_NOT_FOUND;
153 OMXContext *omx_context;
154
155 omx_context = av_mallocz(sizeof(*omx_context));
156 if (!omx_context)
157 return NULL;
158 if (libname) {
159 ret = omx_try_load(omx_context, logctx, libname, prefix, NULL);
160 if (ret < 0) {
161 av_free(omx_context);
162 return NULL;
163 }
164 } else {
165 for (nameptr = libnames; *nameptr; nameptr += 2)
166 if (!(ret = omx_try_load(omx_context, logctx, nameptr[0], prefix, nameptr[1])))
167 break;
168 if (!*nameptr) {
169 av_free(omx_context);
170 return NULL;
171 }
172 }
173
174 if (omx_context->host_init)
175 omx_context->host_init();
176 omx_context->ptr_Init();
177 return omx_context;
178}
179
180static av_cold void omx_deinit(OMXContext *omx_context)
181{
182 if (!omx_context)
183 return;
184 omx_context->ptr_Deinit();
185 dlclose(omx_context->lib);
186 av_free(omx_context);
187}
188
189typedef struct OMXCodecContext {
190 const AVClass *class;
191 char *libname;
192 char *libprefix;
193 OMXContext *omx_context;
194
195 AVCodecContext *avctx;
196
197 char component_name[OMX_MAX_STRINGNAME_SIZE];
198 OMX_VERSIONTYPE version;
199 OMX_HANDLETYPE handle;
200 int in_port, out_port;
201 OMX_COLOR_FORMATTYPE color_format;
202 int stride, plane_size;
203
204 int num_in_buffers, num_out_buffers;
205 OMX_BUFFERHEADERTYPE **in_buffer_headers;
206 OMX_BUFFERHEADERTYPE **out_buffer_headers;
207 int num_free_in_buffers;
208 OMX_BUFFERHEADERTYPE **free_in_buffers;
209 int num_done_out_buffers;
210 OMX_BUFFERHEADERTYPE **done_out_buffers;
211 pthread_mutex_t input_mutex;
212 pthread_cond_t input_cond;
213 pthread_mutex_t output_mutex;
214 pthread_cond_t output_cond;
215
216 pthread_mutex_t state_mutex;
217 pthread_cond_t state_cond;
218 OMX_STATETYPE state;
219 OMX_ERRORTYPE error;
220
221 int mutex_cond_inited;
222
223 int num_in_frames, num_out_frames;
224
225 uint8_t *output_buf;
226 int output_buf_size;
227
228 int input_zerocopy;
229 int profile;
230} OMXCodecContext;
231
232static void append_buffer(pthread_mutex_t *mutex, pthread_cond_t *cond,
233 int* array_size, OMX_BUFFERHEADERTYPE **array,
234 OMX_BUFFERHEADERTYPE *buffer)
235{
236 pthread_mutex_lock(mutex);
237 array[(*array_size)++] = buffer;
238 pthread_cond_broadcast(cond);
239 pthread_mutex_unlock(mutex);
240}
241
242static OMX_BUFFERHEADERTYPE *get_buffer(pthread_mutex_t *mutex, pthread_cond_t *cond,
243 int* array_size, OMX_BUFFERHEADERTYPE **array,
244 int wait)
245{
246 OMX_BUFFERHEADERTYPE *buffer;
247 pthread_mutex_lock(mutex);
248 if (wait) {
249 while (!*array_size)
250 pthread_cond_wait(cond, mutex);
251 }
252 if (*array_size > 0) {
253 buffer = array[0];
254 (*array_size)--;
255 memmove(&array[0], &array[1], (*array_size) * sizeof(OMX_BUFFERHEADERTYPE*));
256 } else {
257 buffer = NULL;
258 }
259 pthread_mutex_unlock(mutex);
260 return buffer;
261}
262
263static OMX_ERRORTYPE event_handler(OMX_HANDLETYPE component, OMX_PTR app_data, OMX_EVENTTYPE event,
264 OMX_U32 data1, OMX_U32 data2, OMX_PTR event_data)
265{
266 OMXCodecContext *s = app_data;
267 // This uses casts in the printfs, since OMX_U32 actually is a typedef for
268 // unsigned long in official header versions (but there are also modified
269 // versions where it is something else).
270 switch (event) {
271 case OMX_EventError:
272 pthread_mutex_lock(&s->state_mutex);
273 av_log(s->avctx, AV_LOG_ERROR, "OMX error %"PRIx32"\n", (uint32_t) data1);
274 s->error = data1;
275 pthread_cond_broadcast(&s->state_cond);
276 pthread_mutex_unlock(&s->state_mutex);
277 break;
278 case OMX_EventCmdComplete:
279 if (data1 == OMX_CommandStateSet) {
280 pthread_mutex_lock(&s->state_mutex);
281 s->state = data2;
282 av_log(s->avctx, AV_LOG_VERBOSE, "OMX state changed to %"PRIu32"\n", (uint32_t) data2);
283 pthread_cond_broadcast(&s->state_cond);
284 pthread_mutex_unlock(&s->state_mutex);
285 } else if (data1 == OMX_CommandPortDisable) {
286 av_log(s->avctx, AV_LOG_VERBOSE, "OMX port %"PRIu32" disabled\n", (uint32_t) data2);
287 } else if (data1 == OMX_CommandPortEnable) {
288 av_log(s->avctx, AV_LOG_VERBOSE, "OMX port %"PRIu32" enabled\n", (uint32_t) data2);
289 } else {
290 av_log(s->avctx, AV_LOG_VERBOSE, "OMX command complete, command %"PRIu32", value %"PRIu32"\n",
291 (uint32_t) data1, (uint32_t) data2);
292 }
293 break;
294 case OMX_EventPortSettingsChanged:
295 av_log(s->avctx, AV_LOG_VERBOSE, "OMX port %"PRIu32" settings changed\n", (uint32_t) data1);
296 break;
297 default:
298 av_log(s->avctx, AV_LOG_VERBOSE, "OMX event %d %"PRIx32" %"PRIx32"\n",
299 event, (uint32_t) data1, (uint32_t) data2);
300 break;
301 }
302 return OMX_ErrorNone;
303}
304
305static OMX_ERRORTYPE empty_buffer_done(OMX_HANDLETYPE component, OMX_PTR app_data,
306 OMX_BUFFERHEADERTYPE *buffer)
307{
308 OMXCodecContext *s = app_data;
309 if (s->input_zerocopy) {
310 if (buffer->pAppPrivate) {
311 if (buffer->pOutputPortPrivate)
312 av_free(buffer->pAppPrivate);
313 else
314 av_frame_free((AVFrame**)&buffer->pAppPrivate);
315 buffer->pAppPrivate = NULL;
316 }
317 }
318 append_buffer(&s->input_mutex, &s->input_cond,
319 &s->num_free_in_buffers, s->free_in_buffers, buffer);
320 return OMX_ErrorNone;
321}
322
323static OMX_ERRORTYPE fill_buffer_done(OMX_HANDLETYPE component, OMX_PTR app_data,
324 OMX_BUFFERHEADERTYPE *buffer)
325{
326 OMXCodecContext *s = app_data;
327 append_buffer(&s->output_mutex, &s->output_cond,
328 &s->num_done_out_buffers, s->done_out_buffers, buffer);
329 return OMX_ErrorNone;
330}
331
332static const OMX_CALLBACKTYPE callbacks = {
333 event_handler,
334 empty_buffer_done,
335 fill_buffer_done
336};
337
338static av_cold int find_component(OMXContext *omx_context, void *logctx,
339 const char *role, char *str, int str_size)
340{
341 OMX_U32 i, num = 0;
342 char **components;
343 int ret = 0;
344
345#if CONFIG_OMX_RPI
346 if (av_strstart(role, "video_encoder.", NULL)) {
347 av_strlcpy(str, "OMX.broadcom.video_encode", str_size);
348 return 0;
349 }
350#endif
351 omx_context->ptr_GetComponentsOfRole((OMX_STRING) role, &num, NULL);
352 if (!num) {
353 av_log(logctx, AV_LOG_WARNING, "No component for role %s found\n", role);
354 return AVERROR_ENCODER_NOT_FOUND;
355 }
356 components = av_mallocz_array(num, sizeof(*components));
357 if (!components)
358 return AVERROR(ENOMEM);
359 for (i = 0; i < num; i++) {
360 components[i] = av_mallocz(OMX_MAX_STRINGNAME_SIZE);
361 if (!components[i]) {
362 ret = AVERROR(ENOMEM);
363 goto end;
364 }
365 }
366 omx_context->ptr_GetComponentsOfRole((OMX_STRING) role, &num, (OMX_U8**) components);
367 av_strlcpy(str, components[0], str_size);
368end:
369 for (i = 0; i < num; i++)
370 av_free(components[i]);
371 av_free(components);
372 return ret;
373}
374
375static av_cold int wait_for_state(OMXCodecContext *s, OMX_STATETYPE state)
376{
377 int ret = 0;
378 pthread_mutex_lock(&s->state_mutex);
379 while (s->state != state && s->error == OMX_ErrorNone)
380 pthread_cond_wait(&s->state_cond, &s->state_mutex);
381 if (s->error != OMX_ErrorNone)
382 ret = AVERROR_ENCODER_NOT_FOUND;
383 pthread_mutex_unlock(&s->state_mutex);
384 return ret;
385}
386
387static av_cold int omx_component_init(AVCodecContext *avctx, const char *role)
388{
389 OMXCodecContext *s = avctx->priv_data;
390 OMX_PARAM_COMPONENTROLETYPE role_params = { 0 };
391 OMX_PORT_PARAM_TYPE video_port_params = { 0 };
392 OMX_PARAM_PORTDEFINITIONTYPE in_port_params = { 0 }, out_port_params = { 0 };
393 OMX_VIDEO_PARAM_PORTFORMATTYPE video_port_format = { 0 };
394 OMX_VIDEO_PARAM_BITRATETYPE vid_param_bitrate = { 0 };
395 OMX_ERRORTYPE err;
396 int i;
397
398 s->version.s.nVersionMajor = 1;
399 s->version.s.nVersionMinor = 1;
400 s->version.s.nRevision = 2;
401
402 err = s->omx_context->ptr_GetHandle(&s->handle, s->component_name, s, (OMX_CALLBACKTYPE*) &callbacks);
403 if (err != OMX_ErrorNone) {
404 av_log(avctx, AV_LOG_ERROR, "OMX_GetHandle(%s) failed: %x\n", s->component_name, err);
405 return AVERROR_UNKNOWN;
406 }
407
408 // This one crashes the mediaserver on qcom, if used over IOMX
409 INIT_STRUCT(role_params);
410 av_strlcpy(role_params.cRole, role, sizeof(role_params.cRole));
411 // Intentionally ignore errors on this one
412 OMX_SetParameter(s->handle, OMX_IndexParamStandardComponentRole, &role_params);
413
414 INIT_STRUCT(video_port_params);
415 err = OMX_GetParameter(s->handle, OMX_IndexParamVideoInit, &video_port_params);
416 CHECK(err);
417
418 s->in_port = s->out_port = -1;
419 for (i = 0; i < video_port_params.nPorts; i++) {
420 int port = video_port_params.nStartPortNumber + i;
421 OMX_PARAM_PORTDEFINITIONTYPE port_params = { 0 };
422 INIT_STRUCT(port_params);
423 port_params.nPortIndex = port;
424 err = OMX_GetParameter(s->handle, OMX_IndexParamPortDefinition, &port_params);
425 if (err != OMX_ErrorNone) {
426 av_log(avctx, AV_LOG_WARNING, "port %d error %x\n", port, err);
427 break;
428 }
429 if (port_params.eDir == OMX_DirInput && s->in_port < 0) {
430 in_port_params = port_params;
431 s->in_port = port;
432 } else if (port_params.eDir == OMX_DirOutput && s->out_port < 0) {
433 out_port_params = port_params;
434 s->out_port = port;
435 }
436 }
437 if (s->in_port < 0 || s->out_port < 0) {
438 av_log(avctx, AV_LOG_ERROR, "No in or out port found (in %d out %d)\n", s->in_port, s->out_port);
439 return AVERROR_UNKNOWN;
440 }
441
442 s->color_format = 0;
443 for (i = 0; ; i++) {
444 INIT_STRUCT(video_port_format);
445 video_port_format.nIndex = i;
446 video_port_format.nPortIndex = s->in_port;
447 if (OMX_GetParameter(s->handle, OMX_IndexParamVideoPortFormat, &video_port_format) != OMX_ErrorNone)
448 break;
449 if (video_port_format.eColorFormat == OMX_COLOR_FormatYUV420Planar ||
450 video_port_format.eColorFormat == OMX_COLOR_FormatYUV420PackedPlanar) {
451 s->color_format = video_port_format.eColorFormat;
452 break;
453 }
454 }
455 if (s->color_format == 0) {
456 av_log(avctx, AV_LOG_ERROR, "No supported pixel formats (%d formats available)\n", i);
457 return AVERROR_UNKNOWN;
458 }
459
460 in_port_params.bEnabled = OMX_TRUE;
461 in_port_params.bPopulated = OMX_FALSE;
462 in_port_params.eDomain = OMX_PortDomainVideo;
463
464 in_port_params.format.video.pNativeRender = NULL;
465 in_port_params.format.video.bFlagErrorConcealment = OMX_FALSE;
466 in_port_params.format.video.eColorFormat = s->color_format;
467 s->stride = avctx->width;
468 s->plane_size = avctx->height;
469 // If specific codecs need to manually override the stride/plane_size,
470 // that can be done here.
471 in_port_params.format.video.nStride = s->stride;
472 in_port_params.format.video.nSliceHeight = s->plane_size;
473 in_port_params.format.video.nFrameWidth = avctx->width;
474 in_port_params.format.video.nFrameHeight = avctx->height;
475 if (avctx->framerate.den > 0 && avctx->framerate.num > 0)
476 in_port_params.format.video.xFramerate = (1 << 16) * avctx->framerate.num / avctx->framerate.den;
477 else
478 in_port_params.format.video.xFramerate = (1 << 16) * avctx->time_base.den / avctx->time_base.num;
479
480 err = OMX_SetParameter(s->handle, OMX_IndexParamPortDefinition, &in_port_params);
481 CHECK(err);
482 err = OMX_GetParameter(s->handle, OMX_IndexParamPortDefinition, &in_port_params);
483 CHECK(err);
484 s->stride = in_port_params.format.video.nStride;
485 s->plane_size = in_port_params.format.video.nSliceHeight;
486 s->num_in_buffers = in_port_params.nBufferCountActual;
487
488 err = OMX_GetParameter(s->handle, OMX_IndexParamPortDefinition, &out_port_params);
489 out_port_params.bEnabled = OMX_TRUE;
490 out_port_params.bPopulated = OMX_FALSE;
491 out_port_params.eDomain = OMX_PortDomainVideo;
492 out_port_params.format.video.pNativeRender = NULL;
493 out_port_params.format.video.nFrameWidth = avctx->width;
494 out_port_params.format.video.nFrameHeight = avctx->height;
495 out_port_params.format.video.nStride = 0;
496 out_port_params.format.video.nSliceHeight = 0;
497 out_port_params.format.video.nBitrate = avctx->bit_rate;
498 out_port_params.format.video.xFramerate = in_port_params.format.video.xFramerate;
499 out_port_params.format.video.bFlagErrorConcealment = OMX_FALSE;
500 if (avctx->codec->id == AV_CODEC_ID_MPEG4)
501 out_port_params.format.video.eCompressionFormat = OMX_VIDEO_CodingMPEG4;
502 else if (avctx->codec->id == AV_CODEC_ID_H264)
503 out_port_params.format.video.eCompressionFormat = OMX_VIDEO_CodingAVC;
504
505 err = OMX_SetParameter(s->handle, OMX_IndexParamPortDefinition, &out_port_params);
506 CHECK(err);
507 err = OMX_GetParameter(s->handle, OMX_IndexParamPortDefinition, &out_port_params);
508 CHECK(err);
509 s->num_out_buffers = out_port_params.nBufferCountActual;
510
511 INIT_STRUCT(vid_param_bitrate);
512 vid_param_bitrate.nPortIndex = s->out_port;
513 vid_param_bitrate.eControlRate = OMX_Video_ControlRateVariable;
514 vid_param_bitrate.nTargetBitrate = avctx->bit_rate;
515 err = OMX_SetParameter(s->handle, OMX_IndexParamVideoBitrate, &vid_param_bitrate);
516 if (err != OMX_ErrorNone)
517 av_log(avctx, AV_LOG_WARNING, "Unable to set video bitrate parameter\n");
518
519 if (avctx->codec->id == AV_CODEC_ID_H264) {
520 OMX_VIDEO_PARAM_AVCTYPE avc = { 0 };
521 INIT_STRUCT(avc);
522 avc.nPortIndex = s->out_port;
523 err = OMX_GetParameter(s->handle, OMX_IndexParamVideoAvc, &avc);
524 CHECK(err);
525 avc.nBFrames = 0;
526 avc.nPFrames = avctx->gop_size - 1;
527 switch (s->profile == FF_PROFILE_UNKNOWN ? avctx->profile : s->profile) {
528 case FF_PROFILE_H264_BASELINE:
529 avc.eProfile = OMX_VIDEO_AVCProfileBaseline;
530 break;
531 case FF_PROFILE_H264_MAIN:
532 avc.eProfile = OMX_VIDEO_AVCProfileMain;
533 break;
534 case FF_PROFILE_H264_HIGH:
535 avc.eProfile = OMX_VIDEO_AVCProfileHigh;
536 break;
537 default:
538 break;
539 }
540 err = OMX_SetParameter(s->handle, OMX_IndexParamVideoAvc, &avc);
541 CHECK(err);
542 }
543
544 err = OMX_SendCommand(s->handle, OMX_CommandStateSet, OMX_StateIdle, NULL);
545 CHECK(err);
546
547 s->in_buffer_headers = av_mallocz(sizeof(OMX_BUFFERHEADERTYPE*) * s->num_in_buffers);
548 s->free_in_buffers = av_mallocz(sizeof(OMX_BUFFERHEADERTYPE*) * s->num_in_buffers);
549 s->out_buffer_headers = av_mallocz(sizeof(OMX_BUFFERHEADERTYPE*) * s->num_out_buffers);
550 s->done_out_buffers = av_mallocz(sizeof(OMX_BUFFERHEADERTYPE*) * s->num_out_buffers);
551 if (!s->in_buffer_headers || !s->free_in_buffers || !s->out_buffer_headers || !s->done_out_buffers)
552 return AVERROR(ENOMEM);
553 for (i = 0; i < s->num_in_buffers && err == OMX_ErrorNone; i++) {
554 if (s->input_zerocopy)
555 err = OMX_UseBuffer(s->handle, &s->in_buffer_headers[i], s->in_port, s, in_port_params.nBufferSize, NULL);
556 else
557 err = OMX_AllocateBuffer(s->handle, &s->in_buffer_headers[i], s->in_port, s, in_port_params.nBufferSize);
558 if (err == OMX_ErrorNone)
559 s->in_buffer_headers[i]->pAppPrivate = s->in_buffer_headers[i]->pOutputPortPrivate = NULL;
560 }
561 CHECK(err);
562 s->num_in_buffers = i;
563 for (i = 0; i < s->num_out_buffers && err == OMX_ErrorNone; i++)
564 err = OMX_AllocateBuffer(s->handle, &s->out_buffer_headers[i], s->out_port, s, out_port_params.nBufferSize);
565 CHECK(err);
566 s->num_out_buffers = i;
567
568 if (wait_for_state(s, OMX_StateIdle) < 0) {
569 av_log(avctx, AV_LOG_ERROR, "Didn't get OMX_StateIdle\n");
570 return AVERROR_UNKNOWN;
571 }
572 err = OMX_SendCommand(s->handle, OMX_CommandStateSet, OMX_StateExecuting, NULL);
573 CHECK(err);
574 if (wait_for_state(s, OMX_StateExecuting) < 0) {
575 av_log(avctx, AV_LOG_ERROR, "Didn't get OMX_StateExecuting\n");
576 return AVERROR_UNKNOWN;
577 }
578
579 for (i = 0; i < s->num_out_buffers && err == OMX_ErrorNone; i++)
580 err = OMX_FillThisBuffer(s->handle, s->out_buffer_headers[i]);
581 if (err != OMX_ErrorNone) {
582 for (; i < s->num_out_buffers; i++)
583 s->done_out_buffers[s->num_done_out_buffers++] = s->out_buffer_headers[i];
584 }
585 for (i = 0; i < s->num_in_buffers; i++)
586 s->free_in_buffers[s->num_free_in_buffers++] = s->in_buffer_headers[i];
587 return err != OMX_ErrorNone ? AVERROR_UNKNOWN : 0;
588}
589
590static av_cold void cleanup(OMXCodecContext *s)
591{
592 int i, executing;
593
594 pthread_mutex_lock(&s->state_mutex);
595 executing = s->state == OMX_StateExecuting;
596 pthread_mutex_unlock(&s->state_mutex);
597
598 if (executing) {
599 OMX_SendCommand(s->handle, OMX_CommandStateSet, OMX_StateIdle, NULL);
600 wait_for_state(s, OMX_StateIdle);
601 OMX_SendCommand(s->handle, OMX_CommandStateSet, OMX_StateLoaded, NULL);
602 for (i = 0; i < s->num_in_buffers; i++) {
603 OMX_BUFFERHEADERTYPE *buffer = get_buffer(&s->input_mutex, &s->input_cond,
604 &s->num_free_in_buffers, s->free_in_buffers, 1);
605 if (s->input_zerocopy)
606 buffer->pBuffer = NULL;
607 OMX_FreeBuffer(s->handle, s->in_port, buffer);
608 }
609 for (i = 0; i < s->num_out_buffers; i++) {
610 OMX_BUFFERHEADERTYPE *buffer = get_buffer(&s->output_mutex, &s->output_cond,
611 &s->num_done_out_buffers, s->done_out_buffers, 1);
612 OMX_FreeBuffer(s->handle, s->out_port, buffer);
613 }
614 wait_for_state(s, OMX_StateLoaded);
615 }
616 if (s->handle) {
617 s->omx_context->ptr_FreeHandle(s->handle);
618 s->handle = NULL;
619 }
620
621 omx_deinit(s->omx_context);
622 s->omx_context = NULL;
623 if (s->mutex_cond_inited) {
624 pthread_cond_destroy(&s->state_cond);
625 pthread_mutex_destroy(&s->state_mutex);
626 pthread_cond_destroy(&s->input_cond);
627 pthread_mutex_destroy(&s->input_mutex);
628 pthread_cond_destroy(&s->output_cond);
629 pthread_mutex_destroy(&s->output_mutex);
630 s->mutex_cond_inited = 0;
631 }
632 av_freep(&s->in_buffer_headers);
633 av_freep(&s->out_buffer_headers);
634 av_freep(&s->free_in_buffers);
635 av_freep(&s->done_out_buffers);
636 av_freep(&s->output_buf);
637}
638
639static av_cold int omx_encode_init(AVCodecContext *avctx)
640{
641 OMXCodecContext *s = avctx->priv_data;
642 int ret = AVERROR_ENCODER_NOT_FOUND;
643 const char *role;
644 OMX_BUFFERHEADERTYPE *buffer;
645 OMX_ERRORTYPE err;
646
647#if CONFIG_OMX_RPI
648 s->input_zerocopy = 1;
649#endif
650
651 s->omx_context = omx_init(avctx, s->libname, s->libprefix);
652 if (!s->omx_context)
653 return AVERROR_ENCODER_NOT_FOUND;
654
655 pthread_mutex_init(&s->state_mutex, NULL);
656 pthread_cond_init(&s->state_cond, NULL);
657 pthread_mutex_init(&s->input_mutex, NULL);
658 pthread_cond_init(&s->input_cond, NULL);
659 pthread_mutex_init(&s->output_mutex, NULL);
660 pthread_cond_init(&s->output_cond, NULL);
661 s->mutex_cond_inited = 1;
662 s->avctx = avctx;
663 s->state = OMX_StateLoaded;
664 s->error = OMX_ErrorNone;
665
666 switch (avctx->codec->id) {
667 case AV_CODEC_ID_MPEG4:
668 role = "video_encoder.mpeg4";
669 break;
670 case AV_CODEC_ID_H264:
671 role = "video_encoder.avc";
672 break;
673 default:
674 return AVERROR(ENOSYS);
675 }
676
677 if ((ret = find_component(s->omx_context, avctx, role, s->component_name, sizeof(s->component_name))) < 0)
678 goto fail;
679
680 av_log(avctx, AV_LOG_INFO, "Using %s\n", s->component_name);
681
682 if ((ret = omx_component_init(avctx, role)) < 0)
683 goto fail;
684
685 if (avctx->flags & AV_CODEC_FLAG_GLOBAL_HEADER) {
686 while (1) {
687 buffer = get_buffer(&s->output_mutex, &s->output_cond,
688 &s->num_done_out_buffers, s->done_out_buffers, 1);
689 if (buffer->nFlags & OMX_BUFFERFLAG_CODECCONFIG) {
690 if ((ret = av_reallocp(&avctx->extradata, avctx->extradata_size + buffer->nFilledLen + AV_INPUT_BUFFER_PADDING_SIZE)) < 0) {
691 avctx->extradata_size = 0;
692 goto fail;
693 }
694 memcpy(avctx->extradata + avctx->extradata_size, buffer->pBuffer + buffer->nOffset, buffer->nFilledLen);
695 avctx->extradata_size += buffer->nFilledLen;
696 memset(avctx->extradata + avctx->extradata_size, 0, AV_INPUT_BUFFER_PADDING_SIZE);
697 }
698 err = OMX_FillThisBuffer(s->handle, buffer);
699 if (err != OMX_ErrorNone) {
700 append_buffer(&s->output_mutex, &s->output_cond,
701 &s->num_done_out_buffers, s->done_out_buffers, buffer);
702 av_log(avctx, AV_LOG_ERROR, "OMX_FillThisBuffer failed: %x\n", err);
703 ret = AVERROR_UNKNOWN;
704 goto fail;
705 }
706 if (avctx->codec->id == AV_CODEC_ID_H264) {
707 // For H.264, the extradata can be returned in two separate buffers
708 // (the videocore encoder on raspberry pi does this);
709 // therefore check that we have got both SPS and PPS before continuing.
710 int nals[32] = { 0 };
711 int i;
712 for (i = 0; i + 4 < avctx->extradata_size; i++) {
713 if (!avctx->extradata[i + 0] &&
714 !avctx->extradata[i + 1] &&
715 !avctx->extradata[i + 2] &&
716 avctx->extradata[i + 3] == 1) {
717 nals[avctx->extradata[i + 4] & 0x1f]++;
718 }
719 }
720 if (nals[H264_NAL_SPS] && nals[H264_NAL_PPS])
721 break;
722 } else {
723 if (avctx->extradata_size > 0)
724 break;
725 }
726 }
727 }
728
729 return 0;
730fail:
731 return ret;
732}
733
734
735static int omx_encode_frame(AVCodecContext *avctx, AVPacket *pkt,
736 const AVFrame *frame, int *got_packet)
737{
738 OMXCodecContext *s = avctx->priv_data;
739 int ret = 0;
740 OMX_BUFFERHEADERTYPE* buffer;
741 OMX_ERRORTYPE err;
742
743 if (frame) {
744 uint8_t *dst[4];
745 int linesize[4];
746 int need_copy;
747 buffer = get_buffer(&s->input_mutex, &s->input_cond,
748 &s->num_free_in_buffers, s->free_in_buffers, 1);
749
750 buffer->nFilledLen = av_image_fill_arrays(dst, linesize, buffer->pBuffer, avctx->pix_fmt, s->stride, s->plane_size, 1);
751
752 if (s->input_zerocopy) {
753 uint8_t *src[4] = { NULL };
754 int src_linesize[4];
755 av_image_fill_arrays(src, src_linesize, frame->data[0], avctx->pix_fmt, s->stride, s->plane_size, 1);
756 if (frame->linesize[0] == src_linesize[0] &&
757 frame->linesize[1] == src_linesize[1] &&
758 frame->linesize[2] == src_linesize[2] &&
759 frame->data[1] == src[1] &&
760 frame->data[2] == src[2]) {
761 // If the input frame happens to have all planes stored contiguously,
762 // with the right strides, just clone the frame and set the OMX
763 // buffer header to point to it
764 AVFrame *local = av_frame_clone(frame);
765 if (!local) {
766 // Return the buffer to the queue so it's not lost
767 append_buffer(&s->input_mutex, &s->input_cond, &s->num_free_in_buffers, s->free_in_buffers, buffer);
768 return AVERROR(ENOMEM);
769 } else {
770 buffer->pAppPrivate = local;
771 buffer->pOutputPortPrivate = NULL;
772 buffer->pBuffer = local->data[0];
773 need_copy = 0;
774 }
775 } else {
776 // If not, we need to allocate a new buffer with the right
777 // size and copy the input frame into it.
778 uint8_t *buf = NULL;
779 int image_buffer_size = av_image_get_buffer_size(avctx->pix_fmt, s->stride, s->plane_size, 1);
780 if (image_buffer_size >= 0)
781 buf = av_malloc(image_buffer_size);
782 if (!buf) {
783 // Return the buffer to the queue so it's not lost
784 append_buffer(&s->input_mutex, &s->input_cond, &s->num_free_in_buffers, s->free_in_buffers, buffer);
785 return AVERROR(ENOMEM);
786 } else {
787 buffer->pAppPrivate = buf;
788 // Mark that pAppPrivate is an av_malloc'ed buffer, not an AVFrame
789 buffer->pOutputPortPrivate = (void*) 1;
790 buffer->pBuffer = buf;
791 need_copy = 1;
792 buffer->nFilledLen = av_image_fill_arrays(dst, linesize, buffer->pBuffer, avctx->pix_fmt, s->stride, s->plane_size, 1);
793 }
794 }
795 } else {
796 need_copy = 1;
797 }
798 if (need_copy)
799 av_image_copy(dst, linesize, (const uint8_t**) frame->data, frame->linesize, avctx->pix_fmt, avctx->width, avctx->height);
800 buffer->nFlags = OMX_BUFFERFLAG_ENDOFFRAME;
801 buffer->nOffset = 0;
802 // Convert the timestamps to microseconds; some encoders can ignore
803 // the framerate and do VFR bit allocation based on timestamps.
804 buffer->nTimeStamp = to_omx_ticks(av_rescale_q(frame->pts, avctx->time_base, AV_TIME_BASE_Q));
805 err = OMX_EmptyThisBuffer(s->handle, buffer);
806 if (err != OMX_ErrorNone) {
807 append_buffer(&s->input_mutex, &s->input_cond, &s->num_free_in_buffers, s->free_in_buffers, buffer);
808 av_log(avctx, AV_LOG_ERROR, "OMX_EmptyThisBuffer failed: %x\n", err);
809 return AVERROR_UNKNOWN;
810 }
811 s->num_in_frames++;
812 }
813
814 while (!*got_packet && ret == 0) {
815 // Only wait for output if flushing and not all frames have been output
816 buffer = get_buffer(&s->output_mutex, &s->output_cond,
817 &s->num_done_out_buffers, s->done_out_buffers,
818 !frame && s->num_out_frames < s->num_in_frames);
819 if (!buffer)
820 break;
821
822 if (buffer->nFlags & OMX_BUFFERFLAG_CODECCONFIG && avctx->flags & AV_CODEC_FLAG_GLOBAL_HEADER) {
823 if ((ret = av_reallocp(&avctx->extradata, avctx->extradata_size + buffer->nFilledLen + AV_INPUT_BUFFER_PADDING_SIZE)) < 0) {
824 avctx->extradata_size = 0;
825 goto end;
826 }
827 memcpy(avctx->extradata + avctx->extradata_size, buffer->pBuffer + buffer->nOffset, buffer->nFilledLen);
828 avctx->extradata_size += buffer->nFilledLen;
829 memset(avctx->extradata + avctx->extradata_size, 0, AV_INPUT_BUFFER_PADDING_SIZE);
830 } else {
831 if (buffer->nFlags & OMX_BUFFERFLAG_ENDOFFRAME)
832 s->num_out_frames++;
833 if (!(buffer->nFlags & OMX_BUFFERFLAG_ENDOFFRAME) || !pkt->data) {
834 // If the output packet isn't preallocated, just concatenate everything in our
835 // own buffer
836 int newsize = s->output_buf_size + buffer->nFilledLen + AV_INPUT_BUFFER_PADDING_SIZE;
837 if ((ret = av_reallocp(&s->output_buf, newsize)) < 0) {
838 s->output_buf_size = 0;
839 goto end;
840 }
841 memcpy(s->output_buf + s->output_buf_size, buffer->pBuffer + buffer->nOffset, buffer->nFilledLen);
842 s->output_buf_size += buffer->nFilledLen;
843 if (buffer->nFlags & OMX_BUFFERFLAG_ENDOFFRAME) {
844 if ((ret = av_packet_from_data(pkt, s->output_buf, s->output_buf_size)) < 0) {
845 av_freep(&s->output_buf);
846 s->output_buf_size = 0;
847 goto end;
848 }
849 s->output_buf = NULL;
850 s->output_buf_size = 0;
851 }
852 } else {
853 // End of frame, and the caller provided a preallocated frame
854 if ((ret = ff_alloc_packet2(avctx, pkt, s->output_buf_size + buffer->nFilledLen, 0)) < 0) {
855 av_log(avctx, AV_LOG_ERROR, "Error getting output packet of size %d.\n",
856 (int)(s->output_buf_size + buffer->nFilledLen));
857 goto end;
858 }
859 memcpy(pkt->data, s->output_buf, s->output_buf_size);
860 memcpy(pkt->data + s->output_buf_size, buffer->pBuffer + buffer->nOffset, buffer->nFilledLen);
861 av_freep(&s->output_buf);
862 s->output_buf_size = 0;
863 }
864 if (buffer->nFlags & OMX_BUFFERFLAG_ENDOFFRAME) {
865 pkt->pts = av_rescale_q(from_omx_ticks(buffer->nTimeStamp), AV_TIME_BASE_Q, avctx->time_base);
866 // We don't currently enable B-frames for the encoders, so set
867 // pkt->dts = pkt->pts. (The calling code behaves worse if the encoder
868 // doesn't set the dts).
869 pkt->dts = pkt->pts;
870 if (buffer->nFlags & OMX_BUFFERFLAG_SYNCFRAME)
871 pkt->flags |= AV_PKT_FLAG_KEY;
872 *got_packet = 1;
873 }
874 }
875end:
876 err = OMX_FillThisBuffer(s->handle, buffer);
877 if (err != OMX_ErrorNone) {
878 append_buffer(&s->output_mutex, &s->output_cond, &s->num_done_out_buffers, s->done_out_buffers, buffer);
879 av_log(avctx, AV_LOG_ERROR, "OMX_FillThisBuffer failed: %x\n", err);
880 ret = AVERROR_UNKNOWN;
881 }
882 }
883 return ret;
884}
885
886static av_cold int omx_encode_end(AVCodecContext *avctx)
887{
888 OMXCodecContext *s = avctx->priv_data;
889
890 cleanup(s);
891 return 0;
892}
893
894#define OFFSET(x) offsetof(OMXCodecContext, x)
895#define VDE AV_OPT_FLAG_VIDEO_PARAM | AV_OPT_FLAG_DECODING_PARAM | AV_OPT_FLAG_ENCODING_PARAM
896#define VE AV_OPT_FLAG_VIDEO_PARAM | AV_OPT_FLAG_ENCODING_PARAM
897static const AVOption options[] = {
898 { "omx_libname", "OpenMAX library name", OFFSET(libname), AV_OPT_TYPE_STRING, { 0 }, 0, 0, VDE },
899 { "omx_libprefix", "OpenMAX library prefix", OFFSET(libprefix), AV_OPT_TYPE_STRING, { 0 }, 0, 0, VDE },
900 { "zerocopy", "Try to avoid copying input frames if possible", OFFSET(input_zerocopy), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, 1, VE },
901 { "profile", "Set the encoding profile", OFFSET(profile), AV_OPT_TYPE_INT, { .i64 = FF_PROFILE_UNKNOWN }, FF_PROFILE_UNKNOWN, FF_PROFILE_H264_HIGH, VE, "profile" },
902 { "baseline", "", 0, AV_OPT_TYPE_CONST, { .i64 = FF_PROFILE_H264_BASELINE }, 0, 0, VE, "profile" },
903 { "main", "", 0, AV_OPT_TYPE_CONST, { .i64 = FF_PROFILE_H264_MAIN }, 0, 0, VE, "profile" },
904 { "high", "", 0, AV_OPT_TYPE_CONST, { .i64 = FF_PROFILE_H264_HIGH }, 0, 0, VE, "profile" },
905 { NULL }
906};
907
908static const enum AVPixelFormat omx_encoder_pix_fmts[] = {
909 AV_PIX_FMT_YUV420P, AV_PIX_FMT_NONE
910};
911
912static const AVClass omx_mpeg4enc_class = {
913 .class_name = "mpeg4_omx",
914 .item_name = av_default_item_name,
915 .option = options,
916 .version = LIBAVUTIL_VERSION_INT,
917};
918AVCodec ff_mpeg4_omx_encoder = {
919 .name = "mpeg4_omx",
920 .long_name = NULL_IF_CONFIG_SMALL("OpenMAX IL MPEG-4 video encoder"),
921 .type = AVMEDIA_TYPE_VIDEO,
922 .id = AV_CODEC_ID_MPEG4,
923 .priv_data_size = sizeof(OMXCodecContext),
924 .init = omx_encode_init,
925 .encode2 = omx_encode_frame,
926 .close = omx_encode_end,
927 .pix_fmts = omx_encoder_pix_fmts,
928 .capabilities = AV_CODEC_CAP_DELAY,
929 .caps_internal = FF_CODEC_CAP_INIT_THREADSAFE | FF_CODEC_CAP_INIT_CLEANUP,
930 .priv_class = &omx_mpeg4enc_class,
931};
932
933static const AVClass omx_h264enc_class = {
934 .class_name = "h264_omx",
935 .item_name = av_default_item_name,
936 .option = options,
937 .version = LIBAVUTIL_VERSION_INT,
938};
939AVCodec ff_h264_omx_encoder = {
940 .name = "h264_omx",
941 .long_name = NULL_IF_CONFIG_SMALL("OpenMAX IL H.264 video encoder"),
942 .type = AVMEDIA_TYPE_VIDEO,
943 .id = AV_CODEC_ID_H264,
944 .priv_data_size = sizeof(OMXCodecContext),
945 .init = omx_encode_init,
946 .encode2 = omx_encode_frame,
947 .close = omx_encode_end,
948 .pix_fmts = omx_encoder_pix_fmts,
949 .capabilities = AV_CODEC_CAP_DELAY,
950 .caps_internal = FF_CODEC_CAP_INIT_THREADSAFE | FF_CODEC_CAP_INIT_CLEANUP,
951 .priv_class = &omx_h264enc_class,
952};
953