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

📄 audio_carbon.c

📁 友善之臂SBC-2410X开发板所使用的测试程序包
💻 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_carbon.c,v 1.4 2004/01/23 09:41:31 rob Exp $ */# ifdef HAVE_CONFIG_H#  include "config.h"# endif# include "global.h"# include <Carbon/Carbon.h># include <mad.h># include "gettext.h"# include "audio.h"static SndChannelPtr channel;static audio_pcmfunc_t *audio_pcm;static unsigned int config_channels;static unsigned int config_speed;static unsigned int config_precision;# define NBUFFERS	16# define NQUEUESAMPLES	1152static struct buffer {  ExtSoundHeader header;  MPSemaphoreID semaphore;  unsigned int pcm_nsamples;  unsigned int pcm_length;  unsigned char pcm_data[MAX_NSAMPLES * 2 * 2];} output[NBUFFERS];static int bindex;enum mode {  QUEUE,  IMMEDIATE};staticint soundcmd(enum mode mode, unsigned short cmd, short param1, long param2){  SndCommand command;  command.cmd    = cmd;  command.param1 = param1;  command.param2 = param2;  switch (mode) {  case IMMEDIATE:    if (SndDoImmediate(channel, &command) != noErr) {      audio_error = _("SndDoImmediate() failed");      return -1;    }    break;  case QUEUE:    if (SndDoCommand(channel, &command, FALSE) != noErr) {      audio_error = _("SndDoCommand() failed");      return -1;    }    break;  }  return 0;}staticvoid callback(SndChannelPtr channel, SndCommand *command){  struct buffer *buffer = (struct buffer *) command->param2;  MPSignalSemaphore(buffer->semaphore);}staticint wait(struct buffer *buffer){  if (MPWaitOnSemaphore(buffer->semaphore, kDurationForever) != noErr) {    audio_error = _("MPWaitOnSemaphore() failed");    return -1;  }  return 0;}staticint init(struct audio_init *init){  int i;  channel = 0;  if (SndNewChannel(&channel, sampledSynth, 0, callback) != noErr) {    audio_error = _("SndNewChannel() failed");    return -1;  }  for (i = 0; i < NBUFFERS; ++i) {    if (MPCreateBinarySemaphore(&output[i].semaphore) != noErr) {      while (i--)	MPDeleteSemaphore(output[i].semaphore);      audio_error = _("failed to create synchronization object");      return -1;    }    output[i].pcm_nsamples = 0;    output[i].pcm_length   = 0;  }  bindex = 0;  return 0;}staticint set_pause(int flag){  static int paused;  if (flag != paused) {    paused = 0;    if (flag) {      if (soundcmd(IMMEDIATE, pauseCmd, 0, 0) == -1 ||	  soundcmd(IMMEDIATE, quietCmd, 0, 0) == -1)	return -1;    }    else if (soundcmd(IMMEDIATE, resumeCmd, 0, 0) == -1)      return -1;    paused = flag;  }  return 0;}staticvoid init_header(struct ExtSoundHeader *header, Ptr samples,		 unsigned int nsamples, unsigned int channels,		 unsigned int speed, unsigned int bits){  double dspeed = speed;  header->samplePtr        = samples;  header->numChannels      = channels;  header->sampleRate       = FixRatio(speed, 1);  header->loopStart        = 0;  header->loopEnd          = 0;  header->encode           = extSH;  header->baseFrequency    = kMiddleC;  header->numFrames        = nsamples;  dtox80(&dspeed, &header->AIFFSampleRate);  header->markerChunk      = 0;  header->instrumentChunks = 0;  header->AESRecording     = 0;  header->sampleSize       = bits;  header->futureUse1       = 0;  header->futureUse2       = 0;  header->futureUse3       = 0;  header->futureUse4       = 0;}staticint queue(struct buffer *buffer){  init_header(&buffer->header, buffer->pcm_data, buffer->pcm_nsamples,	      config_channels, config_speed, config_precision);  if (soundcmd(QUEUE, bufferCmd,   0, (long) &buffer->header) == -1 ||      soundcmd(QUEUE, callBackCmd, 0, (long) buffer) == -1)    return -1;  return 0;}staticint drain(void){  int result = 0;  if (output[bindex].pcm_nsamples) {    if (queue(&output[bindex]) == -1)      result = -1;    bindex = (bindex + 1) % NBUFFERS;    output[bindex].pcm_nsamples = 0;  }  return result;}staticint config(struct audio_config *config){  unsigned int bitdepth;  if (set_pause(0) == -1)    return -1;  bitdepth = config->precision & ~7;  if (bitdepth == 0 || bitdepth > 16)    bitdepth = 16;  if (drain() == -1 ||      soundcmd(QUEUE, reInitCmd, 0, initNoDrop |	       (config->channels == 1 ? initMono : initStereo)) == -1)    return -1;  switch (config->precision = bitdepth) {  case 8:    audio_pcm = audio_pcm_u8;    break;  case 16:    audio_pcm = audio_pcm_s16be;    break;  }  config_channels  = config->channels;  config_speed     = config->speed;  config_precision = config->precision;  return 0;}staticint play(struct audio_play *play){  struct buffer *buffer;  unsigned int len;  if (set_pause(0) == -1)    return -1;  if (output[bindex].pcm_nsamples + play->nsamples > MAX_NSAMPLES &&      drain() == -1)    return -1;  buffer = &output[bindex];  /* wait for block to finish playing */  if (buffer->pcm_nsamples == 0) {    if (wait(buffer) == -1)      return -1;    buffer->pcm_length = 0;  }  /* 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_nsamples += play->nsamples;  buffer->pcm_length   += len;  if (buffer->pcm_nsamples >= NQUEUESAMPLES &&      drain() == -1)    return -1;  return 0;}staticint flush(void){  int i, result = 0;  if (soundcmd(IMMEDIATE, flushCmd, 0, 0) == -1)    result = -1;  for (i = 0; i < NBUFFERS; ++i)    MPSignalSemaphore(output[i].semaphore);  output[bindex].pcm_nsamples = 0;  return result;}staticint stop(struct audio_stop *stop){  int result;  result = set_pause(1);  if (result == 0 && stop->flush && flush() == -1)    result = -1;  return result;}staticint finish(struct audio_finish *finish){  int i, result = 0;  if (set_pause(0) == -1 || drain() == -1)    result = -1;  if (SndDisposeChannel(channel, FALSE) != noErr && result == 0) {    audio_error = _("SndDisposeChannel() failed");    result = -1;  }  for (i = 0; i < NBUFFERS; ++i) {    if (MPDeleteSemaphore(output[i].semaphore) != noErr && result == 0) {      audio_error = _("failed to delete synchronization object");      result = -1;    }  }  return result;}int audio_carbon(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 + -