parent
39808aca1c
commit
e394737800
@ -0,0 +1,178 @@ |
||||
//
|
||||
// Created by frank on 2018/2/2.
|
||||
//
|
||||
|
||||
#include <jni.h> |
||||
#include <string> |
||||
#include <android/log.h> |
||||
|
||||
#define LOG_TAG "FFmpegLive" |
||||
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__) |
||||
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__) |
||||
|
||||
#ifdef __cplusplus |
||||
extern "C" { |
||||
#endif |
||||
#include <libavcodec/avcodec.h> |
||||
#include <libavformat/avformat.h> |
||||
#include <libavfilter/avfilter.h> |
||||
#include <libavutil/time.h> |
||||
|
||||
JNIEXPORT jint JNICALL |
||||
Java_com_frank_ffmpeg_Pusher_pushStream(JNIEnv *env, jobject, jstring filePath, jstring liveUrl) { |
||||
|
||||
AVOutputFormat *output_format = NULL; |
||||
AVFormatContext *in_format = NULL, *out_format = NULL; |
||||
AVPacket packet; |
||||
const char *file_path, *live_url; |
||||
int video_index = -1; |
||||
int ret = 0, i; |
||||
int frame_index = 0; |
||||
int64_t start_time = 0; |
||||
|
||||
file_path = env->GetStringUTFChars(filePath, NULL); |
||||
live_url = env->GetStringUTFChars(liveUrl, NULL); |
||||
|
||||
LOGI("file_path=%s", file_path); |
||||
LOGI("live_url=%s", live_url); |
||||
|
||||
//注册所有组件
|
||||
av_register_all(); |
||||
//初始化网络
|
||||
avformat_network_init(); |
||||
//打开输入文件
|
||||
if((ret = avformat_open_input(&in_format, file_path, 0, 0)) < 0){ |
||||
LOGE("could not open input file..."); |
||||
goto end; |
||||
} |
||||
//寻找流信息
|
||||
if((ret = avformat_find_stream_info(in_format, 0)) < 0){ |
||||
LOGE("could not find stream info..."); |
||||
goto end; |
||||
} |
||||
//
|
||||
for(i=0; i<in_format->nb_streams; i++){ |
||||
//找到视频流
|
||||
if(in_format->streams[i]->codec->codec_type == AVMEDIA_TYPE_VIDEO){ |
||||
video_index = i; |
||||
break; |
||||
} |
||||
} |
||||
av_dump_format(in_format, 0, file_path, 0); |
||||
//分配输出封装格式上下文, rtmp协议支持格式为flv
|
||||
avformat_alloc_output_context2(&out_format, NULL, "flv", live_url); |
||||
if(!out_format){ |
||||
LOGE("could not alloc output context..."); |
||||
ret = AVERROR_UNKNOWN; |
||||
goto end; |
||||
} |
||||
//根据输入流来创建输出流
|
||||
for(i=0; i<in_format->nb_streams; i++){ |
||||
AVStream *in_stream = in_format->streams[i]; |
||||
AVStream *out_stream = avformat_new_stream(out_format, in_stream->codec->codec); |
||||
if(!out_stream){ |
||||
LOGE("could not alloc output stream..."); |
||||
ret = AVERROR_UNKNOWN; |
||||
goto end; |
||||
} |
||||
//复制封装格式上下文
|
||||
ret = avcodec_copy_context(out_stream->codec, in_stream->codec); |
||||
if(ret < 0){ |
||||
LOGE("could not copy context..."); |
||||
goto end; |
||||
} |
||||
out_stream->codec->codec_tag = 0; |
||||
if(out_format->oformat->flags & AVFMT_GLOBALHEADER){ |
||||
out_stream->codec->flags |= CODEC_FLAG_GLOBAL_HEADER; |
||||
} |
||||
} |
||||
//封装格式
|
||||
output_format = out_format->oformat; |
||||
//打开输出文件/URL
|
||||
if(!(output_format->flags & AVFMT_NOFILE)){ |
||||
ret = avio_open(&out_format->pb, live_url, AVIO_FLAG_WRITE); |
||||
if(ret < 0){ |
||||
LOGE("could not open output url '%s'", live_url); |
||||
goto end; |
||||
} |
||||
} |
||||
//写文件头
|
||||
ret = avformat_write_header(out_format, NULL); |
||||
if(ret < 0){ |
||||
LOGE("could not write header..."); |
||||
goto end; |
||||
} |
||||
//获取开始时间
|
||||
start_time = av_gettime(); |
||||
//开始循环读一帧数据
|
||||
while (1){ |
||||
AVStream *in_stream, *out_stream; |
||||
ret = av_read_frame(in_format, &packet); |
||||
if(ret < 0){ |
||||
break; |
||||
} |
||||
//计算帧间隔,参考时钟/采样率
|
||||
if(packet.pts == AV_NOPTS_VALUE){ |
||||
AVRational time_base = in_format->streams[video_index]->time_base; |
||||
int64_t cal_duration = (int64_t)(AV_TIME_BASE/av_q2d(in_format->streams[video_index]->r_frame_rate)); |
||||
packet.pts = (int64_t)((frame_index * cal_duration)/(av_q2d(time_base) * AV_TIME_BASE)); |
||||
packet.dts = packet.pts; |
||||
packet.duration = (int64_t)(cal_duration/(av_q2d(time_base) * AV_TIME_BASE)); |
||||
} |
||||
//视频帧之间延时
|
||||
if(packet.stream_index == video_index){ |
||||
AVRational time_base = in_format->streams[video_index]->time_base; |
||||
AVRational time_base_q = {1, AV_TIME_BASE}; |
||||
int64_t pts_time = av_rescale_q(packet.dts, time_base, time_base_q); |
||||
int64_t now_time = av_gettime() - start_time; |
||||
//延时以保持同步
|
||||
if(pts_time > now_time){ |
||||
av_usleep((unsigned int)(pts_time - now_time)); |
||||
} |
||||
} |
||||
in_stream = in_format->streams[packet.stream_index]; |
||||
out_stream = out_format->streams[packet.stream_index]; |
||||
|
||||
//pts/dts转换
|
||||
packet.pts = av_rescale_q_rnd(packet.pts, in_stream->time_base, out_stream->time_base, |
||||
(AVRounding)(AV_ROUND_NEAR_INF | AV_ROUND_PASS_MINMAX)); |
||||
packet.dts = av_rescale_q_rnd(packet.dts, in_stream->time_base, out_stream->time_base, |
||||
(AVRounding)(AV_ROUND_NEAR_INF | AV_ROUND_PASS_MINMAX)); |
||||
packet.duration = av_rescale_q(packet.duration, in_stream->time_base, out_stream->time_base); |
||||
packet.pos = -1; |
||||
|
||||
//视频帧计数
|
||||
if(packet.stream_index == video_index){ |
||||
frame_index ++; |
||||
LOGI("write frame = %d", frame_index); |
||||
} |
||||
//写一帧数据
|
||||
ret = av_interleaved_write_frame(out_format, &packet); |
||||
if(ret < 0){ |
||||
LOGE("could not write frame..."); |
||||
break; |
||||
} |
||||
//释放包数据内存
|
||||
av_packet_unref(&packet); |
||||
} |
||||
//写文件尾
|
||||
av_write_trailer(out_format); |
||||
|
||||
end: |
||||
avformat_close_input(&in_format); |
||||
if(out_format && !(out_format->flags & AVFMT_NOFILE)){ |
||||
avio_close(out_format->pb); |
||||
} |
||||
avformat_free_context(in_format); |
||||
avformat_free_context(out_format); |
||||
env->ReleaseStringUTFChars(filePath, file_path); |
||||
env->ReleaseStringUTFChars(liveUrl, live_url); |
||||
if(ret < 0 && ret != AVERROR_EOF){ |
||||
return -1; |
||||
} |
||||
return 0; |
||||
} |
||||
|
||||
#ifdef __cplusplus |
||||
} |
||||
#endif |
@ -0,0 +1,16 @@ |
||||
package com.frank.ffmpeg; |
||||
|
||||
/** |
||||
* 使用FFmepg进行推流直播 |
||||
* Created by frank on 2018/2/2. |
||||
*/ |
||||
|
||||
public class Pusher { |
||||
static { |
||||
System.loadLibrary("media-handle"); |
||||
} |
||||
|
||||
//选择本地文件推流到指定平台直播
|
||||
public native int pushStream(String filePath, String liveUrl); |
||||
|
||||
} |
@ -0,0 +1,72 @@ |
||||
package com.frank.ffmpeg.activity; |
||||
|
||||
import android.support.v7.app.AppCompatActivity; |
||||
import android.os.Bundle; |
||||
import android.text.TextUtils; |
||||
import android.util.Log; |
||||
import android.view.View; |
||||
import android.widget.Button; |
||||
import android.widget.EditText; |
||||
import android.widget.Toast; |
||||
|
||||
import com.frank.ffmpeg.Pusher; |
||||
import com.frank.ffmpeg.R; |
||||
|
||||
import java.io.File; |
||||
|
||||
/** |
||||
* 使用ffmpeg推流直播 |
||||
* Created by frank on 2018/2/2. |
||||
*/ |
||||
public class PushActivity extends AppCompatActivity { |
||||
|
||||
private static final String TAG = PushActivity.class.getSimpleName(); |
||||
private static final String FILE_PATH = "storage/emulated/0/hello.flv"; |
||||
private static final String LIVE_URL = "rtmp://192.168.1.104/live/stream"; |
||||
|
||||
@Override |
||||
protected void onCreate(Bundle savedInstanceState) { |
||||
super.onCreate(savedInstanceState); |
||||
setContentView(R.layout.activity_push); |
||||
|
||||
initView(); |
||||
} |
||||
|
||||
private void initView() { |
||||
final EditText edit_file_path = (EditText) findViewById(R.id.edit_file_path); |
||||
final EditText edit_live_url = (EditText) findViewById(R.id.edit_live_url); |
||||
edit_file_path.setText(FILE_PATH); |
||||
edit_live_url.setText(LIVE_URL); |
||||
|
||||
Button btn_push_stream = (Button)findViewById(R.id.btn_push_stream); |
||||
|
||||
btn_push_stream.setOnClickListener(new View.OnClickListener() { |
||||
@Override |
||||
public void onClick(View v) { |
||||
//TODO 视频文件格式为flv
|
||||
final String filePath = edit_file_path.getText().toString(); |
||||
final String liveUrl = edit_live_url.getText().toString(); |
||||
Log.i(TAG, "filePath=" + filePath); |
||||
Log.i(TAG, "liveUrl=" + liveUrl); |
||||
|
||||
if(!TextUtils.isEmpty(filePath) && !TextUtils.isEmpty(filePath)){ |
||||
File file = new File(filePath); |
||||
//判断文件是否存在
|
||||
if(file.exists()){ |
||||
//开启子线程
|
||||
new Thread(new Runnable() { |
||||
@Override |
||||
public void run() { |
||||
//开始推流
|
||||
new Pusher().pushStream(filePath, liveUrl); |
||||
} |
||||
}).start(); |
||||
}else { |
||||
Toast.makeText(PushActivity.this, "文件不存在", Toast.LENGTH_SHORT).show(); |
||||
} |
||||
} |
||||
} |
||||
}); |
||||
} |
||||
|
||||
} |
@ -0,0 +1,35 @@ |
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" |
||||
xmlns:tools="http://schemas.android.com/tools" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="match_parent" |
||||
tools:context="com.frank.ffmpeg.activity.PushActivity"> |
||||
|
||||
<EditText |
||||
android:id="@+id/edit_file_path" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:maxLines="1" |
||||
android:minWidth="200dp" |
||||
android:layout_centerInParent="true"/> |
||||
|
||||
|
||||
<EditText |
||||
android:id="@+id/edit_live_url" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:maxLines="1" |
||||
android:minWidth="200dp" |
||||
android:layout_marginTop="10dp" |
||||
android:layout_below="@+id/edit_file_path"/> |
||||
|
||||
<Button |
||||
android:id="@+id/btn_push_stream" |
||||
android:layout_width="wrap_content" |
||||
android:layout_height="wrap_content" |
||||
android:layout_marginTop="10dp" |
||||
android:layout_below="@+id/edit_live_url" |
||||
android:layout_centerHorizontal="true" |
||||
android:text="@string/video_push"/> |
||||
|
||||
</RelativeLayout> |
After Width: | Height: | Size: 6.7 MiB |
Binary file not shown.
After Width: | Height: | Size: 157 KiB |
Loading…
Reference in new issue