主题 : android下 用C语言进行alsa编程 复制链接 | 浏览器收藏 | 打印
级别: 侠客
UID: 38289
精华: 0
发帖: 54
金钱: 270 两
威望: 54 点
贡献值: 0 点
综合积分: 108 分
注册时间: 2011-02-23
最后登录: 2014-02-12
楼主  发表于: 2012-07-06 20:30

 android下 用C语言进行alsa编程

android下运行的C程序必须要静态编译(-static)才行,为此编译了一个alsa-lib静态库。执行我的alsa测试程序时却报错。不知为何。
以下是alsa-lib编译步骤   ./configure --host=arm-linux --prefix=$PWD/../../output/arm-linux --enable-static --disable-shared  --disable-python  --with-configdir=/usr/local/share --with-plugindir=/usr/local/lib/alsa_lib
make
make install

以下为alsa_test.c源程序
/**alsa play test
*ALSA用户空间编译,ALSA驱动的声卡在用户空间,不宜直接使用
*文件接口中,而应使用alsa-lib
*打开---->设置参数--->读写音频数据 ALSA全部使用alsa-lib中的API
*交叉编译
*export LD_LIBRARY_PATH=$PWD:$LD_LIBRARY_PATH
*arm-linux-gcc -o alsa_play alsa_play_test.c -L. -lasound
*需要交叉编译后的libasound.so库的支持
*
*/
#include <stdio.h>
#include <stdlib.h>
#include "alsa/asoundlib.h"

int main(int argc, char *argv[])
{
    int i;
    int ret;
    int buf[128];
    unsigned int val;
    int dir=0;
    char *buffer;
    int size;
    snd_pcm_uframes_t frames;
    snd_pcm_uframes_t periodsize;
    snd_pcm_t *playback_handle;//PCM设备句柄pcm.h
    snd_pcm_hw_params_t *hw_params;//硬件信息和PCM流配置
    if (argc != 2) {
        printf("error: alsa_play_test [music name]\n");
        exit(1);
    }
    printf("play song %s by wolf\n", argv[1]);
    FILE *fp = fopen(argv[1], "rb");
    if(fp == NULL)
    return 0;
    fseek(fp, 100, SEEK_SET);
    
    //1. 打开PCM,最后一个参数为0意味着标准配置
    ret = snd_pcm_open(&playback_handle, "default", SND_PCM_STREAM_PLAYBACK, 0);
    if (ret < 0) {
        perror("snd_pcm_open");
        exit(1);
    }
    
    //2. 分配snd_pcm_hw_params_t结构体
    ret = snd_pcm_hw_params_malloc(&hw_params);
    if (ret < 0) {
        perror("snd_pcm_hw_params_malloc");
        exit(1);
    }
    //3. 初始化hw_params
    ret = snd_pcm_hw_params_any(playback_handle, hw_params);
    if (ret < 0) {
        perror("snd_pcm_hw_params_any");
        exit(1);
    }
    //4. 初始化访问权限
    ret = snd_pcm_hw_params_set_access(playback_handle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED);
    if (ret < 0) {
        perror("snd_pcm_hw_params_set_access");
        exit(1);
    }
    //5. 初始化采样格式SND_PCM_FORMAT_U8,8位
    ret = snd_pcm_hw_params_set_format(playback_handle, hw_params, SND_PCM_FORMAT_U8);
    if (ret < 0) {
        perror("snd_pcm_hw_params_set_format");
        exit(1);
    }
    //6. 设置采样率,如果硬件不支持我们设置的采样率,将使用最接近的
    //val = 44100,有些录音采样频率固定为8KHz
    

    val = 8000;
    ret = snd_pcm_hw_params_set_rate_near(playback_handle, hw_params, &val, &dir);
    if (ret < 0) {
        perror("snd_pcm_hw_params_set_rate_near");
        exit(1);
    }
    //7. 设置通道数量
    ret = snd_pcm_hw_params_set_channels(playback_handle, hw_params, 2);
    if (ret < 0) {
        perror("snd_pcm_hw_params_set_channels");
        exit(1);
    }
    
    /* Set period size to 32 frames. */
    frames = 32;
    periodsize = frames * 2;
    ret = snd_pcm_hw_params_set_buffer_size_near(playback_handle, hw_params, &periodsize);
    if (ret < 0)
    {
         printf("Unable to set buffer size %li : %s\n", frames * 2, snd_strerror(ret));
        
    }
          periodsize /= 2;

    ret = snd_pcm_hw_params_set_period_size_near(playback_handle, hw_params, &periodsize, 0);
    if (ret < 0)
    {
        printf("Unable to set period size %li : %s\n", periodsize,  snd_strerror(ret));
    }
                                  
    //8. 设置hw_params
    ret = snd_pcm_hw_params(playback_handle, hw_params);
    if (ret < 0) {
        perror("snd_pcm_hw_params");
        exit(1);
    }
    
     /* Use a buffer large enough to hold one period */
    snd_pcm_hw_params_get_period_size(hw_params, &frames, &dir);
                                
    size = frames * 2; /* 2 bytes/sample, 2 channels */
    buffer = (char *) malloc(size);
    fprintf(stderr,
            "size = %d\n",
            size);
    
    while (1)
    {
        ret = fread(buffer, 1, size, fp);
        if(ret == 0)
        {
              fprintf(stderr, "end of file on input\n");
              break;
        }
        else if (ret != size)
        {
        }
        //9. 写音频数据到PCM设备
        while(ret = snd_pcm_writei(playback_handle, buffer, frames)<0)
        {
            usleep(2000);
            if (ret == -EPIPE)
            {
                  /* EPIPE means underrun */
                  fprintf(stderr, "underrun occurred\n");
                  //完成硬件参数设置,使设备准备好
                  snd_pcm_prepare(playback_handle);
            }
            else if (ret < 0)
            {
                  fprintf(stderr,
                      "error from writei: %s\n",
                      snd_strerror(ret));
            }  
        }
        
    }        
    //10. 关闭PCM设备句柄
    snd_pcm_close(playback_handle);
    
    return 0;
}



