3atv精品不卡视频,97人人超碰国产精品最新,中文字幕av一区二区三区人妻少妇,久久久精品波多野结衣,日韩一区二区三区精品

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android 使用FFmpeg3.3.9基于命令实现视频压缩

發布時間:2023/12/3 Android 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android 使用FFmpeg3.3.9基于命令实现视频压缩 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

前言

首先利用linux平臺編譯ffmpeg的so庫,具體詳情請查看文章:Android NDK(ndk-r16b)交叉編譯FFmpeg(3.3.9)_jszlittlecat_720的博客-CSDN博客

1.創建VideoCompress類

?

package com.suoer.ndk.ffmpegtestapplication;public class VideoCompress {/*** compressVideo native 方法* @param compressCommand 壓縮命令* @param callback 壓縮回調*/public native void compressVideo(String[] compressCommand,CompressCallback callback);//CompressCallback 壓縮回調public interface CompressCallback{/*** onCompress* @param current 壓縮的當前進度* @param total 總進度*/public void onCompress(int current,int total);} } compressVideo報紅,鼠標停在上面,左邊會出現紅色小燈泡,點擊紅色小燈泡

?點擊Create JNI function for compressVideo

自動打開native-lib.cpp并創建完成Java_com_suoer_ndk_ffmpegtestapplication_VideoCompress_compressVideo 方法

?在此方法下實現壓縮視頻

2.MainActivity實現點擊TextView實現壓縮視頻

2.1權限問題處理

壓縮視頻需要讀寫文件的權限

打開AndroidManifest.xml

添加讀寫文件權限

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission><uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"></uses-permission>

?權限處理使用rxpermissions

rxpermissions github地址:GitHub - tbruyelle/RxPermissions: Android runtime permissions powered by RxJava2

rxpermissions使用方式

打開項目下的build.gradle添加如下代碼:

maven { url 'https://jitpack.io' }

?打開app下的build.gradle添加如下代碼:

implementation 'com.github.tbruyelle:rxpermissions:0.12'

?點擊同步按鈕,同步文件

?2.2子線程處理耗時操作

壓縮視頻是耗時操作,子線程處理耗時操作。

使用rxandroid實現子線程耗時操作并實現子線程和主線程切換

rxandroid github地址:GitHub - ReactiveX/RxAndroid: RxJava bindings for Android

?rxandroid使用方式:

打開項目下的build.gradle添加如下代碼:

maven { url "https://oss.jfrog.org/libs-snapshot" }

?打開app下的build.gradle添加如下代碼:

implementation 'io.reactivex.rxjava3:rxandroid:3.0.0'

?點擊同步按鈕同步文件

MainActivity.java內容如下:

package com.suoer.ndk.ffmpegtestapplication;import android.Manifest; import android.os.Bundle; import android.os.Environment; import android.util.Log; import android.view.View; import android.widget.TextView;import com.tbruyelle.rxpermissions3.RxPermissions;import java.io.File;import androidx.appcompat.app.AppCompatActivity; import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers; import io.reactivex.rxjava3.core.Observable; import io.reactivex.rxjava3.functions.Consumer; import io.reactivex.rxjava3.functions.Function; import io.reactivex.rxjava3.schedulers.Schedulers;public class MainActivity extends AppCompatActivity {private File mInFile=new File(Environment.getExternalStorageDirectory(),"test.mp4");//mInFile 需要壓縮的文件路徑private File mOutFile=new File(Environment.getExternalStorageDirectory(),"out.mp4");//mOutFile壓縮后的文件路徑// Used to load the 'native-lib' library on application startup.static {System.loadLibrary("native-lib");}@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);// Example of a call to a native methodTextView tv = findViewById(R.id.sample_text);//tv.setText("ffmpeg版本:"+stringFromJNI());tv.setText("壓縮");//tv的點擊事件 點擊按鈕實現視頻壓縮tv.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {// 壓縮文件 需要讀寫文件權限 申請權限RxPermissions rxPermissions=new RxPermissions(MainActivity.this);rxPermissions.request(Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE).subscribe(new Consumer<Boolean>() {@Overridepublic void accept(Boolean aBoolean) throws Throwable {if(aBoolean){//權限已經獲取 壓縮視頻compressVideo();}}});}});}/*** 開啟子線程處理耗時壓縮問題*/private void compressVideo() {//ffmpeg的壓縮命令:ffmpeg -i test.mp4 -b:v 1024k out.mp4//ffmpeg -i test.mp4 -b:v 1024k out.mp4//-b:v 1024k 1024k為碼率 碼率越高視頻越清晰,而且視頻越大//test.mp4需要壓縮的文件//out.mp4 壓縮之后的文件String[] compressCommand={"ffmpeg","-i",mInFile.getAbsolutePath(),"-b:v","1024k",mOutFile.getAbsolutePath()};//壓縮是耗時的,需要子線程處理Observable.just(compressCommand).map(new Function<String[], File>() {@Overridepublic File apply(String[] compressCommand) throws Throwable {VideoCompress videoCompress=new VideoCompress();videoCompress.compressVideo(compressCommand, new VideoCompress.CompressCallback() {@Overridepublic void onCompress(int current, int total) {Log.e("TAG", "onCompress: 壓縮進度:"+current+"/"+total);}});return mOutFile;}}).subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread()).subscribe(new Consumer<File>() {@Overridepublic void accept(File file) throws Throwable {//壓縮完成Log.e("TAG", "accept: 壓縮完成!" );}});}/*** A native method that is implemented by the 'native-lib' native library,* which is packaged with this application.*/public native String stringFromJNI(); }

3.拷貝視頻壓縮使用命令實現需要的其他文件

include目錄下創建compat文件夾把所需要的頭文件?os2threads.h?va_copy.h?w32pthreads.h拷貝此文件夾中

這三個頭文件存在于下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9的compat文件夾中

?在jniLibs文件夾中創建other文件夾

?拷貝linux平臺編譯后的文件cmdutils.h cmdutils_common_opts.h config.h ffmpeg.h 拷貝至other文件夾中

?

?將下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9中的cmdutils.c?

ffmpeg.c?ffmpeg_filter.c?ffmpeg_opt.c 四個文件拷貝至cpp文件夾中

?

?配置CMakeLists.txt文件修改如下內容

?CMakeLists.txt內容如下:

# For more information about using CMake with Android Studio, read the # documentation: https://d.android.com/studio/projects/add-native-code.html# Sets the minimum version of CMake required to build the native library.cmake_minimum_required(VERSION 3.10.2)# Declares and names the project.project("ffmpegtestapplication")#判斷編譯器類型,如果是gcc編譯器,則在編譯選項中加入c++11支持 if(CMAKE_COMPILER_IS_GNUCXX)set(CMAKE_CXX_FLAGS "-std=c++11 ${CMAKE_CXX_FLAGS}")message(STATUS "optional:-std=c++11") endif(CMAKE_COMPILER_IS_GNUCXX)#需要引入我們頭文件,以這個配置的目錄為基準include_directories(${CMAKE_SOURCE_DIR}/../jniLibs/include) include_directories(${CMAKE_SOURCE_DIR}/../jniLibs/other)# Creates and names a library, sets it as either STATIC # or SHARED, and provides the relative paths to its source code. # You can define multiple libraries, and CMake builds them for you. # Gradle automatically packages shared libraries with your APK.add_library( # Sets the name of the library.native-lib# Sets the library as a shared library.SHARED# Provides a relative path to your source file(s).native-lib.cpp#添加額外的c文件cmdutils.cffmpeg.cffmpeg_filter.cffmpeg_opt.c) # 編解碼(最重要的庫) add_library(avcodecSHAREDIMPORTED) set_target_properties(avcodecPROPERTIES IMPORTED_LOCATION${CMAKE_SOURCE_DIR}/../jniLibs/armeabi-v7a/libavcodec-57.so)# 設備信息 add_library(avdeviceSHAREDIMPORTED) set_target_properties(avdevicePROPERTIES IMPORTED_LOCATION${CMAKE_SOURCE_DIR}/../jniLibs/armeabi-v7a/libavdevice-57.so)# 濾鏡特效處理庫 add_library(avfilterSHAREDIMPORTED) set_target_properties(avfilterPROPERTIES IMPORTED_LOCATION${CMAKE_SOURCE_DIR}/../jniLibs/armeabi-v7a/libavfilter-6.so)# 封裝格式處理庫 add_library(avformatSHAREDIMPORTED) set_target_properties(avformatPROPERTIES IMPORTED_LOCATION${CMAKE_SOURCE_DIR}/../jniLibs/armeabi-v7a/libavformat-57.so) # 重采樣處理庫 add_library(avresampleSHAREDIMPORTED) set_target_properties(avresamplePROPERTIES IMPORTED_LOCATION${CMAKE_SOURCE_DIR}/../jniLibs/armeabi-v7a/libavresample-3.so) # 工具庫(大部分庫都需要這個庫的支持) add_library(avutilSHAREDIMPORTED) set_target_properties(avutilPROPERTIES IMPORTED_LOCATION${CMAKE_SOURCE_DIR}/../jniLibs/armeabi-v7a/libavutil-55.so)# 后期處理 add_library(postprocSHAREDIMPORTED) set_target_properties(postprocPROPERTIES IMPORTED_LOCATION${CMAKE_SOURCE_DIR}/../jniLibs/armeabi-v7a/libpostproc-54.so)# 音頻采樣數據格式轉換庫 add_library(swresampleSHAREDIMPORTED) set_target_properties(swresamplePROPERTIES IMPORTED_LOCATION${CMAKE_SOURCE_DIR}/../jniLibs/armeabi-v7a/libswresample-2.so)# 視頻像素數據格式轉換 add_library(swscaleSHAREDIMPORTED) set_target_properties(swscalePROPERTIES IMPORTED_LOCATION${CMAKE_SOURCE_DIR}/../jniLibs/armeabi-v7a/libswscale-4.so)# Searches for a specified prebuilt library and stores the path as a # variable. Because CMake includes system libraries in the search path by # default, you only need to specify the name of the public NDK library # you want to add. CMake verifies that the library exists before # completing its build.find_library( # Sets the name of the path variable.log-lib# Specifies the name of the NDK library that# you want CMake to locate.log )# Specifies libraries CMake should link to your target library. You # can link multiple libraries, such as libraries you define in this # build script, prebuilt third-party libraries, or system libraries.target_link_libraries( # Specifies the target library.native-lib avcodec avdevice avfilter avformat avresample avutil postproc swresample swscale# Links the target library to the log library# included in the NDK.${log-lib} )

運行Run app至手機設備,出現如下如所示錯誤

?D:\suowei\androidproject\FFmpegTestApplication\app\src\main\cpp\cmdutils.c:47:10: fatal error: 'libavutil/libm.h' file not found

解決方式:將下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9中libavutil中的libm.h拷貝至android項目jniLibs下的libavutil中

?

?D:\suowei\androidproject\FFmpegTestApplication\app\src\main\cpp\ffmpeg.c:52:10: fatal error: 'libavutil/internal.h' file not found

?解決方式:將下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9中libavutil中的internal.h拷貝至android項目jniLibs下的libavutil中

?

?重新Run app至手機設備,出現如下圖所示錯誤

?D:\suowei\androidproject\FFmpegTestApplication\app\src\main\cpp\cmdutils.c:58:10: fatal error: 'libavformat/network.h' file not found

解決方式:將下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9中libavformat中的network.h拷貝至android項目jniLibs下的libavformat中

?

?D:\suowei\androidproject\FFmpegTestApplication\app\src\main\jniLibs\include\libavutil\internal.h:42:10: fatal error: 'timer.h' file not found

?解決方式:將下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9中libavutil中的timer.h拷貝至android項目jniLibs下的libavutil中

?

?

?重新Run app至手機設備,出現如下圖所示錯誤

?D:/suowei/androidproject/FFmpegTestApplication/app/src/main/cpp/../jniLibs/include\libavformat/network.h:29:10: fatal error: 'os_support.h' file not found
#include "os_support.h"

??解決方式:將下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9中libavformat中的os_support.h拷貝至android項目jniLibs下的libavformat中

?

?D:/suowei/androidproject/FFmpegTestApplication/app/src/main/cpp/../jniLibs/include\libavutil/timer.h:44:13: fatal error: 'arm/timer.h' file not found

解決方式:android項目jniLibs目錄下的libavutil文件夾下創建arm文件夾

?將下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9中libavutil中的arm文件夾下的timer.h拷貝至android項目jniLibs下的libavutil中的arm文件夾下

?

?重新Run app至手機設備,出現如下圖所示錯誤

?D:/suowei/androidproject/FFmpegTestApplication/app/src/main/cpp/ffmpeg.c:65:10: fatal error: 'libavcodec/mathops.h' file not found

解決方式:將下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9中libavcodec中的mathops.h拷貝至android項目jniLibs下的libavcodec中

?

?重新Run app至手機設備,出現如下圖所示錯誤

?D:/suowei/androidproject/FFmpegTestApplication/app/src/main/cpp/../jniLibs/include\libavformat/network.h:31:10: fatal error: 'url.h' file not found

解決方式:將下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9中libavformat中的url.h拷貝至android項目jniLibs下的libavformat中

?D:/suowei/androidproject/FFmpegTestApplication/app/src/main/cpp/../jniLibs/include\libavcodec/mathops.h:28:10: fatal error: 'libavutil/reverse.h' file not found

解決方式:將下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9中libavutil中的reverse.h拷貝至android項目jniLibs下的libavutil中

?

?

?重新Run app至手機設備,出現如下圖所示錯誤

?D:/suowei/androidproject/FFmpegTestApplication/app/src/main/cpp/../jniLibs/include\libavcodec/mathops.h:40:13: fatal error: 'arm/mathops.h' file not found

解決方式:android項目jniLibs目錄下的libavcodec文件夾下創建arm文件夾

?

?將下載之后的ffmpeg-3.3.9.tar.gz解壓之后的文件夾ffmpeg-3.3.9中libavcodec中的arm文件夾下的mathops.h拷貝至android項目jniLibs下的libavcodec中的arm文件夾下

?

?重新Run app至手機設備,android項目運行成功。

4.編寫native-cpp實現視頻壓縮

?

修改ffmpeg.c,找到main方法修改方法名稱為:run_ffmpeg_command 并注釋掉此行代碼

exit_program(received_nb_signals ? 255 : main_return_code); //命令函數的入口 int run_ffmpeg_command(int argc, char **argv) {int i, ret;int64_t ti;init_dynload();register_exit(ffmpeg_cleanup);setvbuf(stderr,NULL,_IONBF,0); /* win32 runtime needs this */av_log_set_flags(AV_LOG_SKIP_REPEATED);parse_loglevel(argc, argv, options);if(argc>1 && !strcmp(argv[1], "-d")){run_as_daemon=1;av_log_set_callback(log_callback_null);argc--;argv++;}avcodec_register_all(); #if CONFIG_AVDEVICEavdevice_register_all(); #endifavfilter_register_all();av_register_all();avformat_network_init();show_banner(argc, argv, options);/* parse options and open all input/output files */ret = ffmpeg_parse_options(argc, argv);if (ret < 0)exit_program(1);if (nb_output_files <= 0 && nb_input_files == 0) {show_usage();av_log(NULL, AV_LOG_WARNING, "Use -h to get full help or, even better, run 'man %s'\n", program_name);exit_program(1);}/* file converter / grab */if (nb_output_files <= 0) {av_log(NULL, AV_LOG_FATAL, "At least one output file must be specified\n");exit_program(1);}// if (nb_input_files == 0) { // av_log(NULL, AV_LOG_FATAL, "At least one input file must be specified\n"); // exit_program(1); // }for (i = 0; i < nb_output_files; i++) {if (strcmp(output_files[i]->ctx->oformat->name, "rtp"))want_sdp = 0;}current_time = ti = getutime();if (transcode() < 0)exit_program(1);ti = getutime() - ti;if (do_benchmark) {av_log(NULL, AV_LOG_INFO, "bench: utime=%0.3fs\n", ti / 1000000.0);}av_log(NULL, AV_LOG_DEBUG, "%"PRIu64" frames successfully decoded, %"PRIu64" decoding errors\n",decode_error_stat[0], decode_error_stat[1]);if ((decode_error_stat[0] + decode_error_stat[1]) * max_error_rate < decode_error_stat[1])exit_program(69);//注釋掉此行代碼 如果不注釋掉該行代碼,命令執行完會導致app退出//exit_program(received_nb_signals ? 255 : main_return_code);return main_return_code; }

native-lib.cpp代碼如下:

#include <jni.h> #include <string> #include <android/log.h> #define TAG "JNI_TAG" #define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,TAG,__VA_ARGS__); extern "C" { #include "libavutil/avutil.h"//聲明方法 argc命令的個數 argv 二維數組 int run_ffmpeg_command(int argc, char **argv); } extern "C" JNIEXPORT jstring JNICALL Java_com_suoer_ndk_ffmpegtestapplication_MainActivity_stringFromJNI(JNIEnv* env,jobject /* this */) {//std::string hello = "Hello from C++";return env->NewStringUTF(av_version_info()); }extern "C" JNIEXPORT void JNICALL Java_com_suoer_ndk_ffmpegtestapplication_VideoCompress_compressVideo(JNIEnv *env, jobject thiz,jobjectArray compress_command,jobject callback) {//ffmpeg 處理視頻壓縮//arm這個里面的so都是用來處理音視頻的,include都是頭文件//還有幾個沒有被打包編譯成so,因為這些不算是音視頻的處理代碼,只是我們現在支持命令(封裝)//1.獲取命令個數int argc=env->GetArrayLength(compress_command);//2.給char **argv填充數據char **argv=(char **)malloc(sizeof(char*)*argc);for (int i = 0; i <argc ; ++i) {jstring j_param=(jstring)env->GetObjectArrayElement(compress_command,i);argv[i]= (char *)(env->GetStringUTFChars(j_param, NULL));LOGE("參數:%s",argv[i]);}//3.調用命令函數去壓縮run_ffmpeg_command(argc,argv);//4.釋放內存for (int i = 0; i <argc ; ++i) {free(argv[i]);}free(argv); }

視頻壓縮回調處理,視頻壓縮進度回調

修改ffmpeg.c內容如下:

print_report方法

?

?transcode方法

?

?

?run_ffmpeg_command方法

?

?

?ffmpeg.c內容如下:

/** Copyright (c) 2000-2003 Fabrice Bellard** This file is part of FFmpeg.** FFmpeg is free software; you can redistribute it and/or* modify it under the terms of the GNU Lesser General Public* License as published by the Free Software Foundation; either* version 2.1 of the License, or (at your option) any later version.** FFmpeg is distributed in the hope that it will be useful,* but WITHOUT ANY WARRANTY; without even the implied warranty of* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU* Lesser General Public License for more details.** You should have received a copy of the GNU Lesser General Public* License along with FFmpeg; if not, write to the Free Software* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA*//*** @file* multimedia converter based on the FFmpeg libraries*/#include "config.h" #include <ctype.h> #include <string.h> #include <math.h> #include <stdlib.h> #include <errno.h> #include <limits.h> #include <stdatomic.h> #include <stdint.h>#if HAVE_IO_H #include <io.h> #endif #if HAVE_UNISTD_H #include <unistd.h> #endif#include "libavformat/avformat.h" #include "libavdevice/avdevice.h" #include "libswresample/swresample.h" #include "libavutil/opt.h" #include "libavutil/channel_layout.h" #include "libavutil/parseutils.h" #include "libavutil/samplefmt.h" #include "libavutil/fifo.h" #include "libavutil/hwcontext.h" #include "libavutil/internal.h" #include "libavutil/intreadwrite.h" #include "libavutil/dict.h" #include "libavutil/display.h" #include "libavutil/mathematics.h" #include "libavutil/pixdesc.h" #include "libavutil/avstring.h" #include "libavutil/libm.h" #include "libavutil/imgutils.h" #include "libavutil/timestamp.h" #include "libavutil/bprint.h" #include "libavutil/time.h" #include "libavutil/threadmessage.h" #include "libavcodec/mathops.h" #include "libavformat/os_support.h"# include "libavfilter/avfilter.h" # include "libavfilter/buffersrc.h" # include "libavfilter/buffersink.h"#if HAVE_SYS_RESOURCE_H #include <sys/time.h> #include <sys/types.h> #include <sys/resource.h> #elif HAVE_GETPROCESSTIMES #include <windows.h> #endif #if HAVE_GETPROCESSMEMORYINFO #include <windows.h> #include <psapi.h> #endif #if HAVE_SETCONSOLECTRLHANDLER #include <windows.h> #endif#if HAVE_SYS_SELECT_H #include <sys/select.h> #endif#if HAVE_TERMIOS_H #include <fcntl.h> #include <sys/ioctl.h> #include <sys/time.h> #include <termios.h> #elif HAVE_KBHIT #include <conio.h> #endif#if HAVE_PTHREADS #include <pthread.h> #endif#include <time.h> #include <android/log.h>#include "ffmpeg.h" #include "cmdutils.h"#include "libavutil/avassert.h"const char program_name[] = "ffmpeg"; const int program_birth_year = 2000;static FILE *vstats_file;const char *const forced_keyframes_const_names[] = {"n","n_forced","prev_forced_n","prev_forced_t","t",NULL };static void do_video_stats(OutputStream *ost, int frame_size); static int64_t getutime(void); static int64_t getmaxrss(void); static int ifilter_has_all_input_formats(FilterGraph *fg);static int run_as_daemon = 0; static int nb_frames_dup = 0; static unsigned dup_warning = 1000; static int nb_frames_drop = 0; static int64_t decode_error_stat[2];static int want_sdp = 1;static int current_time; AVIOContext *progress_avio = NULL;static uint8_t *subtitle_out;InputStream **input_streams = NULL; int nb_input_streams = 0; InputFile **input_files = NULL; int nb_input_files = 0;OutputStream **output_streams = NULL; int nb_output_streams = 0; OutputFile **output_files = NULL; int nb_output_files = 0;FilterGraph **filtergraphs; int nb_filtergraphs;#if HAVE_TERMIOS_H/* init terminal so that we can grab keys */ static struct termios oldtty; static int restore_tty; #endif#if HAVE_PTHREADS static void free_input_threads(void); #endif/* sub2video hack:Convert subtitles to video with alpha to insert them in filter graphs.This is a temporary solution until libavfilter gets real subtitles support.*/static int sub2video_get_blank_frame(InputStream *ist) {int ret;AVFrame *frame = ist->sub2video.frame;av_frame_unref(frame);ist->sub2video.frame->width = ist->dec_ctx->width ? ist->dec_ctx->width : ist->sub2video.w;ist->sub2video.frame->height = ist->dec_ctx->height ? ist->dec_ctx->height : ist->sub2video.h;ist->sub2video.frame->format = AV_PIX_FMT_RGB32;if ((ret = av_frame_get_buffer(frame, 32)) < 0)return ret;memset(frame->data[0], 0, frame->height * frame->linesize[0]);return 0; }static void sub2video_copy_rect(uint8_t *dst, int dst_linesize, int w, int h,AVSubtitleRect *r) {uint32_t *pal, *dst2;uint8_t *src, *src2;int x, y;if (r->type != SUBTITLE_BITMAP) {av_log(NULL, AV_LOG_WARNING, "sub2video: non-bitmap subtitle\n");return;}if (r->x < 0 || r->x + r->w > w || r->y < 0 || r->y + r->h > h) {av_log(NULL, AV_LOG_WARNING, "sub2video: rectangle (%d %d %d %d) overflowing %d %d\n",r->x, r->y, r->w, r->h, w, h);return;}dst += r->y * dst_linesize + r->x * 4;src = r->data[0];pal = (uint32_t *)r->data[1];for (y = 0; y < r->h; y++) {dst2 = (uint32_t *)dst;src2 = src;for (x = 0; x < r->w; x++)*(dst2++) = pal[*(src2++)];dst += dst_linesize;src += r->linesize[0];} }static void sub2video_push_ref(InputStream *ist, int64_t pts) {AVFrame *frame = ist->sub2video.frame;int i;av_assert1(frame->data[0]);ist->sub2video.last_pts = frame->pts = pts;for (i = 0; i < ist->nb_filters; i++)av_buffersrc_add_frame_flags(ist->filters[i]->filter, frame,AV_BUFFERSRC_FLAG_KEEP_REF |AV_BUFFERSRC_FLAG_PUSH); }void sub2video_update(InputStream *ist, AVSubtitle *sub) {AVFrame *frame = ist->sub2video.frame;int8_t *dst;int dst_linesize;int num_rects, i;int64_t pts, end_pts;if (!frame)return;if (sub) {pts = av_rescale_q(sub->pts + sub->start_display_time * 1000LL,AV_TIME_BASE_Q, ist->st->time_base);end_pts = av_rescale_q(sub->pts + sub->end_display_time * 1000LL,AV_TIME_BASE_Q, ist->st->time_base);num_rects = sub->num_rects;} else {pts = ist->sub2video.end_pts;end_pts = INT64_MAX;num_rects = 0;}if (sub2video_get_blank_frame(ist) < 0) {av_log(ist->dec_ctx, AV_LOG_ERROR,"Impossible to get a blank canvas.\n");return;}dst = frame->data [0];dst_linesize = frame->linesize[0];for (i = 0; i < num_rects; i++)sub2video_copy_rect(dst, dst_linesize, frame->width, frame->height, sub->rects[i]);sub2video_push_ref(ist, pts);ist->sub2video.end_pts = end_pts; }static void sub2video_heartbeat(InputStream *ist, int64_t pts) {InputFile *infile = input_files[ist->file_index];int i, j, nb_reqs;int64_t pts2;/* When a frame is read from a file, examine all sub2video streams inthe same file and send the sub2video frame again. Otherwise, decodedvideo frames could be accumulating in the filter graph while a filter(possibly overlay) is desperately waiting for a subtitle frame. */for (i = 0; i < infile->nb_streams; i++) {InputStream *ist2 = input_streams[infile->ist_index + i];if (!ist2->sub2video.frame)continue;/* subtitles seem to be usually muxed ahead of other streams;if not, subtracting a larger time here is necessary */pts2 = av_rescale_q(pts, ist->st->time_base, ist2->st->time_base) - 1;/* do not send the heartbeat frame if the subtitle is already ahead */if (pts2 <= ist2->sub2video.last_pts)continue;if (pts2 >= ist2->sub2video.end_pts || !ist2->sub2video.frame->data[0])sub2video_update(ist2, NULL);for (j = 0, nb_reqs = 0; j < ist2->nb_filters; j++)nb_reqs += av_buffersrc_get_nb_failed_requests(ist2->filters[j]->filter);if (nb_reqs)sub2video_push_ref(ist2, pts2);} }static void sub2video_flush(InputStream *ist) {int i;if (ist->sub2video.end_pts < INT64_MAX)sub2video_update(ist, NULL);for (i = 0; i < ist->nb_filters; i++)av_buffersrc_add_frame(ist->filters[i]->filter, NULL); }/* end of sub2video hack */static void term_exit_sigsafe(void) { #if HAVE_TERMIOS_Hif(restore_tty)tcsetattr (0, TCSANOW, &oldtty); #endif }void term_exit(void) {av_log(NULL, AV_LOG_QUIET, "%s", "");term_exit_sigsafe(); }static volatile int received_sigterm = 0; static volatile int received_nb_signals = 0; static atomic_int transcode_init_done = ATOMIC_VAR_INIT(0); static volatile int ffmpeg_exited = 0; static int main_return_code = 0;static void sigterm_handler(int sig) {received_sigterm = sig;received_nb_signals++;term_exit_sigsafe();if(received_nb_signals > 3) {write(2/*STDERR_FILENO*/, "Received > 3 system signals, hard exiting\n",strlen("Received > 3 system signals, hard exiting\n"));exit(123);} }#if HAVE_SETCONSOLECTRLHANDLER static BOOL WINAPI CtrlHandler(DWORD fdwCtrlType) {av_log(NULL, AV_LOG_DEBUG, "\nReceived windows signal %ld\n", fdwCtrlType);switch (fdwCtrlType){case CTRL_C_EVENT:case CTRL_BREAK_EVENT:sigterm_handler(SIGINT);return TRUE;case CTRL_CLOSE_EVENT:case CTRL_LOGOFF_EVENT:case CTRL_SHUTDOWN_EVENT:sigterm_handler(SIGTERM);/* Basically, with these 3 events, when we return from this method theprocess is hard terminated, so stall as long as we need toto try and let the main thread(s) clean up and gracefully terminate(we have at most 5 seconds, but should be done far before that). */while (!ffmpeg_exited) {Sleep(0);}return TRUE;default:av_log(NULL, AV_LOG_ERROR, "Received unknown windows signal %ld\n", fdwCtrlType);return FALSE;} } #endifvoid term_init(void) { #if HAVE_TERMIOS_Hif (!run_as_daemon && stdin_interaction) {struct termios tty;if (tcgetattr (0, &tty) == 0) {oldtty = tty;restore_tty = 1;tty.c_iflag &= ~(IGNBRK|BRKINT|PARMRK|ISTRIP|INLCR|IGNCR|ICRNL|IXON);tty.c_oflag |= OPOST;tty.c_lflag &= ~(ECHO|ECHONL|ICANON|IEXTEN);tty.c_cflag &= ~(CSIZE|PARENB);tty.c_cflag |= CS8;tty.c_cc[VMIN] = 1;tty.c_cc[VTIME] = 0;tcsetattr (0, TCSANOW, &tty);}signal(SIGQUIT, sigterm_handler); /* Quit (POSIX). */} #endifsignal(SIGINT , sigterm_handler); /* Interrupt (ANSI). */signal(SIGTERM, sigterm_handler); /* Termination (ANSI). */ #ifdef SIGXCPUsignal(SIGXCPU, sigterm_handler); #endif #if HAVE_SETCONSOLECTRLHANDLERSetConsoleCtrlHandler((PHANDLER_ROUTINE) CtrlHandler, TRUE); #endif }/* read a key without blocking */ static int read_key(void) {unsigned char ch; #if HAVE_TERMIOS_Hint n = 1;struct timeval tv;fd_set rfds;FD_ZERO(&rfds);FD_SET(0, &rfds);tv.tv_sec = 0;tv.tv_usec = 0;n = select(1, &rfds, NULL, NULL, &tv);if (n > 0) {n = read(0, &ch, 1);if (n == 1)return ch;return n;} #elif HAVE_KBHIT # if HAVE_PEEKNAMEDPIPEstatic int is_pipe;static HANDLE input_handle;DWORD dw, nchars;if(!input_handle){input_handle = GetStdHandle(STD_INPUT_HANDLE);is_pipe = !GetConsoleMode(input_handle, &dw);}if (is_pipe) {/* When running under a GUI, you will end here. */if (!PeekNamedPipe(input_handle, NULL, 0, NULL, &nchars, NULL)) {// input pipe may have been closed by the program that ran ffmpegreturn -1;}//Read itif(nchars != 0) {read(0, &ch, 1);return ch;}else{return -1;}} # endifif(kbhit())return(getch()); #endifreturn -1; }static int decode_interrupt_cb(void *ctx) {return received_nb_signals > atomic_load(&transcode_init_done); }const AVIOInterruptCB int_cb = { decode_interrupt_cb, NULL };static void ffmpeg_cleanup(int ret) {int i, j;if (do_benchmark) {int maxrss = getmaxrss() / 1024;av_log(NULL, AV_LOG_INFO, "bench: maxrss=%ikB\n", maxrss);}for (i = 0; i < nb_filtergraphs; i++) {FilterGraph *fg = filtergraphs[i];avfilter_graph_free(&fg->graph);for (j = 0; j < fg->nb_inputs; j++) {while (av_fifo_size(fg->inputs[j]->frame_queue)) {AVFrame *frame;av_fifo_generic_read(fg->inputs[j]->frame_queue, &frame,sizeof(frame), NULL);av_frame_free(&frame);}av_fifo_free(fg->inputs[j]->frame_queue);if (fg->inputs[j]->ist->sub2video.sub_queue) {while (av_fifo_size(fg->inputs[j]->ist->sub2video.sub_queue)) {AVSubtitle sub;av_fifo_generic_read(fg->inputs[j]->ist->sub2video.sub_queue,&sub, sizeof(sub), NULL);avsubtitle_free(&sub);}av_fifo_free(fg->inputs[j]->ist->sub2video.sub_queue);}av_buffer_unref(&fg->inputs[j]->hw_frames_ctx);av_freep(&fg->inputs[j]->name);av_freep(&fg->inputs[j]);}av_freep(&fg->inputs);for (j = 0; j < fg->nb_outputs; j++) {av_freep(&fg->outputs[j]->name);av_freep(&fg->outputs[j]->formats);av_freep(&fg->outputs[j]->channel_layouts);av_freep(&fg->outputs[j]->sample_rates);av_freep(&fg->outputs[j]);}av_freep(&fg->outputs);av_freep(&fg->graph_desc);av_freep(&filtergraphs[i]);}av_freep(&filtergraphs);av_freep(&subtitle_out);/* close files */for (i = 0; i < nb_output_files; i++) {OutputFile *of = output_files[i];AVFormatContext *s;if (!of)continue;s = of->ctx;if (s && s->oformat && !(s->oformat->flags & AVFMT_NOFILE))avio_closep(&s->pb);avformat_free_context(s);av_dict_free(&of->opts);av_freep(&output_files[i]);}for (i = 0; i < nb_output_streams; i++) {OutputStream *ost = output_streams[i];if (!ost)continue;for (j = 0; j < ost->nb_bitstream_filters; j++)av_bsf_free(&ost->bsf_ctx[j]);av_freep(&ost->bsf_ctx);av_freep(&ost->bsf_extradata_updated);av_frame_free(&ost->filtered_frame);av_frame_free(&ost->last_frame);av_dict_free(&ost->encoder_opts);av_parser_close(ost->parser);avcodec_free_context(&ost->parser_avctx);av_freep(&ost->forced_keyframes);av_expr_free(ost->forced_keyframes_pexpr);av_freep(&ost->avfilter);av_freep(&ost->logfile_prefix);av_freep(&ost->audio_channels_map);ost->audio_channels_mapped = 0;av_dict_free(&ost->sws_dict);avcodec_free_context(&ost->enc_ctx);avcodec_parameters_free(&ost->ref_par);if (ost->muxing_queue) {while (av_fifo_size(ost->muxing_queue)) {AVPacket pkt;av_fifo_generic_read(ost->muxing_queue, &pkt, sizeof(pkt), NULL);av_packet_unref(&pkt);}av_fifo_freep(&ost->muxing_queue);}av_freep(&output_streams[i]);} #if HAVE_PTHREADSfree_input_threads(); #endiffor (i = 0; i < nb_input_files; i++) {avformat_close_input(&input_files[i]->ctx);av_freep(&input_files[i]);}for (i = 0; i < nb_input_streams; i++) {InputStream *ist = input_streams[i];av_frame_free(&ist->decoded_frame);av_frame_free(&ist->filter_frame);av_dict_free(&ist->decoder_opts);avsubtitle_free(&ist->prev_sub.subtitle);av_frame_free(&ist->sub2video.frame);av_freep(&ist->filters);av_freep(&ist->hwaccel_device);av_freep(&ist->dts_buffer);avcodec_free_context(&ist->dec_ctx);av_freep(&input_streams[i]);}if (vstats_file) {if (fclose(vstats_file))av_log(NULL, AV_LOG_ERROR,"Error closing vstats file, loss of information possible: %s\n",av_err2str(AVERROR(errno)));}av_freep(&vstats_filename);av_freep(&input_streams);av_freep(&input_files);av_freep(&output_streams);av_freep(&output_files);uninit_opts();avformat_network_deinit();if (received_sigterm) {av_log(NULL, AV_LOG_INFO, "Exiting normally, received signal %d.\n",(int) received_sigterm);} else if (ret && atomic_load(&transcode_init_done)) {av_log(NULL, AV_LOG_INFO, "Conversion failed!\n");}term_exit();ffmpeg_exited = 1; }void remove_avoptions(AVDictionary **a, AVDictionary *b) {AVDictionaryEntry *t = NULL;while ((t = av_dict_get(b, "", t, AV_DICT_IGNORE_SUFFIX))) {av_dict_set(a, t->key, NULL, AV_DICT_MATCH_CASE);} }void assert_avoptions(AVDictionary *m) {AVDictionaryEntry *t;if ((t = av_dict_get(m, "", NULL, AV_DICT_IGNORE_SUFFIX))) {av_log(NULL, AV_LOG_FATAL, "Option %s not found.\n", t->key);exit_program(1);} }static void abort_codec_experimental(AVCodec *c, int encoder) {exit_program(1); }static void update_benchmark(const char *fmt, ...) {if (do_benchmark_all) {int64_t t = getutime();va_list va;char buf[1024];if (fmt) {va_start(va, fmt);vsnprintf(buf, sizeof(buf), fmt, va);va_end(va);av_log(NULL, AV_LOG_INFO, "bench: %8"PRIu64" %s \n", t - current_time, buf);}current_time = t;} }static void close_all_output_streams(OutputStream *ost, OSTFinished this_stream, OSTFinished others) {int i;for (i = 0; i < nb_output_streams; i++) {OutputStream *ost2 = output_streams[i];ost2->finished |= ost == ost2 ? this_stream : others;} }static void write_packet(OutputFile *of, AVPacket *pkt, OutputStream *ost, int unqueue) {AVFormatContext *s = of->ctx;AVStream *st = ost->st;int ret;/** Audio encoders may split the packets -- #frames in != #packets out.* But there is no reordering, so we can limit the number of output packets* by simply dropping them here.* Counting encoded video frames needs to be done separately because of* reordering, see do_video_out().* Do not count the packet when unqueued because it has been counted when queued.*/if (!(st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO && ost->encoding_needed) && !unqueue) {if (ost->frame_number >= ost->max_frames) {av_packet_unref(pkt);return;}ost->frame_number++;}if (!of->header_written) {AVPacket tmp_pkt = {0};/* the muxer is not initialized yet, buffer the packet */if (!av_fifo_space(ost->muxing_queue)) {int new_size = FFMIN(2 * av_fifo_size(ost->muxing_queue),ost->max_muxing_queue_size);if (new_size <= av_fifo_size(ost->muxing_queue)) {av_log(NULL, AV_LOG_ERROR,"Too many packets buffered for output stream %d:%d.\n",ost->file_index, ost->st->index);exit_program(1);}ret = av_fifo_realloc2(ost->muxing_queue, new_size);if (ret < 0)exit_program(1);}ret = av_packet_ref(&tmp_pkt, pkt);if (ret < 0)exit_program(1);av_fifo_generic_write(ost->muxing_queue, &tmp_pkt, sizeof(tmp_pkt), NULL);av_packet_unref(pkt);return;}if ((st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO && video_sync_method == VSYNC_DROP) ||(st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO && audio_sync_method < 0))pkt->pts = pkt->dts = AV_NOPTS_VALUE;if (st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) {int i;uint8_t *sd = av_packet_get_side_data(pkt, AV_PKT_DATA_QUALITY_STATS,NULL);ost->quality = sd ? AV_RL32(sd) : -1;ost->pict_type = sd ? sd[4] : AV_PICTURE_TYPE_NONE;for (i = 0; i<FF_ARRAY_ELEMS(ost->error); i++) {if (sd && i < sd[5])ost->error[i] = AV_RL64(sd + 8 + 8*i);elseost->error[i] = -1;}if (ost->frame_rate.num && ost->is_cfr) {if (pkt->duration > 0)av_log(NULL, AV_LOG_WARNING, "Overriding packet duration by frame rate, this should not happen\n");pkt->duration = av_rescale_q(1, av_inv_q(ost->frame_rate),ost->mux_timebase);}}av_packet_rescale_ts(pkt, ost->mux_timebase, ost->st->time_base);if (!(s->oformat->flags & AVFMT_NOTIMESTAMPS)) {if (pkt->dts != AV_NOPTS_VALUE &&pkt->pts != AV_NOPTS_VALUE &&pkt->dts > pkt->pts) {av_log(s, AV_LOG_WARNING, "Invalid DTS: %"PRId64" PTS: %"PRId64" in output stream %d:%d, replacing by guess\n",pkt->dts, pkt->pts,ost->file_index, ost->st->index);pkt->pts =pkt->dts = pkt->pts + pkt->dts + ost->last_mux_dts + 1- FFMIN3(pkt->pts, pkt->dts, ost->last_mux_dts + 1)- FFMAX3(pkt->pts, pkt->dts, ost->last_mux_dts + 1);}if ((st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO || st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) &&pkt->dts != AV_NOPTS_VALUE &&!(st->codecpar->codec_id == AV_CODEC_ID_VP9 && ost->stream_copy) &&ost->last_mux_dts != AV_NOPTS_VALUE) {int64_t max = ost->last_mux_dts + !(s->oformat->flags & AVFMT_TS_NONSTRICT);if (pkt->dts < max) {int loglevel = max - pkt->dts > 2 || st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO ? AV_LOG_WARNING : AV_LOG_DEBUG;av_log(s, loglevel, "Non-monotonous DTS in output stream ""%d:%d; previous: %"PRId64", current: %"PRId64"; ",ost->file_index, ost->st->index, ost->last_mux_dts, pkt->dts);if (exit_on_error) {av_log(NULL, AV_LOG_FATAL, "aborting.\n");exit_program(1);}av_log(s, loglevel, "changing to %"PRId64". This may result ""in incorrect timestamps in the output file.\n",max);if (pkt->pts >= pkt->dts)pkt->pts = FFMAX(pkt->pts, max);pkt->dts = max;}}}ost->last_mux_dts = pkt->dts;ost->data_size += pkt->size;ost->packets_written++;pkt->stream_index = ost->index;if (debug_ts) {av_log(NULL, AV_LOG_INFO, "muxer <- type:%s ""pkt_pts:%s pkt_pts_time:%s pkt_dts:%s pkt_dts_time:%s size:%d\n",av_get_media_type_string(ost->enc_ctx->codec_type),av_ts2str(pkt->pts), av_ts2timestr(pkt->pts, &ost->st->time_base),av_ts2str(pkt->dts), av_ts2timestr(pkt->dts, &ost->st->time_base),pkt->size);}ret = av_interleaved_write_frame(s, pkt);if (ret < 0) {print_error("av_interleaved_write_frame()", ret);main_return_code = 1;close_all_output_streams(ost, MUXER_FINISHED | ENCODER_FINISHED, ENCODER_FINISHED);}av_packet_unref(pkt); }static void close_output_stream(OutputStream *ost) {OutputFile *of = output_files[ost->file_index];ost->finished |= ENCODER_FINISHED;if (of->shortest) {int64_t end = av_rescale_q(ost->sync_opts - ost->first_pts, ost->enc_ctx->time_base, AV_TIME_BASE_Q);of->recording_time = FFMIN(of->recording_time, end);} }static void output_packet(OutputFile *of, AVPacket *pkt, OutputStream *ost) {int ret = 0;/* apply the output bitstream filters, if any */if (ost->nb_bitstream_filters) {int idx;ret = av_bsf_send_packet(ost->bsf_ctx[0], pkt);if (ret < 0)goto finish;idx = 1;while (idx) {/* get a packet from the previous filter up the chain */ret = av_bsf_receive_packet(ost->bsf_ctx[idx - 1], pkt);if (ret == AVERROR(EAGAIN)) {ret = 0;idx--;continue;} else if (ret < 0)goto finish;/* HACK! - aac_adtstoasc updates extradata after filtering the first frame when* the api states this shouldn't happen after init(). Propagate it here to the* muxer and to the next filters in the chain to workaround this.* TODO/FIXME - Make aac_adtstoasc use new packet side data instead of changing* par_out->extradata and adapt muxers accordingly to get rid of this. */if (!(ost->bsf_extradata_updated[idx - 1] & 1)) {ret = avcodec_parameters_copy(ost->st->codecpar, ost->bsf_ctx[idx - 1]->par_out);if (ret < 0)goto finish;ost->bsf_extradata_updated[idx - 1] |= 1;}/* send it to the next filter down the chain or to the muxer */if (idx < ost->nb_bitstream_filters) {/* HACK/FIXME! - See above */if (!(ost->bsf_extradata_updated[idx] & 2)) {ret = avcodec_parameters_copy(ost->bsf_ctx[idx]->par_out, ost->bsf_ctx[idx - 1]->par_out);if (ret < 0)goto finish;ost->bsf_extradata_updated[idx] |= 2;}ret = av_bsf_send_packet(ost->bsf_ctx[idx], pkt);if (ret < 0)goto finish;idx++;} elsewrite_packet(of, pkt, ost, 0);}} elsewrite_packet(of, pkt, ost, 0);finish:if (ret < 0 && ret != AVERROR_EOF) {av_log(NULL, AV_LOG_ERROR, "Error applying bitstream filters to an output ""packet for stream #%d:%d.\n", ost->file_index, ost->index);if(exit_on_error)exit_program(1);} }static int check_recording_time(OutputStream *ost) {OutputFile *of = output_files[ost->file_index];if (of->recording_time != INT64_MAX &&av_compare_ts(ost->sync_opts - ost->first_pts, ost->enc_ctx->time_base, of->recording_time,AV_TIME_BASE_Q) >= 0) {close_output_stream(ost);return 0;}return 1; }static void do_audio_out(OutputFile *of, OutputStream *ost,AVFrame *frame) {AVCodecContext *enc = ost->enc_ctx;AVPacket pkt;int ret;av_init_packet(&pkt);pkt.data = NULL;pkt.size = 0;if (!check_recording_time(ost))return;if (frame->pts == AV_NOPTS_VALUE || audio_sync_method < 0)frame->pts = ost->sync_opts;ost->sync_opts = frame->pts + frame->nb_samples;ost->samples_encoded += frame->nb_samples;ost->frames_encoded++;av_assert0(pkt.size || !pkt.data);update_benchmark(NULL);if (debug_ts) {av_log(NULL, AV_LOG_INFO, "encoder <- type:audio ""frame_pts:%s frame_pts_time:%s time_base:%d/%d\n",av_ts2str(frame->pts), av_ts2timestr(frame->pts, &enc->time_base),enc->time_base.num, enc->time_base.den);}ret = avcodec_send_frame(enc, frame);if (ret < 0)goto error;while (1) {ret = avcodec_receive_packet(enc, &pkt);if (ret == AVERROR(EAGAIN))break;if (ret < 0)goto error;update_benchmark("encode_audio %d.%d", ost->file_index, ost->index);av_packet_rescale_ts(&pkt, enc->time_base, ost->mux_timebase);if (debug_ts) {av_log(NULL, AV_LOG_INFO, "encoder -> type:audio ""pkt_pts:%s pkt_pts_time:%s pkt_dts:%s pkt_dts_time:%s\n",av_ts2str(pkt.pts), av_ts2timestr(pkt.pts, &enc->time_base),av_ts2str(pkt.dts), av_ts2timestr(pkt.dts, &enc->time_base));}output_packet(of, &pkt, ost);}return; error:av_log(NULL, AV_LOG_FATAL, "Audio encoding failed\n");exit_program(1); }static void do_subtitle_out(OutputFile *of,OutputStream *ost,AVSubtitle *sub) {int subtitle_out_max_size = 1024 * 1024;int subtitle_out_size, nb, i;AVCodecContext *enc;AVPacket pkt;int64_t pts;if (sub->pts == AV_NOPTS_VALUE) {av_log(NULL, AV_LOG_ERROR, "Subtitle packets must have a pts\n");if (exit_on_error)exit_program(1);return;}enc = ost->enc_ctx;if (!subtitle_out) {subtitle_out = av_malloc(subtitle_out_max_size);if (!subtitle_out) {av_log(NULL, AV_LOG_FATAL, "Failed to allocate subtitle_out\n");exit_program(1);}}/* Note: DVB subtitle need one packet to draw them and one otherpacket to clear them *//* XXX: signal it in the codec context ? */if (enc->codec_id == AV_CODEC_ID_DVB_SUBTITLE)nb = 2;elsenb = 1;/* shift timestamp to honor -ss and make check_recording_time() work with -t */pts = sub->pts;if (output_files[ost->file_index]->start_time != AV_NOPTS_VALUE)pts -= output_files[ost->file_index]->start_time;for (i = 0; i < nb; i++) {unsigned save_num_rects = sub->num_rects;ost->sync_opts = av_rescale_q(pts, AV_TIME_BASE_Q, enc->time_base);if (!check_recording_time(ost))return;sub->pts = pts;// start_display_time is required to be 0sub->pts += av_rescale_q(sub->start_display_time, (AVRational){ 1, 1000 }, AV_TIME_BASE_Q);sub->end_display_time -= sub->start_display_time;sub->start_display_time = 0;if (i == 1)sub->num_rects = 0;ost->frames_encoded++;subtitle_out_size = avcodec_encode_subtitle(enc, subtitle_out,subtitle_out_max_size, sub);if (i == 1)sub->num_rects = save_num_rects;if (subtitle_out_size < 0) {av_log(NULL, AV_LOG_FATAL, "Subtitle encoding failed\n");exit_program(1);}av_init_packet(&pkt);pkt.data = subtitle_out;pkt.size = subtitle_out_size;pkt.pts = av_rescale_q(sub->pts, AV_TIME_BASE_Q, ost->mux_timebase);pkt.duration = av_rescale_q(sub->end_display_time, (AVRational){ 1, 1000 }, ost->mux_timebase);if (enc->codec_id == AV_CODEC_ID_DVB_SUBTITLE) {/* XXX: the pts correction is handled here. Maybe handlingit in the codec would be better */if (i == 0)pkt.pts += av_rescale_q(sub->start_display_time, (AVRational){ 1, 1000 }, ost->mux_timebase);elsepkt.pts += av_rescale_q(sub->end_display_time, (AVRational){ 1, 1000 }, ost->mux_timebase);}pkt.dts = pkt.pts;output_packet(of, &pkt, ost);} }static void do_video_out(OutputFile *of,OutputStream *ost,AVFrame *next_picture,double sync_ipts) {int ret, format_video_sync;AVPacket pkt;AVCodecContext *enc = ost->enc_ctx;AVCodecParameters *mux_par = ost->st->codecpar;AVRational frame_rate;int nb_frames, nb0_frames, i;double delta, delta0;double duration = 0;int frame_size = 0;InputStream *ist = NULL;AVFilterContext *filter = ost->filter->filter;if (ost->source_index >= 0)ist = input_streams[ost->source_index];frame_rate = av_buffersink_get_frame_rate(filter);if (frame_rate.num > 0 && frame_rate.den > 0)duration = 1/(av_q2d(frame_rate) * av_q2d(enc->time_base));if(ist && ist->st->start_time != AV_NOPTS_VALUE && ist->st->first_dts != AV_NOPTS_VALUE && ost->frame_rate.num)duration = FFMIN(duration, 1/(av_q2d(ost->frame_rate) * av_q2d(enc->time_base)));if (!ost->filters_script &&!ost->filters &&next_picture &&ist &&lrintf(av_frame_get_pkt_duration(next_picture) * av_q2d(ist->st->time_base) / av_q2d(enc->time_base)) > 0) {duration = lrintf(av_frame_get_pkt_duration(next_picture) * av_q2d(ist->st->time_base) / av_q2d(enc->time_base));}if (!next_picture) {//end, flushingnb0_frames = nb_frames = mid_pred(ost->last_nb0_frames[0],ost->last_nb0_frames[1],ost->last_nb0_frames[2]);} else {delta0 = sync_ipts - ost->sync_opts; // delta0 is the "drift" between the input frame (next_picture) and where it would fall in the output.delta = delta0 + duration;/* by default, we output a single frame */nb0_frames = 0; // tracks the number of times the PREVIOUS frame should be duplicated, mostly for variable framerate (VFR)nb_frames = 1;format_video_sync = video_sync_method;if (format_video_sync == VSYNC_AUTO) {if(!strcmp(of->ctx->oformat->name, "avi")) {format_video_sync = VSYNC_VFR;} elseformat_video_sync = (of->ctx->oformat->flags & AVFMT_VARIABLE_FPS) ? ((of->ctx->oformat->flags & AVFMT_NOTIMESTAMPS) ? VSYNC_PASSTHROUGH : VSYNC_VFR) : VSYNC_CFR;if ( ist&& format_video_sync == VSYNC_CFR&& input_files[ist->file_index]->ctx->nb_streams == 1&& input_files[ist->file_index]->input_ts_offset == 0) {format_video_sync = VSYNC_VSCFR;}if (format_video_sync == VSYNC_CFR && copy_ts) {format_video_sync = VSYNC_VSCFR;}}ost->is_cfr = (format_video_sync == VSYNC_CFR || format_video_sync == VSYNC_VSCFR);if (delta0 < 0 &&delta > 0 &&format_video_sync != VSYNC_PASSTHROUGH &&format_video_sync != VSYNC_DROP) {if (delta0 < -0.6) {av_log(NULL, AV_LOG_WARNING, "Past duration %f too large\n", -delta0);} elseav_log(NULL, AV_LOG_DEBUG, "Clipping frame in rate conversion by %f\n", -delta0);sync_ipts = ost->sync_opts;duration += delta0;delta0 = 0;}switch (format_video_sync) {case VSYNC_VSCFR:if (ost->frame_number == 0 && delta0 >= 0.5) {av_log(NULL, AV_LOG_DEBUG, "Not duplicating %d initial frames\n", (int)lrintf(delta0));delta = duration;delta0 = 0;ost->sync_opts = lrint(sync_ipts);}case VSYNC_CFR:// FIXME set to 0.5 after we fix some dts/pts bugs like in avidec.cif (frame_drop_threshold && delta < frame_drop_threshold && ost->frame_number) {nb_frames = 0;} else if (delta < -1.1)nb_frames = 0;else if (delta > 1.1) {nb_frames = lrintf(delta);if (delta0 > 1.1)nb0_frames = lrintf(delta0 - 0.6);}break;case VSYNC_VFR:if (delta <= -0.6)nb_frames = 0;else if (delta > 0.6)ost->sync_opts = lrint(sync_ipts);break;case VSYNC_DROP:case VSYNC_PASSTHROUGH:ost->sync_opts = lrint(sync_ipts);break;default:av_assert0(0);}}nb_frames = FFMIN(nb_frames, ost->max_frames - ost->frame_number);nb0_frames = FFMIN(nb0_frames, nb_frames);memmove(ost->last_nb0_frames + 1,ost->last_nb0_frames,sizeof(ost->last_nb0_frames[0]) * (FF_ARRAY_ELEMS(ost->last_nb0_frames) - 1));ost->last_nb0_frames[0] = nb0_frames;if (nb0_frames == 0 && ost->last_dropped) {nb_frames_drop++;av_log(NULL, AV_LOG_VERBOSE,"*** dropping frame %d from stream %d at ts %"PRId64"\n",ost->frame_number, ost->st->index, ost->last_frame->pts);}if (nb_frames > (nb0_frames && ost->last_dropped) + (nb_frames > nb0_frames)) {if (nb_frames > dts_error_threshold * 30) {av_log(NULL, AV_LOG_ERROR, "%d frame duplication too large, skipping\n", nb_frames - 1);nb_frames_drop++;return;}nb_frames_dup += nb_frames - (nb0_frames && ost->last_dropped) - (nb_frames > nb0_frames);av_log(NULL, AV_LOG_VERBOSE, "*** %d dup!\n", nb_frames - 1);if (nb_frames_dup > dup_warning) {av_log(NULL, AV_LOG_WARNING, "More than %d frames duplicated\n", dup_warning);dup_warning *= 10;}}ost->last_dropped = nb_frames == nb0_frames && next_picture;/* duplicates frame if needed */for (i = 0; i < nb_frames; i++) {AVFrame *in_picture;av_init_packet(&pkt);pkt.data = NULL;pkt.size = 0;if (i < nb0_frames && ost->last_frame) {in_picture = ost->last_frame;} elsein_picture = next_picture;if (!in_picture)return;in_picture->pts = ost->sync_opts;#if 1if (!check_recording_time(ost)) #elseif (ost->frame_number >= ost->max_frames) #endifreturn;#if FF_API_LAVF_FMT_RAWPICTUREif (of->ctx->oformat->flags & AVFMT_RAWPICTURE &&enc->codec->id == AV_CODEC_ID_RAWVIDEO) {/* raw pictures are written as AVPicture structure toavoid any copies. We support temporarily the oldermethod. */if (in_picture->interlaced_frame)mux_par->field_order = in_picture->top_field_first ? AV_FIELD_TB:AV_FIELD_BT;elsemux_par->field_order = AV_FIELD_PROGRESSIVE;pkt.data = (uint8_t *)in_picture;pkt.size = sizeof(AVPicture);pkt.pts = av_rescale_q(in_picture->pts, enc->time_base, ost->mux_timebase);pkt.flags |= AV_PKT_FLAG_KEY;output_packet(of, &pkt, ost);} else #endif{int forced_keyframe = 0;double pts_time;if (enc->flags & (AV_CODEC_FLAG_INTERLACED_DCT | AV_CODEC_FLAG_INTERLACED_ME) &&ost->top_field_first >= 0)in_picture->top_field_first = !!ost->top_field_first;if (in_picture->interlaced_frame) {if (enc->codec->id == AV_CODEC_ID_MJPEG)mux_par->field_order = in_picture->top_field_first ? AV_FIELD_TT:AV_FIELD_BB;elsemux_par->field_order = in_picture->top_field_first ? AV_FIELD_TB:AV_FIELD_BT;} elsemux_par->field_order = AV_FIELD_PROGRESSIVE;in_picture->quality = enc->global_quality;in_picture->pict_type = 0;pts_time = in_picture->pts != AV_NOPTS_VALUE ?in_picture->pts * av_q2d(enc->time_base) : NAN;if (ost->forced_kf_index < ost->forced_kf_count &&in_picture->pts >= ost->forced_kf_pts[ost->forced_kf_index]) {ost->forced_kf_index++;forced_keyframe = 1;} else if (ost->forced_keyframes_pexpr) {double res;ost->forced_keyframes_expr_const_values[FKF_T] = pts_time;res = av_expr_eval(ost->forced_keyframes_pexpr,ost->forced_keyframes_expr_const_values, NULL);ff_dlog(NULL, "force_key_frame: n:%f n_forced:%f prev_forced_n:%f t:%f prev_forced_t:%f -> res:%f\n",ost->forced_keyframes_expr_const_values[FKF_N],ost->forced_keyframes_expr_const_values[FKF_N_FORCED],ost->forced_keyframes_expr_const_values[FKF_PREV_FORCED_N],ost->forced_keyframes_expr_const_values[FKF_T],ost->forced_keyframes_expr_const_values[FKF_PREV_FORCED_T],res);if (res) {forced_keyframe = 1;ost->forced_keyframes_expr_const_values[FKF_PREV_FORCED_N] =ost->forced_keyframes_expr_const_values[FKF_N];ost->forced_keyframes_expr_const_values[FKF_PREV_FORCED_T] =ost->forced_keyframes_expr_const_values[FKF_T];ost->forced_keyframes_expr_const_values[FKF_N_FORCED] += 1;}ost->forced_keyframes_expr_const_values[FKF_N] += 1;} else if ( ost->forced_keyframes&& !strncmp(ost->forced_keyframes, "source", 6)&& in_picture->key_frame==1) {forced_keyframe = 1;}if (forced_keyframe) {in_picture->pict_type = AV_PICTURE_TYPE_I;av_log(NULL, AV_LOG_DEBUG, "Forced keyframe at time %f\n", pts_time);}update_benchmark(NULL);if (debug_ts) {av_log(NULL, AV_LOG_INFO, "encoder <- type:video ""frame_pts:%s frame_pts_time:%s time_base:%d/%d\n",av_ts2str(in_picture->pts), av_ts2timestr(in_picture->pts, &enc->time_base),enc->time_base.num, enc->time_base.den);}ost->frames_encoded++;ret = avcodec_send_frame(enc, in_picture);if (ret < 0)goto error;while (1) {ret = avcodec_receive_packet(enc, &pkt);update_benchmark("encode_video %d.%d", ost->file_index, ost->index);if (ret == AVERROR(EAGAIN))break;if (ret < 0)goto error;if (debug_ts) {av_log(NULL, AV_LOG_INFO, "encoder -> type:video ""pkt_pts:%s pkt_pts_time:%s pkt_dts:%s pkt_dts_time:%s\n",av_ts2str(pkt.pts), av_ts2timestr(pkt.pts, &enc->time_base),av_ts2str(pkt.dts), av_ts2timestr(pkt.dts, &enc->time_base));}if (pkt.pts == AV_NOPTS_VALUE && !(enc->codec->capabilities & AV_CODEC_CAP_DELAY))pkt.pts = ost->sync_opts;av_packet_rescale_ts(&pkt, enc->time_base, ost->mux_timebase);if (debug_ts) {av_log(NULL, AV_LOG_INFO, "encoder -> type:video ""pkt_pts:%s pkt_pts_time:%s pkt_dts:%s pkt_dts_time:%s\n",av_ts2str(pkt.pts), av_ts2timestr(pkt.pts, &ost->mux_timebase),av_ts2str(pkt.dts), av_ts2timestr(pkt.dts, &ost->mux_timebase));}frame_size = pkt.size;output_packet(of, &pkt, ost);/* if two pass, output log */if (ost->logfile && enc->stats_out) {fprintf(ost->logfile, "%s", enc->stats_out);}}}ost->sync_opts++;/** For video, number of frames in == number of packets out.* But there may be reordering, so we can't throw away frames on encoder* flush, we need to limit them here, before they go into encoder.*/ost->frame_number++;if (vstats_filename && frame_size)do_video_stats(ost, frame_size);}if (!ost->last_frame)ost->last_frame = av_frame_alloc();av_frame_unref(ost->last_frame);if (next_picture && ost->last_frame)av_frame_ref(ost->last_frame, next_picture);elseav_frame_free(&ost->last_frame);return; error:av_log(NULL, AV_LOG_FATAL, "Video encoding failed\n");exit_program(1); }static double psnr(double d) {return -10.0 * log10(d); }static void do_video_stats(OutputStream *ost, int frame_size) {AVCodecContext *enc;int frame_number;double ti1, bitrate, avg_bitrate;/* this is executed just the first time do_video_stats is called */if (!vstats_file) {vstats_file = fopen(vstats_filename, "w");if (!vstats_file) {perror("fopen");exit_program(1);}}enc = ost->enc_ctx;if (enc->codec_type == AVMEDIA_TYPE_VIDEO) {frame_number = ost->st->nb_frames;if (vstats_version <= 1) {fprintf(vstats_file, "frame= %5d q= %2.1f ", frame_number,ost->quality / (float)FF_QP2LAMBDA);} else {fprintf(vstats_file, "out= %2d st= %2d frame= %5d q= %2.1f ", ost->file_index, ost->index, frame_number,ost->quality / (float)FF_QP2LAMBDA);}if (ost->error[0]>=0 && (enc->flags & AV_CODEC_FLAG_PSNR))fprintf(vstats_file, "PSNR= %6.2f ", psnr(ost->error[0] / (enc->width * enc->height * 255.0 * 255.0)));fprintf(vstats_file,"f_size= %6d ", frame_size);/* compute pts value */ti1 = av_stream_get_end_pts(ost->st) * av_q2d(ost->st->time_base);if (ti1 < 0.01)ti1 = 0.01;bitrate = (frame_size * 8) / av_q2d(enc->time_base) / 1000.0;avg_bitrate = (double)(ost->data_size * 8) / ti1 / 1000.0;fprintf(vstats_file, "s_size= %8.0fkB time= %0.3f br= %7.1fkbits/s avg_br= %7.1fkbits/s ",(double)ost->data_size / 1024, ti1, bitrate, avg_bitrate);fprintf(vstats_file, "type= %c\n", av_get_picture_type_char(ost->pict_type));} }static int init_output_stream(OutputStream *ost, char *error, int error_len);static void finish_output_stream(OutputStream *ost) {OutputFile *of = output_files[ost->file_index];int i;ost->finished = ENCODER_FINISHED | MUXER_FINISHED;if (of->shortest) {for (i = 0; i < of->ctx->nb_streams; i++)output_streams[of->ost_index + i]->finished = ENCODER_FINISHED | MUXER_FINISHED;} }/*** Get and encode new output from any of the filtergraphs, without causing* activity.** @return 0 for success, <0 for severe errors*/ static int reap_filters(int flush) {AVFrame *filtered_frame = NULL;int i;/* Reap all buffers present in the buffer sinks */for (i = 0; i < nb_output_streams; i++) {OutputStream *ost = output_streams[i];OutputFile *of = output_files[ost->file_index];AVFilterContext *filter;AVCodecContext *enc = ost->enc_ctx;int ret = 0;if (!ost->filter || !ost->filter->graph->graph)continue;filter = ost->filter->filter;if (!ost->initialized) {char error[1024] = "";ret = init_output_stream(ost, error, sizeof(error));if (ret < 0) {av_log(NULL, AV_LOG_ERROR, "Error initializing output stream %d:%d -- %s\n",ost->file_index, ost->index, error);exit_program(1);}}if (!ost->filtered_frame && !(ost->filtered_frame = av_frame_alloc())) {return AVERROR(ENOMEM);}filtered_frame = ost->filtered_frame;while (1) {double float_pts = AV_NOPTS_VALUE; // this is identical to filtered_frame.pts but with higher precisionret = av_buffersink_get_frame_flags(filter, filtered_frame,AV_BUFFERSINK_FLAG_NO_REQUEST);if (ret < 0) {if (ret != AVERROR(EAGAIN) && ret != AVERROR_EOF) {av_log(NULL, AV_LOG_WARNING,"Error in av_buffersink_get_frame_flags(): %s\n", av_err2str(ret));} else if (flush && ret == AVERROR_EOF) {if (av_buffersink_get_type(filter) == AVMEDIA_TYPE_VIDEO)do_video_out(of, ost, NULL, AV_NOPTS_VALUE);}break;}if (ost->finished) {av_frame_unref(filtered_frame);continue;}if (filtered_frame->pts != AV_NOPTS_VALUE) {int64_t start_time = (of->start_time == AV_NOPTS_VALUE) ? 0 : of->start_time;AVRational filter_tb = av_buffersink_get_time_base(filter);AVRational tb = enc->time_base;int extra_bits = av_clip(29 - av_log2(tb.den), 0, 16);tb.den <<= extra_bits;float_pts =av_rescale_q(filtered_frame->pts, filter_tb, tb) -av_rescale_q(start_time, AV_TIME_BASE_Q, tb);float_pts /= 1 << extra_bits;// avoid exact midoints to reduce the chance of rounding differences, this can be removed in case the fps code is changed to work with integersfloat_pts += FFSIGN(float_pts) * 1.0 / (1<<17);filtered_frame->pts =av_rescale_q(filtered_frame->pts, filter_tb, enc->time_base) -av_rescale_q(start_time, AV_TIME_BASE_Q, enc->time_base);}//if (ost->source_index >= 0)// *filtered_frame= *input_streams[ost->source_index]->decoded_frame; //for me_thresholdswitch (av_buffersink_get_type(filter)) {case AVMEDIA_TYPE_VIDEO:if (!ost->frame_aspect_ratio.num)enc->sample_aspect_ratio = filtered_frame->sample_aspect_ratio;if (debug_ts) {av_log(NULL, AV_LOG_INFO, "filter -> pts:%s pts_time:%s exact:%f time_base:%d/%d\n",av_ts2str(filtered_frame->pts), av_ts2timestr(filtered_frame->pts, &enc->time_base),float_pts,enc->time_base.num, enc->time_base.den);}do_video_out(of, ost, filtered_frame, float_pts);break;case AVMEDIA_TYPE_AUDIO:if (!(enc->codec->capabilities & AV_CODEC_CAP_PARAM_CHANGE) &&enc->channels != av_frame_get_channels(filtered_frame)) {av_log(NULL, AV_LOG_ERROR,"Audio filter graph output is not normalized and encoder does not support parameter changes\n");break;}do_audio_out(of, ost, filtered_frame);break;default:// TODO support subtitle filtersav_assert0(0);}av_frame_unref(filtered_frame);}}return 0; }static void print_final_stats(int64_t total_size) {uint64_t video_size = 0, audio_size = 0, extra_size = 0, other_size = 0;uint64_t subtitle_size = 0;uint64_t data_size = 0;float percent = -1.0;int i, j;int pass1_used = 1;for (i = 0; i < nb_output_streams; i++) {OutputStream *ost = output_streams[i];switch (ost->enc_ctx->codec_type) {case AVMEDIA_TYPE_VIDEO: video_size += ost->data_size; break;case AVMEDIA_TYPE_AUDIO: audio_size += ost->data_size; break;case AVMEDIA_TYPE_SUBTITLE: subtitle_size += ost->data_size; break;default: other_size += ost->data_size; break;}extra_size += ost->enc_ctx->extradata_size;data_size += ost->data_size;if ( (ost->enc_ctx->flags & (AV_CODEC_FLAG_PASS1 | AV_CODEC_FLAG_PASS2))!= AV_CODEC_FLAG_PASS1)pass1_used = 0;}if (data_size && total_size>0 && total_size >= data_size)percent = 100.0 * (total_size - data_size) / data_size;av_log(NULL, AV_LOG_INFO, "video:%1.0fkB audio:%1.0fkB subtitle:%1.0fkB other streams:%1.0fkB global headers:%1.0fkB muxing overhead: ",video_size / 1024.0,audio_size / 1024.0,subtitle_size / 1024.0,other_size / 1024.0,extra_size / 1024.0);if (percent >= 0.0)av_log(NULL, AV_LOG_INFO, "%f%%", percent);elseav_log(NULL, AV_LOG_INFO, "unknown");av_log(NULL, AV_LOG_INFO, "\n");/* print verbose per-stream stats */for (i = 0; i < nb_input_files; i++) {InputFile *f = input_files[i];uint64_t total_packets = 0, total_size = 0;av_log(NULL, AV_LOG_VERBOSE, "Input file #%d (%s):\n",i, f->ctx->filename);for (j = 0; j < f->nb_streams; j++) {InputStream *ist = input_streams[f->ist_index + j];enum AVMediaType type = ist->dec_ctx->codec_type;total_size += ist->data_size;total_packets += ist->nb_packets;av_log(NULL, AV_LOG_VERBOSE, " Input stream #%d:%d (%s): ",i, j, media_type_string(type));av_log(NULL, AV_LOG_VERBOSE, "%"PRIu64" packets read (%"PRIu64" bytes); ",ist->nb_packets, ist->data_size);if (ist->decoding_needed) {av_log(NULL, AV_LOG_VERBOSE, "%"PRIu64" frames decoded",ist->frames_decoded);if (type == AVMEDIA_TYPE_AUDIO)av_log(NULL, AV_LOG_VERBOSE, " (%"PRIu64" samples)", ist->samples_decoded);av_log(NULL, AV_LOG_VERBOSE, "; ");}av_log(NULL, AV_LOG_VERBOSE, "\n");}av_log(NULL, AV_LOG_VERBOSE, " Total: %"PRIu64" packets (%"PRIu64" bytes) demuxed\n",total_packets, total_size);}for (i = 0; i < nb_output_files; i++) {OutputFile *of = output_files[i];uint64_t total_packets = 0, total_size = 0;av_log(NULL, AV_LOG_VERBOSE, "Output file #%d (%s):\n",i, of->ctx->filename);for (j = 0; j < of->ctx->nb_streams; j++) {OutputStream *ost = output_streams[of->ost_index + j];enum AVMediaType type = ost->enc_ctx->codec_type;total_size += ost->data_size;total_packets += ost->packets_written;av_log(NULL, AV_LOG_VERBOSE, " Output stream #%d:%d (%s): ",i, j, media_type_string(type));if (ost->encoding_needed) {av_log(NULL, AV_LOG_VERBOSE, "%"PRIu64" frames encoded",ost->frames_encoded);if (type == AVMEDIA_TYPE_AUDIO)av_log(NULL, AV_LOG_VERBOSE, " (%"PRIu64" samples)", ost->samples_encoded);av_log(NULL, AV_LOG_VERBOSE, "; ");}av_log(NULL, AV_LOG_VERBOSE, "%"PRIu64" packets muxed (%"PRIu64" bytes); ",ost->packets_written, ost->data_size);av_log(NULL, AV_LOG_VERBOSE, "\n");}av_log(NULL, AV_LOG_VERBOSE, " Total: %"PRIu64" packets (%"PRIu64" bytes) muxed\n",total_packets, total_size);}if(video_size + data_size + audio_size + subtitle_size + extra_size == 0){av_log(NULL, AV_LOG_WARNING, "Output file is empty, nothing was encoded ");if (pass1_used) {av_log(NULL, AV_LOG_WARNING, "\n");} else {av_log(NULL, AV_LOG_WARNING, "(check -ss / -t / -frames parameters if used)\n");}} } //打印輸出信息的方法 static void print_report(int is_last_report, int64_t timer_start, int64_t cur_time,void(call_back)(int current,int total)) {char buf[1024];AVBPrint buf_script;OutputStream *ost;AVFormatContext *oc;int64_t total_size;AVCodecContext *enc;int frame_number, vid, i;double bitrate;double speed;int64_t pts = INT64_MIN + 1;static int64_t last_time = -1;static int qp_histogram[52];int hours, mins, secs, us;int ret;float t;if (!print_stats && !is_last_report && !progress_avio)return;if (!is_last_report) {if (last_time == -1) {last_time = cur_time;return;}if ((cur_time - last_time) < 500000)return;last_time = cur_time;}t = (cur_time-timer_start) / 1000000.0;oc = output_files[0]->ctx;total_size = avio_size(oc->pb);if (total_size <= 0) // FIXME improve avio_size() so it works with non seekable output toototal_size = avio_tell(oc->pb);buf[0] = '\0';vid = 0;av_bprint_init(&buf_script, 0, 1);for (i = 0; i < nb_output_streams; i++) {float q = -1;ost = output_streams[i];enc = ost->enc_ctx;if (!ost->stream_copy)q = ost->quality / (float) FF_QP2LAMBDA;if (vid && enc->codec_type == AVMEDIA_TYPE_VIDEO) {snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf), "q=%2.1f ", q);av_bprintf(&buf_script, "stream_%d_%d_q=%.1f\n",ost->file_index, ost->index, q);}if (!vid && enc->codec_type == AVMEDIA_TYPE_VIDEO) {float fps;frame_number = ost->frame_number;fps = t > 1 ? frame_number / t : 0;snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf), "frame=%5d fps=%3.*f q=%3.1f ",frame_number, fps < 9.95, fps, q);av_bprintf(&buf_script, "frame=%d\n", frame_number);av_bprintf(&buf_script, "fps=%.1f\n", fps);av_bprintf(&buf_script, "stream_%d_%d_q=%.1f\n",ost->file_index, ost->index, q);//__android_log_print(ANDROID_LOG_ERROR,"JNI_TAG","當前壓縮幀:%d",frame_number);int total_frame_number=input_files[0]->ctx->streams[0]->nb_frames;call_back(frame_number,total_frame_number);if (is_last_report)snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf), "L");if (qp_hist) {int j;int qp = lrintf(q);if (qp >= 0 && qp < FF_ARRAY_ELEMS(qp_histogram))qp_histogram[qp]++;for (j = 0; j < 32; j++)snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf), "%X", av_log2(qp_histogram[j] + 1));}if ((enc->flags & AV_CODEC_FLAG_PSNR) && (ost->pict_type != AV_PICTURE_TYPE_NONE || is_last_report)) {int j;double error, error_sum = 0;double scale, scale_sum = 0;double p;char type[3] = { 'Y','U','V' };snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf), "PSNR=");for (j = 0; j < 3; j++) {if (is_last_report) {error = enc->error[j];scale = enc->width * enc->height * 255.0 * 255.0 * frame_number;} else {error = ost->error[j];scale = enc->width * enc->height * 255.0 * 255.0;}if (j)scale /= 4;error_sum += error;scale_sum += scale;p = psnr(error / scale);snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf), "%c:%2.2f ", type[j], p);av_bprintf(&buf_script, "stream_%d_%d_psnr_%c=%2.2f\n",ost->file_index, ost->index, type[j] | 32, p);}p = psnr(error_sum / scale_sum);snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf), "*:%2.2f ", psnr(error_sum / scale_sum));av_bprintf(&buf_script, "stream_%d_%d_psnr_all=%2.2f\n",ost->file_index, ost->index, p);}vid = 1;}/* compute min output value */if (av_stream_get_end_pts(ost->st) != AV_NOPTS_VALUE)pts = FFMAX(pts, av_rescale_q(av_stream_get_end_pts(ost->st),ost->st->time_base, AV_TIME_BASE_Q));if (is_last_report)nb_frames_drop += ost->last_dropped;}secs = FFABS(pts) / AV_TIME_BASE;us = FFABS(pts) % AV_TIME_BASE;mins = secs / 60;secs %= 60;hours = mins / 60;mins %= 60;bitrate = pts && total_size >= 0 ? total_size * 8 / (pts / 1000.0) : -1;speed = t != 0.0 ? (double)pts / AV_TIME_BASE / t : -1;if (total_size < 0) snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf),"size=N/A time=");else snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf),"size=%8.0fkB time=", total_size / 1024.0);if (pts < 0)snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf), "-");snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf),"%02d:%02d:%02d.%02d ", hours, mins, secs,(100 * us) / AV_TIME_BASE);if (bitrate < 0) {snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf),"bitrate=N/A");av_bprintf(&buf_script, "bitrate=N/A\n");}else{snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf),"bitrate=%6.1fkbits/s", bitrate);av_bprintf(&buf_script, "bitrate=%6.1fkbits/s\n", bitrate);}if (total_size < 0) av_bprintf(&buf_script, "total_size=N/A\n");else av_bprintf(&buf_script, "total_size=%"PRId64"\n", total_size);av_bprintf(&buf_script, "out_time_ms=%"PRId64"\n", pts);av_bprintf(&buf_script, "out_time=%02d:%02d:%02d.%06d\n",hours, mins, secs, us);if (nb_frames_dup || nb_frames_drop)snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf), " dup=%d drop=%d",nb_frames_dup, nb_frames_drop);av_bprintf(&buf_script, "dup_frames=%d\n", nb_frames_dup);av_bprintf(&buf_script, "drop_frames=%d\n", nb_frames_drop);if (speed < 0) {snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf)," speed=N/A");av_bprintf(&buf_script, "speed=N/A\n");} else {snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf)," speed=%4.3gx", speed);av_bprintf(&buf_script, "speed=%4.3gx\n", speed);}if (print_stats || is_last_report) {const char end = is_last_report ? '\n' : '\r';if (print_stats==1 && AV_LOG_INFO > av_log_get_level()) {fprintf(stderr, "%s %c", buf, end);} elseav_log(NULL, AV_LOG_INFO, "%s %c", buf, end);fflush(stderr);}if (progress_avio) {av_bprintf(&buf_script, "progress=%s\n",is_last_report ? "end" : "continue");avio_write(progress_avio, buf_script.str,FFMIN(buf_script.len, buf_script.size - 1));avio_flush(progress_avio);av_bprint_finalize(&buf_script, NULL);if (is_last_report) {if ((ret = avio_closep(&progress_avio)) < 0)av_log(NULL, AV_LOG_ERROR,"Error closing progress log, loss of information possible: %s\n", av_err2str(ret));}}if (is_last_report)print_final_stats(total_size); }static void flush_encoders(void) {int i, ret;for (i = 0; i < nb_output_streams; i++) {OutputStream *ost = output_streams[i];AVCodecContext *enc = ost->enc_ctx;OutputFile *of = output_files[ost->file_index];if (!ost->encoding_needed)continue;// Try to enable encoding with no input frames.// Maybe we should just let encoding fail instead.if (!ost->initialized) {FilterGraph *fg = ost->filter->graph;char error[1024] = "";av_log(NULL, AV_LOG_WARNING,"Finishing stream %d:%d without any data written to it.\n",ost->file_index, ost->st->index);if (ost->filter && !fg->graph) {int x;for (x = 0; x < fg->nb_inputs; x++) {InputFilter *ifilter = fg->inputs[x];if (ifilter->format < 0) {AVCodecParameters *par = ifilter->ist->st->codecpar;// We never got any input. Set a fake format, which will// come from libavformat.ifilter->format = par->format;ifilter->sample_rate = par->sample_rate;ifilter->channels = par->channels;ifilter->channel_layout = par->channel_layout;ifilter->width = par->width;ifilter->height = par->height;ifilter->sample_aspect_ratio = par->sample_aspect_ratio;}}if (!ifilter_has_all_input_formats(fg))continue;ret = configure_filtergraph(fg);if (ret < 0) {av_log(NULL, AV_LOG_ERROR, "Error configuring filter graph\n");exit_program(1);}finish_output_stream(ost);}ret = init_output_stream(ost, error, sizeof(error));if (ret < 0) {av_log(NULL, AV_LOG_ERROR, "Error initializing output stream %d:%d -- %s\n",ost->file_index, ost->index, error);exit_program(1);}}if (enc->codec_type == AVMEDIA_TYPE_AUDIO && enc->frame_size <= 1)continue; #if FF_API_LAVF_FMT_RAWPICTUREif (enc->codec_type == AVMEDIA_TYPE_VIDEO && (of->ctx->oformat->flags & AVFMT_RAWPICTURE) && enc->codec->id == AV_CODEC_ID_RAWVIDEO)continue; #endifif (enc->codec_type != AVMEDIA_TYPE_VIDEO && enc->codec_type != AVMEDIA_TYPE_AUDIO)continue;for (;;) {const char *desc = NULL;AVPacket pkt;int pkt_size;switch (enc->codec_type) {case AVMEDIA_TYPE_AUDIO:desc = "audio";break;case AVMEDIA_TYPE_VIDEO:desc = "video";break;default:av_assert0(0);}av_init_packet(&pkt);pkt.data = NULL;pkt.size = 0;update_benchmark(NULL);while ((ret = avcodec_receive_packet(enc, &pkt)) == AVERROR(EAGAIN)) {ret = avcodec_send_frame(enc, NULL);if (ret < 0) {av_log(NULL, AV_LOG_FATAL, "%s encoding failed: %s\n",desc,av_err2str(ret));exit_program(1);}}update_benchmark("flush_%s %d.%d", desc, ost->file_index, ost->index);if (ret < 0 && ret != AVERROR_EOF) {av_log(NULL, AV_LOG_FATAL, "%s encoding failed: %s\n",desc,av_err2str(ret));exit_program(1);}if (ost->logfile && enc->stats_out) {fprintf(ost->logfile, "%s", enc->stats_out);}if (ret == AVERROR_EOF) {break;}if (ost->finished & MUXER_FINISHED) {av_packet_unref(&pkt);continue;}av_packet_rescale_ts(&pkt, enc->time_base, ost->mux_timebase);pkt_size = pkt.size;output_packet(of, &pkt, ost);if (ost->enc_ctx->codec_type == AVMEDIA_TYPE_VIDEO && vstats_filename) {do_video_stats(ost, pkt_size);}}} }/** Check whether a packet from ist should be written into ost at this time*/ static int check_output_constraints(InputStream *ist, OutputStream *ost) {OutputFile *of = output_files[ost->file_index];int ist_index = input_files[ist->file_index]->ist_index + ist->st->index;if (ost->source_index != ist_index)return 0;if (ost->finished)return 0;if (of->start_time != AV_NOPTS_VALUE && ist->pts < of->start_time)return 0;return 1; }static void do_streamcopy(InputStream *ist, OutputStream *ost, const AVPacket *pkt) {OutputFile *of = output_files[ost->file_index];InputFile *f = input_files [ist->file_index];int64_t start_time = (of->start_time == AV_NOPTS_VALUE) ? 0 : of->start_time;int64_t ost_tb_start_time = av_rescale_q(start_time, AV_TIME_BASE_Q, ost->mux_timebase);AVPicture pict;AVPacket opkt;av_init_packet(&opkt);if ((!ost->frame_number && !(pkt->flags & AV_PKT_FLAG_KEY)) &&!ost->copy_initial_nonkeyframes)return;if (!ost->frame_number && !ost->copy_prior_start) {int64_t comp_start = start_time;if (copy_ts && f->start_time != AV_NOPTS_VALUE)comp_start = FFMAX(start_time, f->start_time + f->ts_offset);if (pkt->pts == AV_NOPTS_VALUE ?ist->pts < comp_start :pkt->pts < av_rescale_q(comp_start, AV_TIME_BASE_Q, ist->st->time_base))return;}if (of->recording_time != INT64_MAX &&ist->pts >= of->recording_time + start_time) {close_output_stream(ost);return;}if (f->recording_time != INT64_MAX) {start_time = f->ctx->start_time;if (f->start_time != AV_NOPTS_VALUE && copy_ts)start_time += f->start_time;if (ist->pts >= f->recording_time + start_time) {close_output_stream(ost);return;}}/* force the input stream PTS */if (ost->enc_ctx->codec_type == AVMEDIA_TYPE_VIDEO)ost->sync_opts++;if (pkt->pts != AV_NOPTS_VALUE)opkt.pts = av_rescale_q(pkt->pts, ist->st->time_base, ost->mux_timebase) - ost_tb_start_time;elseopkt.pts = AV_NOPTS_VALUE;if (pkt->dts == AV_NOPTS_VALUE)opkt.dts = av_rescale_q(ist->dts, AV_TIME_BASE_Q, ost->mux_timebase);elseopkt.dts = av_rescale_q(pkt->dts, ist->st->time_base, ost->mux_timebase);opkt.dts -= ost_tb_start_time;if (ost->st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO && pkt->dts != AV_NOPTS_VALUE) {int duration = av_get_audio_frame_duration(ist->dec_ctx, pkt->size);if(!duration)duration = ist->dec_ctx->frame_size;opkt.dts = opkt.pts = av_rescale_delta(ist->st->time_base, pkt->dts,(AVRational){1, ist->dec_ctx->sample_rate}, duration, &ist->filter_in_rescale_delta_last,ost->mux_timebase) - ost_tb_start_time;}opkt.duration = av_rescale_q(pkt->duration, ist->st->time_base, ost->mux_timebase);opkt.flags = pkt->flags;// FIXME remove the following 2 lines they shall be replaced by the bitstream filtersif ( ost->st->codecpar->codec_id != AV_CODEC_ID_H264&& ost->st->codecpar->codec_id != AV_CODEC_ID_MPEG1VIDEO&& ost->st->codecpar->codec_id != AV_CODEC_ID_MPEG2VIDEO&& ost->st->codecpar->codec_id != AV_CODEC_ID_VC1) {int ret = av_parser_change(ost->parser, ost->parser_avctx,&opkt.data, &opkt.size,pkt->data, pkt->size,pkt->flags & AV_PKT_FLAG_KEY);if (ret < 0) {av_log(NULL, AV_LOG_FATAL, "av_parser_change failed: %s\n",av_err2str(ret));exit_program(1);}if (ret) {opkt.buf = av_buffer_create(opkt.data, opkt.size, av_buffer_default_free, NULL, 0);if (!opkt.buf)exit_program(1);}} else {opkt.data = pkt->data;opkt.size = pkt->size;}av_copy_packet_side_data(&opkt, pkt);#if FF_API_LAVF_FMT_RAWPICTUREif (ost->st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO &&ost->st->codecpar->codec_id == AV_CODEC_ID_RAWVIDEO &&(of->ctx->oformat->flags & AVFMT_RAWPICTURE)) {/* store AVPicture in AVPacket, as expected by the output format */int ret = avpicture_fill(&pict, opkt.data, ost->st->codecpar->format, ost->st->codecpar->width, ost->st->codecpar->height);if (ret < 0) {av_log(NULL, AV_LOG_FATAL, "avpicture_fill failed: %s\n",av_err2str(ret));exit_program(1);}opkt.data = (uint8_t *)&pict;opkt.size = sizeof(AVPicture);opkt.flags |= AV_PKT_FLAG_KEY;} #endifoutput_packet(of, &opkt, ost); }int guess_input_channel_layout(InputStream *ist) {AVCodecContext *dec = ist->dec_ctx;if (!dec->channel_layout) {char layout_name[256];if (dec->channels > ist->guess_layout_max)return 0;dec->channel_layout = av_get_default_channel_layout(dec->channels);if (!dec->channel_layout)return 0;av_get_channel_layout_string(layout_name, sizeof(layout_name),dec->channels, dec->channel_layout);av_log(NULL, AV_LOG_WARNING, "Guessed Channel Layout for Input Stream ""#%d.%d : %s\n", ist->file_index, ist->st->index, layout_name);}return 1; }static void check_decode_result(InputStream *ist, int *got_output, int ret) {if (*got_output || ret<0)decode_error_stat[ret<0] ++;if (ret < 0 && exit_on_error)exit_program(1);if (exit_on_error && *got_output && ist) {if (av_frame_get_decode_error_flags(ist->decoded_frame) || (ist->decoded_frame->flags & AV_FRAME_FLAG_CORRUPT)) {av_log(NULL, AV_LOG_FATAL, "%s: corrupt decoded frame in stream %d\n", input_files[ist->file_index]->ctx->filename, ist->st->index);exit_program(1);}} }// Filters can be configured only if the formats of all inputs are known. static int ifilter_has_all_input_formats(FilterGraph *fg) {int i;for (i = 0; i < fg->nb_inputs; i++) {if (fg->inputs[i]->format < 0 && (fg->inputs[i]->type == AVMEDIA_TYPE_AUDIO ||fg->inputs[i]->type == AVMEDIA_TYPE_VIDEO))return 0;}return 1; }static int ifilter_send_frame(InputFilter *ifilter, AVFrame *frame) {FilterGraph *fg = ifilter->graph;int need_reinit, ret, i;/* determine if the parameters for this input changed */need_reinit = ifilter->format != frame->format;switch (ifilter->ist->st->codecpar->codec_type) {case AVMEDIA_TYPE_AUDIO:need_reinit |= ifilter->sample_rate != frame->sample_rate ||ifilter->channels != frame->channels ||ifilter->channel_layout != frame->channel_layout;break;case AVMEDIA_TYPE_VIDEO:need_reinit |= ifilter->width != frame->width ||ifilter->height != frame->height;break;}if (!ifilter->ist->reinit_filters && fg->graph)need_reinit = 0;if (!!ifilter->hw_frames_ctx != !!frame->hw_frames_ctx ||(ifilter->hw_frames_ctx && ifilter->hw_frames_ctx->data != frame->hw_frames_ctx->data))need_reinit = 1;if (need_reinit) {ret = ifilter_parameters_from_frame(ifilter, frame);if (ret < 0)return ret;}/* (re)init the graph if possible, otherwise buffer the frame and return */if (need_reinit || !fg->graph) {for (i = 0; i < fg->nb_inputs; i++) {if (!ifilter_has_all_input_formats(fg)) {AVFrame *tmp = av_frame_clone(frame);if (!tmp)return AVERROR(ENOMEM);av_frame_unref(frame);if (!av_fifo_space(ifilter->frame_queue)) {ret = av_fifo_realloc2(ifilter->frame_queue, 2 * av_fifo_size(ifilter->frame_queue));if (ret < 0) {av_frame_free(&tmp);return ret;}}av_fifo_generic_write(ifilter->frame_queue, &tmp, sizeof(tmp), NULL);return 0;}}ret = reap_filters(1);if (ret < 0 && ret != AVERROR_EOF) {char errbuf[128];av_strerror(ret, errbuf, sizeof(errbuf));av_log(NULL, AV_LOG_ERROR, "Error while filtering: %s\n", errbuf);return ret;}ret = configure_filtergraph(fg);if (ret < 0) {av_log(NULL, AV_LOG_ERROR, "Error reinitializing filters!\n");return ret;}}ret = av_buffersrc_add_frame_flags(ifilter->filter, frame, AV_BUFFERSRC_FLAG_PUSH);if (ret < 0) {av_log(NULL, AV_LOG_ERROR, "Error while filtering\n");return ret;}return 0; }static int ifilter_send_eof(InputFilter *ifilter) {int i, j, ret;ifilter->eof = 1;if (ifilter->filter) {ret = av_buffersrc_add_frame_flags(ifilter->filter, NULL, AV_BUFFERSRC_FLAG_PUSH);if (ret < 0)return ret;} else {// the filtergraph was never configuredFilterGraph *fg = ifilter->graph;for (i = 0; i < fg->nb_inputs; i++)if (!fg->inputs[i]->eof)break;if (i == fg->nb_inputs) {// All the input streams have finished without the filtergraph// ever being configured.// Mark the output streams as finished.for (j = 0; j < fg->nb_outputs; j++)finish_output_stream(fg->outputs[j]->ost);}}return 0; }// This does not quite work like avcodec_decode_audio4/avcodec_decode_video2. // There is the following difference: if you got a frame, you must call // it again with pkt=NULL. pkt==NULL is treated differently from pkt.size==0 // (pkt==NULL means get more output, pkt.size==0 is a flush/drain packet) static int decode(AVCodecContext *avctx, AVFrame *frame, int *got_frame, AVPacket *pkt) {int ret;*got_frame = 0;if (pkt) {ret = avcodec_send_packet(avctx, pkt);// In particular, we don't expect AVERROR(EAGAIN), because we read all// decoded frames with avcodec_receive_frame() until done.if (ret < 0 && ret != AVERROR_EOF)return ret;}ret = avcodec_receive_frame(avctx, frame);if (ret < 0 && ret != AVERROR(EAGAIN))return ret;if (ret >= 0)*got_frame = 1;return 0; }static int send_frame_to_filters(InputStream *ist, AVFrame *decoded_frame) {int i, ret;AVFrame *f;av_assert1(ist->nb_filters > 0); /* ensure ret is initialized */for (i = 0; i < ist->nb_filters; i++) {if (i < ist->nb_filters - 1) {f = ist->filter_frame;ret = av_frame_ref(f, decoded_frame);if (ret < 0)break;} elsef = decoded_frame;ret = ifilter_send_frame(ist->filters[i], f);if (ret == AVERROR_EOF)ret = 0; /* ignore */if (ret < 0) {av_log(NULL, AV_LOG_ERROR,"Failed to inject frame into filter network: %s\n", av_err2str(ret));break;}}return ret; }static int decode_audio(InputStream *ist, AVPacket *pkt, int *got_output,int *decode_failed) {AVFrame *decoded_frame;AVCodecContext *avctx = ist->dec_ctx;int ret, err = 0;AVRational decoded_frame_tb;if (!ist->decoded_frame && !(ist->decoded_frame = av_frame_alloc()))return AVERROR(ENOMEM);if (!ist->filter_frame && !(ist->filter_frame = av_frame_alloc()))return AVERROR(ENOMEM);decoded_frame = ist->decoded_frame;update_benchmark(NULL);ret = decode(avctx, decoded_frame, got_output, pkt);update_benchmark("decode_audio %d.%d", ist->file_index, ist->st->index);if (ret < 0)*decode_failed = 1;if (ret >= 0 && avctx->sample_rate <= 0) {av_log(avctx, AV_LOG_ERROR, "Sample rate %d invalid\n", avctx->sample_rate);ret = AVERROR_INVALIDDATA;}if (ret != AVERROR_EOF)check_decode_result(ist, got_output, ret);if (!*got_output || ret < 0)return ret;ist->samples_decoded += decoded_frame->nb_samples;ist->frames_decoded++;#if 1/* increment next_dts to use for the case where the input stream does nothave timestamps or there are multiple frames in the packet */ist->next_pts += ((int64_t)AV_TIME_BASE * decoded_frame->nb_samples) /avctx->sample_rate;ist->next_dts += ((int64_t)AV_TIME_BASE * decoded_frame->nb_samples) /avctx->sample_rate; #endifif (decoded_frame->pts != AV_NOPTS_VALUE) {decoded_frame_tb = ist->st->time_base;} else if (pkt && pkt->pts != AV_NOPTS_VALUE) {decoded_frame->pts = pkt->pts;decoded_frame_tb = ist->st->time_base;}else {decoded_frame->pts = ist->dts;decoded_frame_tb = AV_TIME_BASE_Q;}if (decoded_frame->pts != AV_NOPTS_VALUE)decoded_frame->pts = av_rescale_delta(decoded_frame_tb, decoded_frame->pts,(AVRational){1, avctx->sample_rate}, decoded_frame->nb_samples, &ist->filter_in_rescale_delta_last,(AVRational){1, avctx->sample_rate});ist->nb_samples = decoded_frame->nb_samples;err = send_frame_to_filters(ist, decoded_frame);av_frame_unref(ist->filter_frame);av_frame_unref(decoded_frame);return err < 0 ? err : ret; }static int decode_video(InputStream *ist, AVPacket *pkt, int *got_output, int eof,int *decode_failed) {AVFrame *decoded_frame;int i, ret = 0, err = 0;int64_t best_effort_timestamp;int64_t dts = AV_NOPTS_VALUE;AVPacket avpkt;// With fate-indeo3-2, we're getting 0-sized packets before EOF for some// reason. This seems like a semi-critical bug. Don't trigger EOF, and// skip the packet.if (!eof && pkt && pkt->size == 0)return 0;if (!ist->decoded_frame && !(ist->decoded_frame = av_frame_alloc()))return AVERROR(ENOMEM);if (!ist->filter_frame && !(ist->filter_frame = av_frame_alloc()))return AVERROR(ENOMEM);decoded_frame = ist->decoded_frame;if (ist->dts != AV_NOPTS_VALUE)dts = av_rescale_q(ist->dts, AV_TIME_BASE_Q, ist->st->time_base);if (pkt) {avpkt = *pkt;avpkt.dts = dts; // ffmpeg.c probably shouldn't do this}// The old code used to set dts on the drain packet, which does not work// with the new API anymore.if (eof) {void *new = av_realloc_array(ist->dts_buffer, ist->nb_dts_buffer + 1, sizeof(ist->dts_buffer[0]));if (!new)return AVERROR(ENOMEM);ist->dts_buffer = new;ist->dts_buffer[ist->nb_dts_buffer++] = dts;}update_benchmark(NULL);ret = decode(ist->dec_ctx, decoded_frame, got_output, pkt ? &avpkt : NULL);update_benchmark("decode_video %d.%d", ist->file_index, ist->st->index);if (ret < 0)*decode_failed = 1;// The following line may be required in some cases where there is no parser// or the parser does not has_b_frames correctlyif (ist->st->codecpar->video_delay < ist->dec_ctx->has_b_frames) {if (ist->dec_ctx->codec_id == AV_CODEC_ID_H264) {ist->st->codecpar->video_delay = ist->dec_ctx->has_b_frames;} elseav_log(ist->dec_ctx, AV_LOG_WARNING,"video_delay is larger in decoder than demuxer %d > %d.\n""If you want to help, upload a sample ""of this file to ftp://upload.ffmpeg.org/incoming/ ""and contact the ffmpeg-devel mailing list. (ffmpeg-devel@ffmpeg.org)\n",ist->dec_ctx->has_b_frames,ist->st->codecpar->video_delay);}if (ret != AVERROR_EOF)check_decode_result(ist, got_output, ret);if (*got_output && ret >= 0) {if (ist->dec_ctx->width != decoded_frame->width ||ist->dec_ctx->height != decoded_frame->height ||ist->dec_ctx->pix_fmt != decoded_frame->format) {av_log(NULL, AV_LOG_DEBUG, "Frame parameters mismatch context %d,%d,%d != %d,%d,%d\n",decoded_frame->width,decoded_frame->height,decoded_frame->format,ist->dec_ctx->width,ist->dec_ctx->height,ist->dec_ctx->pix_fmt);}}if (!*got_output || ret < 0)return ret;if(ist->top_field_first>=0)decoded_frame->top_field_first = ist->top_field_first;ist->frames_decoded++;if (ist->hwaccel_retrieve_data && decoded_frame->format == ist->hwaccel_pix_fmt) {err = ist->hwaccel_retrieve_data(ist->dec_ctx, decoded_frame);if (err < 0)goto fail;}ist->hwaccel_retrieved_pix_fmt = decoded_frame->format;best_effort_timestamp= av_frame_get_best_effort_timestamp(decoded_frame);if (ist->framerate.num)best_effort_timestamp = ist->cfr_next_pts++;if (eof && best_effort_timestamp == AV_NOPTS_VALUE && ist->nb_dts_buffer > 0) {best_effort_timestamp = ist->dts_buffer[0];for (i = 0; i < ist->nb_dts_buffer - 1; i++)ist->dts_buffer[i] = ist->dts_buffer[i + 1];ist->nb_dts_buffer--;}if(best_effort_timestamp != AV_NOPTS_VALUE) {int64_t ts = av_rescale_q(decoded_frame->pts = best_effort_timestamp, ist->st->time_base, AV_TIME_BASE_Q);if (ts != AV_NOPTS_VALUE)ist->next_pts = ist->pts = ts;}if (debug_ts) {av_log(NULL, AV_LOG_INFO, "decoder -> ist_index:%d type:video ""frame_pts:%s frame_pts_time:%s best_effort_ts:%"PRId64" best_effort_ts_time:%s keyframe:%d frame_type:%d time_base:%d/%d\n",ist->st->index, av_ts2str(decoded_frame->pts),av_ts2timestr(decoded_frame->pts, &ist->st->time_base),best_effort_timestamp,av_ts2timestr(best_effort_timestamp, &ist->st->time_base),decoded_frame->key_frame, decoded_frame->pict_type,ist->st->time_base.num, ist->st->time_base.den);}if (ist->st->sample_aspect_ratio.num)decoded_frame->sample_aspect_ratio = ist->st->sample_aspect_ratio;err = send_frame_to_filters(ist, decoded_frame);fail:av_frame_unref(ist->filter_frame);av_frame_unref(decoded_frame);return err < 0 ? err : ret; }static int transcode_subtitles(InputStream *ist, AVPacket *pkt, int *got_output,int *decode_failed) {AVSubtitle subtitle;int free_sub = 1;int i, ret = avcodec_decode_subtitle2(ist->dec_ctx,&subtitle, got_output, pkt);check_decode_result(NULL, got_output, ret);if (ret < 0 || !*got_output) {*decode_failed = 1;if (!pkt->size)sub2video_flush(ist);return ret;}if (ist->fix_sub_duration) {int end = 1;if (ist->prev_sub.got_output) {end = av_rescale(subtitle.pts - ist->prev_sub.subtitle.pts,1000, AV_TIME_BASE);if (end < ist->prev_sub.subtitle.end_display_time) {av_log(ist->dec_ctx, AV_LOG_DEBUG,"Subtitle duration reduced from %"PRId32" to %d%s\n",ist->prev_sub.subtitle.end_display_time, end,end <= 0 ? ", dropping it" : "");ist->prev_sub.subtitle.end_display_time = end;}}FFSWAP(int, *got_output, ist->prev_sub.got_output);FFSWAP(int, ret, ist->prev_sub.ret);FFSWAP(AVSubtitle, subtitle, ist->prev_sub.subtitle);if (end <= 0)goto out;}if (!*got_output)return ret;if (ist->sub2video.frame) {sub2video_update(ist, &subtitle);} else if (ist->nb_filters) {if (!ist->sub2video.sub_queue)ist->sub2video.sub_queue = av_fifo_alloc(8 * sizeof(AVSubtitle));if (!ist->sub2video.sub_queue)exit_program(1);if (!av_fifo_space(ist->sub2video.sub_queue)) {ret = av_fifo_realloc2(ist->sub2video.sub_queue, 2 * av_fifo_size(ist->sub2video.sub_queue));if (ret < 0)exit_program(1);}av_fifo_generic_write(ist->sub2video.sub_queue, &subtitle, sizeof(subtitle), NULL);free_sub = 0;}if (!subtitle.num_rects)goto out;ist->frames_decoded++;for (i = 0; i < nb_output_streams; i++) {OutputStream *ost = output_streams[i];if (!check_output_constraints(ist, ost) || !ost->encoding_needed|| ost->enc->type != AVMEDIA_TYPE_SUBTITLE)continue;do_subtitle_out(output_files[ost->file_index], ost, &subtitle);}out:if (free_sub)avsubtitle_free(&subtitle);return ret; }static int send_filter_eof(InputStream *ist) {int i, ret;for (i = 0; i < ist->nb_filters; i++) {ret = ifilter_send_eof(ist->filters[i]);if (ret < 0)return ret;}return 0; }/* pkt = NULL means EOF (needed to flush decoder buffers) */ static int process_input_packet(InputStream *ist, const AVPacket *pkt, int no_eof) {int ret = 0, i;int repeating = 0;int eof_reached = 0;AVPacket avpkt;if (!ist->saw_first_ts) {ist->dts = ist->st->avg_frame_rate.num ? - ist->dec_ctx->has_b_frames * AV_TIME_BASE / av_q2d(ist->st->avg_frame_rate) : 0;ist->pts = 0;if (pkt && pkt->pts != AV_NOPTS_VALUE && !ist->decoding_needed) {ist->dts += av_rescale_q(pkt->pts, ist->st->time_base, AV_TIME_BASE_Q);ist->pts = ist->dts; //unused but better to set it to a value thats not totally wrong}ist->saw_first_ts = 1;}if (ist->next_dts == AV_NOPTS_VALUE)ist->next_dts = ist->dts;if (ist->next_pts == AV_NOPTS_VALUE)ist->next_pts = ist->pts;if (!pkt) {/* EOF handling */av_init_packet(&avpkt);avpkt.data = NULL;avpkt.size = 0;} else {avpkt = *pkt;}if (pkt && pkt->dts != AV_NOPTS_VALUE) {ist->next_dts = ist->dts = av_rescale_q(pkt->dts, ist->st->time_base, AV_TIME_BASE_Q);if (ist->dec_ctx->codec_type != AVMEDIA_TYPE_VIDEO || !ist->decoding_needed)ist->next_pts = ist->pts = ist->dts;}// while we have more to decode or while the decoder did output something on EOFwhile (ist->decoding_needed) {int duration = 0;int got_output = 0;int decode_failed = 0;ist->pts = ist->next_pts;ist->dts = ist->next_dts;switch (ist->dec_ctx->codec_type) {case AVMEDIA_TYPE_AUDIO:ret = decode_audio (ist, repeating ? NULL : &avpkt, &got_output,&decode_failed);break;case AVMEDIA_TYPE_VIDEO:ret = decode_video (ist, repeating ? NULL : &avpkt, &got_output, !pkt,&decode_failed);if (!repeating || !pkt || got_output) {if (pkt && pkt->duration) {duration = av_rescale_q(pkt->duration, ist->st->time_base, AV_TIME_BASE_Q);} else if(ist->dec_ctx->framerate.num != 0 && ist->dec_ctx->framerate.den != 0) {int ticks= av_stream_get_parser(ist->st) ? av_stream_get_parser(ist->st)->repeat_pict+1 : ist->dec_ctx->ticks_per_frame;duration = ((int64_t)AV_TIME_BASE *ist->dec_ctx->framerate.den * ticks) /ist->dec_ctx->framerate.num / ist->dec_ctx->ticks_per_frame;}if(ist->dts != AV_NOPTS_VALUE && duration) {ist->next_dts += duration;}elseist->next_dts = AV_NOPTS_VALUE;}if (got_output)ist->next_pts += duration; //FIXME the duration is not correct in some casesbreak;case AVMEDIA_TYPE_SUBTITLE:if (repeating)break;ret = transcode_subtitles(ist, &avpkt, &got_output, &decode_failed);if (!pkt && ret >= 0)ret = AVERROR_EOF;break;default:return -1;}if (ret == AVERROR_EOF) {eof_reached = 1;break;}if (ret < 0) {if (decode_failed) {av_log(NULL, AV_LOG_ERROR, "Error while decoding stream #%d:%d: %s\n",ist->file_index, ist->st->index, av_err2str(ret));} else {av_log(NULL, AV_LOG_FATAL, "Error while processing the decoded ""data for stream #%d:%d\n", ist->file_index, ist->st->index);}if (!decode_failed || exit_on_error)exit_program(1);break;}if (got_output)ist->got_output = 1;if (!got_output)break;// During draining, we might get multiple output frames in this loop.// ffmpeg.c does not drain the filter chain on configuration changes,// which means if we send multiple frames at once to the filters, and// one of those frames changes configuration, the buffered frames will// be lost. This can upset certain FATE tests.// Decode only 1 frame per call on EOF to appease these FATE tests.// The ideal solution would be to rewrite decoding to use the new// decoding API in a better way.if (!pkt)break;repeating = 1;}/* after flushing, send an EOF on all the filter inputs attached to the stream *//* except when looping we need to flush but not to send an EOF */if (!pkt && ist->decoding_needed && eof_reached && !no_eof) {int ret = send_filter_eof(ist);if (ret < 0) {av_log(NULL, AV_LOG_FATAL, "Error marking filters as finished\n");exit_program(1);}}/* handle stream copy */if (!ist->decoding_needed) {ist->dts = ist->next_dts;switch (ist->dec_ctx->codec_type) {case AVMEDIA_TYPE_AUDIO:if (ist->dec_ctx->sample_rate) {ist->next_dts += ((int64_t)AV_TIME_BASE * ist->dec_ctx->frame_size) /ist->dec_ctx->sample_rate;} else {ist->next_dts += av_rescale_q(pkt->duration, ist->st->time_base, AV_TIME_BASE_Q);}break;case AVMEDIA_TYPE_VIDEO:if (ist->framerate.num) {// TODO: Remove work-around for c99-to-c89 issue 7AVRational time_base_q = AV_TIME_BASE_Q;int64_t next_dts = av_rescale_q(ist->next_dts, time_base_q, av_inv_q(ist->framerate));ist->next_dts = av_rescale_q(next_dts + 1, av_inv_q(ist->framerate), time_base_q);} else if (pkt->duration) {ist->next_dts += av_rescale_q(pkt->duration, ist->st->time_base, AV_TIME_BASE_Q);} else if(ist->dec_ctx->framerate.num != 0) {int ticks= av_stream_get_parser(ist->st) ? av_stream_get_parser(ist->st)->repeat_pict + 1 : ist->dec_ctx->ticks_per_frame;ist->next_dts += ((int64_t)AV_TIME_BASE *ist->dec_ctx->framerate.den * ticks) /ist->dec_ctx->framerate.num / ist->dec_ctx->ticks_per_frame;}break;}ist->pts = ist->dts;ist->next_pts = ist->next_dts;}for (i = 0; pkt && i < nb_output_streams; i++) {OutputStream *ost = output_streams[i];if (!check_output_constraints(ist, ost) || ost->encoding_needed)continue;do_streamcopy(ist, ost, pkt);}return !eof_reached; }static void print_sdp(void) {char sdp[16384];int i;int j;AVIOContext *sdp_pb;AVFormatContext **avc;for (i = 0; i < nb_output_files; i++) {if (!output_files[i]->header_written)return;}avc = av_malloc_array(nb_output_files, sizeof(*avc));if (!avc)exit_program(1);for (i = 0, j = 0; i < nb_output_files; i++) {if (!strcmp(output_files[i]->ctx->oformat->name, "rtp")) {avc[j] = output_files[i]->ctx;j++;}}if (!j)goto fail;av_sdp_create(avc, j, sdp, sizeof(sdp));if (!sdp_filename) {printf("SDP:\n%s\n", sdp);fflush(stdout);} else {if (avio_open2(&sdp_pb, sdp_filename, AVIO_FLAG_WRITE, &int_cb, NULL) < 0) {av_log(NULL, AV_LOG_ERROR, "Failed to open sdp file '%s'\n", sdp_filename);} else {avio_printf(sdp_pb, "SDP:\n%s", sdp);avio_closep(&sdp_pb);av_freep(&sdp_filename);}}fail:av_freep(&avc); }static const HWAccel *get_hwaccel(enum AVPixelFormat pix_fmt) {int i;for (i = 0; hwaccels[i].name; i++)if (hwaccels[i].pix_fmt == pix_fmt)return &hwaccels[i];return NULL; }static enum AVPixelFormat get_format(AVCodecContext *s, const enum AVPixelFormat *pix_fmts) {InputStream *ist = s->opaque;const enum AVPixelFormat *p;int ret;for (p = pix_fmts; *p != -1; p++) {const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(*p);const HWAccel *hwaccel;if (!(desc->flags & AV_PIX_FMT_FLAG_HWACCEL))break;hwaccel = get_hwaccel(*p);if (!hwaccel ||(ist->active_hwaccel_id && ist->active_hwaccel_id != hwaccel->id) ||(ist->hwaccel_id != HWACCEL_AUTO && ist->hwaccel_id != hwaccel->id))continue;ret = hwaccel->init(s);if (ret < 0) {if (ist->hwaccel_id == hwaccel->id) {av_log(NULL, AV_LOG_FATAL,"%s hwaccel requested for input stream #%d:%d, ""but cannot be initialized.\n", hwaccel->name,ist->file_index, ist->st->index);return AV_PIX_FMT_NONE;}continue;}if (ist->hw_frames_ctx) {s->hw_frames_ctx = av_buffer_ref(ist->hw_frames_ctx);if (!s->hw_frames_ctx)return AV_PIX_FMT_NONE;}ist->active_hwaccel_id = hwaccel->id;ist->hwaccel_pix_fmt = *p;break;}return *p; }static int get_buffer(AVCodecContext *s, AVFrame *frame, int flags) {InputStream *ist = s->opaque;if (ist->hwaccel_get_buffer && frame->format == ist->hwaccel_pix_fmt)return ist->hwaccel_get_buffer(s, frame, flags);return avcodec_default_get_buffer2(s, frame, flags); }static int init_input_stream(int ist_index, char *error, int error_len) {int ret;InputStream *ist = input_streams[ist_index];if (ist->decoding_needed) {AVCodec *codec = ist->dec;if (!codec) {snprintf(error, error_len, "Decoder (codec %s) not found for input stream #%d:%d",avcodec_get_name(ist->dec_ctx->codec_id), ist->file_index, ist->st->index);return AVERROR(EINVAL);}ist->dec_ctx->opaque = ist;ist->dec_ctx->get_format = get_format;ist->dec_ctx->get_buffer2 = get_buffer;ist->dec_ctx->thread_safe_callbacks = 1;av_opt_set_int(ist->dec_ctx, "refcounted_frames", 1, 0);if (ist->dec_ctx->codec_id == AV_CODEC_ID_DVB_SUBTITLE &&(ist->decoding_needed & DECODING_FOR_OST)) {av_dict_set(&ist->decoder_opts, "compute_edt", "1", AV_DICT_DONT_OVERWRITE);if (ist->decoding_needed & DECODING_FOR_FILTER)av_log(NULL, AV_LOG_WARNING, "Warning using DVB subtitles for filtering and output at the same time is not fully supported, also see -compute_edt [0|1]\n");}av_dict_set(&ist->decoder_opts, "sub_text_format", "ass", AV_DICT_DONT_OVERWRITE);/* Useful for subtitles retiming by lavf (FIXME), skipping samples in* audio, and video decoders such as cuvid or mediacodec */av_codec_set_pkt_timebase(ist->dec_ctx, ist->st->time_base);if (!av_dict_get(ist->decoder_opts, "threads", NULL, 0))av_dict_set(&ist->decoder_opts, "threads", "auto", 0);if ((ret = avcodec_open2(ist->dec_ctx, codec, &ist->decoder_opts)) < 0) {if (ret == AVERROR_EXPERIMENTAL)abort_codec_experimental(codec, 0);snprintf(error, error_len,"Error while opening decoder for input stream ""#%d:%d : %s",ist->file_index, ist->st->index, av_err2str(ret));return ret;}assert_avoptions(ist->decoder_opts);}ist->next_pts = AV_NOPTS_VALUE;ist->next_dts = AV_NOPTS_VALUE;return 0; }static InputStream *get_input_stream(OutputStream *ost) {if (ost->source_index >= 0)return input_streams[ost->source_index];return NULL; }static int compare_int64(const void *a, const void *b) {return FFDIFFSIGN(*(const int64_t *)a, *(const int64_t *)b); }/* open the muxer when all the streams are initialized */ static int check_init_output_file(OutputFile *of, int file_index) {int ret, i;for (i = 0; i < of->ctx->nb_streams; i++) {OutputStream *ost = output_streams[of->ost_index + i];if (!ost->initialized)return 0;}of->ctx->interrupt_callback = int_cb;ret = avformat_write_header(of->ctx, &of->opts);if (ret < 0) {av_log(NULL, AV_LOG_ERROR,"Could not write header for output file #%d ""(incorrect codec parameters ?): %s\n",file_index, av_err2str(ret));return ret;}//assert_avoptions(of->opts);of->header_written = 1;av_dump_format(of->ctx, file_index, of->ctx->filename, 1);if (sdp_filename || want_sdp)print_sdp();/* flush the muxing queues */for (i = 0; i < of->ctx->nb_streams; i++) {OutputStream *ost = output_streams[of->ost_index + i];/* try to improve muxing time_base (only possible if nothing has been written yet) */if (!av_fifo_size(ost->muxing_queue))ost->mux_timebase = ost->st->time_base;while (av_fifo_size(ost->muxing_queue)) {AVPacket pkt;av_fifo_generic_read(ost->muxing_queue, &pkt, sizeof(pkt), NULL);write_packet(of, &pkt, ost, 1);}}return 0; }static int init_output_bsfs(OutputStream *ost) {AVBSFContext *ctx;int i, ret;if (!ost->nb_bitstream_filters)return 0;for (i = 0; i < ost->nb_bitstream_filters; i++) {ctx = ost->bsf_ctx[i];ret = avcodec_parameters_copy(ctx->par_in,i ? ost->bsf_ctx[i - 1]->par_out : ost->st->codecpar);if (ret < 0)return ret;ctx->time_base_in = i ? ost->bsf_ctx[i - 1]->time_base_out : ost->st->time_base;ret = av_bsf_init(ctx);if (ret < 0) {av_log(NULL, AV_LOG_ERROR, "Error initializing bitstream filter: %s\n",ost->bsf_ctx[i]->filter->name);return ret;}}ctx = ost->bsf_ctx[ost->nb_bitstream_filters - 1];ret = avcodec_parameters_copy(ost->st->codecpar, ctx->par_out);if (ret < 0)return ret;ost->st->time_base = ctx->time_base_out;return 0; }static int init_output_stream_streamcopy(OutputStream *ost) {OutputFile *of = output_files[ost->file_index];InputStream *ist = get_input_stream(ost);AVCodecParameters *par_dst = ost->st->codecpar;AVCodecParameters *par_src = ost->ref_par;AVRational sar;int i, ret;uint32_t codec_tag = par_dst->codec_tag;av_assert0(ist && !ost->filter);ret = avcodec_parameters_to_context(ost->enc_ctx, ist->st->codecpar);if (ret >= 0)ret = av_opt_set_dict(ost->enc_ctx, &ost->encoder_opts);if (ret < 0) {av_log(NULL, AV_LOG_FATAL,"Error setting up codec context options.\n");return ret;}avcodec_parameters_from_context(par_src, ost->enc_ctx);if (!codec_tag) {unsigned int codec_tag_tmp;if (!of->ctx->oformat->codec_tag ||av_codec_get_id (of->ctx->oformat->codec_tag, par_src->codec_tag) == par_src->codec_id ||!av_codec_get_tag2(of->ctx->oformat->codec_tag, par_src->codec_id, &codec_tag_tmp))codec_tag = par_src->codec_tag;}ret = avcodec_parameters_copy(par_dst, par_src);if (ret < 0)return ret;par_dst->codec_tag = codec_tag;if (!ost->frame_rate.num)ost->frame_rate = ist->framerate;ost->st->avg_frame_rate = ost->frame_rate;ret = avformat_transfer_internal_stream_timing_info(of->ctx->oformat, ost->st, ist->st, copy_tb);if (ret < 0)return ret;// copy timebase while removing common factorsif (ost->st->time_base.num <= 0 || ost->st->time_base.den <= 0)ost->st->time_base = av_add_q(av_stream_get_codec_timebase(ost->st), (AVRational){0, 1});// copy estimated duration as a hint to the muxerif (ost->st->duration <= 0 && ist->st->duration > 0)ost->st->duration = av_rescale_q(ist->st->duration, ist->st->time_base, ost->st->time_base);// copy dispositionost->st->disposition = ist->st->disposition;if (ist->st->nb_side_data) {ost->st->side_data = av_realloc_array(NULL, ist->st->nb_side_data,sizeof(*ist->st->side_data));if (!ost->st->side_data)return AVERROR(ENOMEM);ost->st->nb_side_data = 0;for (i = 0; i < ist->st->nb_side_data; i++) {const AVPacketSideData *sd_src = &ist->st->side_data[i];AVPacketSideData *sd_dst = &ost->st->side_data[ost->st->nb_side_data];sd_dst->data = av_malloc(sd_src->size);if (!sd_dst->data)return AVERROR(ENOMEM);memcpy(sd_dst->data, sd_src->data, sd_src->size);sd_dst->size = sd_src->size;sd_dst->type = sd_src->type;ost->st->nb_side_data++;}}if (ost->rotate_overridden) {uint8_t *sd = av_stream_new_side_data(ost->st, AV_PKT_DATA_DISPLAYMATRIX,sizeof(int32_t) * 9);if (sd)av_display_rotation_set((int32_t *)sd, -ost->rotate_override_value);}ost->parser = av_parser_init(par_dst->codec_id);ost->parser_avctx = avcodec_alloc_context3(NULL);if (!ost->parser_avctx)return AVERROR(ENOMEM);switch (par_dst->codec_type) {case AVMEDIA_TYPE_AUDIO:if (audio_volume != 256) {av_log(NULL, AV_LOG_FATAL, "-acodec copy and -vol are incompatible (frames are not decoded)\n");exit_program(1);}if((par_dst->block_align == 1 || par_dst->block_align == 1152 || par_dst->block_align == 576) && par_dst->codec_id == AV_CODEC_ID_MP3)par_dst->block_align= 0;if(par_dst->codec_id == AV_CODEC_ID_AC3)par_dst->block_align= 0;break;case AVMEDIA_TYPE_VIDEO:if (ost->frame_aspect_ratio.num) { // overridden by the -aspect cli optionsar =av_mul_q(ost->frame_aspect_ratio,(AVRational){ par_dst->height, par_dst->width });av_log(NULL, AV_LOG_WARNING, "Overriding aspect ratio ""with stream copy may produce invalid files\n");}else if (ist->st->sample_aspect_ratio.num)sar = ist->st->sample_aspect_ratio;elsesar = par_src->sample_aspect_ratio;ost->st->sample_aspect_ratio = par_dst->sample_aspect_ratio = sar;ost->st->avg_frame_rate = ist->st->avg_frame_rate;ost->st->r_frame_rate = ist->st->r_frame_rate;break;}ost->mux_timebase = ist->st->time_base;return 0; }static void set_encoder_id(OutputFile *of, OutputStream *ost) {AVDictionaryEntry *e;uint8_t *encoder_string;int encoder_string_len;int format_flags = 0;int codec_flags = 0;if (av_dict_get(ost->st->metadata, "encoder", NULL, 0))return;e = av_dict_get(of->opts, "fflags", NULL, 0);if (e) {const AVOption *o = av_opt_find(of->ctx, "fflags", NULL, 0, 0);if (!o)return;av_opt_eval_flags(of->ctx, o, e->value, &format_flags);}e = av_dict_get(ost->encoder_opts, "flags", NULL, 0);if (e) {const AVOption *o = av_opt_find(ost->enc_ctx, "flags", NULL, 0, 0);if (!o)return;av_opt_eval_flags(ost->enc_ctx, o, e->value, &codec_flags);}encoder_string_len = sizeof(LIBAVCODEC_IDENT) + strlen(ost->enc->name) + 2;encoder_string = av_mallocz(encoder_string_len);if (!encoder_string)exit_program(1);if (!(format_flags & AVFMT_FLAG_BITEXACT) && !(codec_flags & AV_CODEC_FLAG_BITEXACT))av_strlcpy(encoder_string, LIBAVCODEC_IDENT " ", encoder_string_len);elseav_strlcpy(encoder_string, "Lavc ", encoder_string_len);av_strlcat(encoder_string, ost->enc->name, encoder_string_len);av_dict_set(&ost->st->metadata, "encoder", encoder_string,AV_DICT_DONT_STRDUP_VAL | AV_DICT_DONT_OVERWRITE); }static void parse_forced_key_frames(char *kf, OutputStream *ost,AVCodecContext *avctx) {char *p;int n = 1, i, size, index = 0;int64_t t, *pts;for (p = kf; *p; p++)if (*p == ',')n++;size = n;pts = av_malloc_array(size, sizeof(*pts));if (!pts) {av_log(NULL, AV_LOG_FATAL, "Could not allocate forced key frames array.\n");exit_program(1);}p = kf;for (i = 0; i < n; i++) {char *next = strchr(p, ',');if (next)*next++ = 0;if (!memcmp(p, "chapters", 8)) {AVFormatContext *avf = output_files[ost->file_index]->ctx;int j;if (avf->nb_chapters > INT_MAX - size ||!(pts = av_realloc_f(pts, size += avf->nb_chapters - 1,sizeof(*pts)))) {av_log(NULL, AV_LOG_FATAL,"Could not allocate forced key frames array.\n");exit_program(1);}t = p[8] ? parse_time_or_die("force_key_frames", p + 8, 1) : 0;t = av_rescale_q(t, AV_TIME_BASE_Q, avctx->time_base);for (j = 0; j < avf->nb_chapters; j++) {AVChapter *c = avf->chapters[j];av_assert1(index < size);pts[index++] = av_rescale_q(c->start, c->time_base,avctx->time_base) + t;}} else {t = parse_time_or_die("force_key_frames", p, 1);av_assert1(index < size);pts[index++] = av_rescale_q(t, AV_TIME_BASE_Q, avctx->time_base);}p = next;}av_assert0(index == size);qsort(pts, size, sizeof(*pts), compare_int64);ost->forced_kf_count = size;ost->forced_kf_pts = pts; }static int init_output_stream_encode(OutputStream *ost) {InputStream *ist = get_input_stream(ost);AVCodecContext *enc_ctx = ost->enc_ctx;AVCodecContext *dec_ctx = NULL;AVFormatContext *oc = output_files[ost->file_index]->ctx;int j, ret;set_encoder_id(output_files[ost->file_index], ost);// Muxers use AV_PKT_DATA_DISPLAYMATRIX to signal rotation. On the other// hand, the legacy API makes demuxers set "rotate" metadata entries,// which have to be filtered out to prevent leaking them to output files.av_dict_set(&ost->st->metadata, "rotate", NULL, 0);if (ist) {ost->st->disposition = ist->st->disposition;dec_ctx = ist->dec_ctx;enc_ctx->chroma_sample_location = dec_ctx->chroma_sample_location;} else {for (j = 0; j < oc->nb_streams; j++) {AVStream *st = oc->streams[j];if (st != ost->st && st->codecpar->codec_type == ost->st->codecpar->codec_type)break;}if (j == oc->nb_streams)if (ost->st->codecpar->codec_type == AVMEDIA_TYPE_AUDIO ||ost->st->codecpar->codec_type == AVMEDIA_TYPE_VIDEO)ost->st->disposition = AV_DISPOSITION_DEFAULT;}if (enc_ctx->codec_type == AVMEDIA_TYPE_VIDEO) {if (!ost->frame_rate.num)ost->frame_rate = av_buffersink_get_frame_rate(ost->filter->filter);if (ist && !ost->frame_rate.num)ost->frame_rate = ist->framerate;if (ist && !ost->frame_rate.num)ost->frame_rate = ist->st->r_frame_rate;if (ist && !ost->frame_rate.num) {ost->frame_rate = (AVRational){25, 1};av_log(NULL, AV_LOG_WARNING,"No information ""about the input framerate is available. Falling ""back to a default value of 25fps for output stream #%d:%d. Use the -r option ""if you want a different framerate.\n",ost->file_index, ost->index);} // ost->frame_rate = ist->st->avg_frame_rate.num ? ist->st->avg_frame_rate : (AVRational){25, 1};if (ost->enc->supported_framerates && !ost->force_fps) {int idx = av_find_nearest_q_idx(ost->frame_rate, ost->enc->supported_framerates);ost->frame_rate = ost->enc->supported_framerates[idx];}// reduce frame rate for mpeg4 to be within the spec limitsif (enc_ctx->codec_id == AV_CODEC_ID_MPEG4) {av_reduce(&ost->frame_rate.num, &ost->frame_rate.den,ost->frame_rate.num, ost->frame_rate.den, 65535);}}switch (enc_ctx->codec_type) {case AVMEDIA_TYPE_AUDIO:enc_ctx->sample_fmt = av_buffersink_get_format(ost->filter->filter);if (dec_ctx)enc_ctx->bits_per_raw_sample = FFMIN(dec_ctx->bits_per_raw_sample,av_get_bytes_per_sample(enc_ctx->sample_fmt) << 3);enc_ctx->sample_rate = av_buffersink_get_sample_rate(ost->filter->filter);enc_ctx->channel_layout = av_buffersink_get_channel_layout(ost->filter->filter);enc_ctx->channels = av_buffersink_get_channels(ost->filter->filter);enc_ctx->time_base = (AVRational){ 1, enc_ctx->sample_rate };break;case AVMEDIA_TYPE_VIDEO:enc_ctx->time_base = av_inv_q(ost->frame_rate);if (!(enc_ctx->time_base.num && enc_ctx->time_base.den))enc_ctx->time_base = av_buffersink_get_time_base(ost->filter->filter);if ( av_q2d(enc_ctx->time_base) < 0.001 && video_sync_method != VSYNC_PASSTHROUGH&& (video_sync_method == VSYNC_CFR || video_sync_method == VSYNC_VSCFR || (video_sync_method == VSYNC_AUTO && !(oc->oformat->flags & AVFMT_VARIABLE_FPS)))){av_log(oc, AV_LOG_WARNING, "Frame rate very high for a muxer not efficiently supporting it.\n""Please consider specifying a lower framerate, a different muxer or -vsync 2\n");}for (j = 0; j < ost->forced_kf_count; j++)ost->forced_kf_pts[j] = av_rescale_q(ost->forced_kf_pts[j],AV_TIME_BASE_Q,enc_ctx->time_base);enc_ctx->width = av_buffersink_get_w(ost->filter->filter);enc_ctx->height = av_buffersink_get_h(ost->filter->filter);enc_ctx->sample_aspect_ratio = ost->st->sample_aspect_ratio =ost->frame_aspect_ratio.num ? // overridden by the -aspect cli optionav_mul_q(ost->frame_aspect_ratio, (AVRational){ enc_ctx->height, enc_ctx->width }) :av_buffersink_get_sample_aspect_ratio(ost->filter->filter);if (!strncmp(ost->enc->name, "libx264", 7) &&enc_ctx->pix_fmt == AV_PIX_FMT_NONE &&av_buffersink_get_format(ost->filter->filter) != AV_PIX_FMT_YUV420P)av_log(NULL, AV_LOG_WARNING,"No pixel format specified, %s for H.264 encoding chosen.\n""Use -pix_fmt yuv420p for compatibility with outdated media players.\n",av_get_pix_fmt_name(av_buffersink_get_format(ost->filter->filter)));if (!strncmp(ost->enc->name, "mpeg2video", 10) &&enc_ctx->pix_fmt == AV_PIX_FMT_NONE &&av_buffersink_get_format(ost->filter->filter) != AV_PIX_FMT_YUV420P)av_log(NULL, AV_LOG_WARNING,"No pixel format specified, %s for MPEG-2 encoding chosen.\n""Use -pix_fmt yuv420p for compatibility with outdated media players.\n",av_get_pix_fmt_name(av_buffersink_get_format(ost->filter->filter)));enc_ctx->pix_fmt = av_buffersink_get_format(ost->filter->filter);if (dec_ctx)enc_ctx->bits_per_raw_sample = FFMIN(dec_ctx->bits_per_raw_sample,av_pix_fmt_desc_get(enc_ctx->pix_fmt)->comp[0].depth);enc_ctx->framerate = ost->frame_rate;ost->st->avg_frame_rate = ost->frame_rate;if (!dec_ctx ||enc_ctx->width != dec_ctx->width ||enc_ctx->height != dec_ctx->height ||enc_ctx->pix_fmt != dec_ctx->pix_fmt) {enc_ctx->bits_per_raw_sample = frame_bits_per_raw_sample;}if (ost->forced_keyframes) {if (!strncmp(ost->forced_keyframes, "expr:", 5)) {ret = av_expr_parse(&ost->forced_keyframes_pexpr, ost->forced_keyframes+5,forced_keyframes_const_names, NULL, NULL, NULL, NULL, 0, NULL);if (ret < 0) {av_log(NULL, AV_LOG_ERROR,"Invalid force_key_frames expression '%s'\n", ost->forced_keyframes+5);return ret;}ost->forced_keyframes_expr_const_values[FKF_N] = 0;ost->forced_keyframes_expr_const_values[FKF_N_FORCED] = 0;ost->forced_keyframes_expr_const_values[FKF_PREV_FORCED_N] = NAN;ost->forced_keyframes_expr_const_values[FKF_PREV_FORCED_T] = NAN;// Don't parse the 'forced_keyframes' in case of 'keep-source-keyframes',// parse it only for static kf timings} else if(strncmp(ost->forced_keyframes, "source", 6)) {parse_forced_key_frames(ost->forced_keyframes, ost, ost->enc_ctx);}}break;case AVMEDIA_TYPE_SUBTITLE:enc_ctx->time_base = AV_TIME_BASE_Q;if (!enc_ctx->width) {enc_ctx->width = input_streams[ost->source_index]->st->codecpar->width;enc_ctx->height = input_streams[ost->source_index]->st->codecpar->height;}break;case AVMEDIA_TYPE_DATA:break;default:abort();break;}ost->mux_timebase = enc_ctx->time_base;return 0; }static int init_output_stream(OutputStream *ost, char *error, int error_len) {int ret = 0;if (ost->encoding_needed) {AVCodec *codec = ost->enc;AVCodecContext *dec = NULL;InputStream *ist;ret = init_output_stream_encode(ost);if (ret < 0)return ret;if ((ist = get_input_stream(ost)))dec = ist->dec_ctx;if (dec && dec->subtitle_header) {/* ASS code assumes this buffer is null terminated so add extra byte. */ost->enc_ctx->subtitle_header = av_mallocz(dec->subtitle_header_size + 1);if (!ost->enc_ctx->subtitle_header)return AVERROR(ENOMEM);memcpy(ost->enc_ctx->subtitle_header, dec->subtitle_header, dec->subtitle_header_size);ost->enc_ctx->subtitle_header_size = dec->subtitle_header_size;}if (!av_dict_get(ost->encoder_opts, "threads", NULL, 0))av_dict_set(&ost->encoder_opts, "threads", "auto", 0);if (ost->enc->type == AVMEDIA_TYPE_AUDIO &&!codec->defaults &&!av_dict_get(ost->encoder_opts, "b", NULL, 0) &&!av_dict_get(ost->encoder_opts, "ab", NULL, 0))av_dict_set(&ost->encoder_opts, "b", "128000", 0);if (ost->filter && av_buffersink_get_hw_frames_ctx(ost->filter->filter) &&((AVHWFramesContext*)av_buffersink_get_hw_frames_ctx(ost->filter->filter)->data)->format ==av_buffersink_get_format(ost->filter->filter)) {ost->enc_ctx->hw_frames_ctx = av_buffer_ref(av_buffersink_get_hw_frames_ctx(ost->filter->filter));if (!ost->enc_ctx->hw_frames_ctx)return AVERROR(ENOMEM);}if ((ret = avcodec_open2(ost->enc_ctx, codec, &ost->encoder_opts)) < 0) {if (ret == AVERROR_EXPERIMENTAL)abort_codec_experimental(codec, 1);snprintf(error, error_len,"Error while opening encoder for output stream #%d:%d - ""maybe incorrect parameters such as bit_rate, rate, width or height",ost->file_index, ost->index);return ret;}if (ost->enc->type == AVMEDIA_TYPE_AUDIO &&!(ost->enc->capabilities & AV_CODEC_CAP_VARIABLE_FRAME_SIZE))av_buffersink_set_frame_size(ost->filter->filter,ost->enc_ctx->frame_size);assert_avoptions(ost->encoder_opts);if (ost->enc_ctx->bit_rate && ost->enc_ctx->bit_rate < 1000)av_log(NULL, AV_LOG_WARNING, "The bitrate parameter is set too low."" It takes bits/s as argument, not kbits/s\n");ret = avcodec_parameters_from_context(ost->st->codecpar, ost->enc_ctx);if (ret < 0) {av_log(NULL, AV_LOG_FATAL,"Error initializing the output stream codec context.\n");exit_program(1);}/** FIXME: ost->st->codec should't be needed here anymore.*/ret = avcodec_copy_context(ost->st->codec, ost->enc_ctx);if (ret < 0)return ret;if (ost->enc_ctx->nb_coded_side_data) {int i;ost->st->side_data = av_realloc_array(NULL, ost->enc_ctx->nb_coded_side_data,sizeof(*ost->st->side_data));if (!ost->st->side_data)return AVERROR(ENOMEM);for (i = 0; i < ost->enc_ctx->nb_coded_side_data; i++) {const AVPacketSideData *sd_src = &ost->enc_ctx->coded_side_data[i];AVPacketSideData *sd_dst = &ost->st->side_data[i];sd_dst->data = av_malloc(sd_src->size);if (!sd_dst->data)return AVERROR(ENOMEM);memcpy(sd_dst->data, sd_src->data, sd_src->size);sd_dst->size = sd_src->size;sd_dst->type = sd_src->type;ost->st->nb_side_data++;}}/** Add global input side data. For now this is naive, and copies it* from the input stream's global side data. All side data should* really be funneled over AVFrame and libavfilter, then added back to* packet side data, and then potentially using the first packet for* global side data.*/if (ist) {int i;for (i = 0; i < ist->st->nb_side_data; i++) {AVPacketSideData *sd = &ist->st->side_data[i];uint8_t *dst = av_stream_new_side_data(ost->st, sd->type, sd->size);if (!dst)return AVERROR(ENOMEM);memcpy(dst, sd->data, sd->size);if (ist->autorotate && sd->type == AV_PKT_DATA_DISPLAYMATRIX)av_display_rotation_set((uint32_t *)dst, 0);}}// copy timebase while removing common factorsif (ost->st->time_base.num <= 0 || ost->st->time_base.den <= 0)ost->st->time_base = av_add_q(ost->enc_ctx->time_base, (AVRational){0, 1});// copy estimated duration as a hint to the muxerif (ost->st->duration <= 0 && ist && ist->st->duration > 0)ost->st->duration = av_rescale_q(ist->st->duration, ist->st->time_base, ost->st->time_base);ost->st->codec->codec= ost->enc_ctx->codec;} else if (ost->stream_copy) {ret = init_output_stream_streamcopy(ost);if (ret < 0)return ret;/** FIXME: will the codec context used by the parser during streamcopy* This should go away with the new parser API.*/ret = avcodec_parameters_to_context(ost->parser_avctx, ost->st->codecpar);if (ret < 0)return ret;}// parse user provided disposition, and update stream valuesif (ost->disposition) {static const AVOption opts[] = {{ "disposition" , NULL, 0, AV_OPT_TYPE_FLAGS, { .i64 = 0 }, INT64_MIN, INT64_MAX, .unit = "flags" },{ "default" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_DEFAULT }, .unit = "flags" },{ "dub" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_DUB }, .unit = "flags" },{ "original" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_ORIGINAL }, .unit = "flags" },{ "comment" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_COMMENT }, .unit = "flags" },{ "lyrics" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_LYRICS }, .unit = "flags" },{ "karaoke" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_KARAOKE }, .unit = "flags" },{ "forced" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_FORCED }, .unit = "flags" },{ "hearing_impaired" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_HEARING_IMPAIRED }, .unit = "flags" },{ "visual_impaired" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_VISUAL_IMPAIRED }, .unit = "flags" },{ "clean_effects" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_CLEAN_EFFECTS }, .unit = "flags" },{ "captions" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_CAPTIONS }, .unit = "flags" },{ "descriptions" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_DESCRIPTIONS }, .unit = "flags" },{ "metadata" , NULL, 0, AV_OPT_TYPE_CONST, { .i64 = AV_DISPOSITION_METADATA }, .unit = "flags" },{ NULL },};static const AVClass class = {.class_name = "",.item_name = av_default_item_name,.option = opts,.version = LIBAVUTIL_VERSION_INT,};const AVClass *pclass = &class;ret = av_opt_eval_flags(&pclass, &opts[0], ost->disposition, &ost->st->disposition);if (ret < 0)return ret;}/* initialize bitstream filters for the output stream* needs to be done here, because the codec id for streamcopy is not* known until now */ret = init_output_bsfs(ost);if (ret < 0)return ret;ost->initialized = 1;ret = check_init_output_file(output_files[ost->file_index], ost->file_index);if (ret < 0)return ret;return ret; }static void report_new_stream(int input_index, AVPacket *pkt) {InputFile *file = input_files[input_index];AVStream *st = file->ctx->streams[pkt->stream_index];if (pkt->stream_index < file->nb_streams_warn)return;av_log(file->ctx, AV_LOG_WARNING,"New %s stream %d:%d at pos:%"PRId64" and DTS:%ss\n",av_get_media_type_string(st->codecpar->codec_type),input_index, pkt->stream_index,pkt->pos, av_ts2timestr(pkt->dts, &st->time_base));file->nb_streams_warn = pkt->stream_index + 1; }static int transcode_init(void) {int ret = 0, i, j, k;AVFormatContext *oc;OutputStream *ost;InputStream *ist;char error[1024] = {0};for (i = 0; i < nb_filtergraphs; i++) {FilterGraph *fg = filtergraphs[i];for (j = 0; j < fg->nb_outputs; j++) {OutputFilter *ofilter = fg->outputs[j];if (!ofilter->ost || ofilter->ost->source_index >= 0)continue;if (fg->nb_inputs != 1)continue;for (k = nb_input_streams-1; k >= 0 ; k--)if (fg->inputs[0]->ist == input_streams[k])break;ofilter->ost->source_index = k;}}/* init framerate emulation */for (i = 0; i < nb_input_files; i++) {InputFile *ifile = input_files[i];if (ifile->rate_emu)for (j = 0; j < ifile->nb_streams; j++)input_streams[j + ifile->ist_index]->start = av_gettime_relative();}/* init input streams */for (i = 0; i < nb_input_streams; i++)if ((ret = init_input_stream(i, error, sizeof(error))) < 0) {for (i = 0; i < nb_output_streams; i++) {ost = output_streams[i];avcodec_close(ost->enc_ctx);}goto dump_format;}/* open each encoder */for (i = 0; i < nb_output_streams; i++) {// skip streams fed from filtergraphs until we have a frame for themif (output_streams[i]->filter)continue;ret = init_output_stream(output_streams[i], error, sizeof(error));if (ret < 0)goto dump_format;}/* discard unused programs */for (i = 0; i < nb_input_files; i++) {InputFile *ifile = input_files[i];for (j = 0; j < ifile->ctx->nb_programs; j++) {AVProgram *p = ifile->ctx->programs[j];int discard = AVDISCARD_ALL;for (k = 0; k < p->nb_stream_indexes; k++)if (!input_streams[ifile->ist_index + p->stream_index[k]]->discard) {discard = AVDISCARD_DEFAULT;break;}p->discard = discard;}}/* write headers for files with no streams */for (i = 0; i < nb_output_files; i++) {oc = output_files[i]->ctx;if (oc->oformat->flags & AVFMT_NOSTREAMS && oc->nb_streams == 0) {ret = check_init_output_file(output_files[i], i);if (ret < 0)goto dump_format;}}dump_format:/* dump the stream mapping */av_log(NULL, AV_LOG_INFO, "Stream mapping:\n");for (i = 0; i < nb_input_streams; i++) {ist = input_streams[i];for (j = 0; j < ist->nb_filters; j++) {if (!filtergraph_is_simple(ist->filters[j]->graph)) {av_log(NULL, AV_LOG_INFO, " Stream #%d:%d (%s) -> %s",ist->file_index, ist->st->index, ist->dec ? ist->dec->name : "?",ist->filters[j]->name);if (nb_filtergraphs > 1)av_log(NULL, AV_LOG_INFO, " (graph %d)", ist->filters[j]->graph->index);av_log(NULL, AV_LOG_INFO, "\n");}}}for (i = 0; i < nb_output_streams; i++) {ost = output_streams[i];if (ost->attachment_filename) {/* an attached file */av_log(NULL, AV_LOG_INFO, " File %s -> Stream #%d:%d\n",ost->attachment_filename, ost->file_index, ost->index);continue;}if (ost->filter && !filtergraph_is_simple(ost->filter->graph)) {/* output from a complex graph */av_log(NULL, AV_LOG_INFO, " %s", ost->filter->name);if (nb_filtergraphs > 1)av_log(NULL, AV_LOG_INFO, " (graph %d)", ost->filter->graph->index);av_log(NULL, AV_LOG_INFO, " -> Stream #%d:%d (%s)\n", ost->file_index,ost->index, ost->enc ? ost->enc->name : "?");continue;}av_log(NULL, AV_LOG_INFO, " Stream #%d:%d -> #%d:%d",input_streams[ost->source_index]->file_index,input_streams[ost->source_index]->st->index,ost->file_index,ost->index);if (ost->sync_ist != input_streams[ost->source_index])av_log(NULL, AV_LOG_INFO, " [sync #%d:%d]",ost->sync_ist->file_index,ost->sync_ist->st->index);if (ost->stream_copy)av_log(NULL, AV_LOG_INFO, " (copy)");else {const AVCodec *in_codec = input_streams[ost->source_index]->dec;const AVCodec *out_codec = ost->enc;const char *decoder_name = "?";const char *in_codec_name = "?";const char *encoder_name = "?";const char *out_codec_name = "?";const AVCodecDescriptor *desc;if (in_codec) {decoder_name = in_codec->name;desc = avcodec_descriptor_get(in_codec->id);if (desc)in_codec_name = desc->name;if (!strcmp(decoder_name, in_codec_name))decoder_name = "native";}if (out_codec) {encoder_name = out_codec->name;desc = avcodec_descriptor_get(out_codec->id);if (desc)out_codec_name = desc->name;if (!strcmp(encoder_name, out_codec_name))encoder_name = "native";}av_log(NULL, AV_LOG_INFO, " (%s (%s) -> %s (%s))",in_codec_name, decoder_name,out_codec_name, encoder_name);}av_log(NULL, AV_LOG_INFO, "\n");}if (ret) {av_log(NULL, AV_LOG_ERROR, "%s\n", error);return ret;}atomic_store(&transcode_init_done, 1);return 0; }/* Return 1 if there remain streams where more output is wanted, 0 otherwise. */ static int need_output(void) {int i;for (i = 0; i < nb_output_streams; i++) {OutputStream *ost = output_streams[i];OutputFile *of = output_files[ost->file_index];AVFormatContext *os = output_files[ost->file_index]->ctx;if (ost->finished ||(os->pb && avio_tell(os->pb) >= of->limit_filesize))continue;if (ost->frame_number >= ost->max_frames) {int j;for (j = 0; j < of->ctx->nb_streams; j++)close_output_stream(output_streams[of->ost_index + j]);continue;}return 1;}return 0; }/*** Select the output stream to process.** @return selected output stream, or NULL if none available*/ static OutputStream *choose_output(void) {int i;int64_t opts_min = INT64_MAX;OutputStream *ost_min = NULL;for (i = 0; i < nb_output_streams; i++) {OutputStream *ost = output_streams[i];int64_t opts = ost->st->cur_dts == AV_NOPTS_VALUE ? INT64_MIN :av_rescale_q(ost->st->cur_dts, ost->st->time_base,AV_TIME_BASE_Q);if (ost->st->cur_dts == AV_NOPTS_VALUE)av_log(NULL, AV_LOG_DEBUG, "cur_dts is invalid (this is harmless if it occurs once at the start per stream)\n");if (!ost->initialized && !ost->inputs_done)return ost;if (!ost->finished && opts < opts_min) {opts_min = opts;ost_min = ost->unavailable ? NULL : ost;}}return ost_min; }static void set_tty_echo(int on) { #if HAVE_TERMIOS_Hstruct termios tty;if (tcgetattr(0, &tty) == 0) {if (on) tty.c_lflag |= ECHO;else tty.c_lflag &= ~ECHO;tcsetattr(0, TCSANOW, &tty);} #endif }static int check_keyboard_interaction(int64_t cur_time) {int i, ret, key;static int64_t last_time;if (received_nb_signals)return AVERROR_EXIT;/* read_key() returns 0 on EOF */if(cur_time - last_time >= 100000 && !run_as_daemon){key = read_key();last_time = cur_time;}elsekey = -1;if (key == 'q')return AVERROR_EXIT;if (key == '+') av_log_set_level(av_log_get_level()+10);if (key == '-') av_log_set_level(av_log_get_level()-10);if (key == 's') qp_hist ^= 1;if (key == 'h'){if (do_hex_dump){do_hex_dump = do_pkt_dump = 0;} else if(do_pkt_dump){do_hex_dump = 1;} elsedo_pkt_dump = 1;av_log_set_level(AV_LOG_DEBUG);}if (key == 'c' || key == 'C'){char buf[4096], target[64], command[256], arg[256] = {0};double time;int k, n = 0;fprintf(stderr, "\nEnter command: <target>|all <time>|-1 <command>[ <argument>]\n");i = 0;set_tty_echo(1);while ((k = read_key()) != '\n' && k != '\r' && i < sizeof(buf)-1)if (k > 0)buf[i++] = k;buf[i] = 0;set_tty_echo(0);fprintf(stderr, "\n");if (k > 0 &&(n = sscanf(buf, "%63[^ ] %lf %255[^ ] %255[^\n]", target, &time, command, arg)) >= 3) {av_log(NULL, AV_LOG_DEBUG, "Processing command target:%s time:%f command:%s arg:%s",target, time, command, arg);for (i = 0; i < nb_filtergraphs; i++) {FilterGraph *fg = filtergraphs[i];if (fg->graph) {if (time < 0) {ret = avfilter_graph_send_command(fg->graph, target, command, arg, buf, sizeof(buf),key == 'c' ? AVFILTER_CMD_FLAG_ONE : 0);fprintf(stderr, "Command reply for stream %d: ret:%d res:\n%s", i, ret, buf);} else if (key == 'c') {fprintf(stderr, "Queuing commands only on filters supporting the specific command is unsupported\n");ret = AVERROR_PATCHWELCOME;} else {ret = avfilter_graph_queue_command(fg->graph, target, command, arg, 0, time);if (ret < 0)fprintf(stderr, "Queuing command failed with error %s\n", av_err2str(ret));}}}} else {av_log(NULL, AV_LOG_ERROR,"Parse error, at least 3 arguments were expected, ""only %d given in string '%s'\n", n, buf);}}if (key == 'd' || key == 'D'){int debug=0;if(key == 'D') {debug = input_streams[0]->st->codec->debug<<1;if(!debug) debug = 1;while(debug & (FF_DEBUG_DCT_COEFF|FF_DEBUG_VIS_QP|FF_DEBUG_VIS_MB_TYPE)) //unsupported, would just crashdebug += debug;}else{char buf[32];int k = 0;i = 0;set_tty_echo(1);while ((k = read_key()) != '\n' && k != '\r' && i < sizeof(buf)-1)if (k > 0)buf[i++] = k;buf[i] = 0;set_tty_echo(0);fprintf(stderr, "\n");if (k <= 0 || sscanf(buf, "%d", &debug)!=1)fprintf(stderr,"error parsing debug value\n");}for(i=0;i<nb_input_streams;i++) {input_streams[i]->st->codec->debug = debug;}for(i=0;i<nb_output_streams;i++) {OutputStream *ost = output_streams[i];ost->enc_ctx->debug = debug;}if(debug) av_log_set_level(AV_LOG_DEBUG);fprintf(stderr,"debug=%d\n", debug);}if (key == '?'){fprintf(stderr, "key function\n""? show this help\n""+ increase verbosity\n""- decrease verbosity\n""c Send command to first matching filter supporting it\n""C Send/Queue command to all matching filters\n""D cycle through available debug modes\n""h dump packets/hex press to cycle through the 3 states\n""q quit\n""s Show QP histogram\n");}return 0; }#if HAVE_PTHREADS static void *input_thread(void *arg) {InputFile *f = arg;unsigned flags = f->non_blocking ? AV_THREAD_MESSAGE_NONBLOCK : 0;int ret = 0;while (1) {AVPacket pkt;ret = av_read_frame(f->ctx, &pkt);if (ret == AVERROR(EAGAIN)) {av_usleep(10000);continue;}if (ret < 0) {av_thread_message_queue_set_err_recv(f->in_thread_queue, ret);break;}ret = av_thread_message_queue_send(f->in_thread_queue, &pkt, flags);if (flags && ret == AVERROR(EAGAIN)) {flags = 0;ret = av_thread_message_queue_send(f->in_thread_queue, &pkt, flags);av_log(f->ctx, AV_LOG_WARNING,"Thread message queue blocking; consider raising the ""thread_queue_size option (current value: %d)\n",f->thread_queue_size);}if (ret < 0) {if (ret != AVERROR_EOF)av_log(f->ctx, AV_LOG_ERROR,"Unable to send packet to main thread: %s\n",av_err2str(ret));av_packet_unref(&pkt);av_thread_message_queue_set_err_recv(f->in_thread_queue, ret);break;}}return NULL; }static void free_input_threads(void) {int i;for (i = 0; i < nb_input_files; i++) {InputFile *f = input_files[i];AVPacket pkt;if (!f || !f->in_thread_queue)continue;av_thread_message_queue_set_err_send(f->in_thread_queue, AVERROR_EOF);while (av_thread_message_queue_recv(f->in_thread_queue, &pkt, 0) >= 0)av_packet_unref(&pkt);pthread_join(f->thread, NULL);f->joined = 1;av_thread_message_queue_free(&f->in_thread_queue);} }static int init_input_threads(void) {int i, ret;if (nb_input_files == 1)return 0;for (i = 0; i < nb_input_files; i++) {InputFile *f = input_files[i];if (f->ctx->pb ? !f->ctx->pb->seekable :strcmp(f->ctx->iformat->name, "lavfi"))f->non_blocking = 1;ret = av_thread_message_queue_alloc(&f->in_thread_queue,f->thread_queue_size, sizeof(AVPacket));if (ret < 0)return ret;if ((ret = pthread_create(&f->thread, NULL, input_thread, f))) {av_log(NULL, AV_LOG_ERROR, "pthread_create failed: %s. Try to increase `ulimit -v` or decrease `ulimit -s`.\n", strerror(ret));av_thread_message_queue_free(&f->in_thread_queue);return AVERROR(ret);}}return 0; }static int get_input_packet_mt(InputFile *f, AVPacket *pkt) {return av_thread_message_queue_recv(f->in_thread_queue, pkt,f->non_blocking ?AV_THREAD_MESSAGE_NONBLOCK : 0); } #endifstatic int get_input_packet(InputFile *f, AVPacket *pkt) {if (f->rate_emu) {int i;for (i = 0; i < f->nb_streams; i++) {InputStream *ist = input_streams[f->ist_index + i];int64_t pts = av_rescale(ist->dts, 1000000, AV_TIME_BASE);int64_t now = av_gettime_relative() - ist->start;if (pts > now)return AVERROR(EAGAIN);}}#if HAVE_PTHREADSif (nb_input_files > 1)return get_input_packet_mt(f, pkt); #endifreturn av_read_frame(f->ctx, pkt); }static int got_eagain(void) {int i;for (i = 0; i < nb_output_streams; i++)if (output_streams[i]->unavailable)return 1;return 0; }static void reset_eagain(void) {int i;for (i = 0; i < nb_input_files; i++)input_files[i]->eagain = 0;for (i = 0; i < nb_output_streams; i++)output_streams[i]->unavailable = 0; }// set duration to max(tmp, duration) in a proper time base and return duration's time_base static AVRational duration_max(int64_t tmp, int64_t *duration, AVRational tmp_time_base,AVRational time_base) {int ret;if (!*duration) {*duration = tmp;return tmp_time_base;}ret = av_compare_ts(*duration, time_base, tmp, tmp_time_base);if (ret < 0) {*duration = tmp;return tmp_time_base;}return time_base; }static int seek_to_start(InputFile *ifile, AVFormatContext *is) {InputStream *ist;AVCodecContext *avctx;int i, ret, has_audio = 0;int64_t duration = 0;ret = av_seek_frame(is, -1, is->start_time, 0);if (ret < 0)return ret;for (i = 0; i < ifile->nb_streams; i++) {ist = input_streams[ifile->ist_index + i];avctx = ist->dec_ctx;// flush decodersif (ist->decoding_needed) {process_input_packet(ist, NULL, 1);avcodec_flush_buffers(avctx);}/* duration is the length of the last frame in a stream* when audio stream is present we don't care about* last video frame length because it's not defined exactly */if (avctx->codec_type == AVMEDIA_TYPE_AUDIO && ist->nb_samples)has_audio = 1;}for (i = 0; i < ifile->nb_streams; i++) {ist = input_streams[ifile->ist_index + i];avctx = ist->dec_ctx;if (has_audio) {if (avctx->codec_type == AVMEDIA_TYPE_AUDIO && ist->nb_samples) {AVRational sample_rate = {1, avctx->sample_rate};duration = av_rescale_q(ist->nb_samples, sample_rate, ist->st->time_base);} elsecontinue;} else {if (ist->framerate.num) {duration = av_rescale_q(1, ist->framerate, ist->st->time_base);} else if (ist->st->avg_frame_rate.num) {duration = av_rescale_q(1, ist->st->avg_frame_rate, ist->st->time_base);} else duration = 1;}if (!ifile->duration)ifile->time_base = ist->st->time_base;/* the total duration of the stream, max_pts - min_pts is* the duration of the stream without the last frame */duration += ist->max_pts - ist->min_pts;ifile->time_base = duration_max(duration, &ifile->duration, ist->st->time_base,ifile->time_base);}if (ifile->loop > 0)ifile->loop--;return ret; }/** Return* - 0 -- one packet was read and processed* - AVERROR(EAGAIN) -- no packets were available for selected file,* this function should be called again* - AVERROR_EOF -- this function should not be called again*/ static int process_input(int file_index) {InputFile *ifile = input_files[file_index];AVFormatContext *is;InputStream *ist;AVPacket pkt;int ret, i, j;int64_t duration;int64_t pkt_dts;is = ifile->ctx;ret = get_input_packet(ifile, &pkt);if (ret == AVERROR(EAGAIN)) {ifile->eagain = 1;return ret;}if (ret < 0 && ifile->loop) {if ((ret = seek_to_start(ifile, is)) < 0)return ret;ret = get_input_packet(ifile, &pkt);if (ret == AVERROR(EAGAIN)) {ifile->eagain = 1;return ret;}}if (ret < 0) {if (ret != AVERROR_EOF) {print_error(is->filename, ret);if (exit_on_error)exit_program(1);}for (i = 0; i < ifile->nb_streams; i++) {ist = input_streams[ifile->ist_index + i];if (ist->decoding_needed) {ret = process_input_packet(ist, NULL, 0);if (ret>0)return 0;}/* mark all outputs that don't go through lavfi as finished */for (j = 0; j < nb_output_streams; j++) {OutputStream *ost = output_streams[j];if (ost->source_index == ifile->ist_index + i &&(ost->stream_copy || ost->enc->type == AVMEDIA_TYPE_SUBTITLE))finish_output_stream(ost);}}ifile->eof_reached = 1;return AVERROR(EAGAIN);}reset_eagain();if (do_pkt_dump) {av_pkt_dump_log2(NULL, AV_LOG_INFO, &pkt, do_hex_dump,is->streams[pkt.stream_index]);}/* the following test is needed in case new streams appeardynamically in stream : we ignore them */if (pkt.stream_index >= ifile->nb_streams) {report_new_stream(file_index, &pkt);goto discard_packet;}ist = input_streams[ifile->ist_index + pkt.stream_index];ist->data_size += pkt.size;ist->nb_packets++;if (ist->discard)goto discard_packet;if (exit_on_error && (pkt.flags & AV_PKT_FLAG_CORRUPT)) {av_log(NULL, AV_LOG_FATAL, "%s: corrupt input packet in stream %d\n", is->filename, pkt.stream_index);exit_program(1);}if (debug_ts) {av_log(NULL, AV_LOG_INFO, "demuxer -> ist_index:%d type:%s ""next_dts:%s next_dts_time:%s next_pts:%s next_pts_time:%s pkt_pts:%s pkt_pts_time:%s pkt_dts:%s pkt_dts_time:%s off:%s off_time:%s\n",ifile->ist_index + pkt.stream_index, av_get_media_type_string(ist->dec_ctx->codec_type),av_ts2str(ist->next_dts), av_ts2timestr(ist->next_dts, &AV_TIME_BASE_Q),av_ts2str(ist->next_pts), av_ts2timestr(ist->next_pts, &AV_TIME_BASE_Q),av_ts2str(pkt.pts), av_ts2timestr(pkt.pts, &ist->st->time_base),av_ts2str(pkt.dts), av_ts2timestr(pkt.dts, &ist->st->time_base),av_ts2str(input_files[ist->file_index]->ts_offset),av_ts2timestr(input_files[ist->file_index]->ts_offset, &AV_TIME_BASE_Q));}if(!ist->wrap_correction_done && is->start_time != AV_NOPTS_VALUE && ist->st->pts_wrap_bits < 64){int64_t stime, stime2;// Correcting starttime based on the enabled streams// FIXME this ideally should be done before the first use of starttime but we do not know which are the enabled streams at that point.// so we instead do it here as part of discontinuity handlingif ( ist->next_dts == AV_NOPTS_VALUE&& ifile->ts_offset == -is->start_time&& (is->iformat->flags & AVFMT_TS_DISCONT)) {int64_t new_start_time = INT64_MAX;for (i=0; i<is->nb_streams; i++) {AVStream *st = is->streams[i];if(st->discard == AVDISCARD_ALL || st->start_time == AV_NOPTS_VALUE)continue;new_start_time = FFMIN(new_start_time, av_rescale_q(st->start_time, st->time_base, AV_TIME_BASE_Q));}if (new_start_time > is->start_time) {av_log(is, AV_LOG_VERBOSE, "Correcting start time by %"PRId64"\n", new_start_time - is->start_time);ifile->ts_offset = -new_start_time;}}stime = av_rescale_q(is->start_time, AV_TIME_BASE_Q, ist->st->time_base);stime2= stime + (1ULL<<ist->st->pts_wrap_bits);ist->wrap_correction_done = 1;if(stime2 > stime && pkt.dts != AV_NOPTS_VALUE && pkt.dts > stime + (1LL<<(ist->st->pts_wrap_bits-1))) {pkt.dts -= 1ULL<<ist->st->pts_wrap_bits;ist->wrap_correction_done = 0;}if(stime2 > stime && pkt.pts != AV_NOPTS_VALUE && pkt.pts > stime + (1LL<<(ist->st->pts_wrap_bits-1))) {pkt.pts -= 1ULL<<ist->st->pts_wrap_bits;ist->wrap_correction_done = 0;}}/* add the stream-global side data to the first packet */if (ist->nb_packets == 1) {for (i = 0; i < ist->st->nb_side_data; i++) {AVPacketSideData *src_sd = &ist->st->side_data[i];uint8_t *dst_data;if (src_sd->type == AV_PKT_DATA_DISPLAYMATRIX)continue;if (av_packet_get_side_data(&pkt, src_sd->type, NULL))continue;dst_data = av_packet_new_side_data(&pkt, src_sd->type, src_sd->size);if (!dst_data)exit_program(1);memcpy(dst_data, src_sd->data, src_sd->size);}}if (pkt.dts != AV_NOPTS_VALUE)pkt.dts += av_rescale_q(ifile->ts_offset, AV_TIME_BASE_Q, ist->st->time_base);if (pkt.pts != AV_NOPTS_VALUE)pkt.pts += av_rescale_q(ifile->ts_offset, AV_TIME_BASE_Q, ist->st->time_base);if (pkt.pts != AV_NOPTS_VALUE)pkt.pts *= ist->ts_scale;if (pkt.dts != AV_NOPTS_VALUE)pkt.dts *= ist->ts_scale;pkt_dts = av_rescale_q_rnd(pkt.dts, ist->st->time_base, AV_TIME_BASE_Q, AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX);if ((ist->dec_ctx->codec_type == AVMEDIA_TYPE_VIDEO ||ist->dec_ctx->codec_type == AVMEDIA_TYPE_AUDIO) &&pkt_dts != AV_NOPTS_VALUE && ist->next_dts == AV_NOPTS_VALUE && !copy_ts&& (is->iformat->flags & AVFMT_TS_DISCONT) && ifile->last_ts != AV_NOPTS_VALUE) {int64_t delta = pkt_dts - ifile->last_ts;if (delta < -1LL*dts_delta_threshold*AV_TIME_BASE ||delta > 1LL*dts_delta_threshold*AV_TIME_BASE){ifile->ts_offset -= delta;av_log(NULL, AV_LOG_DEBUG,"Inter stream timestamp discontinuity %"PRId64", new offset= %"PRId64"\n",delta, ifile->ts_offset);pkt.dts -= av_rescale_q(delta, AV_TIME_BASE_Q, ist->st->time_base);if (pkt.pts != AV_NOPTS_VALUE)pkt.pts -= av_rescale_q(delta, AV_TIME_BASE_Q, ist->st->time_base);}}duration = av_rescale_q(ifile->duration, ifile->time_base, ist->st->time_base);if (pkt.pts != AV_NOPTS_VALUE) {pkt.pts += duration;ist->max_pts = FFMAX(pkt.pts, ist->max_pts);ist->min_pts = FFMIN(pkt.pts, ist->min_pts);}if (pkt.dts != AV_NOPTS_VALUE)pkt.dts += duration;pkt_dts = av_rescale_q_rnd(pkt.dts, ist->st->time_base, AV_TIME_BASE_Q, AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX);if ((ist->dec_ctx->codec_type == AVMEDIA_TYPE_VIDEO ||ist->dec_ctx->codec_type == AVMEDIA_TYPE_AUDIO) &&pkt_dts != AV_NOPTS_VALUE && ist->next_dts != AV_NOPTS_VALUE &&!copy_ts) {int64_t delta = pkt_dts - ist->next_dts;if (is->iformat->flags & AVFMT_TS_DISCONT) {if (delta < -1LL*dts_delta_threshold*AV_TIME_BASE ||delta > 1LL*dts_delta_threshold*AV_TIME_BASE ||pkt_dts + AV_TIME_BASE/10 < FFMAX(ist->pts, ist->dts)) {ifile->ts_offset -= delta;av_log(NULL, AV_LOG_DEBUG,"timestamp discontinuity %"PRId64", new offset= %"PRId64"\n",delta, ifile->ts_offset);pkt.dts -= av_rescale_q(delta, AV_TIME_BASE_Q, ist->st->time_base);if (pkt.pts != AV_NOPTS_VALUE)pkt.pts -= av_rescale_q(delta, AV_TIME_BASE_Q, ist->st->time_base);}} else {if ( delta < -1LL*dts_error_threshold*AV_TIME_BASE ||delta > 1LL*dts_error_threshold*AV_TIME_BASE) {av_log(NULL, AV_LOG_WARNING, "DTS %"PRId64", next:%"PRId64" st:%d invalid dropping\n", pkt.dts, ist->next_dts, pkt.stream_index);pkt.dts = AV_NOPTS_VALUE;}if (pkt.pts != AV_NOPTS_VALUE){int64_t pkt_pts = av_rescale_q(pkt.pts, ist->st->time_base, AV_TIME_BASE_Q);delta = pkt_pts - ist->next_dts;if ( delta < -1LL*dts_error_threshold*AV_TIME_BASE ||delta > 1LL*dts_error_threshold*AV_TIME_BASE) {av_log(NULL, AV_LOG_WARNING, "PTS %"PRId64", next:%"PRId64" invalid dropping st:%d\n", pkt.pts, ist->next_dts, pkt.stream_index);pkt.pts = AV_NOPTS_VALUE;}}}}if (pkt.dts != AV_NOPTS_VALUE)ifile->last_ts = av_rescale_q(pkt.dts, ist->st->time_base, AV_TIME_BASE_Q);if (debug_ts) {av_log(NULL, AV_LOG_INFO, "demuxer+ffmpeg -> ist_index:%d type:%s pkt_pts:%s pkt_pts_time:%s pkt_dts:%s pkt_dts_time:%s off:%s off_time:%s\n",ifile->ist_index + pkt.stream_index, av_get_media_type_string(ist->dec_ctx->codec_type),av_ts2str(pkt.pts), av_ts2timestr(pkt.pts, &ist->st->time_base),av_ts2str(pkt.dts), av_ts2timestr(pkt.dts, &ist->st->time_base),av_ts2str(input_files[ist->file_index]->ts_offset),av_ts2timestr(input_files[ist->file_index]->ts_offset, &AV_TIME_BASE_Q));}sub2video_heartbeat(ist, pkt.pts);process_input_packet(ist, &pkt, 0);discard_packet:av_packet_unref(&pkt);return 0; }/*** Perform a step of transcoding for the specified filter graph.** @param[in] graph filter graph to consider* @param[out] best_ist input stream where a frame would allow to continue* @return 0 for success, <0 for error*/ static int transcode_from_filter(FilterGraph *graph, InputStream **best_ist) {int i, ret;int nb_requests, nb_requests_max = 0;InputFilter *ifilter;InputStream *ist;*best_ist = NULL;ret = avfilter_graph_request_oldest(graph->graph);if (ret >= 0)return reap_filters(0);if (ret == AVERROR_EOF) {ret = reap_filters(1);for (i = 0; i < graph->nb_outputs; i++)close_output_stream(graph->outputs[i]->ost);return ret;}if (ret != AVERROR(EAGAIN))return ret;for (i = 0; i < graph->nb_inputs; i++) {ifilter = graph->inputs[i];ist = ifilter->ist;if (input_files[ist->file_index]->eagain ||input_files[ist->file_index]->eof_reached)continue;nb_requests = av_buffersrc_get_nb_failed_requests(ifilter->filter);if (nb_requests > nb_requests_max) {nb_requests_max = nb_requests;*best_ist = ist;}}if (!*best_ist)for (i = 0; i < graph->nb_outputs; i++)graph->outputs[i]->ost->unavailable = 1;return 0; }/*** Run a single step of transcoding.** @return 0 for success, <0 for error*/ static int transcode_step(void) {OutputStream *ost;InputStream *ist = NULL;int ret;ost = choose_output();if (!ost) {if (got_eagain()) {reset_eagain();av_usleep(10000);return 0;}av_log(NULL, AV_LOG_VERBOSE, "No more inputs to read from, finishing.\n");return AVERROR_EOF;}if (ost->filter && !ost->filter->graph->graph) {if (ifilter_has_all_input_formats(ost->filter->graph)) {ret = configure_filtergraph(ost->filter->graph);if (ret < 0) {av_log(NULL, AV_LOG_ERROR, "Error reinitializing filters!\n");return ret;}}}if (ost->filter && ost->filter->graph->graph) {if ((ret = transcode_from_filter(ost->filter->graph, &ist)) < 0)return ret;if (!ist)return 0;} else if (ost->filter) {int i;for (i = 0; i < ost->filter->graph->nb_inputs; i++) {InputFilter *ifilter = ost->filter->graph->inputs[i];if (!ifilter->ist->got_output && !input_files[ifilter->ist->file_index]->eof_reached) {ist = ifilter->ist;break;}}if (!ist) {ost->inputs_done = 1;return 0;}} else {av_assert0(ost->source_index >= 0);ist = input_streams[ost->source_index];}ret = process_input(ist->file_index);if (ret == AVERROR(EAGAIN)) {if (input_files[ist->file_index]->eagain)ost->unavailable = 1;return 0;}if (ret < 0)return ret == AVERROR_EOF ? 0 : ret;return reap_filters(0); }/** The following code is the main loop of the file converter*/ static int transcode(void(call_back)(int current,int total)) {int ret, i;AVFormatContext *os;OutputStream *ost;InputStream *ist;int64_t timer_start;int64_t total_packets_written = 0;ret = transcode_init();if (ret < 0)goto fail;if (stdin_interaction) {av_log(NULL, AV_LOG_INFO, "Press [q] to stop, [?] for help\n");}timer_start = av_gettime_relative();#if HAVE_PTHREADSif ((ret = init_input_threads()) < 0)goto fail; #endifwhile (!received_sigterm) {int64_t cur_time= av_gettime_relative();/* if 'q' pressed, exits */if (stdin_interaction)if (check_keyboard_interaction(cur_time) < 0)break;/* check if there's any stream where output is still needed */if (!need_output()) {av_log(NULL, AV_LOG_VERBOSE, "No more output streams to write to, finishing.\n");break;}ret = transcode_step();if (ret < 0 && ret != AVERROR_EOF) {char errbuf[128];av_strerror(ret, errbuf, sizeof(errbuf));av_log(NULL, AV_LOG_ERROR, "Error while filtering: %s\n", errbuf);break;}/* dump report by using the output first video and audio streams */print_report(0, timer_start, cur_time,call_back);} #if HAVE_PTHREADSfree_input_threads(); #endif/* at the end of stream, we must flush the decoder buffers */for (i = 0; i < nb_input_streams; i++) {ist = input_streams[i];if (!input_files[ist->file_index]->eof_reached && ist->decoding_needed) {process_input_packet(ist, NULL, 0);}}flush_encoders();term_exit();/* write the trailer if needed and close file */for (i = 0; i < nb_output_files; i++) {os = output_files[i]->ctx;if (!output_files[i]->header_written) {av_log(NULL, AV_LOG_ERROR,"Nothing was written into output file %d (%s), because ""at least one of its streams received no packets.\n",i, os->filename);continue;}if ((ret = av_write_trailer(os)) < 0) {av_log(NULL, AV_LOG_ERROR, "Error writing trailer of %s: %s\n", os->filename, av_err2str(ret));if (exit_on_error)exit_program(1);}}/* dump report by using the first video and audio streams */print_report(1, timer_start, av_gettime_relative(),call_back);/* close each encoder */for (i = 0; i < nb_output_streams; i++) {ost = output_streams[i];if (ost->encoding_needed) {av_freep(&ost->enc_ctx->stats_in);}total_packets_written += ost->packets_written;}if (!total_packets_written && (abort_on_flags & ABORT_ON_FLAG_EMPTY_OUTPUT)) {av_log(NULL, AV_LOG_FATAL, "Empty output\n");exit_program(1);}/* close each decoder */for (i = 0; i < nb_input_streams; i++) {ist = input_streams[i];if (ist->decoding_needed) {avcodec_close(ist->dec_ctx);if (ist->hwaccel_uninit)ist->hwaccel_uninit(ist->dec_ctx);}}av_buffer_unref(&hw_device_ctx);/* finished ! */ret = 0;fail: #if HAVE_PTHREADSfree_input_threads(); #endifif (output_streams) {for (i = 0; i < nb_output_streams; i++) {ost = output_streams[i];if (ost) {if (ost->logfile) {if (fclose(ost->logfile))av_log(NULL, AV_LOG_ERROR,"Error closing logfile, loss of information possible: %s\n",av_err2str(AVERROR(errno)));ost->logfile = NULL;}av_freep(&ost->forced_kf_pts);av_freep(&ost->apad);av_freep(&ost->disposition);av_dict_free(&ost->encoder_opts);av_dict_free(&ost->sws_dict);av_dict_free(&ost->swr_opts);av_dict_free(&ost->resample_opts);}}}return ret; }static int64_t getutime(void) { #if HAVE_GETRUSAGEstruct rusage rusage;getrusage(RUSAGE_SELF, &rusage);return (rusage.ru_utime.tv_sec * 1000000LL) + rusage.ru_utime.tv_usec; #elif HAVE_GETPROCESSTIMESHANDLE proc;FILETIME c, e, k, u;proc = GetCurrentProcess();GetProcessTimes(proc, &c, &e, &k, &u);return ((int64_t) u.dwHighDateTime << 32 | u.dwLowDateTime) / 10; #elsereturn av_gettime_relative(); #endif }static int64_t getmaxrss(void) { #if HAVE_GETRUSAGE && HAVE_STRUCT_RUSAGE_RU_MAXRSSstruct rusage rusage;getrusage(RUSAGE_SELF, &rusage);return (int64_t)rusage.ru_maxrss * 1024; #elif HAVE_GETPROCESSMEMORYINFOHANDLE proc;PROCESS_MEMORY_COUNTERS memcounters;proc = GetCurrentProcess();memcounters.cb = sizeof(memcounters);GetProcessMemoryInfo(proc, &memcounters, sizeof(memcounters));return memcounters.PeakPagefileUsage; #elsereturn 0; #endif }static void log_callback_null(void *ptr, int level, const char *fmt, va_list vl) { } //命令函數的入口 int run_ffmpeg_command(int argc, char **argv,void(call_back)(int current,int total)) {int i, ret;int64_t ti;init_dynload();register_exit(ffmpeg_cleanup);setvbuf(stderr,NULL,_IONBF,0); /* win32 runtime needs this */av_log_set_flags(AV_LOG_SKIP_REPEATED);parse_loglevel(argc, argv, options);if(argc>1 && !strcmp(argv[1], "-d")){run_as_daemon=1;av_log_set_callback(log_callback_null);argc--;argv++;}avcodec_register_all(); #if CONFIG_AVDEVICEavdevice_register_all(); #endifavfilter_register_all();av_register_all();avformat_network_init();show_banner(argc, argv, options);/* parse options and open all input/output files */ret = ffmpeg_parse_options(argc, argv);if (ret < 0)exit_program(1);if (nb_output_files <= 0 && nb_input_files == 0) {show_usage();av_log(NULL, AV_LOG_WARNING, "Use -h to get full help or, even better, run 'man %s'\n", program_name);exit_program(1);}/* file converter / grab */if (nb_output_files <= 0) {av_log(NULL, AV_LOG_FATAL, "At least one output file must be specified\n");exit_program(1);}// if (nb_input_files == 0) { // av_log(NULL, AV_LOG_FATAL, "At least one input file must be specified\n"); // exit_program(1); // }for (i = 0; i < nb_output_files; i++) {if (strcmp(output_files[i]->ctx->oformat->name, "rtp"))want_sdp = 0;}current_time = ti = getutime();if (transcode(call_back) < 0)exit_program(1);ti = getutime() - ti;if (do_benchmark) {av_log(NULL, AV_LOG_INFO, "bench: utime=%0.3fs\n", ti / 1000000.0);}av_log(NULL, AV_LOG_DEBUG, "%"PRIu64" frames successfully decoded, %"PRIu64" decoding errors\n",decode_error_stat[0], decode_error_stat[1]);if ((decode_error_stat[0] + decode_error_stat[1]) * max_error_rate < decode_error_stat[1])exit_program(69);//注釋掉此行代碼 如果不注釋掉該行代碼,命令執行完會導致app退出//exit_program(received_nb_signals ? 255 : main_return_code);//添加如下代碼如果不添加再次執行run_ffmpeg_command 直接崩潰 因為沒有賦初始值nb_filtergraphs = 0;nb_input_streams = 0;nb_input_files = 0;progress_avio = NULL;input_streams = NULL;nb_input_streams = 0;input_files = NULL;nb_input_files = 0;output_streams = NULL;nb_output_streams = 0;output_files = NULL;nb_output_files = 0;return main_return_code; }

修改native-lib.cpp內容

?

?native-lib.cpp內容如下:

#include <jni.h> #include <string> #include <android/log.h> #define TAG "JNI_TAG" #define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,TAG,__VA_ARGS__); extern "C" { #include "libavutil/avutil.h"//聲明方法 argc命令的個數 argv 二維數組 int run_ffmpeg_command(int argc, char **argv,void(call_back)(int current,int total)); //回調函數 static jobject call_back_jobj;//call_back_jobj 對應于VideoCompress類中的CompressCallback static JNIEnv *myEnv; void call_back(int current,int total){//LOGE("壓縮進度:%d/%d",current,total);//把進度回調出去,對象是jobject callbackif(myEnv!=NULL&&call_back_jobj!=NULL){//獲取j_mid是會被多次執行jclass j_clazz=myEnv->GetObjectClass(call_back_jobj);//方法 onCompress對象的方法名 對應VideoCompress類中的CompressCallback接口的onCompress方法//(II)V(方法簽名 利用javap命令也能打印) --(public void onCompress(int current,int total);onCompress的兩個參數都為int類型 onCompress方法的返回值為void)jmethodID j_mid=myEnv->GetMethodID(j_clazz,"onCompress","(II)V");//調用對象的方法 如果寫CallObjectMethod報錯,因為知道是無返回值的方法,所以改成CallVoidMethodmyEnv->CallVoidMethod(call_back_jobj,j_mid,current,total);}} } extern "C" JNIEXPORT jstring JNICALL Java_com_suoer_ndk_ffmpegtestapplication_MainActivity_stringFromJNI(JNIEnv* env,jobject /* this */) {//std::string hello = "Hello from C++";return env->NewStringUTF(av_version_info()); }extern "C" JNIEXPORT void JNICALL Java_com_suoer_ndk_ffmpegtestapplication_VideoCompress_compressVideo(JNIEnv *env, jobject thiz,jobjectArray compress_command,jobject callback) {myEnv=env;call_back_jobj=env->NewGlobalRef(callback);//ffmpeg 處理視頻壓縮//arm這個里面的so都是用來處理音視頻的,include都是頭文件//還有幾個沒有被打包編譯成so,因為這些不算是音視頻的處理代碼,只是我們現在支持命令(封裝)//1.獲取命令個數int argc=env->GetArrayLength(compress_command);//2.給char **argv填充數據char **argv=(char **)malloc(sizeof(char*)*argc);for (int i = 0; i <argc ; ++i) {jstring j_param=(jstring)env->GetObjectArrayElement(compress_command,i);argv[i]= (char *)(env->GetStringUTFChars(j_param, NULL));LOGE("參數:%s",argv[i]);}//3.調用命令函數去壓縮run_ffmpeg_command(argc,argv,call_back);//4.釋放內存for (int i = 0; i <argc ; ++i) {free(argv[i]);}free(argv);env->DeleteGlobalRef(call_back_jobj); }

5.MainActivity界面優化處理

首先需要把assets目錄中的test.mp4放置手機中

?

?修改MainActivity內容:

壓縮之前首先判斷壓縮后的文件out.mp4是否存在,如果已經存在,則刪除此文件。如果壓縮文件已經存在再次壓縮會導致崩潰。

?

?

壓縮過程中展示進度條,壓縮過程中不允許再次點擊壓縮按鈕

?

進度彈窗背景:?bg_4radius.xml

<?xml version="1.0" encoding="utf-8"?> <selector xmlns:android="http://schemas.android.com/apk/res/android"><item><shape android:shape="rectangle"><solid android:color="@color/white" /><corners android:topLeftRadius="4dp" android:topRightRadius="4dp" android:bottomLeftRadius="4dp" android:bottomRightRadius="4dp"/></shape></item> </selector>

進度條進度背景:compress_progressbar.xml

<?xml version="1.0" encoding="utf-8"?> <layer-list xmlns:android="http://schemas.android.com/apk/res/android"><item android:id="@android:id/background"><!--形狀--><shape><!-- 角球--><corners android:radius="6dp" /><solid android:color="#ffeeeeee"></solid></shape></item><item android:id="@android:id/secondaryProgress"><clip><shape><corners android:radius="6dp" /><gradientandroid:type="linear"android:useLevel="true"android:angle="270"android:endColor="@color/purple_500"android:startColor="@color/purple_200" /></shape></clip></item><item android:id="@android:id/progress"><clip><shape><corners android:radius="6dp" /><gradientandroid:type="linear"android:useLevel="true"android:angle="270"android:endColor="@color/purple_500"android:startColor="@color/purple_200" /></shape></clip></item> </layer-list>

進度彈窗布局文件dialog_progress.xml

<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="wrap_content"android:layout_height="320dp"android:layout_gravity="center"android:background="@drawable/bg_4radius"><RelativeLayoutandroid:layout_width="match_parent"android:layout_height="match_parent"><LinearLayoutandroid:layout_centerInParent="true"android:layout_width="match_parent"android:layout_height="wrap_content"android:gravity="center_horizontal"android:orientation="vertical"><TextViewandroid:id="@+id/tv_dialog_compress_progress"android:layout_width="wrap_content"android:layout_height="wrap_content"android:textColor="@color/purple_500"android:textSize="24sp" /><ProgressBarandroid:id="@+id/compress_progressbar"android:layout_marginLeft="40dp"android:layout_marginRight="40dp"style="?android:attr/progressBarStyleHorizontal"android:layout_width="match_parent"android:max="100"android:min="0"android:progressDrawable="@drawable/compress_progressbar"android:layout_height="12dp" /><TextViewandroid:layout_marginBottom="20dp"android:text="視頻正在壓縮,請稍等…"android:layout_marginTop="40dp"android:textSize="14sp"android:textColor="#CCCCCC"android:layout_width="wrap_content"android:layout_height="wrap_content" /></LinearLayout></RelativeLayout></RelativeLayout>

彈窗樣式styles.xml

<?xml version="1.0" encoding="utf-8"?> <resources><style name="normal_dialog" parent="android:Theme.Dialog"><item name="android:background">@android:color/transparent</item><item name="android:windowBackground">@android:color/transparent</item><item name="android:windowNoTitle">true</item><item name="android:backgroundDimAmount">0.8</item></style> </resources>

彈窗ProgressDialog.java

package com.suoer.ndk.ffmpegtestapplication;import android.app.Dialog; import android.content.Context; import android.widget.ProgressBar; import android.widget.TextView;import androidx.annotation.NonNull;public class ProgressDialog extends Dialog {private TextView tvProgress;private ProgressBar compress_progressbar;public ProgressDialog(@NonNull Context context) {this(context, R.style.normal_dialog);setContentView(R.layout.dialog_progress);tvProgress = findViewById(R.id.tv_dialog_compress_progress);compress_progressbar =findViewById(R.id.compress_progressbar);}private ProgressDialog(@NonNull Context context, int themeResId) {super(context, themeResId);}public void setProgress(int progress) {if (progress < 0) {progress = 0;} else if (progress > 100) {progress = 100;}tvProgress.setText(progress + "%");compress_progressbar.setProgress(progress);} }

最終的MainActivity.java

package com.suoer.ndk.ffmpegtestapplication;import android.Manifest; import android.os.Bundle; import android.os.Environment; import android.util.Log; import android.view.View; import android.widget.TextView; import android.widget.Toast;import com.tbruyelle.rxpermissions3.RxPermissions;import java.io.File;import androidx.appcompat.app.AppCompatActivity; import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers; import io.reactivex.rxjava3.core.Observable; import io.reactivex.rxjava3.functions.Consumer; import io.reactivex.rxjava3.functions.Function; import io.reactivex.rxjava3.schedulers.Schedulers;public class MainActivity extends AppCompatActivity {private File mInFile=new File(Environment.getExternalStorageDirectory(),"test.mp4");//mInFile 需要壓縮的文件路徑private File mOutFile=new File(Environment.getExternalStorageDirectory(),"out.mp4");//mOutFile壓縮后的文件路徑private ProgressDialog mProgressDialog;//進度彈窗// Used to load the 'native-lib' library on application startup.static {System.loadLibrary("native-lib");}@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);if (mProgressDialog == null) {mProgressDialog = new ProgressDialog(this);}mProgressDialog.setCancelable(false);mProgressDialog.setProgress(0);// Example of a call to a native methodTextView tv = findViewById(R.id.sample_text);//tv.setText("ffmpeg版本:"+stringFromJNI());tv.setText("壓縮");//tv的點擊事件 點擊按鈕實現視頻壓縮tv.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {// 壓縮文件 需要讀寫文件權限 申請權限RxPermissions rxPermissions=new RxPermissions(MainActivity.this);rxPermissions.request(Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE).subscribe(new Consumer<Boolean>() {@Overridepublic void accept(Boolean aBoolean) throws Throwable {if(aBoolean){//權限已經獲取 壓縮視頻//如果手機中不存在test.mp4的文件則提示if(!mInFile.exists()){Toast.makeText(MainActivity.this,"請把assert目錄中的test.mp4拷貝至手機中!",Toast.LENGTH_LONG).show();return;}if(mOutFile.exists()){mOutFile.delete();}compressVideo();}}});}});}/*** 開啟子線程處理耗時壓縮問題*/private void compressVideo() {//ffmpeg的壓縮命令:ffmpeg -i test.mp4 -b:v 1024k out.mp4//ffmpeg -i test.mp4 -b:v 1024k out.mp4//-b:v 1024k 1024k為碼率 碼率越高視頻越清晰,而且視頻越大//test.mp4需要壓縮的文件//out.mp4 壓縮之后的文件if(mProgressDialog!=null&&!mProgressDialog.isShowing()){mProgressDialog.show();}String[] compressCommand={"ffmpeg","-i",mInFile.getAbsolutePath(),"-b:v","1024k",mOutFile.getAbsolutePath()};//壓縮是耗時的,需要子線程處理Observable.just(compressCommand).map(new Function<String[], File>() {@Overridepublic File apply(String[] compressCommand) throws Throwable {//子線程VideoCompress videoCompress=new VideoCompress();videoCompress.compressVideo(compressCommand, new VideoCompress.CompressCallback() {@Overridepublic void onCompress(int current, int total) {Log.e("TAG", "onCompress: 壓縮進度:"+current+"/"+total);runOnUiThread(new Runnable() {@Overridepublic void run() {if(mProgressDialog!=null){mProgressDialog.setProgress(100 * current / total);}}});}});return mOutFile;}}).subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread()).subscribe(new Consumer<File>() {@Overridepublic void accept(File file) throws Throwable {// 主線程 壓縮完成Log.e("TAG", "accept: 壓縮完成!" );if(mProgressDialog!=null){mProgressDialog.dismiss();}}});}/*** A native method that is implemented by the 'native-lib' native library,* which is packaged with this application.*/public native String stringFromJNI();@Overrideprotected void onDestroy() {super.onDestroy();if (mProgressDialog != null) {mProgressDialog.cancel();mProgressDialog = null;}} }

總結

以上是生活随笔為你收集整理的Android 使用FFmpeg3.3.9基于命令实现视频压缩的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

国产亚洲精品久久久久久 | 高潮毛片无遮挡高清免费视频 | 久久精品人人做人人综合试看 | 亚洲熟女一区二区三区 | 国产激情精品一区二区三区 | 国产舌乚八伦偷品w中 | 国产免费无码一区二区视频 | 国产精品久久久久久久影院 | 国产av人人夜夜澡人人爽麻豆 | 免费视频欧美无人区码 | 国产精品无码成人午夜电影 | 98国产精品综合一区二区三区 | 色欲av亚洲一区无码少妇 | 人妻与老人中文字幕 | 中文久久乱码一区二区 | 国产人妻大战黑人第1集 | 在线观看国产午夜福利片 | 伊人久久大香线蕉av一区二区 | av香港经典三级级 在线 | av人摸人人人澡人人超碰下载 | 免费人成网站视频在线观看 | 内射后入在线观看一区 | 丰满肥臀大屁股熟妇激情视频 | 亚洲国产av精品一区二区蜜芽 | 乌克兰少妇xxxx做受 | 牲欲强的熟妇农村老妇女 | 色狠狠av一区二区三区 | 日欧一片内射va在线影院 | 亚洲爆乳大丰满无码专区 | 98国产精品综合一区二区三区 | 中文无码精品a∨在线观看不卡 | 乱码午夜-极国产极内射 | 亚洲熟悉妇女xxx妇女av | 人人妻人人澡人人爽人人精品 | 中文久久乱码一区二区 | 国产成人无码a区在线观看视频app | 久久精品女人天堂av免费观看 | 欧美怡红院免费全部视频 | 乱中年女人伦av三区 | 成人一区二区免费视频 | 国产明星裸体无码xxxx视频 | 国产人妻精品一区二区三区 | 300部国产真实乱 | 亚洲国产av精品一区二区蜜芽 | 欧美性生交活xxxxxdddd | 久久综合狠狠综合久久综合88 | 国产精品久久久久影院嫩草 | 国语自产偷拍精品视频偷 | 天堂久久天堂av色综合 | 成在人线av无码免费 | 午夜丰满少妇性开放视频 | 人人妻人人澡人人爽精品欧美 | 波多野结衣av一区二区全免费观看 | 欧美人与禽zoz0性伦交 | 亚洲日本一区二区三区在线 | 又紧又大又爽精品一区二区 | 野外少妇愉情中文字幕 | 国产在热线精品视频 | 131美女爱做视频 | 真人与拘做受免费视频一 | 亚洲中文字幕在线无码一区二区 | av无码电影一区二区三区 | 国产一区二区不卡老阿姨 | 人人妻人人澡人人爽欧美一区九九 | 无码人妻出轨黑人中文字幕 | 国产精品资源一区二区 | 国产精品国产自线拍免费软件 | 久久精品中文闷骚内射 | 人人妻人人澡人人爽欧美精品 | 久久精品中文闷骚内射 | 成人免费无码大片a毛片 | 国产sm调教视频在线观看 | 天天拍夜夜添久久精品大 | 亚洲国产精品久久久久久 | 午夜丰满少妇性开放视频 | 乱码午夜-极国产极内射 | 亚洲爆乳精品无码一区二区三区 | 亚洲无人区一区二区三区 | 国产成人一区二区三区别 | 欧美黑人乱大交 | 久久久久久国产精品无码下载 | 国产av一区二区精品久久凹凸 | 波多野结衣高清一区二区三区 | 一本色道久久综合亚洲精品不卡 | 亚洲精品一区国产 | 一本久久伊人热热精品中文字幕 | 内射后入在线观看一区 | 丰满人妻被黑人猛烈进入 | 又大又硬又爽免费视频 | 丰满少妇熟乱xxxxx视频 | 亚洲一区二区三区在线观看网站 | 亚洲另类伦春色综合小说 | 人人妻人人澡人人爽欧美精品 | 波多野结衣av在线观看 | 无码午夜成人1000部免费视频 | 一本久久a久久精品亚洲 | 美女黄网站人色视频免费国产 | 国产真实乱对白精彩久久 | 自拍偷自拍亚洲精品10p | 国产精品理论片在线观看 | 岛国片人妻三上悠亚 | 亚洲国产午夜精品理论片 | 国产精品自产拍在线观看 | 亚洲人成影院在线无码按摩店 | 丰满少妇女裸体bbw | 欧美日韩综合一区二区三区 | 久久精品国产日本波多野结衣 | 红桃av一区二区三区在线无码av | 亚洲色欲色欲天天天www | 午夜理论片yy44880影院 | 成人精品视频一区二区 | 性生交大片免费看女人按摩摩 | 东京热一精品无码av | 99久久久无码国产精品免费 | 少妇被黑人到高潮喷出白浆 | 久久精品国产大片免费观看 | 亚洲精品一区二区三区大桥未久 | 国产后入清纯学生妹 | 亚洲a无码综合a国产av中文 | 1000部啪啪未满十八勿入下载 | 四虎4hu永久免费 | 亚洲一区二区三区香蕉 | 97夜夜澡人人双人人人喊 | 中文字幕乱码人妻二区三区 | 欧美自拍另类欧美综合图片区 | 伊人久久大香线焦av综合影院 | 日韩人妻少妇一区二区三区 | 日韩精品无码一本二本三本色 | 日韩精品无码免费一区二区三区 | 亚洲欧美综合区丁香五月小说 | 女人高潮内射99精品 | 亚洲精品国偷拍自产在线观看蜜桃 | 亚洲国产欧美在线成人 | 久久www免费人成人片 | 亚洲人成影院在线无码按摩店 | 国产农村妇女高潮大叫 | 55夜色66夜色国产精品视频 | 无码毛片视频一区二区本码 | 国产精品第一区揄拍无码 | 欧美兽交xxxx×视频 | 欧美自拍另类欧美综合图片区 | 亚洲熟妇色xxxxx欧美老妇y | 国产卡一卡二卡三 | 一个人看的www免费视频在线观看 | 无码人妻黑人中文字幕 | 欧美人与禽zoz0性伦交 | 亚洲欧洲日本无在线码 | 国产精品-区区久久久狼 | 中文字幕乱码中文乱码51精品 | 亚洲中文字幕av在天堂 | 日韩欧美成人免费观看 | 精品国产乱码久久久久乱码 | 一本精品99久久精品77 | 国产精品无码永久免费888 | 亚洲乱码日产精品bd | 欧美精品免费观看二区 | 久久国产精品二国产精品 | 一本色道久久综合亚洲精品不卡 | 国产乱人偷精品人妻a片 | 波多野结衣av在线观看 | 色一情一乱一伦一视频免费看 | 国产超碰人人爽人人做人人添 | 成人欧美一区二区三区黑人免费 | 色欲综合久久中文字幕网 | 国产成人无码午夜视频在线观看 | 成人免费视频视频在线观看 免费 | 无人区乱码一区二区三区 | 无码人妻久久一区二区三区不卡 | 国产精品视频免费播放 | 永久免费精品精品永久-夜色 | 国产精品美女久久久久av爽李琼 | 一本久道久久综合婷婷五月 | 四虎影视成人永久免费观看视频 | 国产无遮挡又黄又爽免费视频 | 欧美 日韩 亚洲 在线 | 婷婷五月综合激情中文字幕 | 人妻中文无码久热丝袜 | 乌克兰少妇xxxx做受 | 蜜臀av在线观看 在线欧美精品一区二区三区 | 粗大的内捧猛烈进出视频 | 东京热无码av男人的天堂 | 日本精品高清一区二区 | 亚洲最大成人网站 | 中文字幕日韩精品一区二区三区 | 在线播放免费人成毛片乱码 | 国产精品毛片一区二区 | 一本精品99久久精品77 | 99国产欧美久久久精品 | 国产suv精品一区二区五 | 免费无码一区二区三区蜜桃大 | 久久综合给合久久狠狠狠97色 | 老头边吃奶边弄进去呻吟 | 欧美刺激性大交 | 国产精品久久国产精品99 | 99视频精品全部免费免费观看 | 久久午夜无码鲁丝片 | 欧美人与禽zoz0性伦交 | 人妻无码αv中文字幕久久琪琪布 | 小泽玛莉亚一区二区视频在线 | 狠狠综合久久久久综合网 | 丰满岳乱妇在线观看中字无码 | 欧美阿v高清资源不卡在线播放 | 亚洲色大成网站www国产 | 国产人妖乱国产精品人妖 | 国产亚洲tv在线观看 | 亚洲理论电影在线观看 | 牲交欧美兽交欧美 | 亚洲色www成人永久网址 | 综合激情五月综合激情五月激情1 | 又黄又爽又色的视频 | 又大又紧又粉嫩18p少妇 | 成熟女人特级毛片www免费 | 欧美日本日韩 | 色综合久久久久综合一本到桃花网 | 久久精品人人做人人综合 | 乌克兰少妇性做爰 | 国产精品va在线播放 | 亚洲日本一区二区三区在线 | 人妻有码中文字幕在线 | 成年美女黄网站色大免费视频 | 色 综合 欧美 亚洲 国产 | 高清国产亚洲精品自在久久 | 狠狠色丁香久久婷婷综合五月 | 亚洲一区二区三区 | 福利一区二区三区视频在线观看 | 亚洲男人av天堂午夜在 | aⅴ亚洲 日韩 色 图网站 播放 | 一二三四在线观看免费视频 | 国产精品亚洲lv粉色 | 欧美成人免费全部网站 | 亚洲人成网站免费播放 | 久久精品国产一区二区三区肥胖 | 国产精品亚洲五月天高清 | 免费观看黄网站 | 国产精品国产三级国产专播 | 任你躁在线精品免费 | 强奷人妻日本中文字幕 | 国产乱人偷精品人妻a片 | 暴力强奷在线播放无码 | 国产欧美亚洲精品a | 欧美人与物videos另类 | 国产精品免费大片 | 欧美日韩一区二区免费视频 | 国产 精品 自在自线 | 特黄特色大片免费播放器图片 | 色欲av亚洲一区无码少妇 | 亚洲精品中文字幕 | 欧美 亚洲 国产 另类 | 99久久久无码国产aaa精品 | 久久国产精品偷任你爽任你 | 免费乱码人妻系列无码专区 | 亚洲欧美日韩国产精品一区二区 | 图片小说视频一区二区 | 特黄特色大片免费播放器图片 | 精品aⅴ一区二区三区 | 国产真实伦对白全集 | 成人精品天堂一区二区三区 | 国产精品久久久久7777 | 欧美 丝袜 自拍 制服 另类 | 少妇一晚三次一区二区三区 | 成 人影片 免费观看 | 内射欧美老妇wbb | 亚洲精品久久久久avwww潮水 | 伊人久久大香线蕉午夜 | 自拍偷自拍亚洲精品被多人伦好爽 | 国产另类ts人妖一区二区 | 免费男性肉肉影院 | 亚洲色成人中文字幕网站 | 免费人成在线观看网站 | 人妻熟女一区 | 女人被男人躁得好爽免费视频 | 国产精品久久久久无码av色戒 | 风流少妇按摩来高潮 | 亚洲va欧美va天堂v国产综合 | 天堂在线观看www | 亚洲人成网站免费播放 | 性做久久久久久久免费看 | 精品国产aⅴ无码一区二区 | 激情五月综合色婷婷一区二区 | 国产激情无码一区二区app | 一本久久a久久精品vr综合 | 日本在线高清不卡免费播放 | 西西人体www44rt大胆高清 | 无码人妻丰满熟妇区五十路百度 | 日韩精品乱码av一区二区 | 日本熟妇浓毛 | www国产亚洲精品久久网站 | 欧洲极品少妇 | 激情五月综合色婷婷一区二区 | 精品久久久久香蕉网 | 精品夜夜澡人妻无码av蜜桃 | 最近中文2019字幕第二页 | 日日摸日日碰夜夜爽av | 国产精品内射视频免费 | 中文字幕久久久久人妻 | 综合激情五月综合激情五月激情1 | 大地资源网第二页免费观看 | 久久精品国产一区二区三区 | 亚洲精品美女久久久久久久 | 国语自产偷拍精品视频偷 | 国产高潮视频在线观看 | 国产极品视觉盛宴 | 蜜桃视频韩日免费播放 | 97无码免费人妻超级碰碰夜夜 | 成人aaa片一区国产精品 | 三级4级全黄60分钟 | 成熟人妻av无码专区 | 少妇无码吹潮 | 色婷婷久久一区二区三区麻豆 | 国产午夜福利100集发布 | 丰满人妻一区二区三区免费视频 | 老熟妇乱子伦牲交视频 | 日日噜噜噜噜夜夜爽亚洲精品 | 在线看片无码永久免费视频 | 老熟妇乱子伦牲交视频 | 国产亚洲欧美在线专区 | 一个人看的www免费视频在线观看 | 国产在线无码精品电影网 | 日韩亚洲欧美中文高清在线 | 久久精品国产99久久6动漫 | 国产成人无码av在线影院 | 国产片av国语在线观看 | 国产农村乱对白刺激视频 | 成熟女人特级毛片www免费 | 男人的天堂2018无码 | 中文字幕久久久久人妻 | 欧美肥老太牲交大战 | 日韩在线不卡免费视频一区 | 国产亚洲美女精品久久久2020 | 国产av剧情md精品麻豆 | 欧美 日韩 亚洲 在线 | 黑人巨大精品欧美一区二区 | 一本加勒比波多野结衣 | 亚洲精品国偷拍自产在线麻豆 | 国产综合在线观看 | 国产精品亚洲综合色区韩国 | 狂野欧美激情性xxxx | 一本久道高清无码视频 | 欧美兽交xxxx×视频 | 亚洲 日韩 欧美 成人 在线观看 | 乱人伦人妻中文字幕无码久久网 | 免费乱码人妻系列无码专区 | 蜜臀aⅴ国产精品久久久国产老师 | 亚洲日韩av一区二区三区四区 | 国产午夜无码精品免费看 | 亚洲日韩中文字幕在线播放 | 波多野结衣av在线观看 | 国产性生交xxxxx无码 | 色情久久久av熟女人妻网站 | 国产精品永久免费视频 | 亚洲综合精品香蕉久久网 | 乌克兰少妇xxxx做受 | 2020久久香蕉国产线看观看 | 精品人妻人人做人人爽夜夜爽 | 又粗又大又硬又长又爽 | 国产在线精品一区二区三区直播 | 性欧美牲交在线视频 | 色婷婷久久一区二区三区麻豆 | 精品一区二区不卡无码av | 久久综合九色综合欧美狠狠 | 动漫av网站免费观看 | 男女超爽视频免费播放 | 亚洲男女内射在线播放 | 无遮无挡爽爽免费视频 | 国产人妻人伦精品1国产丝袜 | 日本一卡2卡3卡4卡无卡免费网站 国产一区二区三区影院 | 在线天堂新版最新版在线8 | 日本高清一区免费中文视频 | 好屌草这里只有精品 | 国产精品手机免费 | 亚洲理论电影在线观看 | 亚洲精品www久久久 | 人妻少妇精品久久 | 牲欲强的熟妇农村老妇女视频 | 鲁大师影院在线观看 | 国产内射爽爽大片视频社区在线 | 俄罗斯老熟妇色xxxx | 国产农村乱对白刺激视频 | 最新版天堂资源中文官网 | 天天躁日日躁狠狠躁免费麻豆 | 正在播放东北夫妻内射 | 欧美第一黄网免费网站 | 中国女人内谢69xxxxxa片 | 婷婷综合久久中文字幕蜜桃三电影 | 超碰97人人射妻 | 国产xxx69麻豆国语对白 | 国产后入清纯学生妹 | 狠狠色噜噜狠狠狠7777奇米 | 免费网站看v片在线18禁无码 | 在线欧美精品一区二区三区 | 国产又爽又猛又粗的视频a片 | 国产精品久久福利网站 | 久久亚洲中文字幕无码 | 国语精品一区二区三区 | 欧美喷潮久久久xxxxx | 国产激情艳情在线看视频 | 亚洲色大成网站www国产 | 欧美丰满老熟妇xxxxx性 | 免费观看又污又黄的网站 | 亚洲国产精华液网站w | 欧美成人免费全部网站 | 成熟女人特级毛片www免费 | 99久久精品国产一区二区蜜芽 | 欧美 日韩 亚洲 在线 | 久激情内射婷内射蜜桃人妖 | 欧美zoozzooz性欧美 | 日本一卡2卡3卡4卡无卡免费网站 国产一区二区三区影院 | 国产电影无码午夜在线播放 | 内射欧美老妇wbb | 亚洲精品一区二区三区大桥未久 | 激情五月综合色婷婷一区二区 | 亚洲国产一区二区三区在线观看 | 77777熟女视频在线观看 а天堂中文在线官网 | 性史性农村dvd毛片 | 午夜精品一区二区三区在线观看 | 亚洲乱码中文字幕在线 | 久久熟妇人妻午夜寂寞影院 | 日韩欧美中文字幕公布 | 亚洲中文无码av永久不收费 | 日本大香伊一区二区三区 | 国产成人综合美国十次 | 国产福利视频一区二区 | 欧美性猛交内射兽交老熟妇 | 嫩b人妻精品一区二区三区 | 亚洲熟妇色xxxxx亚洲 | 久久久av男人的天堂 | 网友自拍区视频精品 | 欧美日韩一区二区综合 | 国产亚洲精品久久久久久久 | 曰韩无码二三区中文字幕 | 久久综合狠狠综合久久综合88 | 国产av久久久久精东av | 国产性生交xxxxx无码 | 国产成人精品三级麻豆 | 久久精品中文字幕大胸 | 无码人妻丰满熟妇区五十路百度 | 久久亚洲精品成人无码 | 欧美丰满熟妇xxxx | 亚洲娇小与黑人巨大交 | 国产肉丝袜在线观看 | 久久久久99精品国产片 | 日韩av无码一区二区三区不卡 | 天堂在线观看www | 亚洲国产av精品一区二区蜜芽 | 玩弄人妻少妇500系列视频 | 国产激情一区二区三区 | 色欲久久久天天天综合网精品 | 亚洲s码欧洲m码国产av | 久久久久久久久888 | 亚洲综合在线一区二区三区 | 老太婆性杂交欧美肥老太 | 又大又紧又粉嫩18p少妇 | 日韩精品成人一区二区三区 | 久久精品女人天堂av免费观看 | 国产亚洲精品久久久ai换 | 蜜桃av抽搐高潮一区二区 | 亚洲乱码日产精品bd | 亚洲国产精品久久久天堂 | 日韩av无码中文无码电影 | 日本肉体xxxx裸交 | 丰满人妻翻云覆雨呻吟视频 | 一个人看的视频www在线 | 精品国产aⅴ无码一区二区 | 成人影院yy111111在线观看 | 国产午夜无码精品免费看 | 国产美女精品一区二区三区 | 国产av一区二区三区最新精品 | 久久国产自偷自偷免费一区调 | 亚洲性无码av中文字幕 | 一本精品99久久精品77 | 亚洲日本在线电影 | 亚洲熟悉妇女xxx妇女av | 亚洲熟妇色xxxxx欧美老妇y | 国产精品人妻一区二区三区四 | 欧美亚洲国产一区二区三区 | 水蜜桃av无码 | 无码播放一区二区三区 | 国产亚洲精品精品国产亚洲综合 | 乱码av麻豆丝袜熟女系列 | 无码国产色欲xxxxx视频 | 国产69精品久久久久app下载 | 日韩人妻系列无码专区 | 亚洲乱码日产精品bd | 精品久久久中文字幕人妻 | 大胆欧美熟妇xx | 国产乱人偷精品人妻a片 | 免费国产成人高清在线观看网站 | 国产人妻精品一区二区三区 | 久久精品国产99精品亚洲 | 国产精品亚洲а∨无码播放麻豆 | 亚洲精品一区二区三区大桥未久 | 小鲜肉自慰网站xnxx | 国产乱人偷精品人妻a片 | 亚洲人亚洲人成电影网站色 | 久久人人爽人人爽人人片ⅴ | 妺妺窝人体色www婷婷 | 人人妻人人澡人人爽欧美一区九九 | 久久亚洲国产成人精品性色 | 午夜肉伦伦影院 | 久久久国产一区二区三区 | 亚洲欧洲日本综合aⅴ在线 | 乱人伦中文视频在线观看 | 中文字幕人成乱码熟女app | 久久久精品欧美一区二区免费 | 曰韩少妇内射免费播放 | 久久亚洲中文字幕精品一区 | 三上悠亚人妻中文字幕在线 | 人人妻人人澡人人爽人人精品浪潮 | 正在播放老肥熟妇露脸 | 99久久精品国产一区二区蜜芽 | 亚洲精品国产品国语在线观看 | 亚洲欧美综合区丁香五月小说 | 久久午夜无码鲁丝片秋霞 | 特级做a爰片毛片免费69 | 天天av天天av天天透 | 中文字幕人妻无码一夲道 | 中文无码伦av中文字幕 | 日本一卡2卡3卡4卡无卡免费网站 国产一区二区三区影院 | 久久国产劲爆∧v内射 | 久久人妻内射无码一区三区 | 5858s亚洲色大成网站www | 乱码av麻豆丝袜熟女系列 | 亚洲国产av精品一区二区蜜芽 | 国产精品怡红院永久免费 | 日本大香伊一区二区三区 | 久久精品中文字幕大胸 | 东京热无码av男人的天堂 | 黑人巨大精品欧美黑寡妇 | 99麻豆久久久国产精品免费 | 亚洲自偷自拍另类第1页 | 中文字幕av伊人av无码av | 亚洲精品久久久久久一区二区 | 亚洲午夜久久久影院 | 无码av最新清无码专区吞精 | 性开放的女人aaa片 | 国语自产偷拍精品视频偷 | 国产精品久久久一区二区三区 | 国产精品对白交换视频 | 国产成人久久精品流白浆 | 大肉大捧一进一出好爽视频 | 波多野结衣aⅴ在线 | 精品乱码久久久久久久 | 性色欲网站人妻丰满中文久久不卡 | 国产精品亚洲一区二区三区喷水 | 成人欧美一区二区三区黑人免费 | 亚洲精品鲁一鲁一区二区三区 | 成人影院yy111111在线观看 | 麻豆国产丝袜白领秘书在线观看 | 免费无码一区二区三区蜜桃大 | 熟女体下毛毛黑森林 | 在线精品亚洲一区二区 | 久久久精品人妻久久影视 | 人妻插b视频一区二区三区 | 亚洲成av人片在线观看无码不卡 | 精品水蜜桃久久久久久久 | 成人精品视频一区二区三区尤物 | 18黄暴禁片在线观看 | 国产午夜亚洲精品不卡 | 在线观看欧美一区二区三区 | 中文字幕亚洲情99在线 | 日韩 欧美 动漫 国产 制服 | 亚洲中文字幕av在天堂 | 我要看www免费看插插视频 | 日韩精品乱码av一区二区 | 久久精品视频在线看15 | 小sao货水好多真紧h无码视频 | 亚拍精品一区二区三区探花 | 亚洲色www成人永久网址 | 麻豆精产国品 | 无遮挡国产高潮视频免费观看 | 国产精品嫩草久久久久 | 狠狠色色综合网站 | 任你躁国产自任一区二区三区 | www国产亚洲精品久久网站 | 永久免费观看国产裸体美女 | 久久久久久国产精品无码下载 | 欧美35页视频在线观看 | 欧美人与善在线com | 人人爽人人爽人人片av亚洲 | 综合人妻久久一区二区精品 | 日韩精品无码一本二本三本色 | 中文无码伦av中文字幕 | 国产尤物精品视频 | 久久午夜夜伦鲁鲁片无码免费 | 人人妻人人藻人人爽欧美一区 | 亚洲精品国产第一综合99久久 | av香港经典三级级 在线 | 国语自产偷拍精品视频偷 | 两性色午夜免费视频 | 国产人妖乱国产精品人妖 | 任你躁国产自任一区二区三区 | 99re在线播放 | 国产精品igao视频网 | 久久久久人妻一区精品色欧美 | 国产色在线 | 国产 | 一本精品99久久精品77 | 欧美性猛交内射兽交老熟妇 | 青青青爽视频在线观看 | 日本www一道久久久免费榴莲 | 男女爱爱好爽视频免费看 | 久久久国产精品无码免费专区 | 亚洲国产精品成人久久蜜臀 | 国产在线aaa片一区二区99 | 熟妇女人妻丰满少妇中文字幕 | 欧美日韩久久久精品a片 | 人妻有码中文字幕在线 | 日欧一片内射va在线影院 | www成人国产高清内射 | 亚洲自偷精品视频自拍 | 精品无码国产一区二区三区av | 最近的中文字幕在线看视频 | 亚洲狠狠婷婷综合久久 | 国产97在线 | 亚洲 | 亚洲综合在线一区二区三区 | 欧美黑人巨大xxxxx | 国产真实夫妇视频 | 亚洲区小说区激情区图片区 | 日日噜噜噜噜夜夜爽亚洲精品 | 国产午夜福利亚洲第一 | 国产精品久久精品三级 | 熟女俱乐部五十路六十路av | 99久久精品日本一区二区免费 | 国产精品永久免费视频 | 亚洲精品国偷拍自产在线观看蜜桃 | 欧美精品国产综合久久 | 日本熟妇人妻xxxxx人hd | 蜜臀av无码人妻精品 | 亚洲成av人影院在线观看 | 精品人妻人人做人人爽 | 色 综合 欧美 亚洲 国产 | 少妇厨房愉情理9仑片视频 | 特黄特色大片免费播放器图片 | 欧美日本精品一区二区三区 | 久久午夜无码鲁丝片 | 亚洲最大成人网站 | 亚洲国产精品一区二区美利坚 | 久久精品国产亚洲精品 | 青草青草久热国产精品 | 好男人社区资源 | 午夜福利不卡在线视频 | 成熟女人特级毛片www免费 | 亚洲日本一区二区三区在线 | 99久久无码一区人妻 | 欧美freesex黑人又粗又大 | 97se亚洲精品一区 | 图片区 小说区 区 亚洲五月 | 99久久99久久免费精品蜜桃 | 无码人妻久久一区二区三区不卡 | 日日碰狠狠躁久久躁蜜桃 | 国产免费久久久久久无码 | 撕开奶罩揉吮奶头视频 | 日本va欧美va欧美va精品 | 人人澡人人妻人人爽人人蜜桃 | 成人精品一区二区三区中文字幕 | 国产欧美亚洲精品a | 国产亲子乱弄免费视频 | 免费国产成人高清在线观看网站 | 国产sm调教视频在线观看 | 日本欧美一区二区三区乱码 | 国产午夜亚洲精品不卡下载 | 国产三级精品三级男人的天堂 | 日韩精品无码一区二区中文字幕 | 亚洲综合无码一区二区三区 | 在线播放免费人成毛片乱码 | 国产精品va在线播放 | 人人澡人人妻人人爽人人蜜桃 | 亚洲日韩av一区二区三区中文 | 领导边摸边吃奶边做爽在线观看 | 天堂а√在线地址中文在线 | 精品无码国产自产拍在线观看蜜 | 永久免费观看美女裸体的网站 | 18禁黄网站男男禁片免费观看 | 欧美性生交活xxxxxdddd | 成人女人看片免费视频放人 | 欧美日韩一区二区三区自拍 | 中文精品久久久久人妻不卡 | 中文字幕无码av激情不卡 | 欧美精品国产综合久久 | 成人欧美一区二区三区黑人 | 学生妹亚洲一区二区 | 成年美女黄网站色大免费全看 | 国产一区二区三区日韩精品 | 久久精品一区二区三区四区 | 久久综合给久久狠狠97色 | 亚洲成av人综合在线观看 | 男女猛烈xx00免费视频试看 | 亚洲成熟女人毛毛耸耸多 | 又粗又大又硬又长又爽 | 国产精品久久久一区二区三区 | 国产精品永久免费视频 | 成人无码精品1区2区3区免费看 | 中文字幕av无码一区二区三区电影 | 日韩视频 中文字幕 视频一区 | 国产高潮视频在线观看 | 国产网红无码精品视频 | 丰满少妇熟乱xxxxx视频 | 亚洲一区二区三区四区 | 欧美日韩综合一区二区三区 | 欧美阿v高清资源不卡在线播放 | 综合激情五月综合激情五月激情1 | 国产一区二区三区影院 | 久久亚洲国产成人精品性色 | 成人精品视频一区二区三区尤物 | 国产精品丝袜黑色高跟鞋 | 18精品久久久无码午夜福利 | 日韩成人一区二区三区在线观看 | 内射老妇bbwx0c0ck | 久久久久亚洲精品中文字幕 | 乱中年女人伦av三区 | 亚洲欧美国产精品专区久久 | 色婷婷欧美在线播放内射 | 丰满肥臀大屁股熟妇激情视频 | 亚洲男女内射在线播放 | 久久人妻内射无码一区三区 | 水蜜桃亚洲一二三四在线 | 噜噜噜亚洲色成人网站 | 内射白嫩少妇超碰 | 老子影院午夜伦不卡 | 国产成人无码av一区二区 | 牛和人交xxxx欧美 | 成人欧美一区二区三区黑人免费 | 国产精华av午夜在线观看 | 亚洲 高清 成人 动漫 | 亚洲精品综合一区二区三区在线 | 精品乱码久久久久久久 | 亚洲乱码中文字幕在线 | 午夜丰满少妇性开放视频 | 亚洲国精产品一二二线 | 亚洲欧美国产精品专区久久 | 欧美激情内射喷水高潮 | 窝窝午夜理论片影院 | 久久婷婷五月综合色国产香蕉 | 中文精品无码中文字幕无码专区 | 日日鲁鲁鲁夜夜爽爽狠狠 | 扒开双腿疯狂进出爽爽爽视频 | 国产av剧情md精品麻豆 | 99久久亚洲精品无码毛片 | 亚洲色在线无码国产精品不卡 | 亚洲大尺度无码无码专区 | 日日摸夜夜摸狠狠摸婷婷 | 九九综合va免费看 | 日日麻批免费40分钟无码 | 中文字幕 亚洲精品 第1页 | 亚洲自偷自拍另类第1页 | 国语精品一区二区三区 | 人人爽人人澡人人高潮 | 午夜熟女插插xx免费视频 | 亚洲无人区一区二区三区 | 国产精品igao视频网 | 狠狠色丁香久久婷婷综合五月 | 在线天堂新版最新版在线8 | 国产国产精品人在线视 | 少妇高潮一区二区三区99 | 日本一卡2卡3卡4卡无卡免费网站 国产一区二区三区影院 | 爆乳一区二区三区无码 | 亚洲日韩中文字幕在线播放 | 日韩精品成人一区二区三区 | 国产精品久久久久久久影院 | 欧美日韩在线亚洲综合国产人 | 国语自产偷拍精品视频偷 | 一本精品99久久精品77 | 国产激情一区二区三区 | 国产三级久久久精品麻豆三级 | 少妇高潮喷潮久久久影院 | 成人动漫在线观看 | 99久久婷婷国产综合精品青草免费 | 精品一区二区三区波多野结衣 | 欧美激情一区二区三区成人 | 久久精品国产99久久6动漫 | 日韩精品久久久肉伦网站 | 国产精品成人av在线观看 | 又粗又大又硬又长又爽 | 中文字幕无码热在线视频 | 亚洲 欧美 激情 小说 另类 | 天堂亚洲免费视频 | 亚洲国产精华液网站w | 亚洲午夜福利在线观看 | 97夜夜澡人人爽人人喊中国片 | 两性色午夜视频免费播放 | 成人aaa片一区国产精品 | 国产综合在线观看 | 亚洲乱码中文字幕在线 | 亚洲第一无码av无码专区 | 国产极品美女高潮无套在线观看 | 丰满人妻一区二区三区免费视频 | 国产高清av在线播放 | 丰满少妇人妻久久久久久 | 久久99精品久久久久久 | 六十路熟妇乱子伦 | 性色av无码免费一区二区三区 | 中文字幕日韩精品一区二区三区 | 亚洲精品一区国产 | 久久伊人色av天堂九九小黄鸭 | 搡女人真爽免费视频大全 | 久久久精品456亚洲影院 | 亚洲精品鲁一鲁一区二区三区 | 亚洲无人区一区二区三区 | 亚洲一区av无码专区在线观看 | 久久综合狠狠综合久久综合88 | 国产一区二区三区四区五区加勒比 | 午夜福利一区二区三区在线观看 | 日韩人妻无码一区二区三区久久99 | 在线观看国产午夜福利片 | 男女猛烈xx00免费视频试看 | 偷窥日本少妇撒尿chinese | 国产麻豆精品一区二区三区v视界 | 成熟女人特级毛片www免费 | 亚洲精品国产第一综合99久久 | 午夜无码人妻av大片色欲 | 国产精品igao视频网 | 亚洲娇小与黑人巨大交 | 日欧一片内射va在线影院 | 亚洲а∨天堂久久精品2021 | 无遮挡国产高潮视频免费观看 | 亚洲小说图区综合在线 | 欧美乱妇无乱码大黄a片 | 无人区乱码一区二区三区 | 麻豆国产97在线 | 欧洲 | 日产精品99久久久久久 | 亚洲区小说区激情区图片区 | 夜夜高潮次次欢爽av女 | 欧美 亚洲 国产 另类 | 日韩精品无码一区二区中文字幕 | 娇妻被黑人粗大高潮白浆 | 久久综合香蕉国产蜜臀av | 亚洲人成网站在线播放942 | 精品无码一区二区三区的天堂 | 国内少妇偷人精品视频 | 国产亚洲精品久久久ai换 | 国产亚洲视频中文字幕97精品 | 999久久久国产精品消防器材 | 少妇邻居内射在线 | 四虎国产精品一区二区 | 激情内射亚州一区二区三区爱妻 | 国产在线aaa片一区二区99 | 大乳丰满人妻中文字幕日本 | 中文字幕无码av波多野吉衣 | √8天堂资源地址中文在线 | 亚洲s码欧洲m码国产av | 亚洲欧洲日本无在线码 | 成人无码精品一区二区三区 | 久久99精品国产.久久久久 | 无码精品国产va在线观看dvd | 国产成人综合美国十次 | 久久久久久av无码免费看大片 | 久久精品国产99精品亚洲 | 扒开双腿吃奶呻吟做受视频 | 51国偷自产一区二区三区 | 无遮挡国产高潮视频免费观看 | 亚洲精品午夜无码电影网 | 少妇人妻av毛片在线看 | 人妻天天爽夜夜爽一区二区 | 欧美国产日产一区二区 | 亚洲 高清 成人 动漫 | 人人澡人人透人人爽 | 国产亚洲欧美日韩亚洲中文色 | 久久99精品国产.久久久久 | 永久免费观看国产裸体美女 | 无遮挡啪啪摇乳动态图 | 国内少妇偷人精品视频免费 | 大肉大捧一进一出好爽视频 | 成在人线av无码免观看麻豆 | 无码人妻黑人中文字幕 | 欧美精品无码一区二区三区 | 熟妇人妻中文av无码 | 在线欧美精品一区二区三区 | 国产suv精品一区二区五 | 国产精品视频免费播放 | 99久久婷婷国产综合精品青草免费 | 国产亚洲精品久久久ai换 | 亚洲国产精品成人久久蜜臀 | 特级做a爰片毛片免费69 | 日韩欧美群交p片內射中文 | 精品国产aⅴ无码一区二区 | 又紧又大又爽精品一区二区 | 久9re热视频这里只有精品 | 亚洲国产欧美日韩精品一区二区三区 | 中文字幕无码日韩欧毛 | 亚洲欧美国产精品久久 | 日本精品人妻无码免费大全 | 亚欧洲精品在线视频免费观看 | 中文字幕乱码中文乱码51精品 | 国产特级毛片aaaaaa高潮流水 | 精品一区二区不卡无码av | 国产两女互慰高潮视频在线观看 | 日韩无码专区 | 偷窥村妇洗澡毛毛多 | 亚洲精品一区三区三区在线观看 | 欧美丰满熟妇xxxx性ppx人交 | 国产卡一卡二卡三 | 精品欧美一区二区三区久久久 | 中文字幕 亚洲精品 第1页 | 蜜臀av在线观看 在线欧美精品一区二区三区 | 欧美黑人性暴力猛交喷水 | 久久久中文久久久无码 | 爆乳一区二区三区无码 | 特大黑人娇小亚洲女 | 1000部啪啪未满十八勿入下载 | 成人精品一区二区三区中文字幕 | 18禁黄网站男男禁片免费观看 | 永久免费观看国产裸体美女 | 一本精品99久久精品77 | 久久久久成人精品免费播放动漫 | 一本久道久久综合婷婷五月 | 精品无码国产自产拍在线观看蜜 | 大地资源网第二页免费观看 | 免费观看又污又黄的网站 | 国产超级va在线观看视频 | 日本一卡2卡3卡4卡无卡免费网站 国产一区二区三区影院 | 久久99国产综合精品 | 99re在线播放 | 99久久久国产精品无码免费 | 中文字幕 亚洲精品 第1页 | 国产精华av午夜在线观看 | 精品国产一区二区三区四区 | 久久99精品国产麻豆蜜芽 | 美女极度色诱视频国产 | 久久国产精品二国产精品 | 日本熟妇大屁股人妻 | 女人和拘做爰正片视频 | 图片区 小说区 区 亚洲五月 | 黑人粗大猛烈进出高潮视频 | 性史性农村dvd毛片 | 亚洲国产精华液网站w | 少妇被粗大的猛进出69影院 | 中文字幕 亚洲精品 第1页 | 国产精品怡红院永久免费 | 日本爽爽爽爽爽爽在线观看免 | 免费中文字幕日韩欧美 | 国产人妻精品一区二区三区不卡 | 亚洲 a v无 码免 费 成 人 a v | 无码帝国www无码专区色综合 | 欧美日韩一区二区三区自拍 | 国产成人精品三级麻豆 | 乱人伦人妻中文字幕无码久久网 | 亚洲 激情 小说 另类 欧美 | 无码人妻丰满熟妇区毛片18 | 美女极度色诱视频国产 | 天天拍夜夜添久久精品 | 女人和拘做爰正片视频 | 亚洲乱码日产精品bd | 免费网站看v片在线18禁无码 | 无遮挡国产高潮视频免费观看 | 日韩视频 中文字幕 视频一区 | 大地资源网第二页免费观看 | 亚洲欧洲日本综合aⅴ在线 | 99久久精品无码一区二区毛片 | 亚洲自偷自偷在线制服 | 久久国产自偷自偷免费一区调 | 成年美女黄网站色大免费全看 | 久久人人爽人人爽人人片av高清 | 伊人久久大香线焦av综合影院 | 无码免费一区二区三区 | 一本精品99久久精品77 | 婷婷六月久久综合丁香 | 少妇无码一区二区二三区 | 性色av无码免费一区二区三区 | 日本在线高清不卡免费播放 | 丰满人妻被黑人猛烈进入 | 国产精品久久久久影院嫩草 | 人人澡人人透人人爽 | 天堂一区人妻无码 | 国产无遮挡又黄又爽又色 | 无套内谢的新婚少妇国语播放 | 亚洲日韩精品欧美一区二区 | 国产精品福利视频导航 | 青青青手机频在线观看 | 国产精品国产三级国产专播 | 国产精品毛片一区二区 | 国产午夜亚洲精品不卡下载 | 中文字幕乱码亚洲无线三区 | 日韩少妇白浆无码系列 | 麻豆国产人妻欲求不满 | 人人妻人人澡人人爽人人精品浪潮 | 性色欲网站人妻丰满中文久久不卡 | 色婷婷综合中文久久一本 | 精品无码成人片一区二区98 | 国产农村妇女aaaaa视频 撕开奶罩揉吮奶头视频 | 国产精品久久久久9999小说 | 国产偷抇久久精品a片69 | 久久久精品成人免费观看 | 国产成人综合在线女婷五月99播放 | 午夜肉伦伦影院 | 国语自产偷拍精品视频偷 | 国产在线精品一区二区高清不卡 | 国产精品毛多多水多 | 国产精品沙发午睡系列 | 六十路熟妇乱子伦 | 日日碰狠狠躁久久躁蜜桃 | 内射巨臀欧美在线视频 | 婷婷丁香六月激情综合啪 | 亚洲爆乳大丰满无码专区 | 日本丰满熟妇videos | 日本va欧美va欧美va精品 | 国产美女精品一区二区三区 | 蜜臀av无码人妻精品 | 国产午夜福利亚洲第一 | 日韩人妻系列无码专区 | 日韩精品乱码av一区二区 | 牲欲强的熟妇农村老妇女 | 亚洲国产精品无码一区二区三区 | 亚洲自偷精品视频自拍 | 精品人人妻人人澡人人爽人人 | 丰满妇女强制高潮18xxxx | 狠狠cao日日穞夜夜穞av | 国产精品美女久久久久av爽李琼 | 欧美国产亚洲日韩在线二区 | 偷窥日本少妇撒尿chinese | 国产高清av在线播放 | 久久久av男人的天堂 | 亚洲日韩av一区二区三区四区 | 成人欧美一区二区三区黑人免费 | 色欲av亚洲一区无码少妇 | 老熟女重囗味hdxx69 | 乱码av麻豆丝袜熟女系列 | 男人的天堂av网站 | 色婷婷久久一区二区三区麻豆 | 双乳奶水饱满少妇呻吟 | 婷婷丁香六月激情综合啪 | 狠狠cao日日穞夜夜穞av | 国产精品久久久久9999小说 | 两性色午夜免费视频 | 欧美肥老太牲交大战 | 国产精品爱久久久久久久 | 精品久久8x国产免费观看 | 色婷婷久久一区二区三区麻豆 | 亚洲日本在线电影 | 国产 精品 自在自线 | 亚洲の无码国产の无码影院 | 亚洲成av人影院在线观看 | 图片区 小说区 区 亚洲五月 | 久久国产自偷自偷免费一区调 | 日本饥渴人妻欲求不满 | 欧美性生交xxxxx久久久 | 欧美丰满老熟妇xxxxx性 | 国产一区二区三区影院 | 亚洲欧美日韩成人高清在线一区 | 男人和女人高潮免费网站 | 日本高清一区免费中文视频 | 啦啦啦www在线观看免费视频 | 日产精品高潮呻吟av久久 | 天堂无码人妻精品一区二区三区 | 四虎国产精品一区二区 | 久久99精品久久久久婷婷 | 麻豆成人精品国产免费 | 中文字幕人妻无码一夲道 | 中国女人内谢69xxxxxa片 | 婷婷五月综合缴情在线视频 | 国产国产精品人在线视 | 精品久久久中文字幕人妻 | 国产综合久久久久鬼色 | 激情内射日本一区二区三区 | 久久久久久九九精品久 | 久久综合色之久久综合 | 丰满人妻一区二区三区免费视频 | 欧美日韩一区二区三区自拍 | 在线精品国产一区二区三区 | 18无码粉嫩小泬无套在线观看 | 日韩亚洲欧美中文高清在线 | 日韩人妻无码中文字幕视频 | 国产国产精品人在线视 | 小鲜肉自慰网站xnxx | 亚洲国产精品一区二区美利坚 | 久久久久亚洲精品中文字幕 | 日日天日日夜日日摸 | 香港三级日本三级妇三级 | 成年女人永久免费看片 | 国产人妻精品一区二区三区不卡 | 中文字幕无码日韩专区 | 精品久久久中文字幕人妻 | 亚洲另类伦春色综合小说 | 强开小婷嫩苞又嫩又紧视频 | 国产精品久久国产三级国 | 高清无码午夜福利视频 | 小鲜肉自慰网站xnxx | 国产精品高潮呻吟av久久4虎 | 国产成人精品必看 | 亚洲成a人一区二区三区 | 婷婷综合久久中文字幕蜜桃三电影 | 免费男性肉肉影院 | 中文字幕人妻无码一区二区三区 | 国产人妻久久精品二区三区老狼 | 亚洲成a人片在线观看无码3d | 国产性生交xxxxx无码 | 女人高潮内射99精品 | 精品乱子伦一区二区三区 | 成人精品一区二区三区中文字幕 | 亚洲色欲久久久综合网东京热 | 精品亚洲韩国一区二区三区 | 欧美 丝袜 自拍 制服 另类 | 中文字幕+乱码+中文字幕一区 | 国产69精品久久久久app下载 | 久久 国产 尿 小便 嘘嘘 | 性色欲网站人妻丰满中文久久不卡 | 中文字幕乱码人妻无码久久 | 亚洲а∨天堂久久精品2021 | 亚洲国产精品一区二区第一页 | 亚洲精品www久久久 | 免费无码肉片在线观看 | 亚洲а∨天堂久久精品2021 | 国产精品香蕉在线观看 | 无码精品国产va在线观看dvd | 牲欲强的熟妇农村老妇女 | 人妻无码久久精品人妻 | 日本熟妇浓毛 | 国产精品无码久久av | а√天堂www在线天堂小说 | 一二三四在线观看免费视频 | 无码人妻av免费一区二区三区 | 兔费看少妇性l交大片免费 | 亚洲欧美日韩综合久久久 | 天天燥日日燥 | 中文字幕久久久久人妻 | 亚洲色在线无码国产精品不卡 | 爽爽影院免费观看 | 亚洲日韩一区二区 | 久久久久久久女国产乱让韩 | 国产做国产爱免费视频 | a国产一区二区免费入口 | 国产免费观看黄av片 | 99久久精品日本一区二区免费 | 无码午夜成人1000部免费视频 | 国产精品亚洲专区无码不卡 | 性欧美videos高清精品 | 麻豆av传媒蜜桃天美传媒 | 国精产品一区二区三区 | 午夜精品一区二区三区的区别 | 牲欲强的熟妇农村老妇女 | 高潮毛片无遮挡高清免费视频 | 午夜免费福利小电影 | 精品无码一区二区三区爱欲 | 色婷婷av一区二区三区之红樱桃 | 国产美女精品一区二区三区 | 日韩亚洲欧美精品综合 | 色欲av亚洲一区无码少妇 | 午夜熟女插插xx免费视频 | 中文字幕日产无线码一区 | 麻豆精产国品 | 无码播放一区二区三区 | 欧美精品无码一区二区三区 | 国产女主播喷水视频在线观看 | 国产一区二区不卡老阿姨 | 亚洲国产日韩a在线播放 | 久久精品女人天堂av免费观看 | 精品久久久无码中文字幕 | 少妇久久久久久人妻无码 | 午夜无码人妻av大片色欲 | 在线 国产 欧美 亚洲 天堂 | 国产人妻精品午夜福利免费 | 成人精品视频一区二区三区尤物 | 国产成人无码区免费内射一片色欲 | 国产高清不卡无码视频 | 国内少妇偷人精品视频免费 | 色欲人妻aaaaaaa无码 | 四虎永久在线精品免费网址 | 中文字幕无码免费久久99 | 无码av中文字幕免费放 | 国产麻豆精品一区二区三区v视界 | 东京热男人av天堂 | 中国大陆精品视频xxxx | 鲁鲁鲁爽爽爽在线视频观看 | 成人性做爰aaa片免费看不忠 | 国产精品欧美成人 | 伊人色综合久久天天小片 | 久久99精品久久久久久 | 国产成人无码av一区二区 | 图片区 小说区 区 亚洲五月 | a在线观看免费网站大全 | 久久97精品久久久久久久不卡 | 疯狂三人交性欧美 | 欧美兽交xxxx×视频 | 美女黄网站人色视频免费国产 | 国产熟女一区二区三区四区五区 | 色综合久久88色综合天天 | 无码av免费一区二区三区试看 | 国产精品亚洲综合色区韩国 | 骚片av蜜桃精品一区 | 久久婷婷五月综合色国产香蕉 | 免费乱码人妻系列无码专区 | 国产绳艺sm调教室论坛 | 巨爆乳无码视频在线观看 | 亚洲成熟女人毛毛耸耸多 | 99riav国产精品视频 | 综合网日日天干夜夜久久 | 亚洲国产日韩a在线播放 | 国产成人综合美国十次 | 成熟妇人a片免费看网站 | 无码人妻少妇伦在线电影 | 欧美喷潮久久久xxxxx | 永久黄网站色视频免费直播 | 国产亚洲欧美在线专区 | 青草青草久热国产精品 | 亚洲精品一区二区三区四区五区 | 欧美 丝袜 自拍 制服 另类 | 国产亚洲精品久久久久久国模美 | 色婷婷综合中文久久一本 | 久久综合九色综合欧美狠狠 | 97人妻精品一区二区三区 | 亚洲人成无码网www | 无码一区二区三区在线观看 | 国产偷国产偷精品高清尤物 | 麻豆人妻少妇精品无码专区 | 老熟妇仑乱视频一区二区 | 天堂а√在线地址中文在线 | 国产乱人偷精品人妻a片 | 欧美日韩色另类综合 | 精品国产福利一区二区 | 日韩欧美中文字幕在线三区 | 综合网日日天干夜夜久久 | 狠狠色色综合网站 | 国产香蕉尹人视频在线 | 精品国偷自产在线视频 | 色欲久久久天天天综合网精品 | 久久精品国产一区二区三区肥胖 | 亚洲欧洲日本综合aⅴ在线 | 久久综合久久自在自线精品自 | 夜夜躁日日躁狠狠久久av | 熟妇人妻无码xxx视频 | 亚洲s色大片在线观看 | а√资源新版在线天堂 | 无套内谢老熟女 | a在线亚洲男人的天堂 | 国产成人综合色在线观看网站 | 日韩精品一区二区av在线 | 国产在线一区二区三区四区五区 | 久久www免费人成人片 | 色综合久久久无码网中文 | 久久综合给久久狠狠97色 | 国产亚洲美女精品久久久2020 | 中文字幕日韩精品一区二区三区 | 中文字幕无码乱人伦 | 麻豆md0077饥渴少妇 | 午夜理论片yy44880影院 | v一区无码内射国产 | 成 人 网 站国产免费观看 | 亚洲一区二区三区香蕉 | 国产福利视频一区二区 | 亚洲 激情 小说 另类 欧美 | www成人国产高清内射 | 国产绳艺sm调教室论坛 | 免费网站看v片在线18禁无码 | 东京无码熟妇人妻av在线网址 | 婷婷五月综合缴情在线视频 | 精品国产青草久久久久福利 | 亚洲一区二区三区播放 | 亚洲无人区一区二区三区 | 中文字幕无线码 | 乱码午夜-极国产极内射 | 麻豆果冻传媒2021精品传媒一区下载 | 无码国产乱人伦偷精品视频 | 久久无码专区国产精品s | 蜜桃视频韩日免费播放 | 天天躁夜夜躁狠狠是什么心态 | 免费国产成人高清在线观看网站 | 一本大道伊人av久久综合 | 欧美兽交xxxx×视频 | 欧美性色19p | 国产莉萝无码av在线播放 | 少妇无套内谢久久久久 | 一本加勒比波多野结衣 | 国产精品久久精品三级 | 免费观看激色视频网站 | 人妻互换免费中文字幕 | 亚洲经典千人经典日产 | 东北女人啪啪对白 | 午夜成人1000部免费视频 | 蜜臀av无码人妻精品 | 成人av无码一区二区三区 | 亚洲无人区午夜福利码高清完整版 | 亚洲欧美精品aaaaaa片 | 国产精品久久久久9999小说 | 久久99精品国产麻豆 | 国产成人无码午夜视频在线观看 | 欧美亚洲日韩国产人成在线播放 | 特大黑人娇小亚洲女 | 影音先锋中文字幕无码 | 美女极度色诱视频国产 | 国产午夜福利100集发布 | 午夜精品久久久内射近拍高清 | 国产极品美女高潮无套在线观看 | 人妻少妇被猛烈进入中文字幕 | 熟妇人妻无码xxx视频 | 暴力强奷在线播放无码 | 奇米影视888欧美在线观看 | 亚洲欧美国产精品久久 | 国产色xx群视频射精 | 免费观看激色视频网站 | 无套内谢老熟女 | 国产精品亚洲一区二区三区喷水 | 久久国产精品精品国产色婷婷 | 丝袜人妻一区二区三区 | 亚洲色欲色欲欲www在线 | 青草视频在线播放 | 亚洲人交乣女bbw | 国产精品无套呻吟在线 | 扒开双腿疯狂进出爽爽爽视频 | 无码播放一区二区三区 | 少妇被黑人到高潮喷出白浆 | 久久综合久久自在自线精品自 | 131美女爱做视频 | 久久久成人毛片无码 | 强伦人妻一区二区三区视频18 | 日本精品少妇一区二区三区 | 亚洲成a人片在线观看无码3d | 免费无码av一区二区 | 荫蒂被男人添的好舒服爽免费视频 | 在线播放免费人成毛片乱码 | 18禁止看的免费污网站 | 麻豆av传媒蜜桃天美传媒 | 鲁鲁鲁爽爽爽在线视频观看 | 搡女人真爽免费视频大全 | 国产国语老龄妇女a片 | 日韩人妻无码中文字幕视频 | 国产97在线 | 亚洲 | 免费无码午夜福利片69 | 人妻中文无码久热丝袜 | 亚洲精品美女久久久久久久 | 国产乱码精品一品二品 | 一本精品99久久精品77 | 国产亲子乱弄免费视频 | 色情久久久av熟女人妻网站 | 在线亚洲高清揄拍自拍一品区 | 人人妻人人澡人人爽欧美精品 | 久久亚洲中文字幕无码 | 激情五月综合色婷婷一区二区 | 国产精品久久久久久久影院 | 狠狠色丁香久久婷婷综合五月 | 麻豆国产丝袜白领秘书在线观看 | 色婷婷欧美在线播放内射 | 国产亚av手机在线观看 | 男女下面进入的视频免费午夜 | 人人澡人人透人人爽 | 精品久久久中文字幕人妻 | 中文字幕久久久久人妻 | 特黄特色大片免费播放器图片 | 国产手机在线αⅴ片无码观看 | 久久亚洲a片com人成 | 亚洲国产精品一区二区第一页 | 国产乱人伦偷精品视频 | 中文毛片无遮挡高清免费 | 久热国产vs视频在线观看 | 无码人妻久久一区二区三区不卡 | 精品国产青草久久久久福利 | 久久精品中文闷骚内射 | 99riav国产精品视频 | 国内少妇偷人精品视频免费 | 国产精品美女久久久 | 夜夜躁日日躁狠狠久久av | 国产精品美女久久久久av爽李琼 | 日本一区二区更新不卡 | 精品国产一区av天美传媒 | 亚洲精品久久久久avwww潮水 | 国产精品永久免费视频 | 成 人 免费观看网站 | 婷婷综合久久中文字幕蜜桃三电影 | 欧美日韩视频无码一区二区三 | 久久久久久a亚洲欧洲av冫 | 男女爱爱好爽视频免费看 | 十八禁视频网站在线观看 | 国产精品二区一区二区aⅴ污介绍 | 奇米影视7777久久精品 | 国产人成高清在线视频99最全资源 | 欧洲精品码一区二区三区免费看 | 国产精品毛片一区二区 | 国产精品-区区久久久狼 | 狠狠色噜噜狠狠狠狠7777米奇 | 玩弄人妻少妇500系列视频 | 亚洲精品午夜国产va久久成人 | 亚洲欧洲日本无在线码 | 成人免费视频一区二区 | 岛国片人妻三上悠亚 | 真人与拘做受免费视频一 | 国产成人无码一二三区视频 | 日日鲁鲁鲁夜夜爽爽狠狠 | 性欧美疯狂xxxxbbbb | 在线成人www免费观看视频 | 色综合久久网 | 一本久久a久久精品亚洲 | 熟女俱乐部五十路六十路av | 无码播放一区二区三区 | 亚洲精品国产a久久久久久 | 亚洲日韩av一区二区三区四区 | www国产精品内射老师 | 亚洲人成人无码网www国产 | 国产精品二区一区二区aⅴ污介绍 | 免费国产成人高清在线观看网站 | 午夜不卡av免费 一本久久a久久精品vr综合 | 色综合久久久无码中文字幕 | 久久久中文字幕日本无吗 | 久久久亚洲欧洲日产国码αv | 六月丁香婷婷色狠狠久久 | 国产高潮视频在线观看 | 国内精品人妻无码久久久影院蜜桃 | 性欧美疯狂xxxxbbbb | 5858s亚洲色大成网站www | 久久久久久av无码免费看大片 | 久久久成人毛片无码 | 国产精品亚洲专区无码不卡 | 久久人人爽人人爽人人片av高清 | 久久97精品久久久久久久不卡 | 无码毛片视频一区二区本码 | 国产sm调教视频在线观看 | 天堂а√在线中文在线 | 国产国产精品人在线视 | 乱人伦人妻中文字幕无码久久网 | 性做久久久久久久免费看 | 超碰97人人做人人爱少妇 | 黑人巨大精品欧美一区二区 | 国产精品va在线播放 | 亚洲自偷自拍另类第1页 | 久久精品女人天堂av免费观看 | 久久国产36精品色熟妇 | 97资源共享在线视频 | 国产精品无码一区二区桃花视频 | 国产 精品 自在自线 | 国产精品久久国产三级国 | 久久伊人色av天堂九九小黄鸭 | 精品一二三区久久aaa片 | 国产午夜亚洲精品不卡下载 | 丰满护士巨好爽好大乳 | 无遮挡啪啪摇乳动态图 | 丰满肥臀大屁股熟妇激情视频 | 亚洲高清偷拍一区二区三区 | 国产av一区二区精品久久凹凸 | 久久精品无码一区二区三区 | 午夜嘿嘿嘿影院 | 人人妻人人澡人人爽人人精品浪潮 | 狠狠色噜噜狠狠狠狠7777米奇 | 久久久亚洲欧洲日产国码αv | 中文亚洲成a人片在线观看 | 97se亚洲精品一区 | 18禁止看的免费污网站 | 在线精品国产一区二区三区 | 午夜成人1000部免费视频 | 免费国产成人高清在线观看网站 | 激情内射日本一区二区三区 | 一区二区三区乱码在线 | 欧洲 | www国产精品内射老师 | 在线看片无码永久免费视频 | 日韩成人一区二区三区在线观看 | 久久人妻内射无码一区三区 | 麻豆av传媒蜜桃天美传媒 | 久久这里只有精品视频9 | 中文字幕 亚洲精品 第1页 | 国产免费久久久久久无码 | 久久精品99久久香蕉国产色戒 | 亚洲国产精品久久久久久 | 奇米影视7777久久精品人人爽 | 亚洲a无码综合a国产av中文 | 精品国偷自产在线 | 人人妻人人澡人人爽欧美一区 | 亚洲欧洲日本综合aⅴ在线 | 成人精品一区二区三区中文字幕 | 亚洲日本一区二区三区在线 | 无码av最新清无码专区吞精 | 性色av无码免费一区二区三区 | 亚洲区小说区激情区图片区 | 国产舌乚八伦偷品w中 | 国产精品二区一区二区aⅴ污介绍 | 亚洲国产日韩a在线播放 | 日本熟妇浓毛 | 亚洲一区二区三区 | 精品一区二区三区无码免费视频 | 国产成人无码区免费内射一片色欲 | 国产av人人夜夜澡人人爽麻豆 | 久久zyz资源站无码中文动漫 | 精品无码av一区二区三区 | 亚洲精品国产品国语在线观看 | 久久久久久a亚洲欧洲av冫 | 久久国产精品精品国产色婷婷 | 国产av无码专区亚洲awww | 国产极品美女高潮无套在线观看 | 永久黄网站色视频免费直播 | 性色av无码免费一区二区三区 | 亚洲人成影院在线观看 | 国产免费无码一区二区视频 | 领导边摸边吃奶边做爽在线观看 | 牲欲强的熟妇农村老妇女视频 | 熟女俱乐部五十路六十路av | 无码成人精品区在线观看 | 国产成人无码区免费内射一片色欲 | 人人澡人摸人人添 | 1000部夫妻午夜免费 | 精品水蜜桃久久久久久久 | 久久天天躁夜夜躁狠狠 | 久久精品国产一区二区三区肥胖 | 国产精品鲁鲁鲁 | 九九在线中文字幕无码 | 粗大的内捧猛烈进出视频 | 国产真实乱对白精彩久久 | 国产乡下妇女做爰 | 日本一区二区更新不卡 | 好男人社区资源 | 国产精品99久久精品爆乳 | 色综合天天综合狠狠爱 | 久久午夜无码鲁丝片秋霞 | 麻豆蜜桃av蜜臀av色欲av | 成人片黄网站色大片免费观看 | 日本熟妇人妻xxxxx人hd | 国产午夜无码视频在线观看 | 久久久久久久久888 | 成熟妇人a片免费看网站 | 国产香蕉97碰碰久久人人 | 亚洲人成网站免费播放 | 在线观看欧美一区二区三区 | 人人超人人超碰超国产 | 亚洲国产精品一区二区美利坚 | 三上悠亚人妻中文字幕在线 | 国产无遮挡又黄又爽又色 | 国产精品嫩草久久久久 | 动漫av一区二区在线观看 | 天天摸天天碰天天添 | 四虎影视成人永久免费观看视频 | 欧美大屁股xxxxhd黑色 | a片免费视频在线观看 | 无码精品国产va在线观看dvd | 国产成人精品优优av | 综合人妻久久一区二区精品 | 天堂在线观看www | 国产午夜精品一区二区三区嫩草 | 性开放的女人aaa片 | 亚洲成色在线综合网站 | 成人女人看片免费视频放人 | 国语精品一区二区三区 | 欧美日本精品一区二区三区 | 国产一精品一av一免费 | 牲欲强的熟妇农村老妇女 | 国产激情艳情在线看视频 | 无码帝国www无码专区色综合 | 国产一区二区三区精品视频 | 国产免费久久精品国产传媒 | 99久久婷婷国产综合精品青草免费 | 亚洲乱码中文字幕在线 | 成人免费视频视频在线观看 免费 | 国产另类ts人妖一区二区 | 乌克兰少妇性做爰 | 青草青草久热国产精品 | 中文字幕 人妻熟女 | 激情人妻另类人妻伦 | 亚洲国产精品久久人人爱 | 男女猛烈xx00免费视频试看 | 在线观看国产一区二区三区 | 沈阳熟女露脸对白视频 | 国产超级va在线观看视频 | 国产亚洲精品久久久久久久久动漫 | 欧美 亚洲 国产 另类 | 2020久久超碰国产精品最新 | 少女韩国电视剧在线观看完整 | 国产suv精品一区二区五 | 图片小说视频一区二区 | 亚洲精品久久久久久久久久久 | 亚洲精品久久久久中文第一幕 | 巨爆乳无码视频在线观看 | 精品 日韩 国产 欧美 视频 | 永久免费精品精品永久-夜色 | 亚洲日本va午夜在线电影 | 欧美xxxx黑人又粗又长 | 俺去俺来也www色官网 | 亚洲精品国产精品乱码视色 | 澳门永久av免费网站 | 麻豆成人精品国产免费 | 精品国产aⅴ无码一区二区 | 鲁大师影院在线观看 | 蜜臀av在线观看 在线欧美精品一区二区三区 | 欧美老妇与禽交 | 伊人久久大香线焦av综合影院 | 欧美日韩久久久精品a片 | 国产特级毛片aaaaaa高潮流水 | 国产av剧情md精品麻豆 | 国产香蕉尹人综合在线观看 | 亚洲伊人久久精品影院 | 精品亚洲韩国一区二区三区 | 老司机亚洲精品影院无码 | 日本护士毛茸茸高潮 | 波多野42部无码喷潮在线 | 国产精品亚洲一区二区三区喷水 | 国产精品亚洲lv粉色 | 人妻无码αv中文字幕久久琪琪布 | 日本精品久久久久中文字幕 | 在线天堂新版最新版在线8 | 中文字幕人妻无码一夲道 | 中文字幕人成乱码熟女app | 少妇性俱乐部纵欲狂欢电影 | 爆乳一区二区三区无码 | 美女毛片一区二区三区四区 | 日韩少妇内射免费播放 | 亚洲码国产精品高潮在线 | 人人妻人人澡人人爽人人精品浪潮 | 国产精品多人p群无码 | 国产精品美女久久久久av爽李琼 | аⅴ资源天堂资源库在线 | 色五月丁香五月综合五月 | 国产熟妇另类久久久久 | 久久精品国产一区二区三区肥胖 | 国产精品无码一区二区三区不卡 | 亚洲欧美综合区丁香五月小说 | 荫蒂被男人添的好舒服爽免费视频 | 人妻有码中文字幕在线 | 亚洲第一无码av无码专区 | 激情内射亚州一区二区三区爱妻 | 国产香蕉97碰碰久久人人 | 免费观看黄网站 | 亚洲精品无码国产 | 亚洲人成无码网www | 四虎4hu永久免费 | 日韩人妻少妇一区二区三区 | 丰满少妇弄高潮了www | 日本熟妇大屁股人妻 | 久久综合给合久久狠狠狠97色 | 欧美成人午夜精品久久久 | 免费无码一区二区三区蜜桃大 | 国内精品人妻无码久久久影院 | a国产一区二区免费入口 | 亚洲狠狠色丁香婷婷综合 | 午夜精品久久久内射近拍高清 | 少妇被粗大的猛进出69影院 | 捆绑白丝粉色jk震动捧喷白浆 | 中国女人内谢69xxxx | 免费乱码人妻系列无码专区 | 久热国产vs视频在线观看 | 四十如虎的丰满熟妇啪啪 | 麻豆av传媒蜜桃天美传媒 | 国产精品办公室沙发 | 人妻互换免费中文字幕 | 国产精品久久久久久亚洲毛片 | 美女扒开屁股让男人桶 | 亚洲日本va中文字幕 | 少妇太爽了在线观看 | 日本精品高清一区二区 | 亚洲中文字幕无码中字 | 性欧美牲交xxxxx视频 | 亚洲精品成a人在线观看 | 未满小14洗澡无码视频网站 | 久久久久99精品成人片 | 强辱丰满人妻hd中文字幕 | 欧美野外疯狂做受xxxx高潮 | 亚洲色欲久久久综合网东京热 | 台湾无码一区二区 | 精品人妻中文字幕有码在线 | 国产精品亚洲专区无码不卡 | 亚洲综合另类小说色区 | 色综合久久网 | 色综合久久久无码中文字幕 | 久久久精品欧美一区二区免费 | 性欧美牲交在线视频 | 久久综合九色综合欧美狠狠 | 一本精品99久久精品77 | 日韩欧美中文字幕公布 | 日本护士xxxxhd少妇 | 国产亚洲日韩欧美另类第八页 | 夜夜夜高潮夜夜爽夜夜爰爰 | 国产精品资源一区二区 | 国产精品亚洲专区无码不卡 | 天天摸天天透天天添 | 老子影院午夜精品无码 | 99久久精品日本一区二区免费 | 中文字幕久久久久人妻 | 日本一区二区更新不卡 | 狠狠综合久久久久综合网 | 5858s亚洲色大成网站www | 西西人体www44rt大胆高清 | 国产婷婷色一区二区三区在线 | 精品久久久无码中文字幕 | 内射爽无广熟女亚洲 | 成人性做爰aaa片免费看不忠 | 国产精品资源一区二区 | 在教室伦流澡到高潮hnp视频 | 国产亚洲视频中文字幕97精品 | 国产精品鲁鲁鲁 | 骚片av蜜桃精品一区 | 国产免费观看黄av片 | 中文字幕乱码亚洲无线三区 | 午夜无码区在线观看 | 日韩精品一区二区av在线 | 免费无码av一区二区 | 色综合久久久无码中文字幕 |