libgui/source/sounds/OggDecoder.cpp

135 lines
3.3 KiB
C++
Raw Normal View History

2017-10-29 10:28:14 +01:00
/***************************************************************************
* Copyright (C) 2010
* by Dimok
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any
* damages arising from the use of this software.
*
* Permission is granted to anyone to use this software for any
* purpose, including commercial applications, and to alter it and
* redistribute it freely, subject to the following restrictions:
*
* 1. The origin of this software must not be misrepresented; you
* must not claim that you wrote the original software. If you use
* this software in a product, an acknowledgment in the product
* documentation would be appreciated but is not required.
*
* 2. Altered source versions must be plainly marked as such, and
* must not be misrepresented as being the original software.
*
* 3. This notice may not be removed or altered from any source
* distribution.
*
* for WiiXplorer 2010
***************************************************************************/
#include <unistd.h>
#include <malloc.h>
2018-06-21 20:44:58 +02:00
#include <coreinit/time.h>
#include <coreinit/thread.h>
2019-08-14 23:24:55 +02:00
#include <gui/sounds/OggDecoder.hpp>
#include "fs/CFile.hpp"
2017-10-29 10:28:14 +01:00
2020-08-13 12:38:07 +02:00
static int ogg_read(void *punt, int bytes, int blocks, int *f) {
return ((CFile *) f)->read((uint8_t *) punt, bytes * blocks);
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
static int ogg_seek(int *f, ogg_int64_t offset, int mode) {
return ((CFile *) f)->seek((uint64_t) offset, mode);
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
static int ogg_close(int *f) {
((CFile *) f)->close();
return 0;
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
static long ogg_tell(int *f) {
return (long) ((CFile *) f)->tell();
2017-10-29 10:28:14 +01:00
}
static ov_callbacks callbacks = {
2020-08-13 12:38:07 +02:00
(size_t (*)(void *, size_t, size_t, void *)) ogg_read,
(int (*)(void *, ogg_int64_t, int)) ogg_seek,
(int (*)(void *)) ogg_close,
(long (*)(void *)) ogg_tell
2017-10-29 10:28:14 +01:00
};
2020-08-13 12:38:07 +02:00
OggDecoder::OggDecoder(const char *filepath)
: SoundDecoder(filepath) {
2018-06-21 20:44:58 +02:00
SoundType = SOUND_OGG;
2017-10-29 10:28:14 +01:00
2020-08-13 12:58:19 +02:00
if (!file_fd) {
2018-06-21 20:44:58 +02:00
return;
2020-08-13 12:58:19 +02:00
}
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
OpenFile();
2017-10-29 10:28:14 +01:00
}
2020-08-13 12:38:07 +02:00
OggDecoder::OggDecoder(const uint8_t *snd, int32_t len)
: SoundDecoder(snd, len) {
2018-06-21 20:44:58 +02:00
SoundType = SOUND_OGG;
2017-10-29 10:28:14 +01:00
2020-08-13 12:58:19 +02:00
if (!file_fd) {
2018-06-21 20:44:58 +02:00
return;
2020-08-13 12:58:19 +02:00
}
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
OpenFile();
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
OggDecoder::~OggDecoder() {
ExitRequested = true;
2020-08-13 12:38:07 +02:00
while (Decoding)
2018-06-21 20:44:58 +02:00
OSSleepTicks(OSMicrosecondsToTicks(100));
2017-10-29 10:28:14 +01:00
2020-08-13 12:58:19 +02:00
if (file_fd) {
2018-06-21 20:44:58 +02:00
ov_clear(&ogg_file);
2020-08-13 12:58:19 +02:00
}
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
void OggDecoder::OpenFile() {
if (ov_open_callbacks(file_fd, &ogg_file, NULL, 0, callbacks) < 0) {
delete file_fd;
file_fd = NULL;
return;
}
ogg_info = ov_info(&ogg_file, -1);
2020-08-13 12:38:07 +02:00
if (!ogg_info) {
2018-06-21 20:44:58 +02:00
ov_clear(&ogg_file);
delete file_fd;
file_fd = NULL;
return;
}
Format = ((ogg_info->channels == 2) ? (FORMAT_PCM_16_BIT | CHANNELS_STEREO) : (FORMAT_PCM_16_BIT | CHANNELS_MONO));
SampleRate = ogg_info->rate;
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
int32_t OggDecoder::Rewind() {
2020-08-13 12:58:19 +02:00
if (!file_fd) {
2018-06-21 20:44:58 +02:00
return -1;
2020-08-13 12:58:19 +02:00
}
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
int32_t ret = ov_time_seek(&ogg_file, 0);
CurPos = 0;
EndOfFile = false;
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
return ret;
2017-10-29 10:28:14 +01:00
}
2020-08-13 12:38:07 +02:00
int32_t OggDecoder::Read(uint8_t *buffer, int32_t buffer_size, int32_t pos) {
2020-08-13 12:58:19 +02:00
if (!file_fd) {
2018-06-21 20:44:58 +02:00
return -1;
2020-08-13 12:58:19 +02:00
}
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
int32_t bitstream = 0;
2017-10-29 10:28:14 +01:00
2020-08-13 12:38:07 +02:00
int32_t read = (int32_t) ov_read(&ogg_file, (char *) buffer, (int) buffer_size, (int *) &bitstream);
2017-10-29 10:28:14 +01:00
2020-08-13 12:58:19 +02:00
if (read > 0) {
2018-06-21 20:44:58 +02:00
CurPos += read;
2020-08-13 12:58:19 +02:00
}
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
return read;
2017-10-29 10:28:14 +01:00
}