OpenAL 应用程序 crashes/hangs 当我将非 nullptr 从本机代码传递给 `alBufferData()` 时
OpenAL application crashes/hangs when I pass non-nullptr to `alBufferData()` from native code
我正在尝试使用 OpenAL 和 C/C++ 在 Android 上播放一些声音。我正在使用 SDL 框架。
即使我的代码很简单,alBufferData()
也总是会出现莫名其妙的崩溃:
ALuint buf, src;
alGenBuffers(1, &buf);
alGenSources(1, &src);
uint8_t data[8000]{};
alBufferData(buf, AL_FORMAT_MONO8, data, 8000, 8000);
alSourcei(src, AL_BUFFER, buf);
alSourcePlay(src);
程序每次到达 alBufferData()
时都会崩溃或挂起。我没有收到任何类型的错误消息。
令人惊讶的是,如果我传递 0
而不是指向 alBufferData()
的实际指针,那么程序运行良好并播放一些随机噪音。
我完全不知道出了什么问题。
MCVE:
#include <cstdlib>
#include <cmath>
#include <string>
#include <SDL2/SDL.h>
#include <AL/al.h>
#include <AL/alc.h>
#if !defined(ANDROID) && !defined(__ANDROID__)
#define GLEW_STATIC
#include <GL/glew.h>
#else
#include <GLES2/gl2.h>
#endif
void Msg(const char *txt)
{
SDL_ShowSimpleMessageBox(SDL_MESSAGEBOX_INFORMATION, "test app", txt, 0);
}
void Err(const char *txt)
{
Msg(txt);
std::exit(0);
}
int SDL_main(int, char **)
{
Msg("Running build compiled at " __DATE__ " " __TIME__);
if (SDL_Init(SDL_INIT_EVENTS | SDL_INIT_VIDEO))
Err("SDL init failed");
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 2);
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 0);
SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_ES);
#if !defined(ANDROID) && !defined(__ANDROID__)
SDL_Window *win = SDL_CreateWindow("test app", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, 800, 600, SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE);
#else
SDL_Window *win = SDL_CreateWindow("test app", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, 800, 600, SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE | SDL_WINDOW_FULLSCREEN);
#endif
if (!win)
Err("Window creation failed");
SDL_GLContext con = SDL_GL_CreateContext(win);
if (!con)
Err("OpenGL context creation failed");
#if !defined(ANDROID) && !defined(__ANDROID__)
glewExperimental = 1;
if (glewInit() != GLEW_OK)
Msg("GLEW init failed");
while (glGetError()) {}
#endif
SDL_GL_SetSwapInterval(1);
ALCdevice *device = alcOpenDevice(0);
if (!device)
Err("OpenAL init failed");
const ALCint config_array[] = {ALC_FREQUENCY, 44100, ALC_MONO_SOURCES, 4, ALC_STEREO_SOURCES, 4, 0};
ALCcontext *context = alcCreateContext(device, config_array);
if (!context)
Err("OpenAL context creation failed");
if (!alcMakeContextCurrent(context))
Err("OpenAL context switching failed");
uint8_t sound_wave[8000];
for (int i = 0; i < 8000; i++)
{
float pi = std::atan(1)*4;
sound_wave[i] = int(std::pow(std::sin(i * pi / 180 * 4),0.5) * 127) + 127;
auto smoothstep = [](float x){return 2*x*x*x-3*x*x;};
if (i < 1000)
sound_wave[i] *= smoothstep(i / 1000.f);
else if (i >= 7000)
sound_wave[i] *= smoothstep((8000 - i) / 1000.f);
}
ALuint buf, src;
int frames = 0;
while (1)
{
SDL_Event event;
bool click = 0;
while (SDL_PollEvent(&event))
{
switch (event.type)
{
case SDL_QUIT:
std::exit(0);
break;
case SDL_KEYDOWN:
click = 1;
break;
}
}
switch (frames)
{
case 3: Msg("Creating a buffer"); break;
case 4: alGenBuffers(1, &buf); if (!buf) Err("No buffer"); break;
case 5: Msg("Creating a source"); break;
case 6: alGenSources(1, &src); if (!src) Err("No source"); break;
case 7: Msg("Setting buffer data"); break;
case 8:
alBufferData(buf, AL_FORMAT_MONO8, sound_wave, 8000, 8000);
if (int err = alGetError()) // This code is not reached
Msg((std::string("Error code: ") + std::to_string(err)).c_str());
break;
case 9: Msg("Attaching buffer"); break;
case 10: alSourcei(src, AL_BUFFER, buf); break;
case 11: Msg("Done"); break;
}
if (frames < 12)
frames++;
if (frames == 12 && click)
alSourcePlay(src);
SDL_GL_SwapWindow(win);
}
}
这是我的 Application.mk:
APP_ABI := armeabi armeabi-v7a x86 mips
APP_STL := c++_shared
LOCAL_SHARED_LIBRARIES := c++_shared
APP_CFLAGS += -w
APP_CPPFLAGS += -fexceptions -frtti -I../lib/include
APP_CPPFLAGS += -std=c++14 -O3 -s
NDK_TOOLCHAIN_VERSION := clang
我使用的是使用独立工具链预构建的 OpenAL。我不知道这是否重要,但这是我的构建命令(这个命令用于 armv7-a,对于其他 ABI,有不同的命令):
cmake -D CMAKE_SYSTEM_NAME:string=android -D CMAKE_C_COMPILER:filepath="Y:/clang_3.8_android_api12_androideabi/bin/arm-linux-androideabi-clang.cmd" ^
-D CMAKE_CXX_COMPILER:filepath="Y:/clang_3.8_android_api12_androideabi/bin/arm-linux-androideabi-clang++.cmd" ^
-D CMAKE_C_FLAGS:string="-w -O3 -mthumb -march=armv7-a -mfloat-abi=softfp -mfpu=vfpv3-d16" ^
-D CMAKE_CXX_FLAGS:string="-w -O3 -mthumb -march=armv7-a -mfloat-abi=softfp -mfpu=vfpv3-d16" ^
-D CMAKE_SHARED_LINKER_FLAGS:string="-lc++_shared" ^
-D CMAKE_MODULE_LINKER_FLAGS:string="-lc++_shared" ^
-D CMAKE_BUILD_TYPE:string="Release" ^
-G "MinGW Makefiles" ^
../../..
mingw32-make.exe
编辑:
现在越来越有趣了。我设法使它工作了一次,但在对源代码进行了一些无关紧要的更改后,它又崩溃了。
某处一定有某种未定义的行为,但我确定它不在我这边。
这是 OpenAL 中的错误吗?我使用的确切版本是 openal-soft-1.17.1
.
编辑:
已更新至 OpenAL 1.17.2。运气不好,同样的错误。
编辑:
Here is logcat output for the mcve from the emulator.
它在外部托管,因为它有 2000 多行。
如果我没理解错的话,日志显示存在堆损坏并且该应用程序已在 dlmalloc()
中进行了 sigsegv。但我不知道如何解决它。
我尝试将 SDL_INIT_AUDIO
标志添加到 SDL_Init()
- 同样的错误。
好像是a bug in OpenAL,这也影响了MacOS。应该已经修复了。
此外,它仅在库以“Release”或“RelWithDebInfo”模式构建时崩溃。切换到任意随机字符串或“调试”可解决此问题。
你好:)对我来说
alBufferData(buf, AL_FORMAT_MONO8, sound_wave, 8000, 8000);
应该是问题的一部分。
来自文档:
void alBufferData(
ALuint buffer,
ALenum format,
const ALvoid *data,
ALsizei size,
ALsizei freq
);
size : 音频数据的字节大小
freq : 音频数据的频率
我的第一点是您的频率 (8000) 可能不受实现支持,请尝试将 44100 作为上下文频率。您需要扩大 sound_wave
阵列。如果可行,您可以通过对 5/6 个样本使用相同的样本值来获得“8k”频率效果;)
我的第二点是尺寸,使用 al.h
中的类型 ALubyte
可能更好,对于 size
类似 sizeof(sound_wave)
或 samples_count * sizeof(ALubyte)
确保始终匹配代码更改。
但是对于用 NULL
代替 data
的文档,将设置 AL_INVALID_VALUE
错误状态(如 alGetError()
returns)
希望对您有所帮助
我正在尝试使用 OpenAL 和 C/C++ 在 Android 上播放一些声音。我正在使用 SDL 框架。
即使我的代码很简单,alBufferData()
也总是会出现莫名其妙的崩溃:
ALuint buf, src;
alGenBuffers(1, &buf);
alGenSources(1, &src);
uint8_t data[8000]{};
alBufferData(buf, AL_FORMAT_MONO8, data, 8000, 8000);
alSourcei(src, AL_BUFFER, buf);
alSourcePlay(src);
程序每次到达 alBufferData()
时都会崩溃或挂起。我没有收到任何类型的错误消息。
令人惊讶的是,如果我传递 0
而不是指向 alBufferData()
的实际指针,那么程序运行良好并播放一些随机噪音。
我完全不知道出了什么问题。
MCVE:
#include <cstdlib>
#include <cmath>
#include <string>
#include <SDL2/SDL.h>
#include <AL/al.h>
#include <AL/alc.h>
#if !defined(ANDROID) && !defined(__ANDROID__)
#define GLEW_STATIC
#include <GL/glew.h>
#else
#include <GLES2/gl2.h>
#endif
void Msg(const char *txt)
{
SDL_ShowSimpleMessageBox(SDL_MESSAGEBOX_INFORMATION, "test app", txt, 0);
}
void Err(const char *txt)
{
Msg(txt);
std::exit(0);
}
int SDL_main(int, char **)
{
Msg("Running build compiled at " __DATE__ " " __TIME__);
if (SDL_Init(SDL_INIT_EVENTS | SDL_INIT_VIDEO))
Err("SDL init failed");
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 2);
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 0);
SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_ES);
#if !defined(ANDROID) && !defined(__ANDROID__)
SDL_Window *win = SDL_CreateWindow("test app", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, 800, 600, SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE);
#else
SDL_Window *win = SDL_CreateWindow("test app", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, 800, 600, SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE | SDL_WINDOW_FULLSCREEN);
#endif
if (!win)
Err("Window creation failed");
SDL_GLContext con = SDL_GL_CreateContext(win);
if (!con)
Err("OpenGL context creation failed");
#if !defined(ANDROID) && !defined(__ANDROID__)
glewExperimental = 1;
if (glewInit() != GLEW_OK)
Msg("GLEW init failed");
while (glGetError()) {}
#endif
SDL_GL_SetSwapInterval(1);
ALCdevice *device = alcOpenDevice(0);
if (!device)
Err("OpenAL init failed");
const ALCint config_array[] = {ALC_FREQUENCY, 44100, ALC_MONO_SOURCES, 4, ALC_STEREO_SOURCES, 4, 0};
ALCcontext *context = alcCreateContext(device, config_array);
if (!context)
Err("OpenAL context creation failed");
if (!alcMakeContextCurrent(context))
Err("OpenAL context switching failed");
uint8_t sound_wave[8000];
for (int i = 0; i < 8000; i++)
{
float pi = std::atan(1)*4;
sound_wave[i] = int(std::pow(std::sin(i * pi / 180 * 4),0.5) * 127) + 127;
auto smoothstep = [](float x){return 2*x*x*x-3*x*x;};
if (i < 1000)
sound_wave[i] *= smoothstep(i / 1000.f);
else if (i >= 7000)
sound_wave[i] *= smoothstep((8000 - i) / 1000.f);
}
ALuint buf, src;
int frames = 0;
while (1)
{
SDL_Event event;
bool click = 0;
while (SDL_PollEvent(&event))
{
switch (event.type)
{
case SDL_QUIT:
std::exit(0);
break;
case SDL_KEYDOWN:
click = 1;
break;
}
}
switch (frames)
{
case 3: Msg("Creating a buffer"); break;
case 4: alGenBuffers(1, &buf); if (!buf) Err("No buffer"); break;
case 5: Msg("Creating a source"); break;
case 6: alGenSources(1, &src); if (!src) Err("No source"); break;
case 7: Msg("Setting buffer data"); break;
case 8:
alBufferData(buf, AL_FORMAT_MONO8, sound_wave, 8000, 8000);
if (int err = alGetError()) // This code is not reached
Msg((std::string("Error code: ") + std::to_string(err)).c_str());
break;
case 9: Msg("Attaching buffer"); break;
case 10: alSourcei(src, AL_BUFFER, buf); break;
case 11: Msg("Done"); break;
}
if (frames < 12)
frames++;
if (frames == 12 && click)
alSourcePlay(src);
SDL_GL_SwapWindow(win);
}
}
这是我的 Application.mk:
APP_ABI := armeabi armeabi-v7a x86 mips
APP_STL := c++_shared
LOCAL_SHARED_LIBRARIES := c++_shared
APP_CFLAGS += -w
APP_CPPFLAGS += -fexceptions -frtti -I../lib/include
APP_CPPFLAGS += -std=c++14 -O3 -s
NDK_TOOLCHAIN_VERSION := clang
我使用的是使用独立工具链预构建的 OpenAL。我不知道这是否重要,但这是我的构建命令(这个命令用于 armv7-a,对于其他 ABI,有不同的命令):
cmake -D CMAKE_SYSTEM_NAME:string=android -D CMAKE_C_COMPILER:filepath="Y:/clang_3.8_android_api12_androideabi/bin/arm-linux-androideabi-clang.cmd" ^
-D CMAKE_CXX_COMPILER:filepath="Y:/clang_3.8_android_api12_androideabi/bin/arm-linux-androideabi-clang++.cmd" ^
-D CMAKE_C_FLAGS:string="-w -O3 -mthumb -march=armv7-a -mfloat-abi=softfp -mfpu=vfpv3-d16" ^
-D CMAKE_CXX_FLAGS:string="-w -O3 -mthumb -march=armv7-a -mfloat-abi=softfp -mfpu=vfpv3-d16" ^
-D CMAKE_SHARED_LINKER_FLAGS:string="-lc++_shared" ^
-D CMAKE_MODULE_LINKER_FLAGS:string="-lc++_shared" ^
-D CMAKE_BUILD_TYPE:string="Release" ^
-G "MinGW Makefiles" ^
../../..
mingw32-make.exe
编辑:
现在越来越有趣了。我设法使它工作了一次,但在对源代码进行了一些无关紧要的更改后,它又崩溃了。
某处一定有某种未定义的行为,但我确定它不在我这边。
这是 OpenAL 中的错误吗?我使用的确切版本是 openal-soft-1.17.1
.
编辑:
已更新至 OpenAL 1.17.2。运气不好,同样的错误。
编辑:
Here is logcat output for the mcve from the emulator.
它在外部托管,因为它有 2000 多行。
如果我没理解错的话,日志显示存在堆损坏并且该应用程序已在 dlmalloc()
中进行了 sigsegv。但我不知道如何解决它。
我尝试将 SDL_INIT_AUDIO
标志添加到 SDL_Init()
- 同样的错误。
好像是a bug in OpenAL,这也影响了MacOS。应该已经修复了。
此外,它仅在库以“Release”或“RelWithDebInfo”模式构建时崩溃。切换到任意随机字符串或“调试”可解决此问题。
你好:)对我来说
alBufferData(buf, AL_FORMAT_MONO8, sound_wave, 8000, 8000);
应该是问题的一部分。 来自文档:
void alBufferData(
ALuint buffer,
ALenum format,
const ALvoid *data,
ALsizei size,
ALsizei freq
);
size : 音频数据的字节大小 freq : 音频数据的频率
我的第一点是您的频率 (8000) 可能不受实现支持,请尝试将 44100 作为上下文频率。您需要扩大 sound_wave
阵列。如果可行,您可以通过对 5/6 个样本使用相同的样本值来获得“8k”频率效果;)
我的第二点是尺寸,使用 al.h
中的类型 ALubyte
可能更好,对于 size
类似 sizeof(sound_wave)
或 samples_count * sizeof(ALubyte)
确保始终匹配代码更改。
但是对于用 NULL
代替 data
的文档,将设置 AL_INVALID_VALUE
错误状态(如 alGetError()
returns)
希望对您有所帮助