native-lib.cpp

#include <jni.h>
#include <string> #include <SLES/OpenSLES.h>
#include <SLES/OpenSLES_Android.h> #include <android/log.h> #define LOGD(FORMAT,...) __android_log_print(ANDROID_LOG_ERROR,"xp.chen",FORMAT,##__VA_ARGS__); void bufferQueueCallback(SLAndroidSimpleBufferQueueItf bufferQueue, void *pContext) {
static FILE *fp = NULL;
static char *buf = NULL;
if (!buf) {
buf = new char[*];
}
if (!fp) {
fp = fopen("/sdcard/test.pcm", "rb");
}
if (!fp) return;
if (feof(fp) == ) {
int len = fread(buf, , , fp);
if (len > )
(*bufferQueue)->Enqueue(bufferQueue, buf, len);
}
} extern "C" JNIEXPORT jstring JNICALL
Java_com_yongdaimi_android_androidapitest_OpenSLESApiUseDemoActivity_stringFromJNI(JNIEnv* env, jobject object)
{
std::string hello = "Hello from C++"; SLresult re;
SLObjectItf engineObject;
SLEngineItf slAudioEngine; // 1. Create and init audio engine
re = slCreateEngine(&engineObject, , NULL, , NULL, NULL);
if (re != SL_RESULT_SUCCESS) {
LOGD("slCreateEngine() failed");
}
re = (*engineObject)->Realize(engineObject, SL_BOOLEAN_FALSE);
if (re != SL_RESULT_SUCCESS) {
LOGD("engineObject Realize failed");
}
re = (*engineObject)->GetInterface(engineObject, SL_IID_ENGINE, &slAudioEngine);
if (re != SL_RESULT_SUCCESS) {
LOGD("engineObject GetInterface SL_IID_ENGINE failed");
} // 2. Set output mix
SLObjectItf outputMix;
re = (*slAudioEngine)->CreateOutputMix(slAudioEngine, &outputMix, , NULL, NULL);
if (re != SL_RESULT_SUCCESS) {
LOGD("CreateOutputMix() failed");
}
re = (*outputMix)->Realize(outputMix, SL_BOOLEAN_FALSE);
if (re != SL_RESULT_SUCCESS) {
LOGD("outputMix Realize failed");
} // 3. Configuring the input data source
SLDataLocator_AndroidSimpleBufferQueue inputBuffQueueLocator = {SL_DATALOCATOR_ANDROIDSIMPLEBUFFERQUEUE, };
SLDataFormat_PCM input_format_pcm = {
SL_DATAFORMAT_PCM, // <<< 输入的音频格式,PCM
, // <<< 输入的声道数,2(立体声)
SL_SAMPLINGRATE_44_1, // <<< 输入的采样率,44100hz
SL_PCMSAMPLEFORMAT_FIXED_16, // <<< 输入的采样位数,16bit
SL_PCMSAMPLEFORMAT_FIXED_16, // <<< 容器大小,同上
SL_SPEAKER_FRONT_LEFT|SL_SPEAKER_FRONT_RIGHT, // <<< 声道标记,这里使用左前声道和右前声道
SL_BYTEORDER_LITTLEENDIAN // <<< 输入的字节序,小端
};
SLDataSource dataSource = {&inputBuffQueueLocator, &input_format_pcm}; SLDataLocator_OutputMix outputMixLocator = {SL_DATALOCATOR_OUTPUTMIX, outputMix};
SLDataSink dataSink = {&outputMixLocator, }; // 4. Create Audio Player
SLObjectItf audioPlayer;
SLAndroidSimpleBufferQueueItf pcmBufferQueue;
SLPlayItf playInterface;
SLInterfaceID audioPlayerInterfaceIDs[] = {SL_IID_ANDROIDSIMPLEBUFFERQUEUE};
SLboolean audioPlayerInterfaceRequired[] = {SL_BOOLEAN_TRUE}; re = (*slAudioEngine)->CreateAudioPlayer(slAudioEngine, &audioPlayer, &dataSource, &dataSink, , audioPlayerInterfaceIDs, audioPlayerInterfaceRequired);
if (re != SL_RESULT_SUCCESS) {
LOGD("CreateAudioPlayer() failed");
}
re = (*audioPlayer)->Realize(audioPlayer, SL_BOOLEAN_FALSE);
if (re != SL_RESULT_SUCCESS) {
LOGD("AudioPlayer Realize failed");
}
re = (*audioPlayer)->GetInterface(audioPlayer, SL_IID_PLAY, &playInterface);
if (re != SL_RESULT_SUCCESS) {
LOGD("AudioPlayer GetInterface SL_IID_PLAY failed");
}
re = (*audioPlayer)->GetInterface(audioPlayer, SL_IID_BUFFERQUEUE, &pcmBufferQueue);
if (re != SL_RESULT_SUCCESS) {
LOGD("AudioPlayer GetInterface SL_IID_BUFFERQUEUE failed");
} (*pcmBufferQueue)->RegisterCallback(pcmBufferQueue, bufferQueueCallback, NULL);
(*playInterface)->SetPlayState(playInterface, SL_PLAYSTATE_PLAYING); // Start queue callback
(*pcmBufferQueue)->Enqueue(pcmBufferQueue, "", );
return env->NewStringUTF(hello.c_str());
}

