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

📄 audio.c

📁 xmms-1.2.10.tar.gz学习使用的就下吧
💻 C
字号:
/* *  Copyright (C) 2001  CubeSoft Communications, Inc. *  <http://www.csoft.org> * *  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. *//* FIXME: g_error() is used several places, it will exit xmms */#include <errno.h>#include "libxmms/util.h"#include "sun.h"#include "resample.h"static int sun_bps(int, int, int);static int sun_format(AFormat);static void sun_setformat(AFormat, int, int);static void sun_setparams(void);static void *sun_loop(void *);static int sun_downsample(gpointer, guint, guint, guint);static gboolean	prebuffer, remove_prebuffer;static pthread_t buffer_thread;static int (*sun_convert)(void **, int);static int realtime;static int rd_index, wr_index;static int buffer_size;static int prebuffer_size;static int output_time_offset;static int device_buffer_used;static int blocksize;static char *buffer;static guint64 output_bytes;static guint64 written;/* * The format of the data from the input plugin * This will never change during a song.  */struct sun_format input;/* * The format we get from the effect plugin. * This will be different from input if the effect plugin does * some kind of format conversion. */struct sun_format effect;/* * The format of the data we actually send to the soundcard. * This might be different from effect if we need to resample or do * some other format conversion. */struct sun_format output;static int sun_bps(int sunfmt, int rate, int nch){	int bitrate;	bitrate = rate * nch;	switch (sunfmt)	{		case AUDIO_ENCODING_ULINEAR_BE:		case AUDIO_ENCODING_ULINEAR_LE:		case AUDIO_ENCODING_SLINEAR_BE:		case AUDIO_ENCODING_SLINEAR_LE:			bitrate *= 2;			break;	}	return (bitrate);}static int sun_format(AFormat fmt){	switch (fmt)	{		case FMT_U8:			return (AUDIO_ENCODING_PCM8);		case FMT_S8:			return (AUDIO_ENCODING_SLINEAR);		case FMT_U16_LE:			return (AUDIO_ENCODING_ULINEAR_LE);		case FMT_U16_BE:			return (AUDIO_ENCODING_ULINEAR_BE);		case FMT_U16_NE:#ifdef WORDS_BIGENDIAN			return (AUDIO_ENCODING_ULINEAR_BE);#else			return (AUDIO_ENCODING_ULINEAR_LE);#endif		case FMT_S16_LE:			return (AUDIO_ENCODING_SLINEAR_LE);		case FMT_S16_BE:			return (AUDIO_ENCODING_SLINEAR_BE);		case FMT_S16_NE:#ifdef WORDS_BIGENDIAN			return (AUDIO_ENCODING_SLINEAR_BE);#else			return (AUDIO_ENCODING_SLINEAR_LE);#endif	}	return -1;}static void sun_setformat(AFormat fmt, int rate, int nch){	int sun;		sun = sun_format(fmt);	effect.format.sun = sun;	effect.format.xmms = fmt;	effect.frequency = rate;	effect.channels = nch;	effect.bps = sun_bps(sun, rate, nch);	output.format.sun = sun;	output.format.xmms = fmt;	output.frequency = rate;	output.channels = nch;	sun_setparams();	output.bps = sun_bps(output.format.sun, output.frequency,			     output.channels);	audio.input = &input;	audio.output = &output;	audio.effect = &effect;}void sun_setparams(void){	audio_info_t info;	audio_encoding_t enc;	AUDIO_INITINFO(&info);	info.mode = AUMODE_PLAY;	if (ioctl(audio.fd, AUDIO_SETINFO, &info) != 0)	{		g_error("%s: cannot play (%s)", audio.devaudio,			strerror(errno));		return;	}	/*	 * Pass 1: try the preferred encoding, if it is supported.	 */	enc.index = 0;	while (ioctl(audio.fd, AUDIO_GETENC, &enc) == 0 &&	       enc.encoding != output.format.sun)		enc.index++;	info.play.encoding = enc.encoding;	info.play.precision = enc.precision;	strcpy(output.name, enc.name);	if (ioctl(audio.fd, AUDIO_SETINFO, &info) != 0)	{		g_error("%s: unsupported encoding: %s (%s)", audio.devaudio,			output.name, strerror(errno));		return;	}	info.play.channels = output.channels;	ioctl(audio.fd, AUDIO_SETINFO, &info);	info.play.sample_rate = output.frequency;	if (ioctl(audio.fd, AUDIO_SETINFO, &info) < 0)	{		g_error("%s: cannot handle %i Hz (%s)", audio.devaudio,			output.frequency, strerror(errno));		return;	}	if (ioctl(audio.fd, AUDIO_GETINFO, &info) != 0)	{		blocksize = SUN_DEFAULT_BLOCKSIZE;		output.channels = info.play.channels;	}	else	{		blocksize = blocksize;	}	sun_convert = sun_get_convert_func(output.format.sun,					   sun_format(effect.format.xmms));#if 0	if (sun_convert != NULL)	{		g_warning("audio conversion (output=0x%x effect=0x%x)",		    output.format.sun, sun_format(effect.format.xmms));	}#endif}static inline void sun_bufused(void){	audio_offset_t ooffs;	if (audio.paused)		device_buffer_used = 0;	else if (ioctl(audio.fd, AUDIO_GETOOFFS, &ooffs) == 0)		device_buffer_used = ooffs.offset;}int sun_written_time(void){	if (!audio.going)		return 0;	return ((written * 1000) / effect.bps);}int sun_output_time(void){	guint64 bytes;	if (!audio.fd || !audio.going)		return 0;	if (realtime)		sun_bufused();	bytes = output_bytes < device_buffer_used ?		0 : output_bytes - device_buffer_used;	return (output_time_offset + ((bytes * 1000) / output.bps));}static inline int sun_used(void){	if (realtime)		return 0;		if (wr_index >= rd_index)		return (wr_index - rd_index);	return (buffer_size - (rd_index - wr_index));}int sun_playing(void){	if (!audio.going)		return 0;	if (realtime)		sun_bufused();	if (!sun_used() && (device_buffer_used - (3 * blocksize)) <= 0)		return (FALSE);	return (TRUE);}int sun_free(void){	if (realtime)		return (audio.paused ? 0 : 1000000);		if (remove_prebuffer && prebuffer)	{		prebuffer = FALSE;		remove_prebuffer = FALSE;	}	if (prebuffer)		remove_prebuffer = TRUE;	if (rd_index > wr_index)		return ((rd_index - wr_index) - blocksize - 1);	return ((buffer_size - (wr_index - rd_index)) - blocksize - 1);}static inline ssize_t write_all(int fd, const void *buf, size_t count){	static ssize_t done;	for (done = 0; count > done; )	{		static ssize_t n;		n = write(fd, buf, count - done);		if (n == -1)		{			if (errno == EINTR)				continue;			else				break;		}		done += n;	}	return (done);}static inline void sun_write_audio(gpointer data, int length){	AFormat new_format;	EffectPlugin *ep;	int new_frequency, new_channels;	new_format = input.format.xmms;	new_frequency = input.frequency;	new_channels = input.channels;	ep = get_current_effect_plugin();	if (effects_enabled() && ep && ep->query_format)		ep->query_format(&new_format, &new_frequency, &new_channels);	if (new_format != effect.format.xmms || 	    new_frequency != effect.frequency ||	    new_channels != effect.channels)	{		output_time_offset += (output_bytes * 1000) / output.bps;		output_bytes = 0;		close(audio.fd);		audio.fd = open(audio.devaudio, O_RDWR);		sun_setformat(new_format, new_frequency, new_channels);	}	if (effects_enabled() && ep && ep->mod_samples)	{		length = ep->mod_samples(&data, length, input.format.xmms,					 input.frequency, input.channels);	}	if (sun_convert != NULL)		length = sun_convert(&data, length);	if (effect.frequency == output.frequency)	{		output_bytes += write_all(audio.fd, data, length);	}	else	{		output_bytes += sun_downsample(data, length,		    effect.frequency, output.frequency);	}}static void sun_bswap16(guint16 *data, int len){	int i;	for (i = 0; i < len; i += 2, data++)		*data = GUINT16_SWAP_LE_BE(*data);}static int sun_downsample(gpointer ob, guint length, guint speed, guint espeed){	guint w = 0;	static gpointer nbuffer = NULL;	static int nbuffer_size = 0;	switch (output.format.sun) {		case AUDIO_ENCODING_SLINEAR_BE:		case AUDIO_ENCODING_SLINEAR_LE:			if (output.channels == 2)				RESAMPLE_STEREO(gint16);			else				RESAMPLE_MONO(gint16);			break;		case AUDIO_ENCODING_ULINEAR_BE:		case AUDIO_ENCODING_ULINEAR_LE:			if (output.channels == 2)				RESAMPLE_STEREO(guint16);			else				RESAMPLE_MONO(guint16);			break;		case AUDIO_ENCODING_SLINEAR:			if (output.channels == 2)				RESAMPLE_STEREO(gint8);			else				RESAMPLE_MONO(gint8);			break;		case AUDIO_ENCODING_ULINEAR:			if (output.channels == 2)				RESAMPLE_STEREO(guint8);			else				RESAMPLE_MONO(guint8);			break;	}	return (w);}void sun_write(gpointer ptr, int length){	int cnt, off = 0;	if (realtime)	{		if (audio.paused)			return;		sun_write_audio(ptr, length);		written += length;		return;	}	remove_prebuffer = FALSE;	written += length;	while (length > 0)	{		cnt = MIN(length, buffer_size - wr_index);		memcpy(buffer + wr_index, (char *) ptr + off, cnt);		wr_index = (wr_index + cnt) % buffer_size;		length -= cnt;		off += cnt;	}}void sun_close(void){	if (!audio.going)		return;	audio.going = 0;	if (realtime)	{		ioctl(audio.fd, AUDIO_FLUSH, NULL);		close(audio.fd);	}	else	{		pthread_join(buffer_thread, NULL);	}	sun_get_convert_buffer(0);	wr_index = 0;	rd_index = 0;}void sun_flush(int time){	ioctl(audio.fd, AUDIO_FLUSH, NULL);	output_time_offset = time;	written = (guint16)(time / 10) * (guint64)(input.bps / 100);	output_bytes = 0;}void sun_pause(short p){	if (!realtime)	{		if (p == TRUE)			audio.do_pause = TRUE;		else			audio.unpause = TRUE;	}	else		audio.paused = p;}static void* sun_loop(void *arg){	struct timeval tv;	int length, cnt;	fd_set set;	while (audio.going)	{		if (sun_used() > prebuffer_size)			prebuffer = FALSE;		if (sun_used() > 0 && !audio.paused && !prebuffer)		{			tv.tv_sec = 0;			tv.tv_usec = 10000;			FD_ZERO(&set);			FD_SET(audio.fd, &set);			if (select(audio.fd + 1, NULL, &set, NULL, &tv) > 0)			{				length = MIN(blocksize, sun_used());				while (length > 0)				{					cnt = MIN(length,					    buffer_size - rd_index);					sun_write_audio(					    buffer + rd_index, cnt);					rd_index = (rd_index + cnt) %					    buffer_size;					length -= cnt;				}			}		}		else			xmms_usleep(10000);		sun_bufused();		if (audio.do_pause && !audio.paused)		{			audio.do_pause = FALSE;			audio.paused = TRUE;			rd_index -= device_buffer_used;			output_bytes -= device_buffer_used;			if (rd_index < 0)				rd_index += buffer_size;			ioctl(audio.fd, AUDIO_FLUSH, NULL);		}		else if (audio.unpause && audio.paused)		{			audio.unpause = FALSE;			close(audio.fd);			audio.fd = open(audio.devaudio, O_RDWR);			sun_setparams();			audio.paused = FALSE;		}	}	close(audio.fd);	g_free(buffer);	pthread_exit(NULL);}int sun_open(AFormat fmt, int rate, int nch){	audio_info_t info;	AUDIO_INITINFO(&info);	if ((audio.fd = open(audio.devaudio, O_RDWR)) < 0)	{		g_error("%s: %s", audio.devaudio, strerror(errno));		return 0;	}	input.format.xmms = fmt;	input.frequency = rate;	input.channels = nch;	input.bps = sun_bps(sun_format(fmt), rate, nch);	sun_setformat(fmt, rate, nch);	realtime = xmms_check_realtime_priority();	if (ioctl(audio.fd, AUDIO_GETINFO, &info) != 0)		blocksize = SUN_DEFAULT_BLOCKSIZE;	else		blocksize = info.blocksize;	if (!realtime)	{		buffer_size = audio.req_buffer_size;		if (buffer_size < SUN_MIN_BUFFER_SIZE)			buffer_size = SUN_MIN_BUFFER_SIZE;		prebuffer_size = (buffer_size * audio.req_prebuffer_size) / 100;		buffer_size += blocksize;		buffer = g_malloc0(buffer_size);	}	prebuffer = TRUE;	wr_index = 0;	rd_index = 0;	output_time_offset = 0;	written = 0;	output_bytes = 0;	audio.paused = FALSE;	audio.do_pause = FALSE;	audio.unpause = FALSE;	remove_prebuffer = FALSE;	audio.going++;	if (!realtime)		pthread_create(&buffer_thread, NULL, sun_loop, NULL);	return 1;}

⌨️ 快捷键说明

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