linux音频的播放

以下是一个基于QT上面做的一个linux下的一个音频的播放


int sound::play(QString filename)

{

int id,fd,i,j;

char buf[512];

fd = open("/dev/dsp",O_WRONLY);//只写方式打开设备文件/dev/dsp

if(fd<0)

{

perror("Couldn't open the file /dev/audio:");

return -1;

}

QByteArray ba=filename.toLatin1();

const char *c_str2=ba.data();

id = open(c_str2,O_RDWR | O_CREAT,755);//打开要播放的文件

if(id<0)

{

perror("Couldn't open the file test.wav:");

return -1;

}

/*************Set the ioctl********************/

i =0;

if(ioctl(fd,SNDCTL_DSP_RESET,(char *)&i)==-1)perror("reset error:");

if(ioctl(fd,SNDCTL_DSP_SYNC,(char *)&i)==-1)perror("sync error:");

i =1;

if(ioctl(fd,SNDCTL_DSP_NONBLOCK,(char *)&i)==-1)perror("nonblock error:");

i=RATE;

if(ioctl(fd,SNDCTL_DSP_SPEED,(char *)&i)==-1){perror("speed error:");return -1;}

i=1;

if(ioctl(fd,SNDCTL_DSP_CHANNELS,(char *)&i)==-1)perror("channel error:");

i=AFMT_S16_NE;

if(ioctl(fd,SNDCTL_DSP_SETFMT,(char *)&i)==-1){perror("setfmt error:");return -1;}

if(i!=AFMT_S16_NE){printf("the device is not suppor the AFMT_S16_NE");return -1;}

i=3;

if(ioctl(fd,SNDCTL_DSP_SETTRIGGER,(char *)&i)==-1)perror("settrigger error:");

i=3;

if(ioctl(fd,SNDCTL_DSP_SETFRAGMENT,(char *)&i)==-1)perror("setfragment error");

i=1;

if(ioctl(fd,SNDCTL_DSP_PROFILE,(char *)&i)==-1)perror("profile error");

/*****************TO Work*********************/

i=0;

for(j=0;j<8000;)

{

i=read(id,buf,12);//这里的第三个参数是缓冲区大小,太小了会播放得太慢了,太大了就播放快了,还不能有单数,主要是看你的那个采样频率和那个采样位数;

printf("In the %dth time it had read %d bytes\n",j,i);//这一句可以用来做一下稍微的延时,不加的话就不行了,还想大神指导怎么去改善啊呀

if(i>0)

{

write(fd,buf,i);

//ioctl(fd,SNDCTL_DSP_SYNC);

j++;

}

}

close(fd);

close(id);

return 0;

}



  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Linux系统中,PCM(Pulse-code modulation)是一种数字音频编码方式,它将模拟音频信号转换为数字信号。PCM音频数据可以通过ALSA(Advanced Linux Sound Architecture)库进行播放。 ALSA是Linux系统中的一个音频驱动程序,它提供了一组API,用于管理音频设备和处理音频数据。在ALSA中,PCM音频数据可以通过snd_pcm_writei()函数进行播放。该函数将PCM音频数据写入到PCM设备的缓冲区中,并且在缓冲区满时自动触发音频播放。 以下是一个简单的PCM音频播放示例: ```c #include <alsa/asoundlib.h> int main() { int err; snd_pcm_t *pcm_handle; snd_pcm_hw_params_t *params; unsigned int sample_rate = 44100; unsigned int channels = 2; unsigned int buffer_size = 1024; char *buffer; // 打开PCM设备 err = snd_pcm_open(&pcm_handle, "default", SND_PCM_STREAM_PLAYBACK, 0); if (err < 0) { printf("Error opening PCM device: %s\n", snd_strerror(err)); return -1; } // 分配PCM参数对象 snd_pcm_hw_params_alloca(&params); // 初始化PCM参数 err = snd_pcm_hw_params_any(pcm_handle, params); if (err < 0) { printf("Error initializing PCM parameters: %s\n", snd_strerror(err)); return -1; } // 设置PCM参数 err = snd_pcm_hw_params_set_access(pcm_handle, params, SND_PCM_ACCESS_RW_INTERLEAVED); err |= snd_pcm_hw_params_set_format(pcm_handle, params, SND_PCM_FORMAT_S16_LE); err |= snd_pcm_hw_params_set_channels(pcm_handle, params, channels); err |= snd_pcm_hw_params_set_rate_near(pcm_handle, params, &sample_rate, 0); err |= snd_pcm_hw_params_set_buffer_size_near(pcm_handle, params, &buffer_size); if (err < 0) { printf("Error setting PCM parameters: %s\n", snd_strerror(err)); return -1; } // 应用PCM参数 err = snd_pcm_hw_params(pcm_handle, params); if (err < 0) { printf("Error applying PCM parameters: %s\n", snd_strerror(err)); return -1; } // 分配PCM缓冲区 buffer = (char *)malloc(buffer_size * channels * 2); // 播放PCM音频数据 while (1) { // 从文件或其他数据源读取PCM音频数据到缓冲区中 // ... // 将PCM音频数据写入到PCM设备的缓冲区中 err = snd_pcm_writei(pcm_handle, buffer, buffer_size); if (err < 0) { printf("Error writing PCM data: %s\n", snd_strerror(err)); break; } } // 关闭PCM设备 snd_pcm_close(pcm_handle); return 0; } ``` 在上面的示例中,我们首先打开PCM设备,然后初始化和设置PCM参数,接着分配PCM缓冲区,并且不断地从文件或其他数据源读取PCM音频数据到缓冲区中,最后将PCM音频数据写入到PCM设备的缓冲区中进行播放。需要注意的是,我们需要在程序结束时关闭PCM设备。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值