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

📄 decoder.c

📁 完成MP3播放功能
💻 C
字号:
/* * libmad - MPEG audio decoder library * Copyright (C) 2000-2004 Underbit Technologies, Inc. * * 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: decoder.c,v 1.22 2004/01/23 09:41:32 rob Exp $ */# ifdef HAVE_CONFIG_H#  include "config.h"# endif# include "global.h"# ifdef HAVE_SYS_TYPES_H#  include <sys/types.h># endif# ifdef HAVE_SYS_WAIT_H#  include <sys/wait.h># endif# ifdef HAVE_UNISTD_H#  include <unistd.h># endif# ifdef HAVE_FCNTL_H#  include <fcntl.h># endif# include <stdlib.h># ifdef HAVE_ERRNO_H#  include <errno.h># endif# include "stream.h"# include "frame.h"# include "synth.h"# include "decoder.h"/* NAME:	decoder->init() DESCRIPTION:	initialize a decoder object with callback routines */void mad_decoder_init(struct mad_decoder *decoder, void *data,		      enum mad_flow (*input_func)(void *,struct mad_stream *),		      enum mad_flow (*header_func)(void *, struct mad_header const *),		      enum mad_flow (*filter_func)(void *, struct mad_stream const *,struct mad_frame *),		      enum mad_flow (*output_func)(void *, struct mad_header const *,struct mad_pcm *),		      enum mad_flow (*error_func)(void *,struct mad_stream *, struct mad_frame *),		      enum mad_flow (*message_func)(void *,void *, unsigned int *)){ decoder->mode         = -1;  decoder->options      = 0;  decoder->async.pid    = 0;  decoder->async.in     = -1;  decoder->async.out    = -1;  decoder->sync         = 0;  decoder->cb_data      = data;		///////////////player--->decoder->cb_data  decoder->input_func   = input_func;  decoder->header_func  = header_func;  decoder->filter_func  = filter_func;  decoder->output_func  = output_func;  decoder->error_func   = error_func;  decoder->message_func = message_func;}int mad_decoder_finish(struct mad_decoder *decoder){# if defined(USE_ASYNC)  if (decoder->mode == MAD_DECODER_MODE_ASYNC && decoder->async.pid) {    pid_t pid;    int status;    close(decoder->async.in);    do      pid = waitpid(decoder->async.pid, &status, 0);    while (pid == -1 && errno == EINTR);    decoder->mode = -1;    close(decoder->async.out);    decoder->async.pid = 0;    decoder->async.in  = -1;    decoder->async.out = -1;    if (pid == -1)      return -1;    return (!WIFEXITED(status) || WEXITSTATUS(status)) ? -1 : 0;  }# endif  return 0;}# if defined(USE_ASYNC)staticenum mad_flow send_io(int fd, void const *data, size_t len){  char const *ptr = data;  ssize_t count;  while (len) {    do      count = write(fd, ptr, len);    while (count == -1 && errno == EINTR);    if (count == -1)      return MAD_FLOW_BREAK;    len -= count;    ptr += count;  }  return MAD_FLOW_CONTINUE;}staticenum mad_flow receive_io(int fd, void *buffer, size_t len){  char *ptr = buffer;  ssize_t count;  while (len) {    do      count = read(fd, ptr, len);    while (count == -1 && errno == EINTR);    if (count == -1)      return (errno == EAGAIN) ? MAD_FLOW_IGNORE : MAD_FLOW_BREAK;    else if (count == 0)      return MAD_FLOW_STOP;    len -= count;    ptr += count;  }  return MAD_FLOW_CONTINUE;}staticenum mad_flow receive_io_blocking(int fd, void *buffer, size_t len){  int flags, blocking;  enum mad_flow result;  flags = fcntl(fd, F_GETFL);  if (flags == -1)    return MAD_FLOW_BREAK;  blocking = flags & ~O_NONBLOCK;  if (blocking != flags &&      fcntl(fd, F_SETFL, blocking) == -1)    return MAD_FLOW_BREAK;  result = receive_io(fd, buffer, len);  if (flags != blocking &&      fcntl(fd, F_SETFL, flags) == -1)    return MAD_FLOW_BREAK;  return result;}staticenum mad_flow send(int fd, void const *message, unsigned int size){  enum mad_flow result;  /* send size */  result = send_io(fd, &size, sizeof(size));  /* send message */  if (result == MAD_FLOW_CONTINUE)    result = send_io(fd, message, size);  return result;}staticenum mad_flow receive(int fd, void **message, unsigned int *size){  enum mad_flow result;  unsigned int actual;  if (*message == 0)    *size = 0;  /* receive size */  result = receive_io(fd, &actual, sizeof(actual));  /* receive message */  if (result == MAD_FLOW_CONTINUE) {    if (actual > *size)      actual -= *size;    else {      *size  = actual;      actual = 0;    }    if (*size > 0) {      if (*message == 0) {	*message = malloc(*size);	if (*message == 0)	  return MAD_FLOW_BREAK;      }      result = receive_io_blocking(fd, *message, *size);    }    /* throw away remainder of message */    while (actual && result == MAD_FLOW_CONTINUE) {      char sink[256];      unsigned int len;      len = actual > sizeof(sink) ? sizeof(sink) : actual;      result = receive_io_blocking(fd, sink, len);      actual -= len;    }  }  return result;}staticenum mad_flow check_message(struct mad_decoder *decoder){  enum mad_flow result;  void *message = 0;  unsigned int size;  result = receive(decoder->async.in, &message, &size);  if (result == MAD_FLOW_CONTINUE) {    if (decoder->message_func == 0)      size = 0;    else {      result = decoder->message_func(decoder->cb_data, message, &size);      if (result == MAD_FLOW_IGNORE ||	  result == MAD_FLOW_BREAK)	size = 0;    }    if (send(decoder->async.out, message, size) != MAD_FLOW_CONTINUE)      result = MAD_FLOW_BREAK;  }  if (message)    free(message);  return result;}# endifstaticenum mad_flow error_default(void *data, struct mad_stream *stream,			    struct mad_frame *frame){  int *bad_last_frame = data;  switch (stream->error) {  case MAD_ERROR_BADCRC:    if (*bad_last_frame)      mad_frame_mute(frame);    else      *bad_last_frame = 1;    return MAD_FLOW_IGNORE;  default:    return MAD_FLOW_CONTINUE;  }}int num=1;			//出错帧计数//被mad_decoder_run()调用static int run_sync(struct mad_decoder *decoder)//解码主体{ static framee=0;  enum mad_flow (*error_func)(void *, struct mad_stream *, struct mad_frame *);  void *error_data;  int bad_last_frame = 0;  struct mad_stream *stream;  struct mad_frame *frame;  struct mad_synth *synth;  int result = 0;  if (decoder->input_func == 0)    return 0;  if (decoder->error_func)   { error_func = decoder->error_func;    error_data = decoder->cb_data;  }  else   { error_func = error_default;    error_data = &bad_last_frame;  }  stream = &decoder->sync->stream;  frame  = &decoder->sync->frame;  synth  = &decoder->sync->synth;  mad_stream_init(stream);  mad_frame_init(frame);  mad_synth_init(synth);  mad_stream_options(stream, decoder->options);  do //一直到文件结束  { switch (decoder->input_func(decoder->cb_data, stream))  //读40000字节到input->fd    {  	case MAD_FLOW_STOP:     			goto done;    		case MAD_FLOW_BREAK:      		goto fail;    		case MAD_FLOW_IGNORE:      		continue;    		case MAD_FLOW_CONTINUE:    		break;    }    while (1) 	//一直到结束或出错    {      # if defined(USE_ASYNC)      if (decoder->mode == MAD_DECODER_MODE_ASYNC)       {	switch (check_message(decoder))       	{	case MAD_FLOW_IGNORE:      		case MAD_FLOW_CONTINUE:			break;					case MAD_FLOW_BREAK:				goto fail;					case MAD_FLOW_STOP:					goto done;				}      }			# endif	  printf("%d%c",framee++,0xd);			//自己加的      if (decoder->header_func)       {	if (mad_header_decode(&frame->header, stream) == -1) 		//解码帧头,第一次调用mad_bit_read()    	  {	  if (!MAD_RECOVERABLE(stream->error))  			break;				    switch (error_func(error_data, stream, frame)) 				    {	  case MAD_FLOW_STOP:	    					goto done;	  				  	case MAD_FLOW_BREAK:    					goto fail;	  					  case MAD_FLOW_IGNORE:	  					  case MAD_FLOW_CONTINUE:	 						  default:						    					continue;	  			  }			  }			  switch (decoder->header_func(decoder->cb_data, &frame->header)) 			  {		case MAD_FLOW_STOP:	  				goto done;				  	case MAD_FLOW_BREAK:  				goto fail;					  case MAD_FLOW_IGNORE:  				continue;					  case MAD_FLOW_CONTINUE:				break;			  }      }      if (mad_frame_decode(frame, stream) == -1) 						//解码数据      { //printf("\nThis is the %d frame\n",num++);					//显示出错的帧数  			if (!MAD_RECOVERABLE(stream->error))		break;	  		switch (error_func(error_data, stream, frame)) 		  	{	case MAD_FLOW_STOP:	  			goto done;			  	case MAD_FLOW_BREAK:  			goto fail;				  case MAD_FLOW_IGNORE:  			break;				  case MAD_FLOW_CONTINUE:							  default:				  	  			 continue;			  }      }      else		bad_last_frame = 0;      if (decoder->filter_func)       {	switch (decoder->filter_func(decoder->cb_data, stream, frame))     	  {	case MAD_FLOW_STOP:	  			goto done;				  case MAD_FLOW_BREAK:  			goto fail;				  case MAD_FLOW_IGNORE:  			continue;				  case MAD_FLOW_CONTINUE:			break;			  }      }      mad_synth_frame(synth, frame);      if (decoder->output_func)       {	switch (decoder->output_func(decoder->cb_data,&frame->header, &synth->pcm))       	{	case MAD_FLOW_STOP:	  			goto done;	  			case MAD_FLOW_BREAK:  			goto fail;		  		case MAD_FLOW_IGNORE:			  	case MAD_FLOW_CONTINUE:			break;			  }      }    }//end of while(1)	}  while (stream->error == MAD_ERROR_BUFLEN); fail:  result = -1; done:  mad_synth_finish(synth);			  mad_frame_finish(frame);  			mad_stream_finish(stream); return result;}# if defined(USE_ASYNC)staticint run_async(struct mad_decoder *decoder){  pid_t pid;  int ptoc[2], ctop[2], flags;  if (pipe(ptoc) == -1)    return -1;  if (pipe(ctop) == -1) {    close(ptoc[0]);    close(ptoc[1]);    return -1;  }  flags = fcntl(ptoc[0], F_GETFL);  if (flags == -1 ||      fcntl(ptoc[0], F_SETFL, flags | O_NONBLOCK) == -1) {    close(ctop[0]);    close(ctop[1]);    close(ptoc[0]);    close(ptoc[1]);    return -1;  }  pid = fork();  if (pid == -1) {    close(ctop[0]);    close(ctop[1]);    close(ptoc[0]);    close(ptoc[1]);    return -1;  }  decoder->async.pid = pid;  if (pid) {    /* parent */    close(ptoc[0]);    close(ctop[1]);    decoder->async.in  = ctop[0];    decoder->async.out = ptoc[1];    return 0;  }  /* child */  close(ptoc[1]);  close(ctop[0]);  decoder->async.in  = ptoc[0];  decoder->async.out = ctop[1];  _exit(run_sync(decoder));  /* not reached */  return -1;}# endif/* NAME:	decoder->run() DESCRIPTION:	run the decoder thread either synchronously or asynchronously */ //被player.c中的decode()调用int mad_decoder_run(struct mad_decoder *decoder, enum mad_decoder_mode mode){ int result;  int (*run)(struct mad_decoder *) = 0;  switch (decoder->mode = mode)   { case MAD_DECODER_MODE_SYNC:   			run = run_sync;    		break;  	case MAD_DECODER_MODE_ASYNC:				# if defined(USE_ASYNC)    		run = run_async;				# endif    		break;  }  if (run == 0)    return -1;  decoder->sync = malloc(sizeof(*decoder->sync));  if (decoder->sync == 0)    return -1;  result = run(decoder);  free(decoder->sync);  decoder->sync = 0;  return result;}/* * NAME:	decoder->message() * DESCRIPTION:	send a message to and receive a reply from the decoder process */int mad_decoder_message(struct mad_decoder *decoder,			void *message, unsigned int *len){# if defined(USE_ASYNC)  if (decoder->mode != MAD_DECODER_MODE_ASYNC ||      send(decoder->async.out, message, *len) != MAD_FLOW_CONTINUE ||      receive(decoder->async.in, &message, len) != MAD_FLOW_CONTINUE)    return -1;  return 0;# else  return -1;# endif}

⌨️ 快捷键说明

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