SDL播放PCM文件
C语言文件打开方式
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
extern "C"
{
#include <SDL.h>
}
#define PCM_BUFFER_SIZE (2 * 1024 * 2 * 2)
static Uint8 *s_audio_buf = NULL;
static Uint8 *s_audio_pos = NULL;
static Uint8 *s_audio_end = NULL;
void fill_audio_pcm(void *udata, Uint8 *stream, int len)
{
SDL_memset(stream, 0, len);
if (s_audio_pos >= s_audio_end)
{
return;
}
int remain_buffer_len = s_audio_end - s_audio_pos;
len = (len < remain_buffer_len) ? len : remain_buffer_len;
SDL_MixAudio(stream, s_audio_pos, len, SDL_MIX_MAXVOLUME / 8);
printf("use %d bytes\n", len);
s_audio_pos += len;
}
#undef main
int main()
{
int ret = -1;
FILE *audio_fd = NULL;
SDL_AudioSpec spec;
const char *path = "./suiyue_44100_2_s16le.pcm";
size_t read_buffer_len = 0;
if (SDL_Init(SDL_INIT_AUDIO))
{
fprintf(stderr, "Could not initialize SDL - %s\n", SDL_GetError());
return ret;
}
audio_fd = fopen(path, "rb");
if (!audio_fd)
{
fprintf(stderr, "Failed to open pcm file!\n");
goto _FAIL;
}
s_audio_buf = (uint8_t *)malloc(PCM_BUFFER_SIZE);
spec.freq = 44100;
spec.format = AUDIO_S16SYS;
spec.channels = 2;
spec.silence = 0;
spec.samples = 1024;
spec.callback = fill_audio_pcm;
spec.userdata = NULL;
if (SDL_OpenAudio(&spec, NULL))
{
fprintf(stderr, "Failed to open audio device, %s\n", SDL_GetError());
goto _FAIL;
}
SDL_PauseAudio(0);
int data_count = 0;
while (1)
{
read_buffer_len = fread(s_audio_buf, 1, PCM_BUFFER_SIZE, audio_fd);
if (read_buffer_len == 0)
{
break;
}
data_count += read_buffer_len;
printf("read %10d bytes data\n", data_count);
s_audio_end = s_audio_buf + read_buffer_len;
s_audio_pos = s_audio_buf;
while (s_audio_pos < s_audio_end)
{
SDL_Delay(10);
}
}
printf("play PCM finish\n");
SDL_CloseAudio();
_FAIL:
if (s_audio_buf)
free(s_audio_buf);
if (audio_fd)
fclose(audio_fd);
SDL_Quit();
return 0;
}
C++文件打开方式
-
- 初始化SDL SDL_Init
-
- 创建音频播放回调函数,打开音频设备 SDL_OpenAudio
-
- 开始播放 SDL_PauseAudio
-
- 打开文件循环读取数据到缓冲
-
- 关闭音频设备 SDL_CloseAudio
-
- 释放资源,退出 SDL_Quit
#include <iostream>
#include <fstream>
#include <cmath>
extern "C"
{
#include <SDL.h>
}
#pragma comment(lib, "SDL2.lib")
Uint8* g_audio_chunk;
Uint32 g_audio_len;
Uint8* g_audio_pos;
void read_audio_data_cb(void* udata, Uint8* stream, int len)
{
SDL_memset(stream, 0x00, len);
if (g_audio_len == 0)
return;
len = SDL_min(len, static_cast<int>(g_audio_len));
std::cout << "use " << len << " bytes" << std::endl;
SDL_MixAudio(stream, g_audio_pos, len, SDL_MIX_MAXVOLUME);
g_audio_pos += len;
g_audio_len -= len;
}
#undef main
int main()
{
int nRet = SDL_Init(SDL_INIT_AUDIO);
if (nRet < 0)
{
std::cout << "SDL Error: " << SDL_GetError() << std::endl;
return -1;
}
SDL_AudioSpec spec;
spec.freq = 44100;
spec.channels = 2;
spec.format = AUDIO_S16SYS;
spec.samples = 1024;
spec.callback = read_audio_data_cb;
spec.userdata = NULL;
if (SDL_OpenAudio(&spec, NULL) < 0)
{
return -1;
}
std::ifstream pcmFile("./suiyue_44100_2_s16le.pcm", std::ios::in | std::ios::binary);
if (!pcmFile.is_open())
{
return -1;
}
SDL_PauseAudio(0);
char* buffer = (char*)malloc(2 * 1024 * 2 * 2);
while (!pcmFile.eof())
{
pcmFile.read(buffer, 1024 * 2 * 2 * 2);
if (pcmFile.bad())
{
return -1;
}
g_audio_chunk = reinterpret_cast<Uint8*>(buffer);
g_audio_len = pcmFile.gcount();
g_audio_pos = g_audio_chunk;
std::cout << "read " << g_audio_len << " bytes" << std::endl;
while (g_audio_len > 0)
{
SDL_Delay(10);
}
}
SDL_CloseAudio();
free(buffer);
buffer = NULL;
SDL_Quit();
return 0;
}