MyMinimad ── Linux下用libmad写的mp3解码播放程序

来源:互联网 发布:杀人游戏和狼人杀 知乎 编辑:程序博客网 时间:2024/05/22 11:03

 首先这里有篇介绍libmad的技术文章:http://www.ibm.com/developerworks/cn/linux/l-cn-libmadmp3player/index.html,介绍了大部分libmad技术。

/*****************************************************************/

平台:ubuntu-8.04.1-amd64-desktop
声卡驱动:oss
安装依赖软件包: sudo apt-get install libmad0 libmad0-dev libid3tag0 libid3tag0-dev

libmad的官网上没有文档,有两个例子:minimad(high-level api),madlld(low-level api)。low-level api用起来太复杂,所以选择minimad为本程序的蓝本。

打开音频设备:

  static int sfd;
  if((sfd=open("/dev/dsp",O_WRONLY))<0)
  {
    printf("can not open device!!!/n");
    return 5;
  }

在output函数中设置音频设备。在mp3字节流解码后得到pcm音频的采样率,声道和音频格式。

一般mp3文件都是立体声(有2个声道),由于要把pcm采样后并处理的数据放入一个char型的数组,而并行的左右声道的每个采样要在字符数组中处理成2个,所以字符数组中的数据的个数应该是pcm音频采样数的4倍。又因为把左右声道的数据合在一个字符数组里串行处理,所以播放的速度应该是pcm音频采样率的两倍。代码如下:

  unsigned char Output[6912],*OutputPtr;
  int fmt,wrote,speed;
    fmt=AFMT_S16_LE;
  speed=pcm->samplerate*2;
  ioctl(sfd, SNDCTL_DSP_SPEED, &(speed));
  ioctl(sfd, SNDCTL_DSP_SETFMT, &fmt);
  ioctl(sfd, SNDCTL_DSP_CHANNELS, &(pcm->channels));
 
  OutputPtr=Output;

  while (nsamples--) {
    signed int sample;

    /* output sample(s) in 16-bit signed little-endian PCM */

    sample = scale(*left_ch++);
    *(OutputPtr++)=sample >> 0;
    *(OutputPtr++)=sample >> 8;

    if (nchannels == 2) {
      sample = scale(*right_ch++);
      *(OutputPtr++)=sample >> 0;
      *(OutputPtr++)=sample >> 8;
    }
  }
  n*=4;

然后,就是把数据写入/dev/dsp中播放了:
 
  OutputPtr=Output;

  while(n)
  {
    wrote=write(sfd,OutputPtr,n);
    if(wrote==-1) printf("wrote ====  -1");
    OutputPtr+=wrote;
    n-=wrote;
    
  }

minimad是用的从标准输入得到mp3文件字节流,感觉不爽,改了下,用mmap把文件映射到内存

  struct stat stat;
  void *fdm;
  char const *file;
  int fd;
 
  file=argv[1];
  fd=open(file,O_RDONLY);
  if (fstat(fd, &stat) == -1 ||
      stat.st_size == 0)
    return 2;

  fdm = mmap(0, stat.st_size, PROT_READ, MAP_SHARED, fd, 0);

/************************  end  *************************************/

完整源代码