Blame libao/audio_out_win.c

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