android 移植ffmpeg后so库的使用

  只需要将我们编译好的ffmpeg的so包(在/obj/local/armeabi/libffmpeg.so)copy到所在ndk下的\platforms\android-8\arch-arm\usr\lib文件夹下就可以了。
  再看下我们的代码吧:
  1. public class HelloJni extends Activity
  2. {
  3. /** Called when the activity is first created. */
  4. @Override
  5. public void onCreate(Bundle savedInstanceState)
  6. {
  7. super.onCreate(savedInstanceState);
  8. /* Create a TextView and set its content.
  9. * the text is retrieved by calling a native
  10. * function.
  11. */
  12. TextView  tv = new TextView(this);
  13. String str = String.valueOf(stringFromJNI());
  14. str = str +"\n"+getVoice();
  15. tv.setText(str);
  16. setContentView(tv);
  17. }
  18. public native int getVoice();
  19. public native String  stringFromJNI();
  20. public native String  unimplementedStringFromJNI();
  21. static {
  22. System.loadLibrary("ffmpeg");
  23. System.loadLibrary("hello-jni");
  24. }
  25. }

这个是Android上层的使用。下面附上c层的代码

  1. /*
  2. * Copyright (C) 2009 The Android Open Source Project
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. *      http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. *
  16. */
  17. #include <string.h>
  18. #include <jni.h>
  19. #include <android/log.h>
  20. #include <ffmpeg/libavcodec/avcodec.h>
  21. #include <ffmpeg/libavformat/avformat.h>
  22. #include <ffmpeg/libswscale/swscale.h>
  23. #include <string.h>
  24. #define LOG_TAG    "***************************lixingyun*****************" // 这个是自定义的LOG的标识
  25. #undef LOG // 取消默认的LOG
  26. #define LOGD(...)  __android_log_print(ANDROID_LOG_DEBUG,LOG_TAG,__VA_ARGS__) // 定义LOG类型
  27. #define LOGI(...)  __android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__) // 定义LOG类型
  28. #define LOGW(...)  __android_log_print(ANDROID_LOG_WARN,LOG_TAG,__VA_ARGS__) // 定义LOG类型
  29. #define LOGE(...)  __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__) // 定义LOG类型
  30. #define LOGF(...)  __android_log_print(ANDROID_LOG_FATAL,LOG_TAG,__VA_ARGS__) // 定义LOG类型
  31. /* This is a trivial JNI example where we use a native method
  32. * to return a new VM String. See the corresponding Java source
  33. * file located at:
  34. *
  35. *   apps/samples/hello-jni/project/src/com/example/HelloJni/HelloJni.java
  36. */
  37. JNIEXPORT jstring Java_com_example_hellojni_HelloJni_stringFromJNI(JNIEnv* env,
  38. jobject thiz) {
  39. char str[25];
  40. sprintf(str, "%d", avcodec_version());
  41. return (*env)->NewStringUTF(env, str);
  42. // return (*env)->NewStringUTF(env, "Hello from JNI !");0
  43. }
  44. JNIEXPORT jint Java_com_example_hellojni_HelloJni_getVoice(JNIEnv* env,
  45. jobject thiz) {
  46. int flag = 0;
  47. AVPacket aacpkt;
  48. aacpkt.data = NULL;
  49. aacpkt.size = 0;
  50. const char *filename = "file:/sdcard/test.mp3";
  51. av_init_packet(&aacpkt);
  52. av_register_all(); //注册所有可解码类型
  53. AVFormatContext *pInFmtCtx = NULL; //文件格式
  54. AVCodecContext *pInCodecCtx = NULL; //编码格式
  55. pInFmtCtx = avformat_alloc_context();
  56. if (av_open_input_file(&pInFmtCtx, filename, NULL, 0, NULL) != 0) //获取文件格式
  57. LOGE("av_open_input_file error\n");
  58. if (av_find_stream_info(pInFmtCtx) < 0) //获取文件内音视频流的信息
  59. LOGE("av_find_stream_info error\n");
  60. unsigned int j;
  61. // Find the first audio stream
  62. int audioStream = -1;
  63. // Dump information about file onto standard error
  64. dump_format(pInFmtCtx, 0, filename, 0);
  65. //从FormatdContext的中找到对应流对应的编码类型,若是CODEC_TYPE_AUDIO则表示是音频
  66. for (j = 0; j < pInFmtCtx->nb_streams; j++) //找到音频对应的stream
  67. if (pInFmtCtx->streams[j]->codec->codec_type == CODEC_TYPE_AUDIO) {
  68. audioStream = j;
  69. break;
  70. }
  71. if (audioStream == -1) {
  72. printf("input file has no audio stream\n");
  73. return 0; // Didn't find a audio stream
  74. }
  75. LOGE("audio stream num: %d\n", audioStream);
  76. pInCodecCtx = pInFmtCtx->streams[audioStream]->codec; //音频的编码上下文
  77. AVCodec *pInCodec = NULL;
  78. /* FILE *file3 = fopen("codec_private_data_size.txt","w");
  79. for(int i = 0; i <200000; i++)
  80. {
  81. pInCodec = avcodec_find_decoder((CodecID)i);
  82. if (pInCodec!=NULL)
  83. {
  84. fprintf(file3,"%s %d\n",pInCodec->name,pInCodec->priv_data_size );
  85. }
  86. }
  87. fclose(file3);
  88. system("pause");
  89. */
  90. pInCodec = avcodec_find_decoder(pInCodecCtx->codec_id); //根据编码ID找到用于解码的结构体
  91. if (pInCodec == NULL) {
  92. printf("error no Codec found\n");
  93. return -1; // Codec not found
  94. }
  95. //使用test的代替pInCodecCtx也可以完成解码,可以看出只要获取以下几个重要信息就可以实现解码和重采样
  96. AVCodecContext *test = avcodec_alloc_context();
  97. test->bit_rate = pInCodecCtx->bit_rate; //重采样用
  98. test->sample_rate = pInCodecCtx->sample_rate; //重采样用
  99. test->channels = pInCodecCtx->channels; //重采样用
  100. test->extradata = pInCodecCtx->extradata; //若有则必有
  101. test->extradata_size = pInCodecCtx->extradata_size; //若有则必要
  102. test->codec_type = CODEC_TYPE_AUDIO; //不必要
  103. test->block_align = pInCodecCtx->block_align; //必要
  104. if (avcodec_open(test, pInCodec) < 0) //将两者结合以便在下面的解码函数中调用pInCodec中的对应解码函数
  105. {
  106. printf("error avcodec_open failed.\n");
  107. return -1; // Could not open codec
  108. }
  109. if (avcodec_open(pInCodecCtx, pInCodec) < 0) {
  110. printf("error avcodec_open failed.\n");
  111. return -1; // Could not open codec
  112. }
  113. static AVPacket packet;
  114. LOGI(" bit_rate = %d \r\n", pInCodecCtx->bit_rate);
  115. LOGI(" sample_rate = %d \r\n", pInCodecCtx->sample_rate);
  116. LOGI(" channels = %d \r\n", pInCodecCtx->channels);
  117. LOGI(" code_name = %s \r\n", pInCodecCtx->codec->name);
  118. //LOGI("extra data size: %d :data%x %x %x %x\n",pInCodecCtx->extradata_size,pInCodecCtx->extradata[0]
  119. // ,pInCodecCtx->extradata[1],pInCodecCtx->extradata[2],pInCodecCtx->extradata[3]);
  120. LOGI(" block_align = %d\n", pInCodecCtx->block_align);
  121. /********************************************************/
  122. unsigned char *outbuf = NULL;
  123. outbuf = (unsigned char *) malloc(AVCODEC_MAX_AUDIO_FRAME_SIZE * 2);
  124. int outsize;
  125. outsize = AVCODEC_MAX_AUDIO_FRAME_SIZE * 2;
  126. int len = 0;
  127. int size = 0;
  128. FILE *out_fp;
  129. out_fp = fopen("out.dat", "wb+");
  130. while (av_read_frame(pInFmtCtx, &aacpkt) >= 0) {
  131. LOGI("***************************");
  132. if (aacpkt.stream_index == audioStream) {
  133. int declen = 0;
  134. size = aacpkt.size;
  135. while (aacpkt.size > 0) {
  136. outsize = AVCODEC_MAX_AUDIO_FRAME_SIZE * 2;
  137. memset(outbuf, 0, outsize);
  138. len = avcodec_decode_audio3(pInCodecCtx, (int16_t *) outbuf,
  139. &outsize, &aacpkt);
  140. if (len < 0) {
  141. printf("avcodec_decode_audio3 failed!\n");
  142. break;
  143. }
  144. if (outsize > 0) {
  145. fwrite(outbuf, 1, outsize, out_fp);
  146. printf("write %d bytes\n", outsize);
  147. }
  148. declen += len;
  149. if (declen == size) {
  150. av_free_packet(&aacpkt);
  151. printf("packet decoded succeed!\n");
  152. break;
  153. } else if (declen < size) {
  154. aacpkt.size -= len;
  155. aacpkt.data += len;
  156. } else {
  157. printf("decode error!\n");
  158. break;
  159. }
  160. }
  161. }
  162. }
  163. fclose(out_fp);
  164. // Close the codec
  165. avcodec_close(pInCodecCtx);
  166. // Close the video file
  167. av_close_input_file(pInFmtCtx);
  168. return flag;
  169. }
  其中的LOGE那几个就是Android下的哦。
  再附上Android.mk的东西吧:

  1. LOCAL_PATH := $(call my-dir)
  2. include $(CLEAR_VARS)
  3. PATH_TO_FFMPEG_SOURCE:=$(LOCAL_PATH)/ffmpeg
  4. LOCAL_C_INCLUDES += $(PATH_TO_FFMPEG_SOURCE)
  5. LOCAL_LDLIBS := -lffmpeg -llog
  6. LOCAL_MODULE    := hello-jni
  7. LOCAL_SRC_FILES := hello-jni.c
  8. include $(BUILD_SHARED_LIBRARY)
 