代码中的一些概念补充:

1.结构体中的 numInterfaces , pInterfaceIds , pInterfaceRequired ,这里以创建播放器所调用的 CreateAudioPlayer 函数为例说明:

SLresult (*CreateAudioPlayer) (
SLEngineItf self,
SLObjectItf * pPlayer,
SLDataSource *pAudioSrc,
SLDataSink *pAudioSnk,
SLuint32 numInterfaces,
const SLInterfaceID * pInterfaceIds,
const SLboolean * pInterfaceRequired
);

各参数含义如下:

  • SLEngineItf C语言不像C++,没有this指针,只能通过每次调用SLEngineItf的方法的时候手动传入
  • SLObjectItf 用于保存创建出来的AudioPlayerObject
  • SLDataSource 输入数据源的信息
  • SLDataSink 输出的信息
  • numInterfaces 与下面的SLInterfaceID和SLboolean配合使用,用于标记SLInterfaceID数组和SLboolean数组的大小
  • SLInterfaceID 这里需要传入一个数组,指定创建的AudioPlayerObject需要包含哪些Interface
  • SLboolean 这里也是一个数组,用来标记每个需要包含的Interface在AudioPlayerObject不支持的情况下,是不是需要在创建AudioPlayerObject时返回失败。

最后的三个参数用于指定AudioPlayerObject需要包含哪些Interface,如果不包含是不是要直接创建失败。之前也提到过,并不是每个系统上都实现了 OpenSL ES 为 Object 定义的所有 Interface,所以在获取 Interface 的时候需要做一些选择和判断,如果创建成功的话我们就能使用AudioPlayerObject的GetInterface方法获取到这些Interface了。

2. DataSouce和DataSink

OpenSL ES 里面,这两个结构体均是作为创建 Media Object 对象时的参数而存在的,data source 代表着输入源的信息,即数据从哪儿来、输入的数据参数是怎样的;而 data sink 则代表着输出的信息,即数据输出到哪儿、以什么样的参数来输出。

