TDME2 1.9.121
Sound.cpp
Go to the documentation of this file.
1#include <tdme/audio/Sound.h>
2
3#if defined(__APPLE__)
4 #include <OpenAL/al.h>
5#elif defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__linux__) || defined(_WIN32) || defined(__HAIKU__)
6 #include <AL/al.h>
7#endif
8
9#include <string>
10
11#include <tdme/tdme.h>
15#include <tdme/audio/Audio.h>
18#include <tdme/math/Vector3.h>
23
24using std::string;
25using std::to_string;
26
37
38Sound::Sound(const string& id, const string& pathName, const string& fileName) : AudioEntity(id)
39{
40 initiated = false;
41 this->bufferId = pathName + "/" + fileName;
42 this->pathName = pathName;
43 this->fileName = fileName;
46}
47
49{
50 ALint state;
51 alGetSourcei(alSourceId, AL_SOURCE_STATE, &state);
52 return (state == AL_PLAYING);
53}
54
56{
57 if (initiated == false)
58 return;
59
60 alSourceRewind(alSourceId);
61 if (alGetError() != AL_NO_ERROR) {
62 Console::println(string("Audio sound: '" + id + "': Could not rewind"));
63 }
64}
65
67{
68 if (initiated == false)
69 return;
70 // update
71 update();
72 // play
73 alSourcePlay(alSourceId);
74 if (alGetError() != AL_NO_ERROR) {
75 Console::println(string("Audio sound: '" + id + "': Could not play"));
76 }
77}
78
80{
81 if (initiated == false)
82 return;
83
84 alSourcePause(alSourceId);
85 if (alGetError() != AL_NO_ERROR) {
86 Console::println(string("Audio sound: '" + id+ "': Could not pause"));
87 }
88}
89
91{
92 if (initiated == false)
93 return;
94
95 alSourceStop(alSourceId);
96 if (alGetError() != AL_NO_ERROR) {
97 Console::println(string("Audio sound: '" + id + "': Could not stop"));
98 }
99}
100
102{
103 // check if we already have this buffer
104 auto audioBufferManaged = Audio::instance->audioBufferManager.addAudioBuffer(bufferId);
105 if (audioBufferManaged->alId == Audio::ALBUFFERID_NONE) {
106 // nope, generate al buffer
107 int alError;
108 alGenBuffers(1, &alBufferId);
109 if (alGetError() != AL_NO_ERROR) {
110 Console::println(string("Audio sound: '" + id + "': Could not generate buffer"));
111 return false;
112 }
113 // set up al id in audio buffer managed
114 audioBufferManaged->setAlId(alBufferId);
115 auto format = -1;
116 auto frequency = -1;
117
118 // decode audio sound
119 VorbisDecoder decoder;
120 ByteBuffer* data = nullptr;
121 try {
122 // decode ogg vorbis
123 decoder.openFile(pathName, fileName);
124 Console::println(
125 string(
126 "Audio sound: '" +
127 id +
128 "' with " +
129 to_string(decoder.getBitsPerSample()) +
130 " bits per sample, " +
131 to_string(decoder.getChannels()) +
132 " channels, " +
133 to_string(decoder.getSampleRate()) +
134 " samplerate"
135 )
136 );
137 frequency = decoder.getSampleRate();
138 switch (decoder.getChannels()) {
139 case(1): format = AL_FORMAT_MONO16; break;
140 case(2): format = AL_FORMAT_STEREO16; break;
141 default:
142 Console::println(string("Audio sound: '" + id + "': Unsupported number of channels"));
143 }
144 data = ByteBuffer::allocate(2 * 2 * decoder.getSamples());
145 if (decoder.readFromStream(data) == 0) throw new AudioDecoderException("no audio data was decoded");
146 Console::println(
147 string(
148 "Audio sound: '" +
149 id +
150 "' with length " +
151 to_string((float)data->getPosition() / 2.0f / (float)decoder.getChannels() / (float)decoder.getSampleRate()) +
152 " seconds" +
153 "(" +
154 to_string(data->getPosition()) +
155 " bytes)"
156 )
157 );
158 } catch (FileSystemException& fse) {
159 Console::println(string("Audio sound: '" + (id) + "': " + fse.what()));
160 decoder.close();
161 dispose();
162 return false;
163 } catch (AudioDecoderException& ade) {
164 Console::println(string("Audio sound: '" + (id) + "': " + ade.what()));
165 decoder.close();
166 dispose();
167 return false;
168 }
169 decoder.close();
170 // check for valid format and frequency
171 if (format == -1 || frequency == -1) {
172 Console::println(string("Audio sound: '" + id + "': Format or frequency invalid"));
173 dispose();
174 return false;
175 }
176 // upload to al
177 alBufferData(alBufferId, format, data->getBuffer(), data->getPosition(), frequency);
178 if (alGetError() != AL_NO_ERROR) {
179 Console::println(string("Audio sound: '" + id + "': Could not upload buffer data"));
180 dispose();
181 return false;
182 }
183 delete data;
184 } else {
185 alBufferId = audioBufferManaged->alId;
186 }
187
188 // create source
189 alGenSources(1, &alSourceId);
190 if (alGetError() != AL_NO_ERROR) {
191 Console::println(string("Audio sound: '" + id + "': Could not generate source"));
192 dispose();
193 return false;
194 }
195
196 // initiate sound properties
197 alSourcei(alSourceId, AL_BUFFER, alBufferId);
198 update();
199
200 //
201 initiated = true;
202 return true;
203}
204
206{
207 // update sound properties
208 alSourcef(alSourceId, AL_PITCH, pitch);
209 alSourcef(alSourceId, AL_GAIN, gain);
210 alSourcefv(alSourceId, AL_POSITION, sourcePosition.getArray().data());
211 alSourcefv(alSourceId, AL_DIRECTION, sourceDirection.getArray().data());
212 alSourcefv(alSourceId, AL_VELOCITY, sourceVelocity.getArray().data());
213 alSourcei(alSourceId, AL_LOOPING, looping ? AL_TRUE : AL_FALSE);
214 if (fixed == true) {
215 alSourcef(alSourceId, AL_ROLLOFF_FACTOR, 0.0f);
216 alSourcei(alSourceId, AL_SOURCE_RELATIVE, AL_TRUE);
217 } else {
218 alSourcef(alSourceId, AL_ROLLOFF_FACTOR, 1.0f);
219 alSourcei(alSourceId, AL_SOURCE_RELATIVE, AL_FALSE);
220 }
221}
222
224{
226 alDeleteSources(1, &alSourceId);
227 if (alGetError() != AL_NO_ERROR) {
228 Console::println(string("Audio sound: '" + id + "': Could not delete source"));
229 }
231 }
232 if (alBufferId != Audio::ALBUFFERID_NONE && Audio::instance->audioBufferManager.removeAudioBuffer(bufferId) == true) {
233 alDeleteBuffers(1, &alBufferId);
234 if (alGetError() != AL_NO_ERROR) {
235 Console::println(string("Audio sound: '" + id + "': Could not delete buffers"));
236 }
238 }
239 initiated = false;
240}
AudioBufferManager_AudioBufferManaged * addAudioBuffer(const string &id)
Adds a audio buffer to manager / open al stack.
Audio entity base class.
Definition: AudioEntity.h:20
Interface to audio module.
Definition: Audio.h:30
static constexpr int32_t ALSOURCEID_NONE
Definition: Audio.h:37
static constexpr int32_t ALBUFFERID_NONE
Definition: Audio.h:36
static STATIC_DLL_IMPEXT Audio * instance
Definition: Audio.h:38
AudioBufferManager audioBufferManager
Definition: Audio.h:45
Sound audio entity implementation.
Definition: Sound.h:20
void play() override
Plays this audio entity.
Definition: Sound.cpp:66
void rewind() override
Rewinds this audio entity.
Definition: Sound.cpp:55
void dispose() override
Dispose this entity from OpenAL.
Definition: Sound.cpp:223
uint32_t alSourceId
Definition: Sound.h:29
void update() override
Commits properties to OpenAl.
Definition: Sound.cpp:205
bool initiated
Definition: Sound.h:24
bool initialize() override
Initiates this OpenAL entity to OpenAl.
Definition: Sound.cpp:101
string fileName
Definition: Sound.h:26
string pathName
Definition: Sound.h:25
bool isPlaying() override
Definition: Sound.cpp:48
uint32_t alBufferId
Definition: Sound.h:28
void stop() override
Stops this audio entity.
Definition: Sound.cpp:90
string bufferId
Definition: Sound.h:27
void pause() override
Pauses this audio entity.
Definition: Sound.cpp:79
OGG/Vorbis audio decoder.
Definition: VorbisDecoder.h:29
virtual void openFile(const string &pathName, const string &fileName)
Open a local file.
virtual void close()
Closes the audio file.
virtual int32_t readFromStream(ByteBuffer *data)
Read raw PCM data from stream.
3D vector 3 class
Definition: Vector3.h:22
array< float, 3 > & getArray() const
Definition: Vector3.h:171
uint8_t * getBuffer()
Definition: Buffer.h:131
virtual int32_t getPosition()
Definition: Buffer.h:84
Byte buffer class.
Definition: ByteBuffer.h:24
Console class.
Definition: Console.h:26