哈哈,这样就Ok了。
源码也附上吧.
 

FFmpeg在Android使用3的更多相关文章

  1. How to Build FFmpeg for Android

    http://www.roman10.net/how-to-build-ffmpeg-for-android/ ffmpeg is an open-source platform for record ...

  2. [原]ffmpeg编译android 硬解码支持库 libstagefright

    最近花了一天时间将ffmpeg/tools/build_stagefright执行成功,主要是交叉编译所需要的各种动态库的支持没链接上,导致各种报错,基本上网络上问到的问题我都碰到了,特此记录下来. ...

  3. FFmpeg在Android上的移植之第一步

    http://blog.sina.com.cn/s/blog_69a04cf40100x1fr.html 从事多媒体软件开发的人几乎没有不知道FFmpeg的,很多视频播放器都是基于FFmpeg开发的. ...

  4. FFmpeg的Android平台移植—编译篇

    摘要:本文主要介绍将FFmpeg音视频编解码库移植到Android平台上的编译和基本测试过程. 环境准备: ubuntu-12.04.5 android-ndk64-r10-linux-x86_64. ...

  5. FFmpeg在Android上的移植优化步骤

    http://blog.csdn.net/feixiang_john/article/details/7894188 从事多媒体软件开发的人几乎没有不知道FFmpeg的,很多视频播放器都是基于FFmp ...

  6. FFmpeg for Android compiled with x264, libass, fontconfig, freetype and fribidi

    android下打算使用ffmpeg的 drawtext ,不过需要 --enable-libfreetype  但是freetype是个第三方库,所以需要先编译freetype,然后再编译ffmpe ...

  7. FFmpeg编译Android版本

    FFmpeg是很好用的一个音视频库,功能强大,但是用起来并不是很方便.之前一直不想用FFmpeg,因为感觉编译太麻烦,但是到了不得不用的时候了,没办法,参考了网上大神的方法,在这里自己也记录一下方便以 ...

  8. FFmpeg 移植 Android

    近期项目需要解析苹果的HLS流媒体协议,而FFmpeg从0.11.1“Happiness”版本开始,才增加了对HLS协议的支持.目前网上关于FFmpeg编译移植的文章有很多,但大多都是对旧版本的说明. ...

  9. Mac下通过FFMpeg实现Android手机推流和播放

    一.Mac下搭建推流服务器(Nginx+RTMP+FFMpeg) 安装x264 git clone git://git.videolan.org/x264.git cd x264 ./configur ...

