|
Packit |
64f477 |
/*
|
|
Packit |
64f477 |
* audio_out_win.c
|
|
Packit |
64f477 |
* Copyright (C) 2000-2002 Michel Lespinasse <walken@zoy.org>
|
|
Packit |
64f477 |
* Copyright (C) 1999-2000 Aaron Holtzman <aholtzma@ess.engr.uvic.ca>
|
|
Packit |
64f477 |
*
|
|
Packit |
64f477 |
* This file is part of a52dec, a free ATSC A-52 stream decoder.
|
|
Packit |
64f477 |
* See http://liba52.sourceforge.net/ for updates.
|
|
Packit |
64f477 |
*
|
|
Packit |
64f477 |
* a52dec is free software; you can redistribute it and/or modify
|
|
Packit |
64f477 |
* it under the terms of the GNU General Public License as published by
|
|
Packit |
64f477 |
* the Free Software Foundation; either version 2 of the License, or
|
|
Packit |
64f477 |
* (at your option) any later version.
|
|
Packit |
64f477 |
*
|
|
Packit |
64f477 |
* a52dec is distributed in the hope that it will be useful,
|
|
Packit |
64f477 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
64f477 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
Packit |
64f477 |
* GNU General Public License for more details.
|
|
Packit |
64f477 |
*
|
|
Packit |
64f477 |
* You should have received a copy of the GNU General Public License
|
|
Packit |
64f477 |
* along with this program; if not, write to the Free Software
|
|
Packit |
64f477 |
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
Packit |
64f477 |
*/
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
#include "config.h"
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
#ifdef LIBAO_WIN
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
#include <stdio.h>
|
|
Packit |
64f477 |
#include <stdlib.h>
|
|
Packit |
64f477 |
#include <fcntl.h>
|
|
Packit |
64f477 |
#include <windows.h>
|
|
Packit |
64f477 |
#include <mmsystem.h>
|
|
Packit |
64f477 |
#include <inttypes.h>
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
#include "a52.h"
|
|
Packit |
64f477 |
#include "audio_out.h"
|
|
Packit |
64f477 |
#include "audio_out_internal.h"
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
static void win_close (ao_instance_t * _instance);
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
/* each buffer has a size of 256 samples. Using 40 of them will give us about
|
|
Packit |
64f477 |
* 1/5 sec of buffered sound for a 48000hz stream */
|
|
Packit |
64f477 |
#define NUMBUF 40
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
typedef struct win_instance_s {
|
|
Packit |
64f477 |
ao_instance_t ao;
|
|
Packit |
64f477 |
HWAVEOUT h_waveout;
|
|
Packit |
64f477 |
WAVEHDR waveheader[NUMBUF];
|
|
Packit |
64f477 |
int16_t int16_samples[NUMBUF][256*2];
|
|
Packit |
64f477 |
int current_buffer;
|
|
Packit |
64f477 |
int sample_rate;
|
|
Packit |
64f477 |
int set_params;
|
|
Packit |
64f477 |
int flags;
|
|
Packit |
64f477 |
} win_instance_t;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
static int win_setup (ao_instance_t * _instance, int sample_rate, int * flags,
|
|
Packit |
64f477 |
sample_t * level, sample_t * bias)
|
|
Packit |
64f477 |
{
|
|
Packit |
64f477 |
win_instance_t * instance = (win_instance_t *) _instance;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
if ((instance->set_params == 0) && (instance->sample_rate != sample_rate))
|
|
Packit |
64f477 |
return 1;
|
|
Packit |
64f477 |
instance->sample_rate = sample_rate;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
*flags = instance->flags;
|
|
Packit |
64f477 |
*level = 1;
|
|
Packit |
64f477 |
*bias = 384;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
return 0;
|
|
Packit |
64f477 |
}
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
static int win_play (ao_instance_t * _instance, int flags, sample_t * _samples)
|
|
Packit |
64f477 |
{
|
|
Packit |
64f477 |
win_instance_t * instance = (win_instance_t *) _instance;
|
|
Packit |
64f477 |
int current_buffer;
|
|
Packit |
64f477 |
MMRESULT result;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
#ifdef LIBA52_DOUBLE
|
|
Packit |
64f477 |
float samples[256 * 2];
|
|
Packit |
64f477 |
int i;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
for (i = 0; i < 256 * 2; i++)
|
|
Packit |
64f477 |
samples[i] = _samples[i];
|
|
Packit |
64f477 |
#else
|
|
Packit |
64f477 |
float * samples = _samples;
|
|
Packit |
64f477 |
#endif
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
flags &= A52_CHANNEL_MASK | A52_LFE;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
if (instance->set_params) {
|
|
Packit |
64f477 |
WAVEFORMATEX waveformat;
|
|
Packit |
64f477 |
MMRESULT result;
|
|
Packit |
64f477 |
int i;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
waveformat.wFormatTag = WAVE_FORMAT_PCM;
|
|
Packit |
64f477 |
waveformat.nChannels = 2;
|
|
Packit |
64f477 |
waveformat.nSamplesPerSec = instance->sample_rate;
|
|
Packit |
64f477 |
waveformat.wBitsPerSample = 16;
|
|
Packit |
64f477 |
waveformat.nBlockAlign = 4;
|
|
Packit |
64f477 |
waveformat.nAvgBytesPerSec = 4 * instance->sample_rate;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
result = waveOutOpen (&instance->h_waveout, WAVE_MAPPER, &waveformat,
|
|
Packit |
64f477 |
0 /*callback*/, 0 /*data*/, CALLBACK_NULL);
|
|
Packit |
64f477 |
if (result != MMSYSERR_NOERROR) {
|
|
Packit |
64f477 |
fprintf (stderr, "Can not open waveOut device\n");
|
|
Packit |
64f477 |
return 1;
|
|
Packit |
64f477 |
}
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
for (i = 0; i < NUMBUF; i++) {
|
|
Packit |
64f477 |
instance->waveheader[i].lpData = (LPSTR)instance->int16_samples[i];
|
|
Packit |
64f477 |
instance->waveheader[i].dwBufferLength = 256 * 2 * sizeof(int16_t);
|
|
Packit |
64f477 |
instance->waveheader[i].dwFlags = WHDR_DONE;
|
|
Packit |
64f477 |
}
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
instance->flags = flags;
|
|
Packit |
64f477 |
instance->set_params = 0;
|
|
Packit |
64f477 |
} else if ((flags == A52_DOLBY) && (instance->flags == A52_STEREO)) {
|
|
Packit |
64f477 |
fprintf (stderr, "Switching from stereo to dolby surround\n");
|
|
Packit |
64f477 |
instance->flags = A52_DOLBY;
|
|
Packit |
64f477 |
} else if ((flags == A52_STEREO) && (instance->flags == A52_DOLBY)) {
|
|
Packit |
64f477 |
fprintf (stderr, "Switching from dolby surround to stereo\n");
|
|
Packit |
64f477 |
instance->flags = A52_STEREO;
|
|
Packit |
64f477 |
} else if (flags != instance->flags)
|
|
Packit |
64f477 |
return 1;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
current_buffer = instance->current_buffer;
|
|
Packit |
64f477 |
instance->current_buffer = (current_buffer + 1) % NUMBUF;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
while (!(instance->waveheader[current_buffer].dwFlags & WHDR_DONE))
|
|
Packit |
64f477 |
Sleep (1000 * 256 / instance->sample_rate);
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
result = waveOutUnprepareHeader (instance->h_waveout,
|
|
Packit |
64f477 |
&instance->waveheader[current_buffer],
|
|
Packit |
64f477 |
sizeof(WAVEHDR));
|
|
Packit |
64f477 |
if (result != MMSYSERR_NOERROR) {
|
|
Packit |
64f477 |
fprintf (stderr, "waveOutUnprepareHeader failed\n");
|
|
Packit |
64f477 |
return 1;
|
|
Packit |
64f477 |
}
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
instance->waveheader[current_buffer].dwFlags = 0;
|
|
Packit |
64f477 |
result = waveOutPrepareHeader (instance->h_waveout,
|
|
Packit |
64f477 |
&instance->waveheader[current_buffer],
|
|
Packit |
64f477 |
sizeof(WAVEHDR));
|
|
Packit |
64f477 |
if (result != MMSYSERR_NOERROR) {
|
|
Packit |
64f477 |
fprintf (stderr, "waveOutPrepareHeader failed\n");
|
|
Packit |
64f477 |
return 1;
|
|
Packit |
64f477 |
}
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
float2s16_2 (samples, instance->int16_samples[current_buffer]);
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
result = waveOutWrite (instance->h_waveout,
|
|
Packit |
64f477 |
&instance->waveheader[current_buffer],
|
|
Packit |
64f477 |
sizeof(WAVEHDR));
|
|
Packit |
64f477 |
if (result != MMSYSERR_NOERROR) {
|
|
Packit |
64f477 |
fprintf (stderr, "waveOutWrite failed\n");
|
|
Packit |
64f477 |
return 1;
|
|
Packit |
64f477 |
}
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
return 0;
|
|
Packit |
64f477 |
}
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
static void win_close (ao_instance_t * _instance)
|
|
Packit |
64f477 |
{
|
|
Packit |
64f477 |
win_instance_t * instance = (win_instance_t *) _instance;
|
|
Packit |
64f477 |
int i;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
waveOutReset (instance->h_waveout);
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
for (i = 0; i < NUMBUF; i++)
|
|
Packit |
64f477 |
waveOutUnprepareHeader (instance->h_waveout,
|
|
Packit |
64f477 |
&instance->waveheader[i],
|
|
Packit |
64f477 |
sizeof(WAVEHDR));
|
|
Packit |
64f477 |
waveOutClose (instance->h_waveout);
|
|
Packit |
64f477 |
}
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
static ao_instance_t * win_open (int flags)
|
|
Packit |
64f477 |
{
|
|
Packit |
64f477 |
win_instance_t * instance;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
instance = malloc (sizeof (win_instance_t));
|
|
Packit |
64f477 |
if (instance == NULL)
|
|
Packit |
64f477 |
return NULL;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
instance->ao.setup = win_setup;
|
|
Packit |
64f477 |
instance->ao.play = win_play;
|
|
Packit |
64f477 |
instance->ao.close = win_close;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
instance->sample_rate = 0;
|
|
Packit |
64f477 |
instance->set_params = 1;
|
|
Packit |
64f477 |
instance->flags = flags;
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
return (ao_instance_t *) instance;
|
|
Packit |
64f477 |
}
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
ao_instance_t * ao_win_open (void)
|
|
Packit |
64f477 |
{
|
|
Packit |
64f477 |
return win_open (A52_STEREO);
|
|
Packit |
64f477 |
}
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
ao_instance_t * ao_windolby_open (void)
|
|
Packit |
64f477 |
{
|
|
Packit |
64f477 |
return win_open (A52_DOLBY);
|
|
Packit |
64f477 |
}
|
|
Packit |
64f477 |
|
|
Packit |
64f477 |
#endif
|