[Android开发]利用Cmake在AndroidStudio来使用lame库

之前项目需要将录音的数据转换成MP3文件,考虑一番后决定使用Lame库进行转换。网上多采用ndk-build对其进行构建,但AndroidStudio支持Cmake有一段时间,此次使用Cmake来试试。

准备:
1.在AndroidSudio上装好Cmake和NDK 可以参考之前的文章
2.下载Lame的源码 官方地址需要自备梯子

一、修改Lame的部分内容
将Lame的源码解压后,把libmp3lame文件夹下除了.h和.c的文件都去掉,vector和i386文件夹也都去掉。并将在libmp3lame里剩下的文件,都复制到AS的cpp目录下。同时还要将lame-3.99.5\include\lame.h这个头文件也复制过去。到为了好管理,可以在cpp下新建一个文件夹把这些源码也放在一起。


需要修改的部分:
1 、util.h中574行将里面的一行 extern ieee754_float32_t fast_log2(ieee754_float32_t x); 改為 extern float fast_log2(float x); 因为Android下并不支持该类型

2、在id3tag.c和machine.h两个文件里,將HAVE_STRCHR和HAVE_MEMCPY的ifdef结构体注释掉。

#ifdef STDC_HEADERS
# include <stdlib.h>
# include <string.h>
#else
/*# ifndef HAVE_STRCHR
#  define strchr index
#  define strrchr rindex
# endif*/
char   *strchr(), *strrchr();
/*# ifndef HAVE_MEMCPY
#  define memcpy(d, s, n) bcopy ((s), (d), (n))
#  define memmove(d, s, n) bcopy ((s), (d), (n))
# endif*/
#endif

3、fft.c中47行将vector/lame_intrin.h这个头文件注释了或者去掉

#ifdef HAVE_CONFIG_H
# include <config.h>
#endif

#include "lame.h"
#include "machine.h"
#include "encoder.h"
#include "util.h"
#include "fft.h"

//#include "vector/lame_intrin.h"

4、set_get.h中24行将include <lame.h>改为include "lame.h"

#ifndef __SET_GET_H__
#define __SET_GET_H__

#include "lame.h"

二、编写CmakeList.txt
本例中lame的源码和头文件都放在了src/main/cpp/lamemp3这个文件夹下。在添加自定义库既add_library时,可以先指定一个目录,然后将这个目录传入add_library中,避免一个个文件添加这么麻烦。另外lame本身并没有jni的api,需要自己去编写,本例中在SimpleLame.cpp中调用lame的api进行简单的编码转换,让PCM文件换为MP3文件。


cmake_minimum_required(VERSION 3.4.1)

#设置变量SRC_DIR为lamemp3的所在路径
set(SRC_DIR src/main/cpp/lamemp3)

#指定头文件所在,可以多次调用,指定多个路径
include_directories(src/main/cpp/lamemp3)

#添加自自定义的so库时,有两种方式,一种添加一个目录,一种一个个文件添加

#设定一个目录
aux_source_directory(src/main/cpp/lamemp3 SRC_LIST)

#将前面目录下所有的文件都添加进去
add_library(lamemp3 SHARED src/main/cpp/SimpleLame.cpp ${SRC_LIST})

#一个个文件的加
#add_library(lame-mp3
#            SHARED
#            ${SRC_DIR}/bitstream.c
#            ${SRC_DIR}/encoder.c
#            ${SRC_DIR}/fft.c
#            ${SRC_DIR}/gain_analysis.c
#            ${SRC_DIR}/id3tag.c
#            ${SRC_DIR}/lame.c
#            ${SRC_DIR}/mpglib_interface.c
#            ${SRC_DIR}/newmdct.c
#            ${SRC_DIR}/presets.c
#            ${SRC_DIR}/psymodel.c
#            ${SRC_DIR}/quantize.c
#            ${SRC_DIR}/quantize_pvt.c
#            ${SRC_DIR}/reservoir.c
#            ${SRC_DIR}/set_get.c
#            ${SRC_DIR}/tables.c
#            ${SRC_DIR}/takehiro.c
#            ${SRC_DIR}/util.c
#            ${SRC_DIR}/vbrquantize.c
#            ${SRC_DIR}/VbrTag.c
#            ${SRC_DIR}/version.c
#            )

find_library(log-lib log )

三、调用Lame的本地代码,SimpleLame.cpp

#include <cwchar>
#include "SimpleLame.h"
#include "lamemp3/lame.h"