随机推荐

  1. JS内存管理测试

    打开调试器,切换到timer,点击左下角的record按钮开始,切换到memory视图,在文档中点击鼠标左右键,看股价走势图 function Allocate(kbs){ this.mem = ne ...

  2. Kafka之ReplicaManager(1)

    基于Kafka 0.9.0版 ReplicaManager需要做什么 Replicated Logs Kafka的partition可以看成是一个replicated log, 每个replica就是 ...

  3. C++ char*,const char*,string,int 的相互转换

    C++ char*,const char*,string,int 的相互转换   1. string转const char* string s ="abc";const char* ...

  4. Oracle日期函数

    Oracle日期函数用于对Oracle数据库中的日期及时间进行处理. (1)ADD_MONTHS Oracle日期函数返回一个具有与所提供日期相差月份的日期,函数中给出了未来或以前的月份数.语法如下: ...

  5. hdu1022 Train Problem I

    http://acm.hdu.edu.cn/showproblem.php?pid=1022 #include<iostream> #include<stdio.h> #inc ...

  6. 欧拉工程第62题:Cubic permutations

    题目链接 找出最小的立方数,它的各位数的排列能够形成五个立方数 解决关键点: 这五个数的由相同的数组成的 可以用HashMap,Key是由各位数字形成的key,value记录由这几个数组成的立方数出现 ...

  7. JavaScript基础精华01(变量,语法,数据类型)

    JavaScript是一种脚本语言. 脚本,一条条的文字命令.执行时由系统的一个解释器,将其一条条的翻译成机器可识别的指令,然后执行 JavaScript基本组成 1.基本语法(浏览器基本都支持,有统 ...

  8. NSSize

    #import <Foundation/Foundation.h>   int main(int argc, const char * argv[]) {    @autoreleasep ...

  9. sql server 自定义函数的使用

    sql server 自定义函数的使用 自定义函数 用户定义自定义函数像内置函数一样返回标量值,也可以将结果集用表格变量返回 用户自定义函数的类型: 标量函数:返回一个标量值 表格值函数{内联表格值函 ...

  10. Intellij IDEA新建一个EJB工程(二)

    从博文:http://www.cnblogs.com/yangyquin/p/5328344.html 中可以知道如何利用Intellij IDEA建立一个EJB Module,还有新建一个测试Mod ...