/***************************************************************************** * CloseDecoder: Close the decoder instance *****************************************************************************/ static void CloseDecoder(vlc_object_t *p_this) { decoder_t *p_dec = (decoder_t *)p_this; decoder_sys_t *p_sys = p_dec->p_sys; if (!p_sys) return; StopMediaCodec(p_dec); CSDFree(p_dec); p_sys->api->clean(p_sys->api); if (p_dec->fmt_in.i_cat == VIDEO_ES) { ArchitectureSpecificCopyHooksDestroy(p_sys->u.video.i_pixel_format, &p_sys->u.video.ascd); free(p_sys->u.video.pp_inflight_pictures); if (p_sys->u.video.timestamp_fifo) timestamp_FifoRelease(p_sys->u.video.timestamp_fifo); if (p_sys->u.video.p_awh) AWindowHandler_destroy(p_sys->u.video.p_awh); } free(p_sys->api); free(p_sys->psz_name); free(p_sys); }
static void Close(vlc_object_t *p_this) { vout_display_t *vd = (vout_display_t *)p_this; vout_display_sys_t *sys = vd->sys; if (!sys) return; /* Check if SPU regions have been properly cleared, and clear them if they * were not. */ if (sys->b_has_subpictures) { SubpicturePrepare(vd, NULL); if (sys->b_sub_pic_locked) AndroidWindow_UnlockPicture(sys, sys->p_sub_window, sys->p_sub_pic, true); } if (sys->pool) picture_pool_Release(sys->pool); if (sys->p_window) AndroidWindow_Destroy(vd, sys->p_window); if (sys->p_sub_pic) picture_Release(sys->p_sub_pic); if (sys->p_spu_blend) filter_DeleteBlend(sys->p_spu_blend); free(sys->p_sub_buffer_bounds); if (sys->p_sub_window) AndroidWindow_Destroy(vd, sys->p_sub_window); if (sys->p_awh) AWindowHandler_destroy(sys->p_awh); free(sys); }
static void Close(vlc_object_t *p_this) { vout_display_t *vd = (vout_display_t *)p_this; vout_display_sys_t *sys = vd->sys; if (!sys) return; if (sys->pool) picture_pool_Release(sys->pool); if (sys->p_window) AndroidWindow_Destroy(vd, sys->p_window); if (sys->p_sub_pic) picture_Release(sys->p_sub_pic); if (sys->p_spu_blend) filter_DeleteBlend(sys->p_spu_blend); free(sys->p_sub_buffer_bounds); if (sys->p_sub_window) AndroidWindow_Destroy(vd, sys->p_sub_window); if (sys->p_awh) AWindowHandler_destroy(sys->p_awh); free(sys); }
/** * Create an Android native window. */ static int Open(vout_window_t *wnd, const vout_window_cfg_t *cfg) { ANativeWindow *p_anw; if (cfg->type != VOUT_WINDOW_TYPE_INVALID && cfg->type != VOUT_WINDOW_TYPE_ANDROID_NATIVE) return VLC_EGENERIC; vout_window_sys_t *p_sys = malloc(sizeof (*p_sys)); if (p_sys == NULL) return VLC_ENOMEM; p_sys->p_awh = AWindowHandler_new(VLC_OBJECT(wnd)); if (!p_sys->p_awh) goto error; p_anw = AWindowHandler_getANativeWindow(p_sys->p_awh, AWindow_Video); if (!p_anw) goto error; wnd->type = VOUT_WINDOW_TYPE_ANDROID_NATIVE; wnd->handle.anativewindow = p_anw; wnd->control = Control; wnd->sys = p_sys; // Set the Java surface size. AWindowHandler_setWindowLayout(p_sys->p_awh, cfg->width, cfg->height, cfg->width, cfg->height, 1, 1); return VLC_SUCCESS; error: if (p_sys->p_awh) AWindowHandler_destroy(p_sys->p_awh); free(p_sys); return VLC_EGENERIC; }
/** * Destroys the Android native window. */ static void Close(vout_window_t *wnd) { vout_window_sys_t *p_sys = wnd->sys; AWindowHandler_destroy(p_sys->p_awh); free (p_sys); }
/***************************************************************************** * StartMediaCodec: Create the mediacodec instance *****************************************************************************/ static int StartMediaCodec(decoder_t *p_dec) { decoder_sys_t *p_sys = p_dec->p_sys; int i_ret = 0; union mc_api_args args; if (p_dec->fmt_in.i_extra && !p_sys->pp_csd) { /* Try first to configure specific Video CSD */ if (p_dec->fmt_in.i_cat == VIDEO_ES) i_ret = ParseVideoExtra(p_dec, p_dec->fmt_in.p_extra, p_dec->fmt_in.i_extra); if (i_ret != VLC_SUCCESS) return i_ret; /* Set default CSD if ParseVideoExtra failed to configure one */ if (!p_sys->pp_csd) { struct csd csd; csd.p_buf = p_dec->fmt_in.p_extra; csd.i_size = p_dec->fmt_in.i_extra; CSDDup(p_dec, &csd, 1); } } if (p_dec->fmt_in.i_cat == VIDEO_ES) { if (!p_sys->u.video.i_width || !p_sys->u.video.i_height) { msg_Err(p_dec, "invalid size, abort MediaCodec"); return VLC_EGENERIC; } args.video.i_width = p_sys->u.video.i_width; args.video.i_height = p_sys->u.video.i_height; switch (p_dec->fmt_in.video.orientation) { case ORIENT_ROTATED_90: args.video.i_angle = 90; break; case ORIENT_ROTATED_180: args.video.i_angle = 180; break; case ORIENT_ROTATED_270: args.video.i_angle = 270; break; default: args.video.i_angle = 0; } /* Check again the codec name if h264 profile changed */ if (p_dec->fmt_in.i_codec == VLC_CODEC_H264 && !p_sys->u.video.i_h264_profile) { h264_get_profile_level(&p_dec->fmt_in, &p_sys->u.video.i_h264_profile, NULL, NULL); if (p_sys->u.video.i_h264_profile) { free(p_sys->psz_name); p_sys->psz_name = MediaCodec_GetName(VLC_OBJECT(p_dec), p_sys->mime, p_sys->u.video.i_h264_profile); if (!p_sys->psz_name) return VLC_EGENERIC; } } if (!p_sys->u.video.p_awh && var_InheritBool(p_dec, CFG_PREFIX "dr")) { if ((p_sys->u.video.p_awh = AWindowHandler_new(VLC_OBJECT(p_dec)))) { /* Direct rendering: * The surface must be released by the Vout before calling * start. Request a valid OPAQUE Vout to release any non-OPAQUE * Vout that will release the surface. */ p_dec->fmt_out.video.i_width = p_sys->u.video.i_width; p_dec->fmt_out.video.i_height = p_sys->u.video.i_height; p_dec->fmt_out.i_codec = VLC_CODEC_ANDROID_OPAQUE; if (decoder_UpdateVideoFormat(p_dec) != 0) { msg_Err(p_dec, "Opaque Vout request failed: " "fallback to non opaque"); AWindowHandler_destroy(p_sys->u.video.p_awh); p_sys->u.video.p_awh = NULL; } } } args.video.p_awh = p_sys->u.video.p_awh; } else { date_Set(&p_sys->u.audio.i_end_date, VLC_TS_INVALID); args.audio.i_sample_rate = p_dec->fmt_in.audio.i_rate; args.audio.i_channel_count = p_dec->p_sys->u.audio.i_channels; } return p_sys->api->start(p_sys->api, p_sys->psz_name, p_sys->mime, &args); }