c55sound continued

This commit is contained in:
Perttu Ahola 2012-03-23 15:29:30 +02:00
parent 0ac2ce7dea
commit 8c2f3bb378
9 changed files with 266 additions and 52 deletions

@ -42,7 +42,7 @@ if(ENABLE_AUDIO)
find_package(Vorbis) find_package(Vorbis)
if (VORBIS_FOUND) if (VORBIS_FOUND)
set(USE_AUDIO 1) set(USE_AUDIO 1)
set(audio_SRCS sound.cpp sound_openal.cpp) set(audio_SRCS sound_openal.cpp)
set(AUDIO_INCLUDE_DIRS set(AUDIO_INCLUDE_DIRS
${OPENAL_INCLUDE_DIR} ${OPENAL_INCLUDE_DIR}
${VORBIS_INCLUDE_DIR} ${VORBIS_INCLUDE_DIR}
@ -125,6 +125,7 @@ configure_file(
) )
set(common_SRCS set(common_SRCS
sound.cpp
quicktune.cpp quicktune.cpp
subgame.cpp subgame.cpp
inventorymanager.cpp inventorymanager.cpp
@ -234,7 +235,7 @@ include_directories(
${CMAKE_BUILD_TYPE} ${CMAKE_BUILD_TYPE}
${PNG_INCLUDE_DIR} ${PNG_INCLUDE_DIR}
${GETTEXT_INCLUDE_DIR} ${GETTEXT_INCLUDE_DIR}
${AUDIO_INLCUDE_DIR} ${AUDIO_INLCUDE_DIRS}
${JTHREAD_INCLUDE_DIR} ${JTHREAD_INCLUDE_DIR}
${SQLITE3_INCLUDE_DIR} ${SQLITE3_INCLUDE_DIR}
${LUA_INCLUDE_DIR} ${LUA_INCLUDE_DIR}

@ -30,8 +30,11 @@ with this program; if not, write to the Free Software Foundation, Inc.,
#include "settings.h" #include "settings.h"
#include "itemdef.h" // For wield visualization #include "itemdef.h" // For wield visualization
#include "noise.h" // easeCurve #include "noise.h" // easeCurve
#include "gamedef.h"
#include "sound.h"
Camera::Camera(scene::ISceneManager* smgr, MapDrawControl& draw_control): Camera::Camera(scene::ISceneManager* smgr, MapDrawControl& draw_control,
IGameDef *gamedef):
m_smgr(smgr), m_smgr(smgr),
m_playernode(NULL), m_playernode(NULL),
m_headnode(NULL), m_headnode(NULL),
@ -42,6 +45,7 @@ Camera::Camera(scene::ISceneManager* smgr, MapDrawControl& draw_control):
m_wieldlight(0), m_wieldlight(0),
m_draw_control(draw_control), m_draw_control(draw_control),
m_gamedef(gamedef),
m_camera_position(0,0,0), m_camera_position(0,0,0),
m_camera_direction(0,0,0), m_camera_direction(0,0,0),
@ -168,7 +172,13 @@ void Camera::step(f32 dtime)
} }
else else
{ {
float was = m_view_bobbing_anim;
m_view_bobbing_anim = my_modf(m_view_bobbing_anim + offset); m_view_bobbing_anim = my_modf(m_view_bobbing_anim + offset);
bool step = (was == 0 ||
(was < 0.5f && m_view_bobbing_anim >= 0.5f) ||
(was > 0.5f && m_view_bobbing_anim <= 0.5f));
if(step)
m_gamedef->sound()->playSound("default_grass_walk", false, 1.0);
} }
} }
@ -180,6 +190,7 @@ void Camera::step(f32 dtime)
{ {
m_digging_anim = 0; m_digging_anim = 0;
m_digging_button = -1; m_digging_button = -1;
m_gamedef->sound()->playSound("dig", false, 1.0);
} }
} }
} }
@ -481,9 +492,9 @@ void Camera::setDigging(s32 button)
m_digging_button = button; m_digging_button = button;
} }
void Camera::wield(const ItemStack &item, IGameDef *gamedef) void Camera::wield(const ItemStack &item)
{ {
IItemDefManager *idef = gamedef->idef(); IItemDefManager *idef = m_gamedef->idef();
scene::IMesh *wield_mesh = item.getDefinition(idef).wield_mesh; scene::IMesh *wield_mesh = item.getDefinition(idef).wield_mesh;
if(wield_mesh) if(wield_mesh)
{ {

@ -39,7 +39,8 @@ class IGameDef;
class Camera class Camera
{ {
public: public:
Camera(scene::ISceneManager* smgr, MapDrawControl& draw_control); Camera(scene::ISceneManager* smgr, MapDrawControl& draw_control,
IGameDef *gamedef);
~Camera(); ~Camera();
// Get player scene node. // Get player scene node.
@ -116,7 +117,7 @@ public:
void setDigging(s32 button); void setDigging(s32 button);
// Replace the wielded item mesh // Replace the wielded item mesh
void wield(const ItemStack &item, IGameDef *gamedef); void wield(const ItemStack &item);
// Draw the wielded tool. // Draw the wielded tool.
// This has to happen *after* the main scene is drawn. // This has to happen *after* the main scene is drawn.
@ -137,6 +138,8 @@ private:
// draw control // draw control
MapDrawControl& m_draw_control; MapDrawControl& m_draw_control;
IGameDef *m_gamedef;
// Absolute camera position // Absolute camera position
v3f m_camera_position; v3f m_camera_position;
// Absolute camera direction // Absolute camera direction

@ -225,11 +225,13 @@ Client::Client(
MapDrawControl &control, MapDrawControl &control,
IWritableTextureSource *tsrc, IWritableTextureSource *tsrc,
IWritableItemDefManager *itemdef, IWritableItemDefManager *itemdef,
IWritableNodeDefManager *nodedef IWritableNodeDefManager *nodedef,
ISoundManager *sound
): ):
m_tsrc(tsrc), m_tsrc(tsrc),
m_itemdef(itemdef), m_itemdef(itemdef),
m_nodedef(nodedef), m_nodedef(nodedef),
m_sound(sound),
m_mesh_update_thread(this), m_mesh_update_thread(this),
m_env( m_env(
new ClientMap(this, this, control, new ClientMap(this, this, control,
@ -2326,6 +2328,6 @@ u16 Client::allocateUnknownNodeId(const std::string &name)
} }
ISoundManager* Client::getSoundManager() ISoundManager* Client::getSoundManager()
{ {
return &dummySoundManager; return m_sound;
} }

@ -174,7 +174,8 @@ public:
MapDrawControl &control, MapDrawControl &control,
IWritableTextureSource *tsrc, IWritableTextureSource *tsrc,
IWritableItemDefManager *itemdef, IWritableItemDefManager *itemdef,
IWritableNodeDefManager *nodedef IWritableNodeDefManager *nodedef,
ISoundManager *sound
); );
~Client(); ~Client();
@ -333,6 +334,7 @@ private:
IWritableTextureSource *m_tsrc; IWritableTextureSource *m_tsrc;
IWritableItemDefManager *m_itemdef; IWritableItemDefManager *m_itemdef;
IWritableNodeDefManager *m_nodedef; IWritableNodeDefManager *m_nodedef;
ISoundManager *m_sound;
MeshUpdateThread m_mesh_update_thread; MeshUpdateThread m_mesh_update_thread;
ClientEnvironment m_env; ClientEnvironment m_env;
con::Connection m_con; con::Connection m_con;

@ -56,6 +56,9 @@ with this program; if not, write to the Free Software Foundation, Inc.,
#include "clientmap.h" #include "clientmap.h"
#include "sky.h" #include "sky.h"
#include "sound.h" #include "sound.h"
#if USE_AUDIO
#include "sound_openal.h"
#endif
#include <list> #include <list>
/* /*
@ -824,6 +827,27 @@ void the_game(
// Create node definition manager // Create node definition manager
IWritableNodeDefManager *nodedef = createNodeDefManager(); IWritableNodeDefManager *nodedef = createNodeDefManager();
// Sound manager
ISoundManager *sound = NULL;
bool sound_is_dummy = false;
#if USE_AUDIO
infostream<<"Attempting to use OpenAL audio"<<std::endl;
sound = createOpenALSoundManager(NULL);
if(!sound)
infostream<<"Failed to initialize OpenAL audio"<<std::endl;
#endif
if(!sound){
infostream<<"Using dummy audio."<<std::endl;
sound = &dummySoundManager;
sound_is_dummy = true;
}
// Test sounds
sound->loadSound("default_grass_walk", porting::path_share + DIR_DELIM
+ "sounds" + DIR_DELIM + "default_grass_walk3_mono.ogg");
//sound->playSound("default_grass_walk", false, 1.0);
//sound->playSoundAt("default_grass_walk", true, 1.0, v3f(0,10,0)*BS);
// Add chat log output for errors to be shown in chat // Add chat log output for errors to be shown in chat
LogOutputBuffer chat_log_error_buf(LMT_ERROR); LogOutputBuffer chat_log_error_buf(LMT_ERROR);
@ -855,7 +879,7 @@ void the_game(
MapDrawControl draw_control; MapDrawControl draw_control;
Client client(device, playername.c_str(), password, draw_control, Client client(device, playername.c_str(), password, draw_control,
tsrc, itemdef, nodedef); tsrc, itemdef, nodedef, sound);
// Client acts as our GameDef // Client acts as our GameDef
IGameDef *gamedef = &client; IGameDef *gamedef = &client;
@ -1020,7 +1044,7 @@ void the_game(
/* /*
Create the camera node Create the camera node
*/ */
Camera camera(smgr, draw_control); Camera camera(smgr, draw_control, gamedef);
if (!camera.successfullyCreated(error_message)) if (!camera.successfullyCreated(error_message))
return; return;
@ -1924,8 +1948,11 @@ void the_game(
camera_direction, camera_fov); camera_direction, camera_fov);
} }
//timer2.stop(); // Update sound listener
//TimeTaker //timer3("//timer3"); sound->updateListener(camera.getCameraNode()->getPosition(),
v3f(0,0,0), // velocity
camera.getCameraNode()->getTarget(),
camera.getCameraNode()->getUpVector());
/* /*
Calculate what block is the crosshair pointing to Calculate what block is the crosshair pointing to
@ -2536,7 +2563,7 @@ void the_game(
ItemStack item; ItemStack item;
if(mlist != NULL) if(mlist != NULL)
item = mlist->getItem(client.getPlayerItem()); item = mlist->getItem(client.getPlayerItem());
camera.wield(item, gamedef); camera.wield(item);
} }
/* /*
@ -2730,9 +2757,11 @@ void the_game(
// Client scope (client is destructed before destructing *def and tsrc) // Client scope (client is destructed before destructing *def and tsrc)
}while(0); }while(0);
delete tsrc; if(!sound_is_dummy)
delete sound;
delete nodedef; delete nodedef;
delete itemdef; delete itemdef;
delete tsrc;
} }

@ -28,8 +28,9 @@ with this program; if not, write to the Free Software Foundation, Inc.,
class OnDemandSoundFetcher class OnDemandSoundFetcher
{ {
public: public:
virtual void getSoundFilenames(const std::string &name, virtual void fetchSounds(const std::string &name,
std::set<std::string> &dst); std::set<std::string> &dst_paths,
std::set<std::vector<char> > &dst_datas) = 0;
}; };
class ISoundManager class ISoundManager
@ -48,10 +49,10 @@ public:
virtual void updateListener(v3f pos, v3f vel, v3f at, v3f up) = 0; virtual void updateListener(v3f pos, v3f vel, v3f at, v3f up) = 0;
// playSound functions return -1 on failure, otherwise a handle to the // playSound functions return -1 on failure, otherwise a handle to the
// sound // sound
virtual int playSound(const std::string &name, int loopcount, virtual int playSound(const std::string &name, bool loop,
float volume) = 0; float volume) = 0;
virtual int playSoundAt(const std::string &name, int loopcount, virtual int playSoundAt(const std::string &name, bool loop,
v3f pos, float volume) = 0; float volume, v3f pos) = 0;
virtual void stopSound(int sound) = 0; virtual void stopSound(int sound) = 0;
}; };
@ -63,10 +64,10 @@ public:
virtual bool loadSound(const std::string &name, virtual bool loadSound(const std::string &name,
const std::vector<char> &filedata) {return true;} const std::vector<char> &filedata) {return true;}
void updateListener(v3f pos, v3f vel, v3f at, v3f up) {} void updateListener(v3f pos, v3f vel, v3f at, v3f up) {}
int playSound(const std::string &name, int loopcount, int playSound(const std::string &name, bool loop,
float volume) {return 0;} float volume) {return 0;}
int playSoundAt(const std::string &name, int loopcount, int playSoundAt(const std::string &name, bool loop,
v3f pos, float volume) {return 0;} float volume, v3f pos) {return 0;}
void stopSound(int sound) {} void stopSound(int sound) {}
}; };

@ -84,6 +84,14 @@ static const char *alErrorString(ALenum err)
} }
} }
static ALenum warn_if_error(ALenum err, const char *desc)
{
if(err == AL_NO_ERROR)
return err;
errorstream<<"WARNING: "<<desc<<": "<<alErrorString(err)<<std::endl;
return err;
}
void f3_set(ALfloat *f3, v3f v) void f3_set(ALfloat *f3, v3f v)
{ {
f3[0] = v.X; f3[0] = v.X;
@ -95,7 +103,7 @@ struct SoundBuffer
{ {
ALenum format; ALenum format;
ALsizei freq; ALsizei freq;
ALuint bufferID; ALuint buffer_id;
std::vector<char> buffer; std::vector<char> buffer;
}; };
@ -146,8 +154,8 @@ SoundBuffer* loadOggFile(const std::string &filepath)
snd->buffer.insert(snd->buffer.end(), array, array + bytes); snd->buffer.insert(snd->buffer.end(), array, array + bytes);
} while (bytes > 0); } while (bytes > 0);
alGenBuffers(1, &snd->bufferID); alGenBuffers(1, &snd->buffer_id);
alBufferData(snd->bufferID, snd->format, alBufferData(snd->buffer_id, snd->format,
&(snd->buffer[0]), snd->buffer.size(), &(snd->buffer[0]), snd->buffer.size(),
snd->freq); snd->freq);
@ -168,19 +176,24 @@ SoundBuffer* loadOggFile(const std::string &filepath)
struct PlayingSound struct PlayingSound
{ {
ALuint source_id;
bool loop;
}; };
class OpenALSoundManager: public ISoundManager class OpenALSoundManager: public ISoundManager
{ {
private: private:
OnDemandSoundFetcher *m_fetcher;
ALCdevice *m_device; ALCdevice *m_device;
ALCcontext *m_context; ALCcontext *m_context;
bool m_can_vorbis; bool m_can_vorbis;
int m_next_id; int m_next_id;
std::map<std::string, std::vector<SoundBuffer*> > m_buffers; std::map<std::string, std::vector<SoundBuffer*> > m_buffers;
std::map<int, PlayingSound*> m_sounds_playing; std::map<int, PlayingSound*> m_sounds_playing;
v3f m_listener_pos;
public: public:
OpenALSoundManager(): OpenALSoundManager(OnDemandSoundFetcher *fetcher):
m_fetcher(fetcher),
m_device(NULL), m_device(NULL),
m_context(NULL), m_context(NULL),
m_can_vorbis(false), m_can_vorbis(false),
@ -258,6 +271,7 @@ public:
} }
std::vector<SoundBuffer*> bufs; std::vector<SoundBuffer*> bufs;
bufs.push_back(buf); bufs.push_back(buf);
m_buffers[name] = bufs;
return; return;
} }
@ -272,18 +286,6 @@ public:
return bufs[j]; return bufs[j];
} }
void updateListener(v3f pos, v3f vel, v3f at, v3f up)
{
ALfloat f[6];
f3_set(f, pos);
alListenerfv(AL_POSITION, f);
f3_set(f, vel);
alListenerfv(AL_VELOCITY, f);
f3_set(f, at);
f3_set(f+3, up);
alListenerfv(AL_ORIENTATION, f);
}
bool loadSound(const std::string &name, bool loadSound(const std::string &name,
const std::string &filepath) const std::string &filepath)
{ {
@ -300,23 +302,186 @@ public:
return false; return false;
} }
int playSound(const std::string &name, int loopcount, PlayingSound* createPlayingSound(SoundBuffer *buf, bool loop,
float volume) float volume)
{ {
infostream<<"OpenALSoundManager: Creating playing sound"<<std::endl;
assert(buf);
PlayingSound *sound = new PlayingSound;
assert(sound);
warn_if_error(alGetError(), "before createPlayingSound");
alGenSources(1, &sound->source_id);
alSourcei(sound->source_id, AL_BUFFER, buf->buffer_id);
alSourcei(sound->source_id, AL_SOURCE_RELATIVE, true);
alSource3f(sound->source_id, AL_POSITION, 0, 0, 0);
alSource3f(sound->source_id, AL_VELOCITY, 0, 0, 0);
alSourcei(sound->source_id, AL_LOOPING, loop ? AL_TRUE : AL_FALSE);
volume = MYMAX(0.0, volume);
alSourcef(sound->source_id, AL_GAIN, volume);
alSourcePlay(sound->source_id);
warn_if_error(alGetError(), "createPlayingSound");
return sound;
}
PlayingSound* createPlayingSoundAt(SoundBuffer *buf, bool loop,
float volume, v3f pos)
{
infostream<<"OpenALSoundManager: Creating positional playing sound"
<<std::endl;
assert(buf);
PlayingSound *sound = new PlayingSound;
assert(sound);
warn_if_error(alGetError(), "before createPlayingSoundAt");
alGenSources(1, &sound->source_id);
alSourcei(sound->source_id, AL_BUFFER, buf->buffer_id);
alSourcei(sound->source_id, AL_SOURCE_RELATIVE, false);
alSource3f(sound->source_id, AL_POSITION, pos.X, pos.Y, pos.Z);
alSource3f(sound->source_id, AL_VELOCITY, 0, 0, 0);
//alSourcef(sound->source_id, AL_ROLLOFF_FACTOR, 0.7);
alSourcef(sound->source_id, AL_REFERENCE_DISTANCE, 30.0);
alSourcei(sound->source_id, AL_LOOPING, loop ? AL_TRUE : AL_FALSE);
volume = MYMAX(0.0, volume);
alSourcef(sound->source_id, AL_GAIN, volume);
alSourcePlay(sound->source_id);
warn_if_error(alGetError(), "createPlayingSoundAt");
return sound;
}
int playSoundRaw(SoundBuffer *buf, bool loop, float volume)
{
assert(buf);
PlayingSound *sound = createPlayingSound(buf, loop, volume);
if(!sound)
return -1;
int id = m_next_id++;
m_sounds_playing[id] = sound;
return id;
}
int playSoundRawAt(SoundBuffer *buf, bool loop, float volume, v3f pos)
{
assert(buf);
PlayingSound *sound = createPlayingSoundAt(buf, loop, volume, pos);
if(!sound)
return -1;
int id = m_next_id++;
m_sounds_playing[id] = sound;
return id;
}
void deleteSound(int id)
{
std::map<int, PlayingSound*>::iterator i =
m_sounds_playing.find(id);
if(i == m_sounds_playing.end())
return;
PlayingSound *sound = i->second;
alDeleteSources(1, &sound->source_id);
delete sound;
m_sounds_playing.erase(id);
}
/* If buffer does not exist, consult the fetcher */
SoundBuffer* getFetchBuffer(const std::string name)
{
SoundBuffer *buf = getBuffer(name);
if(buf)
return buf;
if(!m_fetcher)
return NULL;
std::set<std::string> paths;
std::set<std::vector<char> > datas;
m_fetcher->fetchSounds(name, paths, datas);
for(std::set<std::string>::iterator i = paths.begin();
i != paths.end(); i++){
loadSound(name, *i);
}
for(std::set<std::vector<char> >::iterator i = datas.begin();
i != datas.end(); i++){
loadSound(name, *i);
}
return getBuffer(name);
}
// Remove stopped sounds
void maintain()
{
verbosestream<<"OpenALSoundManager::maintain(): "
<<m_sounds_playing.size()<<" playing sounds, "
<<m_buffers.size()<<" sound names loaded"<<std::endl;
std::set<int> del_list;
for(std::map<int, PlayingSound*>::iterator
i = m_sounds_playing.begin();
i != m_sounds_playing.end(); i++)
{
int id = i->first;
PlayingSound *sound = i->second;
// If not playing, remove it
{
ALint state;
alGetSourcei(sound->source_id, AL_SOURCE_STATE, &state);
if(state != AL_PLAYING){
del_list.insert(id);
}
}
}
if(del_list.size() != 0)
verbosestream<<"OpenALSoundManager::maintain(): deleting "
<<del_list.size()<<" playing sounds"<<std::endl;
for(std::set<int>::iterator i = del_list.begin();
i != del_list.end(); i++)
{
deleteSound(*i);
}
}
/* Interface */
void updateListener(v3f pos, v3f vel, v3f at, v3f up)
{
m_listener_pos = pos;
alListener3f(AL_POSITION, pos.X, pos.Y, pos.Z);
alListener3f(AL_VELOCITY, vel.X, vel.Y, vel.Z);
ALfloat f[6];
f3_set(f, at);
f3_set(f+3, up);
alListenerfv(AL_ORIENTATION, f);
warn_if_error(alGetError(), "updateListener");
}
int playSound(const std::string &name, bool loop, float volume)
{
maintain();
SoundBuffer *buf = getFetchBuffer(name);
if(!buf){
infostream<<"OpenALSoundManager: \""<<name<<"\" not found."
<<std::endl;
return -1; return -1;
} }
int playSoundAt(const std::string &name, int loopcount, return playSoundRaw(buf, loop, volume);
v3f pos, float volume) }
int playSoundAt(const std::string &name, bool loop, float volume, v3f pos)
{ {
maintain();
SoundBuffer *buf = getFetchBuffer(name);
if(!buf){
infostream<<"OpenALSoundManager: \""<<name<<"\" not found."
<<std::endl;
return -1; return -1;
} }
return playSoundRawAt(buf, loop, volume, pos);
}
void stopSound(int sound) void stopSound(int sound)
{ {
maintain();
deleteSound(sound);
} }
}; };
ISoundManager *createSoundManager() ISoundManager *createOpenALSoundManager(OnDemandSoundFetcher *fetcher)
{ {
return new OpenALSoundManager(); return new OpenALSoundManager(fetcher);
}; };

@ -22,7 +22,7 @@ with this program; if not, write to the Free Software Foundation, Inc.,
#include "sound.h" #include "sound.h"
ISoundManager *createOpenALSoundManager(); ISoundManager *createOpenALSoundManager(OnDemandSoundFetcher *fetcher);
#endif #endif