Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: orxonox.OLD/trunk/src/lib/sound/sound_engine.cc @ 6176

Last change on this file since 6176 was 6142, checked in by bensch, 19 years ago

orxonox/trunk: merge the ObjectManager to the trunk
merged with command:
svn merge -r6082:HEAD objectmanager/ ../trunk/

conflicts resolution was easy this time :)
but specially merged the world to network_world

File size: 10.1 KB
RevLine 
[4597]1/*
[4504]2   orxonox - the future of 3D-vertical-scrollers
3
4   Copyright (C) 2004 orx
5
6   This program is free software; you can redistribute it and/or modify
7   it under the terms of the GNU General Public License as published by
8   the Free Software Foundation; either version 2, or (at your option)
9   any later version.
10
11   ### File Specific:
12   main-programmer: Benjamin Grauer
13   co-programmer: ...
14
15   code has been taken from http://www.devmaster.net/articles.php?catID=6
16   The code has been applied to our needs, and many things have been changed.
17*/
18
[5386]19#define DEBUG_SPECIAL_MODULE DEBUG_MODULE_SOUND
[4504]20
21#include "sound_engine.h"
22
[4960]23#include "class_list.h"
[4504]24
25#include "p_node.h"
26#include "list.h"
27#include "resource_manager.h"
28#include "debug.h"
[5944]29#include "parser/ini_parser/ini_parser.h"
[5427]30#include "globals.h"
[4504]31
32using namespace std;
33
[4597]34
[4504]35//////////////////
36/* SOUND-ENGINE */
37//////////////////
38/**
[4836]39 *  standard constructor
[4504]40*/
[4597]41SoundEngine::SoundEngine ()
[4504]42{
[4597]43  this->setClassID(CL_SOUND_ENGINE, "SoundEngine");
44  this->setName("SoundEngine");
45
[4504]46  this->listener = NULL;
[4960]47  this->bufferList = NULL;
48  this->sourceList = NULL;
[5930]49
50  this->device = NULL;
51  this->context = NULL;
52
53  this->maxSourceCount = 32;
[4504]54}
55
56/**
[4836]57 *  the singleton reference to this class
[4504]58*/
59SoundEngine* SoundEngine::singletonRef = NULL;
60
61/**
[4836]62 *  standard deconstructor
[5293]63 */
[4597]64SoundEngine::~SoundEngine ()
[4504]65{
66  // deleting all the SoundSources
[4960]67  if(this->sourceList != NULL)
68  {
[5779]69    while (this->sourceList->size() > 0)
70      delete dynamic_cast<SoundSource*>(this->sourceList->front());
[4960]71  }
[4504]72
[5930]73  while(!this->ALSources.empty())
74  {
75    alDeleteSources(1, &this->ALSources.top());
76    this->ALSources.pop();
77  }
78
[4504]79  // deleting all the SoundBuffers
[4960]80  if (this->bufferList != NULL)
81  {
[5779]82    while(this->bufferList->size() > 0)
83      ResourceManager::getInstance()->unload(dynamic_cast<SoundBuffer*>(this->bufferList->front()));
[4960]84  }
[5293]85
[4504]86  // removing openAL from AudioResource
[5473]87  //! @todo this should be terminated through alc
88  //alutExit();
[5293]89
90  SoundEngine::singletonRef = NULL;
[4504]91}
92
93/**
[4985]94 * loads the settings of the SoundEngine from an ini-file
95 * @param iniParser the IniParser of the inifile
96 */
97void SoundEngine::loadSettings(IniParser* iniParser)
98{
[5953]99  const char* channels = iniParser->getVar(CONFIG_NAME_AUDIO_CHANNELS, CONFIG_SECTION_AUDIO, "32");
100  this->maxSourceCount = atoi(channels);
[4985]101  const char* musicVolume = iniParser->getVar(CONFIG_NAME_MUSIC_VOLUME, CONFIG_SECTION_AUDIO, "80");
102  this->musicVolume = atof(musicVolume)/100.0;
103
104  const char* effectsVolume = iniParser->getVar(CONFIG_NAME_EFFECTS_VOLUME, CONFIG_SECTION_AUDIO, "80");
105  this->effectsVolume = atof(effectsVolume)/100.0;
106}
107
108/**
[4836]109 *  creates a new SoundSource.
110 * @param fileName The Name to load the SoundBuffer from
111 * @param sourceNode The sourceNode to bind this SoundSource to.
112 * @returns The newly created SoundSource
[4504]113
114   acctualy this is nothing more than a wrapper around the ResourceManager.
115*/
116SoundSource* SoundEngine::createSource(const char* fileName, PNode* sourceNode)
117{
[4885]118  return new SoundSource(sourceNode, (SoundBuffer*)ResourceManager::getInstance()->load(fileName, WAV, RP_LEVEL));
[4504]119}
120
121/**
[4836]122 *  Sets the doppler values of openAL
123 * @param dopplerFactor the extent of the doppler-effect
124 * @param dopplerVelocity the Speed the sound travels
[4504]125*/
126void SoundEngine::setDopplerValues(ALfloat dopplerFactor, ALfloat dopplerVelocity)
127{
128  alDopplerFactor(dopplerFactor);
129  alDopplerVelocity(dopplerVelocity);
130}
131
132
[5930]133void SoundEngine::popALSource(ALuint& source)
[4504]134{
[5930]135  if (source != 0)
136    return;
137  else
138  {
[4504]139
[5930]140    /// @TODO try to create more sources if needed
141    if (!this->ALSources.empty())
[4504]142    {
[5930]143      source = this->ALSources.top();
144      this->ALSources.pop();
[4504]145    }
[5779]146  }
[4504]147}
148
149
150/**
[4836]151 *  updates all The positions, Directions and Velocities of all Sounds
[4504]152*/
[4746]153void SoundEngine::update()
[4504]154{
155
156  // updating the Listeners Position
157  if (likely(this->listener != NULL))
158    {
159      alListener3f(AL_POSITION,
[4597]160                   this->listener->getAbsCoor().x,
161                   this->listener->getAbsCoor().y,
162                   this->listener->getAbsCoor().z);
[4504]163      alListener3f(AL_VELOCITY,
[4597]164                   this->listener->getVelocity().x,
165                   this->listener->getVelocity().y,
166                   this->listener->getVelocity().z);
[4504]167      Vector absDirV = this->listener->getAbsDirV();
168      ALfloat orientation [6] = {1,0,0, absDirV.x, absDirV.y, absDirV.z};
169      alListenerfv(AL_ORIENTATION, orientation);
170    }
171  else
172    PRINTF(2)("no listener defined\n");
173
174  // updating all the Sources positions
[5930]175  if (likely(this->sourceList != NULL || (this->sourceList = ClassList::getList(CL_SOUND_SOURCE)) != NULL))
[4960]176  {
[5885]177    list<BaseObject*>::const_iterator sourceIT;
[5779]178    SoundSource* source;
179    for (sourceIT = this->sourceList->begin(); sourceIT != this->sourceList->end(); sourceIT++)
[4504]180    {
[5779]181      source = static_cast<SoundSource*>(*sourceIT);
[5930]182      if (source->isPlaying())
[4504]183      {
[6073]184        int play = 0x000;
[5930]185        alGetSourcei(source->getID(), AL_SOURCE_STATE, &play);
186        if(play == AL_PLAYING)
187        {
188          if (likely(source->getNode() != NULL))
189          {
190            alSource3f(source->getID(), AL_POSITION,
191                       source->getNode()->getAbsCoor().x,
192                       source->getNode()->getAbsCoor().y,
193                       source->getNode()->getAbsCoor().z);
194            alSource3f(source->getID(), AL_VELOCITY,
195                       source->getNode()->getVelocity().x,
196                       source->getNode()->getVelocity().y,
197                       source->getNode()->getVelocity().z);
198          }
199
200        }
201        else
202        {
203          source->stop();
204        }
[4504]205      }
206    }
[4960]207  }
[4504]208}
209
210/**
[4836]211 *  Removes all the Buffers that are not anymore needed by any Sources
[4504]212*/
[4746]213void SoundEngine::flushUnusedBuffers()
[4504]214{
[5819]215  /// FIXME
[5779]216/*  if(this->sourceList && this->bufferList)
[4960]217  {
218    tIterator<BaseObject>* bufferIterator = this->bufferList->getIterator();
[5115]219    SoundBuffer* enumBuffer = (SoundBuffer*)bufferIterator->firstElement();
[4960]220    while (enumBuffer)
[4504]221    {
[4960]222      tIterator<BaseObject>* sourceIterator = this->sourceList->getIterator();
[5115]223      SoundSource* enumSource = (SoundSource*)sourceIterator->firstElement();
[4504]224      while (enumSource)
[4960]225      {
226        if (enumBuffer == enumSource->getBuffer())
227          break;
228        enumSource = (SoundSource*)sourceIterator->nextElement();
229      }
[4504]230      delete sourceIterator;
231      if (enumSource == NULL)
[4597]232        ResourceManager::getInstance()->unload(enumBuffer);
[4960]233      enumBuffer = (SoundBuffer*)bufferIterator->nextElement();
[4504]234    }
[4960]235    delete bufferIterator;
[5779]236}*/ /// FIXME
[4504]237}
238
239/**
[5293]240 * flushes all the Buffers
241 * deletes them from the BufferList, and also removes them via the ResourceManager.
242 */
[4746]243void SoundEngine::flushAllBuffers()
[4504]244{
[4960]245  if (this->bufferList)
246  {
[5779]247    while (this->bufferList->size() > 0)
248      ResourceManager::getInstance()->unload(static_cast<SoundBuffer*>(this->bufferList->front()), RP_LEVEL);
[4960]249  }
[4504]250}
251
252/**
[5293]253 * deletes all the Sources.
[4830]254 */
255void SoundEngine::flushAllSources()
256{
[4960]257  if (this->sourceList)
[4830]258  {
[5779]259    while(this->sourceList->size() > 0)
260      delete this->sourceList->front();
[4830]261  }
262}
263
264/**
[4836]265 *  initializes Audio in general
[4504]266*/
[4746]267bool SoundEngine::initAudio()
[4504]268{
269  ALenum result;
[5385]270  PRINTF(3)("Initialisazing openAL sound engine\n");
271  const char* defaultDevice =(const char*) alcGetString(NULL, ALC_DEFAULT_DEVICE_SPECIFIER);
272  const char* deviceList = (const char*)alcGetString(NULL,ALC_DEVICE_SPECIFIER);
273  const char* devWalk = deviceList;
[5416]274//  if (alcIsExtensionPresent(NULL, (const ALCchar*)"ALC_ENUMERATION_EXT") == AL_TRUE)
275{ // try out enumeration extension
[5386]276    PRINTF(3)("Enumeration-extension found\n");
[4504]277
[5386]278    PRINTF(3)("Default device: %s\n", defaultDevice);
[5385]279    do
280    {
[5386]281      PRINTF(3)("%s\n", devWalk);
[5385]282      devWalk += strlen(devWalk)+1;
283    } while (devWalk[0] != '\0');
[4504]284
[5385]285
[5982]286
[5385]287  }
[5819]288  // INITIALIZING THE DEVICE:
[5996]289
[6076]290#ifdef AL_VERSION_1_1
[5915]291  ALCchar deviceName[] =
[5996]292
[5819]293#ifdef __WIN32__
[5915]294      "Direct3D";
[5819]295#else
[5915]296      "'( ( devices '( native null ) ) )";
[5819]297#endif
298  //
299  this->device = alcOpenDevice(deviceName);
[5385]300
[5819]301  this->context = alcCreateContext(this->device, NULL);
302
[5996]303  alcMakeContextCurrent(this->context);
[6076]304#else
305  alutInit(0, NULL);
306#endif
[5819]307
[4504]308  if ((result = alGetError()) != AL_NO_ERROR)
[5930]309    PRINTF(2)("%s\n", SoundEngine::getALErrorString(result));
[4504]310
311  this->setDopplerValues(SOUND_DOPPLER_FACTOR, SOUND_DOPPLER_VELOCITY);
312}
313
[5917]314
[4504]315/**
[5917]316 * Allocates openAL sources
317 * @param count how many sources to allocate
318 * @returns true on success, false if at least one source could not be allocated
319 */
320bool SoundEngine::allocateSources(unsigned int count)
321{
322  ALenum result;
[5930]323  // Setting default values.
324  for (unsigned int i = 0; i < count; i++)
[5917]325  {
[5930]326    ALuint source;
[5917]327
[5930]328    alGenSources(1, &source);
329    if ((result = alGetError()) != AL_NO_ERROR)
330      PRINTF(1)("Error Generating Sources: '%s'\n", SoundEngine::getALErrorString(result));
[5917]331
[5930]332    alSourcef (source, AL_PITCH,    1.0      );
333    alSourcef (source, AL_GAIN,     this->getEffectsVolume() );
334    alSourcei (source, AL_LOOPING,  AL_FALSE );
335    this->ALSources.push(source);
[5917]336  }
337  return true;
338}
339
340/**
[4836]341 *  Transforms AL-errors into something readable
342 * @param err The error found
[4504]343*/
[5930]344const char* SoundEngine::getALErrorString(ALenum err)
[4504]345{
346  switch(err)
347    {
348    case AL_NO_ERROR:
[5930]349      return ("AL_NO_ERROR");
[4504]350    case AL_INVALID_NAME:
[5930]351      return ("AL_INVALID_NAME");
[4504]352    case AL_INVALID_ENUM:
[5930]353      return ("AL_INVALID_ENUM");
[4504]354    case AL_INVALID_VALUE:
[5930]355      return ("AL_INVALID_VALUE");
[4504]356    case AL_INVALID_OPERATION:
[5930]357      return ("AL_INVALID_OPERATION");
[4504]358    case AL_OUT_OF_MEMORY:
[5930]359      return ("AL_OUT_OF_MEMORY");
[4504]360    };
361}
362
[4959]363void SoundEngine::listDevices()
364{
365  printf("%s\n",(const char*)alcGetString(NULL, ALC_DEVICE_SPECIFIER));
366}
367
[4504]368/*
369void SoundEngine::PrintALCErrorString(ALenum err)
370{
371  switch(err)
372    {
373    case ALC_NO_ERROR:
374      PRINTF(4)("AL_NO_ERROR\n");
375      break;
376
377    case ALC_INVALID_DEVICE:
378      PRINTF(2)("ALC_INVALID_DEVICE\n");
379      break;
380
381    case ALC_INVALID_CONTEXT:
382      PRINTF(2)("ALC_INVALID_CONTEXT\n");
383      break;
384
385    case ALC_INVALID_ENUM:
386      PRINTF(2)("ALC_INVALID_ENUM\n");
387      break;
388
389    case ALC_INVALID_VALUE:
390      PRINTF(2)("ALC_INVALID_VALUE\n");
391      break;
392
393    case ALC_OUT_OF_MEMORY:
394      PRINTF(2)("ALC_OUT_OF_MEMORY\n");
395      break;
396    };
397}
398*/
Note: See TracBrowser for help on using the repository browser.