下边是运行结果

ALSA lib control.c:870:(snd_ctl_open_conf) symbol _snd_ctl_hw_open is not
defined inside (null)
ALSA lib confmisc.c:674:(snd_determine_driver) could not open control for card 0
ALSA lib conf.c:3286:(snd_config_hooks_call) function
snd_config_hook_load_for_all_cards returned error: No such device or address
ALSA lib control.c:870:(snd_ctl_open_conf) symbol _snd_ctl_hw_open is not
defined inside (null)

请大神们看看这是为什么呢? 在linux下不加静态编译是没有问题的,难道静态调用libasound.a不行?
^很多问题的背后都是简单的原因......
级别: 荣誉会员
UID: 34780
精华: 0
发帖: 1219
金钱: 6230 两
威望: 1246 点
贡献值: 0 点
综合积分: 2438 分
注册时间: 2010-12-21
最后登录: 2017-09-18
1楼  发表于: 2012-07-09 09:45
在你的安装后的alsa lib目录,把usr/share/下的内容打包放到板上相应的目录,再试试
zee
级别: 侠客
UID: 75453
精华: 0
发帖: 133
金钱: 665 两
威望: 133 点
贡献值: 0 点
综合积分: 266 分
注册时间: 2012-08-08
最后登录: 2013-04-01
2楼  发表于: 2012-11-20 09:19

 回 1楼(911gt3) 的帖子

这样看来是可以在android下用C开发alsa应用程序了,不一定用android系统的接口也可以的?
级别: 侠客
UID: 38289
精华: 0
发帖: 54
金钱: 270 两
威望: 54 点
贡献值: 0 点
综合积分: 108 分
注册时间: 2011-02-23
最后登录: 2014-02-12
3楼  发表于: 2012-12-04 15:31
我们现在做法就是调“stop media”命令,这样Android就不会抢占音频设备。用NDK生成可执行文件
我不是固执 我只是为了他还在坚持( ⊙ o ⊙ )。。。
级别: 新手上路
UID: 83781
精华: 0
发帖: 23
金钱: 125 两
威望: 25 点
贡献值: 0 点
综合积分: 46 分
注册时间: 2012-12-12
最后登录: 2019-08-14
4楼  发表于: 2013-01-03 10:13

 回 楼主(优才) 的帖子

我也在做ALSA,先移植了一下alsa_lib库,跟楼主的步骤一样,只是--enable-static --disable-shared 变成--enable-shared ,为什么我主函数的#include "asoundlib.h"编译时总说fatal error: asoundlib.h: No such file or directory
compilation terminated.不解  求赐教
级别: 新手上路
UID: 4247
精华: 0
发帖: 31
金钱: 160 两
威望: 32 点
贡献值: 0 点
综合积分: 62 分
注册时间: 2009-03-05
最后登录: 2013-07-10
5楼  发表于: 2013-06-26 20:26
应该使用"alsa/asoundlib.h"才可以
级别: 新手上路
UID: 4247
精华: 0
发帖: 31
金钱: 160 两
威望: 32 点
贡献值: 0 点
综合积分: 62 分
注册时间: 2009-03-05
最后登录: 2013-07-10
6楼  发表于: 2013-06-26 20:28
同时用-I/你的ALSA LIB头文件目录