⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 audio_win32.c

📁 linux下MPEG播放器
💻 C
字号:
/* * madplay - MPEG audio decoder and player * Copyright (C) 2000-2004 Robert Leslie * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA * * $Id: audio_win32.c,v 1.57 2004/01/23 09:41:31 rob Exp $ */# ifdef HAVE_CONFIG_H#  include "config.h"# endif# include "global.h"# include <windows.h># include <mmsystem.h># include <mad.h># include "gettext.h"# include "audio.h"static int opened;static HWAVEOUT wave_handle;static audio_pcmfunc_t *audio_pcm;static unsigned int samplerate, samplesize;# define NBUFFERS  2static struct buffer {  WAVEHDR wave_header;  HANDLE event_handle;  int playing;  unsigned int pcm_length;  unsigned char pcm_data[48000 * 4 * 2];} output[NBUFFERS];static int bindex;staticchar const *error_text(MMRESULT result){  static char buffer[MAXERRORLENGTH];  if (waveOutGetErrorText(result, buffer, sizeof(buffer)) != MMSYSERR_NOERROR)    return _("error getting error text");  return buffer;}staticint init(struct audio_init *init){  int i;  opened = 0;  for (i = 0; i < NBUFFERS; ++i) {    output[i].event_handle = CreateEvent(0, FALSE /* manual reset */,					 FALSE /* initial state */, 0);    if (output[i].event_handle == NULL) {      while (i--)	CloseHandle(output[i].event_handle);      audio_error = _("failed to create synchronization object");      return -1;    }    output[i].playing    = 0;    output[i].pcm_length = 0;  }  bindex = 0;  /* try to obtain high priority status */  SetPriorityClass(GetCurrentProcess(), HIGH_PRIORITY_CLASS);  SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_HIGHEST);  return 0;}staticvoid CALLBACK callback(HWAVEOUT handle, UINT message, DWORD data,		       DWORD param1, DWORD param2){  WAVEHDR *header;  struct buffer *buffer;  switch (message) {  case WOM_DONE:    header = (WAVEHDR *) param1;    buffer = (struct buffer *) header->dwUser;    if (SetEvent(buffer->event_handle) == 0) {      /* error */    }    break;  case WOM_OPEN:  case WOM_CLOSE:    break;  }}staticvoid set_format(WAVEFORMATEX *format, unsigned int channels,		unsigned int speed, unsigned int bits){  unsigned int block_al;  unsigned long bytes_ps;  block_al = channels * bits / 8;  bytes_ps = speed * block_al;  format->wFormatTag      = WAVE_FORMAT_PCM;  format->nChannels       = channels;  format->nSamplesPerSec  = speed;  format->nAvgBytesPerSec = bytes_ps;  format->nBlockAlign     = block_al;  format->wBitsPerSample  = bits;  format->cbSize          = 0;}staticint open_dev(HWAVEOUT *handle, unsigned int channels, unsigned int speed,	     unsigned int depth){  WAVEFORMATEX format;  MMRESULT error;  set_format(&format, channels, speed, depth);  error = waveOutOpen(handle, WAVE_MAPPER, &format,		      (DWORD) callback, 0, CALLBACK_FUNCTION);  if (error != MMSYSERR_NOERROR) {    audio_error = error_text(error);    return -1;  }  opened = 1;  return 0;}staticint close_dev(HWAVEOUT handle){  MMRESULT error;  opened = 0;  error = waveOutClose(handle);  if (error != MMSYSERR_NOERROR) {    audio_error = error_text(error);    return -1;  }  return 0;}staticint set_pause(int flag){  static int paused;  MMRESULT error;  if (flag && !paused) {    error = waveOutPause(wave_handle);    if (error != MMSYSERR_NOERROR) {      audio_error = error_text(error);      return -1;    }  }  else if (!flag && paused) {    error = waveOutRestart(wave_handle);    if (error != MMSYSERR_NOERROR) {      audio_error = error_text(error);      return -1;    }  }  paused = flag;  return 0;}staticint write_dev(HWAVEOUT handle, struct buffer *buffer){  MMRESULT error;  buffer->wave_header.lpData         = buffer->pcm_data;  buffer->wave_header.dwBufferLength = buffer->pcm_length;  buffer->wave_header.dwUser         = (DWORD) buffer;  buffer->wave_header.dwFlags        = 0;  error = waveOutPrepareHeader(handle, &buffer->wave_header,			       sizeof(buffer->wave_header));  if (error != MMSYSERR_NOERROR) {    audio_error = error_text(error);    return -1;  }  error = waveOutWrite(handle, &buffer->wave_header,		       sizeof(buffer->wave_header));  if (error != MMSYSERR_NOERROR) {    audio_error = error_text(error);    return -1;  }  buffer->playing = 1;  return 0;}staticint wait(struct buffer *buffer){  MMRESULT error;  if (!buffer->playing)    return 0;  switch (WaitForSingleObject(buffer->event_handle, INFINITE)) {  case WAIT_OBJECT_0:    break;  case WAIT_ABANDONED:    audio_error = _("wait abandoned");    return -1;  case WAIT_TIMEOUT:    audio_error = _("wait timeout");    return -1;  case WAIT_FAILED:  default:    audio_error = _("wait failed");    return -1;  }  buffer->playing = 0;  error = waveOutUnprepareHeader(wave_handle, &buffer->wave_header,				 sizeof(buffer->wave_header));  if (error != MMSYSERR_NOERROR) {    audio_error = error_text(error);    return -1;  }  return 0;}staticint drain(void){  int i, result = 0;  if (set_pause(0) == -1)    result = -1;  if (output[bindex].pcm_length &&      write_dev(wave_handle, &output[bindex]) == -1)    result = -1;  for (i = 0; i < NBUFFERS; ++i) {    if (wait(&output[i]) == -1)      result = -1;  }  output[bindex].pcm_length = 0;  return result;}staticint config(struct audio_config *config){  unsigned int bitdepth;  bitdepth = config->precision & ~7;  if (bitdepth == 0)    bitdepth = 16;  else if (bitdepth > 32)    bitdepth = 32;  if (opened) {    if (drain() == -1)      return -1;    close_dev(wave_handle);  }  if (open_dev(&wave_handle, config->channels, config->speed, bitdepth) == -1)    return -1;  switch (config->precision = bitdepth) {  case 8:    audio_pcm = audio_pcm_u8;    break;  case 16:    audio_pcm = audio_pcm_s16le;    break;  case 24:    audio_pcm = audio_pcm_s24le;    break;  case 32:    audio_pcm = audio_pcm_s32le;    break;  }  samplerate = config->speed;  samplesize = bitdepth / 8;  return 0;}staticint play(struct audio_play *play){  struct buffer *buffer;  unsigned int len;  if (set_pause(0) == -1)    return -1;  buffer = &output[bindex];  /* wait for block to finish playing */  if (buffer->pcm_length == 0 && wait(buffer) == -1)    return -1;  /* prepare block */  len = audio_pcm(&buffer->pcm_data[buffer->pcm_length], play->nsamples,		  play->samples[0], play->samples[1], play->mode, play->stats);  buffer->pcm_length += len;  if (buffer->pcm_length + MAX_NSAMPLES * samplesize * 2 >      samplerate * samplesize * 2) {    write_dev(wave_handle, buffer);    bindex = (bindex + 1) % NBUFFERS;    output[bindex].pcm_length = 0;  }  return 0;}staticint stop(struct audio_stop *stop){  int result = 0;  MMRESULT error;  if (stop->flush) {    error = waveOutReset(wave_handle);    if (error != MMSYSERR_NOERROR) {      audio_error = error_text(error);      result = -1;    }    output[bindex].pcm_length = 0;    return result;  }  return set_pause(1);}staticint finish(struct audio_finish *finish){  int i, result = 0;  if (opened) {    if (drain() == -1)      result = -1;    if (close_dev(wave_handle) == -1)      result = -1;  }  /* restore priority status */  SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_NORMAL);  SetPriorityClass(GetCurrentProcess(), NORMAL_PRIORITY_CLASS);  for (i = 0; i < NBUFFERS; ++i) {    if (CloseHandle(output[i].event_handle) == 0 && result == 0) {      audio_error = _("failed to close synchronization object");      result = -1;    }  }  return result;}int audio_win32(union audio_control *control){  audio_error = 0;  switch (control->command) {  case AUDIO_COMMAND_INIT:    return init(&control->init);  case AUDIO_COMMAND_CONFIG:    return config(&control->config);  case AUDIO_COMMAND_PLAY:    return play(&control->play);  case AUDIO_COMMAND_STOP:    return stop(&control->stop);  case AUDIO_COMMAND_FINISH:    return finish(&control->finish);  }  return 0;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -