Mercurial > fife-parpg
view engine/core/audio/soundclip.cpp @ 181:56ac89189bc4
fixed time handling in fife:
- the time scaling method was unreliable when changing the time factors (jumping in the moment of the change)
- replaced much calls to SDL_GetTicks with TimerManager::instance()->getTime()
- some other time related cleanup
author | spq@33b003aa-7bff-0310-803a-e67f0ece8222 |
---|---|
date | Sat, 31 Jan 2009 04:15:43 +0000 |
parents | 90005975cdbb |
children | 3804348fe3fb |
line wrap: on
line source
/*************************************************************************** * Copyright (C) 2005-2008 by the FIFE team * * http://www.fifengine.de * * This file is part of FIFE. * * * * FIFE is free software; you can redistribute it and/or * * modify it under the terms of the GNU Lesser General Public * * License as published by the Free Software Foundation; either * * version 2.1 of the License, or (at your option) any later version. * * * * This library is distributed in the hope that it will be useful, * * but WITHOUT ANY WARRANTY; without even the implied warranty of * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * * Lesser General Public License for more details. * * * * You should have received a copy of the GNU Lesser General Public * * License along with this library; if not, write to the * * Free Software Foundation, Inc., * * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * ***************************************************************************/ // Standard C++ library includes // Platform specific includes // 3rd party library includes // FIFE includes // These includes are split up in two parts, separated by one empty line // First block: files included from the FIFE root src directory // Second block: files included from the same folder #include "util/base/exception.h" #include "util/log/logger.h" #include "soundclip.h" namespace FIFE { static Logger _log(LM_AUDIO); SoundClip::SoundClip(SoundDecoder* decptr, bool deletedecoder) : m_location(ResourceLocation("")), m_isstream(decptr->needsStreaming()), m_decoder(decptr), m_deletedecoder(deletedecoder) { if (!m_isstream) { // only for non-streaming buffers SoundBufferEntry* ptr = new SoundBufferEntry(); // iterate the bufs and fill them with data for (int i = 0; i < BUFFER_NUM; i++) { if (m_decoder->decode(BUFFER_LEN)) { // EOF or error break; } // generate buffer and fill it with data alGenBuffers(1, &ptr->buffers[i]); alBufferData(ptr->buffers[i], m_decoder->getALFormat(), m_decoder->getBuffer(), m_decoder->getBufferSize(), m_decoder->getSampleRate()); CHECK_OPENAL_LOG(_log, LogManager::LEVEL_ERROR, "error copying data to buffers") ptr->usedbufs++; } m_decoder->releaseBuffer(); // push the buffer information to the vector m_buffervec.push_back(ptr); } } unsigned int SoundClip::beginStreaming() { // create new sound buffer entry SoundBufferEntry* ptr = new SoundBufferEntry(); ptr->usedbufs=0; alGenBuffers(BUFFER_NUM, ptr->buffers); CHECK_OPENAL_LOG(_log, LogManager::LEVEL_ERROR, "error creating streaming-buffers") m_buffervec.push_back(ptr); return m_buffervec.size()-1; } bool SoundClip::setStreamPos(unsigned int streamid, SoundPositionType type, float value) { unsigned long pos = 0; // convert position to bytes switch (type) { case SD_BYTE_POS: pos = static_cast<unsigned long>(value); break; case SD_TIME_POS: value *= m_decoder->getSampleRate(); case SD_SAMPLE_POS: pos = static_cast<unsigned long>((m_decoder->getBitResolution() / 8) * (m_decoder->isStereo() ? 2 : 1) * value); break; } if (pos > m_decoder->getDecodedLength()) { // EOF! return true; } // set cursor position m_buffervec.at(streamid)->deccursor = pos; return false; } float SoundClip::getStreamPos(unsigned int streamid, SoundPositionType type) { unsigned long pos = m_buffervec.at(streamid)->deccursor; switch(type) { case SD_BYTE_POS: return pos; case SD_SAMPLE_POS: return pos / (m_decoder->getBitResolution() / 8 * (m_decoder->isStereo() ? 2 : 1)); case SD_TIME_POS: return pos / (m_decoder->getBitResolution() / 8 * (m_decoder->isStereo() ? 2 : 1) * m_decoder->getSampleRate()); } return 0.0f; } void SoundClip::acquireStream(unsigned int streamid) { SoundBufferEntry* ptr = m_buffervec.at(streamid); for (int i = 0; i < BUFFER_NUM; i++) { getStream(streamid, ptr->buffers[i]); } } bool SoundClip::getStream(unsigned int streamid, ALuint buffer) { SoundBufferEntry* ptr = m_buffervec.at(streamid); if (ptr->deccursor >= m_decoder->getDecodedLength()) { // EOF! return true; } // set cursor of decoder m_decoder->setCursor(ptr->deccursor); // Error while decoding file? if (m_decoder->decode(BUFFER_LEN)) { throw Exception("error while reading from audio file"); } // fill the buffer with data alBufferData(buffer, m_decoder->getALFormat(), m_decoder->getBuffer(), m_decoder->getBufferSize(), m_decoder->getSampleRate()); m_decoder->releaseBuffer(); // update cursor ptr->deccursor += BUFFER_LEN; CHECK_OPENAL_LOG(_log, LogManager::LEVEL_ERROR, "error catching stream") return false; } void SoundClip::quitStreaming(unsigned int streamid) { // release the buffers SoundBufferEntry* ptr = m_buffervec.at(streamid); alDeleteBuffers(BUFFER_NUM, ptr->buffers); ptr->buffers[0] = 0; } SoundClip::~SoundClip() { if (m_isstream) { // erase all elements from the list std::vector<SoundBufferEntry*>::iterator it; for (it = m_buffervec.begin(); it != m_buffervec.end(); ++it) { if ((*it)->buffers[0] != 0) { alDeleteBuffers(BUFFER_NUM, (*it)->buffers); } delete (*it); } m_buffervec.clear(); } else { // for non-streaming soundclips SoundBufferEntry* ptr = m_buffervec.at(0); for(unsigned int i = 0; i < ptr->usedbufs; i++) { alDeleteBuffers(1, &ptr->buffers[i]); } } // delete decoder if (m_deletedecoder && m_decoder != NULL) { delete m_decoder; } } }