OpenSL ES: 利用OpenSL ES播放一个存在于SDcard上的PCM文件的更多相关文章

  1. Android OpenSL ES 开发:OpenSL ES利用SoundTouch实现PCM音频的变速和变调

    缘由 OpenSL ES 学习到现在已经知道 OpenSL ES 不仅能播放和录制PCM音频数据,还能改变声音大小.设置左声道或右声道播放.还能变速播放,可谓是播放音频的王者.但是变速有一点不好的就是 ...

  2. ES 记录之如何创建一个索引映射,以及一些设置

    ElasticSearch 系列文章 1 ES 入门之一 安装ElasticSearcha 2 ES 记录之如何创建一个索引映射 3 ElasticSearch 学习记录之Text keyword 两 ...

  3. 利用videojs自动播放下一个

    利用videojs自动播放下一个 一.总结 一句话总结: 在视频放完的ended方法里面,指定video的src,然后this.play()放视频就好 vue来控制视频的链接也是蛮不错的 this.o ...

  4. OpenGL ES: (1) OpenGL ES的由来 (转)

    1. 电脑是做什么用的? 电脑又被称为计算机,那么最重要的工作就是计算.看过三体的同学都知道, 电脑中有无数纳米级别的计算单元,通过 0 和 1 的转换,完成加减乘除的操作. 2. 是什么使电脑工作? ...

  5. es笔记---新建es索引

    es对索引的一堆操作都是用restful api去进行的,参数时一堆json,一年前边查边写搞过一次,这回搞迁移,发现es都到6.0版本了,也变化了很多,写个小笔记记录一下. 创建一个es索引很简单, ...

  6. [OpenGL ES 02]OpenGL ES渲染管线与着色器

    [OpenGL ES 02]OpenGL ES渲染管线与着色器 罗朝辉 (http://www.cnblogs.com/kesalin/) 本文遵循"署名-非商业用途-保持一致"创 ...

  7. 利用SCI做的一个足球答题系统

    SCI,异步串行通信接口,内置独立的波特率产生电路和SCI收发器,可以选择发送8或9个数据位(其中一位可以指定为奇或偶校验位). SCI是全双工异步串行通信接口,主要用于MCU与其他计算机或设备之间的 ...

  8. iOS Dev (20) 用 AVAudioPlayer 播放一个本地音频文件

    iOS Dev (20) 用 AVAudioPlayer 播放一个本地音频文件 作者:CSDN 大锐哥 博客:http://blog.csdn.net/prevention 步骤 第一步:在 Proj ...

  9. iOS Dev (21) 用 AVPlayer 播放一个本地音频文件

    iOS Dev (21) 用 AVPlayer 播放一个本地音频文件 作者:CSDN 大锐哥 博客:http://blog.csdn.net/prevention 前言 这篇文章与上一篇极其相似,要注 ...

随机推荐

  1. SSM - SpringBoot - SpringCloud

    SSM框架 Spring + Spring MVC + MyBatis:标准MVC模式 继 SSH (Struts+Spring+Hibernate)之后,主流的 Java EE企业级 Web应用程序 ...

  2. Android笔记(二十五) ListView的缓存机制与BaseAdapter

    之前接触了ListView和Adapter,Adapter将数据源和View连接起来,实际应用中,我们要显示的数据往往有很多,而屏幕只有那么大,系统只能屏幕所能显示的内容,当我们滑动屏幕,会将旧的内容 ...

  3. LTS秘钥协商算法分析

    1.根据RCF文档说法 在1-RTT中有两种密钥协商算法(1-RTT ECDHE和 1-RTT PSK  )和4中0-RTT密钥协商方式(0-RTT PSK, 0-RTT ECDH ,0-RTT EC ...

  4. vim配置笔记

    1. vim两种配置方法 1)配置文件 全局配置文件:/etc/vim/vimrc或者/etc/vimrc 用户个人配置文件:~/.vimrc 2)命令模式 命令行模式下直接输入配置命令即可.如:se ...

  5. java 从上至下打印二叉树

    从上往下打印二叉树题目描述: 从上往下打印出二叉树的每个节点,同层节点从左至右打印. 输入: 输入可能包含多个测试样例. 对于每个测试案例,输入的第一行一个整数n(1<=n<=1000, ...

  6. Codeforces #364 (Div. 2) D. As Fa(数学公式推导 或者二分)

    数学推导的博客 http://codeforces.com/contest/701/problem/D  题目 推导的思路就是 : 让每个人乘车的时间相等 ,让每个人走路的时间相等. 在图上可以这么表 ...

  7. 【HDU-1045,Fire Net-纯暴力简单DFS】

    原题链接:点击!   大致题意:白块表示可以放置炮台的位置——每个炮台可以攻击到上下左右的直线上的炮台(也就是说在它的上下左右直线上不可以再放置炮台,避免引起互相攻击),黑块表示隔离墙的位置——不可放 ...

  8. 解决在macOS下安装了python却没有pip命令的问题【经验总结】

    可以使用brew直接安装python,但是安装完成了之后没有pip命令. pip是常用的python包管理工具,类似于java的maven.第一反应brew install pip,却提示没这货. 可 ...

  9. CQOI2005 三角形面积并 和 POJ1177 Picture

    1845: [Cqoi2005] 三角形面积并 Time Limit: 3 Sec  Memory Limit: 64 MBSubmit: 1664  Solved: 443[Submit][Stat ...

  10. WCF之MSMQ消息队列

    一.MSMQ简介 MSMQ(微软消息队列)是Windows操作系统中消息应用程序的基础,是用于创建分布式.松散连接的消息通讯应用程序的开发工具. MSMQ与XML Web Services和.Net ...