Skip to content
This repository has been archived by the owner on Feb 11, 2021. It is now read-only.

Latest commit

 

History

History
418 lines (347 loc) · 11.8 KB

SDL_mintaudio_mcsn.c

File metadata and controls

418 lines (347 loc) · 11.8 KB
 
1
2
/*
SDL - Simple DirectMedia Layer
Jan 4, 2004
Jan 4, 2004
3
Copyright (C) 1997-2004 Sam Lantinga
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation; either
version 2 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
Library General Public License for more details.
You should have received a copy of the GNU Library General Public
License along with this library; if not, write to the Free
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
Sam Lantinga
slouken@libsdl.org
*/
Feb 21, 2006
Feb 21, 2006
22
#include "SDL_config.h"
23
24
25
26
27
28
29
30
/*
MiNT audio driver
using XBIOS functions (MacSound compatible driver)
Patrice Mandin
*/
Aug 10, 2005
Aug 10, 2005
31
#include <support.h>
32
33
34
35
36
37
38
/* Mint includes */
#include <mint/osbind.h>
#include <mint/falcon.h>
#include <mint/cookie.h>
#include "SDL_audio.h"
Feb 16, 2006
Feb 16, 2006
39
40
#include "../SDL_audio_c.h"
#include "../SDL_sysaudio.h"
41
Feb 21, 2006
Feb 21, 2006
42
#include "../../video/ataricommon/SDL_atarimxalloc_c.h"
43
44
45
46
47
48
49
50
51
52
#include "SDL_mintaudio.h"
#include "SDL_mintaudio_mcsn.h"
/*--- Defines ---*/
#define MINT_AUDIO_DRIVER_NAME "mint_mcsn"
/* Debug print info */
#define DEBUG_NAME "audio:mcsn: "
Oct 29, 2004
Oct 29, 2004
53
#if 0
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
#define DEBUG_PRINT(what) \
{ \
printf what; \
}
#else
#define DEBUG_PRINT(what)
#endif
/*--- Static variables ---*/
static unsigned long cookie_snd, cookie_mch;
static cookie_mcsn_t *cookie_mcsn;
/*--- Audio driver functions ---*/
static void Mint_CloseAudio(_THIS);
Jul 10, 2006
Jul 10, 2006
70
static int Mint_OpenAudio(_THIS, SDL_AudioSpec * spec);
71
72
73
74
static void Mint_LockAudio(_THIS);
static void Mint_UnlockAudio(_THIS);
/* To check/init hardware audio */
Jul 10, 2006
Jul 10, 2006
75
76
static int Mint_CheckAudio(_THIS, SDL_AudioSpec * spec);
static void Mint_InitAudio(_THIS, SDL_AudioSpec * spec);
77
78
79
/*--- Audio driver bootstrap functions ---*/
Jul 10, 2006
Jul 10, 2006
80
81
static int
Audio_Available(void)
82
{
Jul 10, 2006
Jul 10, 2006
83
84
unsigned long dummy;
const char *envr = SDL_getenv("SDL_AUDIODRIVER");
85
Jul 10, 2006
Jul 10, 2006
86
SDL_MintAudio_mint_present = (Getcookie(C_MiNT, &dummy) == C_FOUND);
Jul 20, 2005
Jul 20, 2005
87
Jul 10, 2006
Jul 10, 2006
88
89
90
91
/* We can't use XBIOS in interrupt with Magic, don't know about thread */
if (Getcookie(C_MagX, &dummy) == C_FOUND) {
return (0);
}
Jul 29, 2005
Jul 29, 2005
92
Jul 10, 2006
Jul 10, 2006
93
94
95
96
97
/* Check if user asked a different audio driver */
if ((envr) && (SDL_strcmp(envr, MINT_AUDIO_DRIVER_NAME) != 0)) {
DEBUG_PRINT((DEBUG_NAME "user asked a different audio driver\n"));
return (0);
}
98
Jul 10, 2006
Jul 10, 2006
99
100
101
102
/* Cookie _MCH present ? if not, assume ST machine */
if (Getcookie(C__MCH, &cookie_mch) == C_NOTFOUND) {
cookie_mch = MCH_ST;
}
103
Jul 10, 2006
Jul 10, 2006
104
105
106
107
/* Cookie _SND present ? if not, assume ST machine */
if (Getcookie(C__SND, &cookie_snd) == C_NOTFOUND) {
cookie_snd = SND_PSG;
}
108
Jul 10, 2006
Jul 10, 2006
109
110
111
112
113
/* Check if we have 16 bits audio */
if ((cookie_snd & SND_16BIT) == 0) {
DEBUG_PRINT((DEBUG_NAME "no 16 bits sound\n"));
return (0);
}
114
Jul 10, 2006
Jul 10, 2006
115
116
117
118
119
/* Cookie MCSN present ? */
if (Getcookie(C_McSn, (long *) &cookie_mcsn) != C_FOUND) {
DEBUG_PRINT((DEBUG_NAME "no MCSN audio\n"));
return (0);
}
120
Jul 10, 2006
Jul 10, 2006
121
122
123
124
125
/* Check if interrupt at end of replay */
if (cookie_mcsn->pint == 0) {
DEBUG_PRINT((DEBUG_NAME "no interrupt at end of replay\n"));
return (0);
}
126
Jul 10, 2006
Jul 10, 2006
127
128
129
130
131
/* Check if audio is lockable */
if (Locksnd() != 1) {
DEBUG_PRINT((DEBUG_NAME "audio locked by other application\n"));
return (0);
}
132
Jul 10, 2006
Jul 10, 2006
133
Unlocksnd();
134
Jul 10, 2006
Jul 10, 2006
135
136
DEBUG_PRINT((DEBUG_NAME "MCSN audio available!\n"));
return (1);
137
138
}
Jul 10, 2006
Jul 10, 2006
139
140
static void
Audio_DeleteDevice(SDL_AudioDevice * device)
141
{
Feb 7, 2006
Feb 7, 2006
142
143
SDL_free(device->hidden);
SDL_free(device);
144
145
}
Jul 10, 2006
Jul 10, 2006
146
147
static SDL_AudioDevice *
Audio_CreateDevice(int devindex)
148
{
Jul 10, 2006
Jul 10, 2006
149
SDL_AudioDevice *this;
150
Jul 10, 2006
Jul 10, 2006
151
152
153
/* Initialize all variables that we clean on shutdown */
this = (SDL_AudioDevice *) SDL_malloc(sizeof(SDL_AudioDevice));
if (this) {
Feb 7, 2006
Feb 7, 2006
154
SDL_memset(this, 0, (sizeof *this));
155
this->hidden = (struct SDL_PrivateAudioData *)
Jul 10, 2006
Jul 10, 2006
156
SDL_malloc((sizeof *this->hidden));
157
}
Jul 10, 2006
Jul 10, 2006
158
if ((this == NULL) || (this->hidden == NULL)) {
159
SDL_OutOfMemory();
Jul 10, 2006
Jul 10, 2006
160
if (this) {
Feb 7, 2006
Feb 7, 2006
161
SDL_free(this);
162
}
Jul 10, 2006
Jul 10, 2006
163
return (0);
164
}
Feb 7, 2006
Feb 7, 2006
165
SDL_memset(this->hidden, 0, (sizeof *this->hidden));
166
167
/* Set the function pointers */
Jul 10, 2006
Jul 10, 2006
168
169
170
this->OpenAudio = Mint_OpenAudio;
this->CloseAudio = Mint_CloseAudio;
this->LockAudio = Mint_LockAudio;
171
this->UnlockAudio = Mint_UnlockAudio;
Jul 10, 2006
Jul 10, 2006
172
this->free = Audio_DeleteDevice;
173
174
175
176
177
return this;
}
AudioBootStrap MINTAUDIO_MCSN_bootstrap = {
Jul 10, 2006
Jul 10, 2006
178
179
MINT_AUDIO_DRIVER_NAME, "MiNT MCSN audio driver",
Audio_Available, Audio_CreateDevice
180
181
};
Jul 10, 2006
Jul 10, 2006
182
183
static void
Mint_LockAudio(_THIS)
184
{
Jul 10, 2006
Jul 10, 2006
185
186
/* Stop replay */
Buffoper(0);
187
188
}
Jul 10, 2006
Jul 10, 2006
189
190
static void
Mint_UnlockAudio(_THIS)
191
{
Jul 10, 2006
Jul 10, 2006
192
193
/* Restart replay */
Buffoper(SB_PLA_ENA | SB_PLA_RPT);
194
195
}
Jul 10, 2006
Jul 10, 2006
196
197
static void
Mint_CloseAudio(_THIS)
198
{
Jul 10, 2006
Jul 10, 2006
199
200
201
/* Stop replay */
SDL_MintAudio_WaitThread();
Buffoper(0);
202
Jul 10, 2006
Jul 10, 2006
203
204
205
206
if (!SDL_MintAudio_mint_present) {
/* Uninstall interrupt */
Jdisint(MFP_DMASOUND);
}
207
Jul 10, 2006
Jul 10, 2006
208
209
210
/* Wait if currently playing sound */
while (SDL_MintAudio_mutex != 0) {
}
211
Jul 10, 2006
Jul 10, 2006
212
213
214
215
216
/* Clear buffers */
if (SDL_MintAudio_audiobuf[0]) {
Mfree(SDL_MintAudio_audiobuf[0]);
SDL_MintAudio_audiobuf[0] = SDL_MintAudio_audiobuf[1] = NULL;
}
217
Jul 10, 2006
Jul 10, 2006
218
219
/* Unlock sound system */
Unlocksnd();
220
221
}
Jul 10, 2006
Jul 10, 2006
222
223
static int
Mint_CheckAudio(_THIS, SDL_AudioSpec * spec)
224
{
Jul 10, 2006
Jul 10, 2006
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
int i;
unsigned long masterclock, masterprediv;
DEBUG_PRINT((DEBUG_NAME "asked: %d bits, ", spec->format & 0x00ff));
DEBUG_PRINT(("signed=%d, ", ((spec->format & 0x8000) != 0)));
DEBUG_PRINT(("big endian=%d, ", ((spec->format & 0x1000) != 0)));
DEBUG_PRINT(("channels=%d, ", spec->channels));
DEBUG_PRINT(("freq=%d\n", spec->freq));
/* Check formats available */
MINTAUDIO_freqcount = 0;
switch (cookie_mcsn->play) {
case MCSN_ST:
spec->channels = 1;
spec->format = 8; /* FIXME: is it signed or unsigned ? */
SDL_MintAudio_AddFrequency(this, 12500, 0, 0, -1);
break;
case MCSN_TT: /* Also STE, Mega STE */
spec->format = AUDIO_S8;
masterclock = MASTERCLOCK_STE;
masterprediv = MASTERPREDIV_STE;
if ((cookie_mch >> 16) == MCH_TT) {
masterclock = MASTERCLOCK_TT;
masterprediv = MASTERPREDIV_TT;
}
for (i = 0; i < 4; i++) {
SDL_MintAudio_AddFrequency(this,
masterclock / (masterprediv *
(1 << i)),
masterclock, 3 - i, -1);
}
break;
case MCSN_FALCON: /* Also Mac */
for (i = 1; i < 12; i++) {
/* Remove unusable Falcon codec predivisors */
if ((i == 6) || (i == 8) || (i == 10)) {
continue;
}
SDL_MintAudio_AddFrequency(this,
MASTERCLOCK_FALCON1 /
(MASTERPREDIV_FALCON * (i + 1)),
CLK25M, i + 1, -1);
}
if (cookie_mcsn->res1 != 0) {
for (i = 1; i < 4; i++) {
SDL_MintAudio_AddFrequency(this,
(cookie_mcsn->res1) /
(MASTERPREDIV_FALCON *
(1 << i)), CLKEXT,
(1 << i) - 1, -1);
}
}
spec->format |= 0x8000; /* Audio is always signed */
if ((spec->format & 0x00ff) == 16) {
spec->format |= 0x1000; /* Audio is always big endian */
spec->channels = 2; /* 16 bits always stereo */
}
break;
}
284
Oct 29, 2004
Oct 29, 2004
285
#if 1
Jul 10, 2006
Jul 10, 2006
286
287
288
289
290
291
for (i = 0; i < MINTAUDIO_freqcount; i++) {
DEBUG_PRINT((DEBUG_NAME "freq %d: %lu Hz, clock %lu, prediv %d\n",
i, MINTAUDIO_frequencies[i].frequency,
MINTAUDIO_frequencies[i].masterclock,
MINTAUDIO_frequencies[i].predivisor));
}
Oct 29, 2004
Oct 29, 2004
292
293
#endif
Jul 10, 2006
Jul 10, 2006
294
295
MINTAUDIO_numfreq = SDL_MintAudio_SearchFrequency(this, spec->freq);
spec->freq = MINTAUDIO_frequencies[MINTAUDIO_numfreq].frequency;
296
Jul 10, 2006
Jul 10, 2006
297
298
299
300
301
DEBUG_PRINT((DEBUG_NAME "obtained: %d bits, ", spec->format & 0x00ff));
DEBUG_PRINT(("signed=%d, ", ((spec->format & 0x8000) != 0)));
DEBUG_PRINT(("big endian=%d, ", ((spec->format & 0x1000) != 0)));
DEBUG_PRINT(("channels=%d, ", spec->channels));
DEBUG_PRINT(("freq=%d\n", spec->freq));
302
Jul 10, 2006
Jul 10, 2006
303
return 0;
304
305
}
Jul 10, 2006
Jul 10, 2006
306
307
static void
Mint_InitAudio(_THIS, SDL_AudioSpec * spec)
308
{
Jul 10, 2006
Jul 10, 2006
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
int channels_mode, prediv, dmaclock;
void *buffer;
/* Stop currently playing sound */
SDL_MintAudio_quit_thread = SDL_FALSE;
SDL_MintAudio_thread_finished = SDL_TRUE;
SDL_MintAudio_WaitThread();
Buffoper(0);
/* Set replay tracks */
Settracks(0, 0);
Setmontracks(0);
/* Select replay format */
channels_mode = STEREO16;
switch (spec->format & 0xff) {
case 8:
if (spec->channels == 2) {
channels_mode = STEREO8;
} else {
channels_mode = MONO8;
}
break;
}
if (Setmode(channels_mode) < 0) {
DEBUG_PRINT((DEBUG_NAME "Setmode() failed\n"));
}
336
Jul 10, 2006
Jul 10, 2006
337
338
339
340
341
342
343
344
345
346
347
348
349
dmaclock = MINTAUDIO_frequencies[MINTAUDIO_numfreq].masterclock;
prediv = MINTAUDIO_frequencies[MINTAUDIO_numfreq].predivisor;
switch (cookie_mcsn->play) {
case MCSN_TT:
Devconnect(DMAPLAY, DAC, CLK25M, CLKOLD, 1);
Soundcmd(SETPRESCALE, prediv);
DEBUG_PRINT((DEBUG_NAME "STE/TT prescaler selected\n"));
break;
case MCSN_FALCON:
Devconnect(DMAPLAY, DAC, dmaclock, prediv, 1);
DEBUG_PRINT((DEBUG_NAME "Falcon prescaler selected\n"));
break;
}
350
Jul 10, 2006
Jul 10, 2006
351
352
353
354
355
/* Set buffer */
buffer = SDL_MintAudio_audiobuf[SDL_MintAudio_numbuf];
if (Setbuffer(0, buffer, buffer + spec->size) < 0) {
DEBUG_PRINT((DEBUG_NAME "Setbuffer() failed\n"));
}
356
Jul 10, 2006
Jul 10, 2006
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
if (SDL_MintAudio_mint_present) {
SDL_MintAudio_thread_pid = tfork(SDL_MintAudio_Thread, 0);
} else {
/* Install interrupt */
Jdisint(MFP_DMASOUND);
Xbtimer(XB_TIMERA, 8, 1, SDL_MintAudio_XbiosInterrupt);
Jenabint(MFP_DMASOUND);
if (Setinterrupt(SI_TIMERA, SI_PLAY) < 0) {
DEBUG_PRINT((DEBUG_NAME "Setinterrupt() failed\n"));
}
}
/* Go */
Buffoper(SB_PLA_ENA | SB_PLA_RPT);
DEBUG_PRINT((DEBUG_NAME "hardware initialized\n"));
373
374
}
Jul 10, 2006
Jul 10, 2006
375
376
static int
Mint_OpenAudio(_THIS, SDL_AudioSpec * spec)
377
{
Jul 10, 2006
Jul 10, 2006
378
379
380
381
382
/* Lock sound system */
if (Locksnd() != 1) {
SDL_SetError("Mint_OpenAudio: Audio system already in use");
return (-1);
}
383
Jul 10, 2006
Jul 10, 2006
384
SDL_MintAudio_device = this;
385
Jul 10, 2006
Jul 10, 2006
386
387
388
389
/* Check audio capabilities */
if (Mint_CheckAudio(this, spec) == -1) {
return -1;
}
390
Jul 10, 2006
Jul 10, 2006
391
SDL_CalculateAudioSpec(spec);
392
Jul 10, 2006
Jul 10, 2006
393
394
/* Allocate memory for audio buffers in DMA-able RAM */
DEBUG_PRINT((DEBUG_NAME "buffer size=%d\n", spec->size));
395
Jul 10, 2006
Jul 10, 2006
396
397
398
399
400
401
402
403
404
405
SDL_MintAudio_audiobuf[0] = Atari_SysMalloc(spec->size * 2, MX_STRAM);
if (SDL_MintAudio_audiobuf[0] == NULL) {
SDL_SetError("MINT_OpenAudio: Not enough memory for audio buffer");
return (-1);
}
SDL_MintAudio_audiobuf[1] = SDL_MintAudio_audiobuf[0] + spec->size;
SDL_MintAudio_numbuf = 0;
SDL_memset(SDL_MintAudio_audiobuf[0], spec->silence, spec->size * 2);
SDL_MintAudio_audiosize = spec->size;
SDL_MintAudio_mutex = 0;
406
Jul 10, 2006
Jul 10, 2006
407
408
409
410
DEBUG_PRINT((DEBUG_NAME "buffer 0 at 0x%08x\n",
SDL_MintAudio_audiobuf[0]));
DEBUG_PRINT((DEBUG_NAME "buffer 1 at 0x%08x\n",
SDL_MintAudio_audiobuf[1]));
411
Jul 10, 2006
Jul 10, 2006
412
413
/* Setup audio hardware */
Mint_InitAudio(this, spec);
414
Jul 10, 2006
Jul 10, 2006
415
return (1); /* We don't use SDL threaded audio */
416
}
Jul 10, 2006
Jul 10, 2006
417
418
/* vi: set ts=4 sw=4 expandtab: */