Skip to content

Latest commit

 

History

History
163 lines (128 loc) · 4.4 KB

load_ogg.c

File metadata and controls

163 lines (128 loc) · 4.4 KB
 
Dec 31, 2011
Dec 31, 2011
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
SDL_mixer: An audio mixer library based on the SDL library
Copyright (C) 1997-2012 Sam Lantinga <slouken@libsdl.org>
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.
This is the source needed to decode an Ogg Vorbis into a waveform.
This file by Vaclav Slavik (vaclav.slavik@matfyz.cz).
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
*/
/* $Id$ */
#ifdef OGG_MUSIC
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "SDL_mutex.h"
#include "SDL_endian.h"
#include "SDL_timer.h"
#include "SDL_mixer.h"
May 12, 2006
May 12, 2006
38
#include "dynamic_ogg.h"
39
40
41
42
43
44
45
46
47
#include "load_ogg.h"
static size_t sdl_read_func(void *ptr, size_t size, size_t nmemb, void *datasource)
{
return SDL_RWread((SDL_RWops*)datasource, ptr, size, nmemb);
}
static int sdl_seek_func(void *datasource, ogg_int64_t offset, int whence)
{
Feb 26, 2003
Feb 26, 2003
48
return SDL_RWseek((SDL_RWops*)datasource, (int)offset, whence);
49
50
51
52
53
54
55
56
57
}
static int sdl_close_func_freesrc(void *datasource)
{
return SDL_RWclose((SDL_RWops*)datasource);
}
static int sdl_close_func_nofreesrc(void *datasource)
{
Nov 8, 2009
Nov 8, 2009
58
return SDL_RWseek((SDL_RWops*)datasource, 0, RW_SEEK_SET);
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
}
static long sdl_tell_func(void *datasource)
{
return SDL_RWtell((SDL_RWops*)datasource);
}
/* don't call this directly; use Mix_LoadWAV_RW() for now. */
SDL_AudioSpec *Mix_LoadOGG_RW (SDL_RWops *src, int freesrc,
SDL_AudioSpec *spec, Uint8 **audio_buf, Uint32 *audio_len)
{
OggVorbis_File vf;
ov_callbacks callbacks;
vorbis_info *info;
Uint8 *buf;
int bitstream = -1;
long samplesize;
long samples;
int read, to_read;
int must_close = 1;
int was_error = 1;
if ( (!src) || (!audio_buf) || (!audio_len) ) /* sanity checks. */
goto done;
Nov 8, 2009
Nov 8, 2009
85
if ( !Mix_Init(MIX_INIT_OGG) )
May 12, 2006
May 12, 2006
86
87
goto done;
88
89
90
91
92
93
callbacks.read_func = sdl_read_func;
callbacks.seek_func = sdl_seek_func;
callbacks.tell_func = sdl_tell_func;
callbacks.close_func = freesrc ?
sdl_close_func_freesrc : sdl_close_func_nofreesrc;
May 12, 2006
May 12, 2006
94
if (vorbis.ov_open_callbacks(src, &vf, NULL, 0, callbacks) != 0)
95
96
97
98
99
100
101
{
SDL_SetError("OGG bitstream is not valid Vorbis stream!");
goto done;
}
must_close = 0;
May 12, 2006
May 12, 2006
102
info = vorbis.ov_info(&vf, -1);
103
104
105
106
107
108
109
110
111
112
*audio_buf = NULL;
*audio_len = 0;
memset(spec, '\0', sizeof (SDL_AudioSpec));
spec->format = AUDIO_S16;
spec->channels = info->channels;
spec->freq = info->rate;
spec->samples = 4096; /* buffer size */
May 12, 2006
May 12, 2006
113
samples = (long)vorbis.ov_pcm_total(&vf, -1);
114
115
116
117
118
119
120
121
*audio_len = spec->size = samples * spec->channels * 2;
*audio_buf = malloc(*audio_len);
if (*audio_buf == NULL)
goto done;
buf = *audio_buf;
to_read = *audio_len;
Jul 15, 2007
Jul 15, 2007
122
123
124
125
126
#ifdef OGG_USE_TREMOR
for (read = vorbis.ov_read(&vf, (char *)buf, to_read, &bitstream);
read > 0;
read = vorbis.ov_read(&vf, (char *)buf, to_read, &bitstream))
#else
May 12, 2006
May 12, 2006
127
for (read = vorbis.ov_read(&vf, (char *)buf, to_read, 0/*LE*/, 2/*16bit*/, 1/*signed*/, &bitstream);
128
read > 0;
May 12, 2006
May 12, 2006
129
read = vorbis.ov_read(&vf, (char *)buf, to_read, 0, 2, 1, &bitstream))
Jul 15, 2007
Jul 15, 2007
130
#endif
131
132
133
134
135
136
137
138
{
if (read == OV_HOLE || read == OV_EBADLINK)
break; /* error */
to_read -= read;
buf += read;
}
May 12, 2006
May 12, 2006
139
vorbis.ov_clear(&vf);
140
141
142
143
144
145
146
147
148
149
150
151
was_error = 0;
/* Don't return a buffer that isn't a multiple of samplesize */
samplesize = ((spec->format & 0xFF)/8)*spec->channels;
*audio_len &= ~(samplesize-1);
done:
if (src && must_close)
{
if (freesrc)
SDL_RWclose(src);
else
Nov 8, 2009
Nov 8, 2009
152
SDL_RWseek(src, 0, RW_SEEK_SET);
153
154
155
156
157
158
159
160
161
162
163
}
if ( was_error )
spec = NULL;
return(spec);
} /* Mix_LoadOGG_RW */
/* end of load_ogg.c ... */
#endif