static lame_global_flags *glf = NULL;

void Java_com_clam314_lame_SimpleLame_close(JNIEnv *env, jclass type){
    lame_close(glf);
    glf = NULL;
}

jint Java_com_clam314_lame_SimpleLame_encode(JNIEnv *env, jclass type, jshortArray buffer_l_,
                                        jshortArray buffer_r_, jint samples, jbyteArray mp3buf_) {
    jshort *buffer_l = env->GetShortArrayElements(buffer_l_, NULL);
    jshort *buffer_r = env->GetShortArrayElements(buffer_r_, NULL);
    jbyte *mp3buf = env->GetByteArrayElements(mp3buf_, NULL);

    const jsize mp3buf_size = env->GetArrayLength(mp3buf_);

    int result =lame_encode_buffer(glf, buffer_l, buffer_r, samples, (u_char*)mp3buf, mp3buf_size);

    env->ReleaseShortArrayElements(buffer_l_, buffer_l, 0);
    env->ReleaseShortArrayElements(buffer_r_, buffer_r, 0);
    env->ReleaseByteArrayElements(mp3buf_, mp3buf, 0);

    return result;
}

jint Java_com_clam314_lame_SimpleLame_flush(JNIEnv *env, jclass type, jbyteArray mp3buf_) {
    jbyte *mp3buf = env->GetByteArrayElements(mp3buf_, NULL);

    const jsize  mp3buf_size = env->GetArrayLength(mp3buf_);

    int result = lame_encode_flush(glf, (u_char*)mp3buf, mp3buf_size);

    env->ReleaseByteArrayElements(mp3buf_, mp3buf, 0);

    return result;
}

void Java_com_clam314_lame_SimpleLame_init__IIIII(JNIEnv *env, jclass type, jint inSampleRate, jint outChannel,
                                             jint outSampleRate, jint outBitrate, jint quality) {
    if(glf != NULL){
        lame_close(glf);
        glf = NULL;
    }
    glf = lame_init();
    lame_set_in_samplerate(glf, inSampleRate);
    lame_set_num_channels(glf, outChannel);
    lame_set_out_samplerate(glf, outSampleRate);
    lame_set_brate(glf, outBitrate);
    lame_set_quality(glf, quality);
    lame_init_params(glf);
}

四、调用Lame对应的Java代码

public class SimpleLame {

    public native static void close();

    public native static int encode(short[] buffer_l, short[] buffer_r, int samples, byte[] mp3buf);

    public native static int flush(byte[] mp3buf);

    public native static void init(int inSampleRate, int outChannel, int outSampleRate, int outBitrate, int quality);

    public static void init(int inSampleRate, int outChannel, int outSampleRate, int outBitrate) {
        init(inSampleRate, outChannel, outSampleRate, outBitrate, 7);
    }
}

这里重点介绍Lame库的移植,完整的实现录音-转码-播放的代码放到github上了
项目地址:https://github.com/clam314/LameMp3ForAndroid

推荐阅读更多精彩内容

  • 貌似许多人都是从lame库开始入门Android NDK开发的,在网上一搜一大堆详细教程。本篇的亮点是采用Goog...
    Javine_Kuang阅读 1,139评论 2 7
  • 1.安装 $sudo apt-get install cmake 2.示例:简单的文件目录 sample |—...
    荷包蛋酱阅读 27,926评论 0 15
  • 在项目中要实装lame的开源库实现录音转码mp3的功能。期间遇到诸多问题,度娘了N多博客,最终总结了接入了lame...
    hahafei阅读 6,697评论 11 10
  • 继承是OO语言中的一个最为人津津乐道的概念。许多OO语言都支持两种继承方式:接口继承和实现继承。接口继承只继承方法...
    frankisbaby阅读 56评论 0 0
  • 今天看到一位朋友在朋友圈里发了一句成者为王败者寇,本来还没怎么上心,结果静下心来这么一想,竟有醍醐灌顶之感。一直以...
    骚人搁笔费评章阅读 701评论 0 0
  • 瞬间的回忆 刹那间的记忆停留 最深刻 无故怀念的过往 终究无法再驻足 只有时间和热泪 能让我自己消化和平静 最后一...
    小红帆阅读 43评论 0 0
  • 今天是端午小长假第一天,按照原计划一大早:上海→仪征,去看闺蜜和我两个月零两天的干女儿。所有的车票都在一周之前预定...
    my彩色沙漠阅读 157评论 3 5