亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? pcm_bluetooth.c

?? 基于LINUX內核驅動的開發
?? C
?? 第 1 頁 / 共 3 頁
字號:
/* * *  BlueZ - Bluetooth protocol stack for Linux * *  Copyright (C) 2006-2007  Nokia Corporation *  Copyright (C) 2004-2008  Marcel Holtmann <marcel@holtmann.org> * * *  This library is free software; you can redistribute it and/or *  modify it under the terms of the GNU Lesser General Public *  License as published by the Free Software Foundation; either *  version 2.1 of the License, or (at your option) any later version. * *  This library 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 *  Lesser General Public License for more details. * *  You should have received a copy of the GNU Lesser General Public *  License along with this library; if not, write to the Free Software *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA * */#ifdef HAVE_CONFIG_H#include <config.h>#endif#include <stdint.h>#include <sys/socket.h>#include <sys/un.h>#include <time.h>#include <sys/time.h>#include <pthread.h>#include <signal.h>#include <limits.h>#include <netinet/in.h>#include <alsa/asoundlib.h>#include <alsa/pcm_external.h>#include "ipc.h"#include "sbc.h"#include "rtp.h"//#define ENABLE_DEBUG#define UINT_SECS_MAX (UINT_MAX / 1000000 - 1)#define MIN_PERIOD_TIME 1#define BUFFER_SIZE 2048#ifdef ENABLE_DEBUG#define DBG(fmt, arg...)  printf("DEBUG: %s: " fmt "\n" , __FUNCTION__ , ## arg)#else#define DBG(fmt, arg...)#endif#ifndef SOL_SCO#define SOL_SCO 17#endif#ifndef SCO_TXBUFS#define SCO_TXBUFS 0x03#endif#ifndef SCO_RXBUFS#define SCO_RXBUFS 0x04#endif#ifndef MIN# define MIN(x, y) ((x) < (y) ? (x) : (y))#endif#ifndef MAX# define MAX(x, y) ((x) > (y) ? (x) : (y))#endif#define MAX_BITPOOL 64#define MIN_BITPOOL 2/* adapted from glibc sys/time.h timersub() macro */#define priv_timespecsub(a, b, result)					\	do {								\		(result)->tv_sec = (a)->tv_sec - (b)->tv_sec;		\		(result)->tv_nsec = (a)->tv_nsec - (b)->tv_nsec;	\		if ((result)->tv_nsec < 0) {				\			--(result)->tv_sec;				\			(result)->tv_nsec += 1000000000;		\		}							\	} while (0)struct bluetooth_a2dp {	sbc_capabilities_t sbc_capabilities;	sbc_t sbc;				/* Codec data */	int sbc_initialized;			/* Keep track if the encoder is initialized */	int codesize;				/* SBC codesize */	int samples;				/* Number of encoded samples */	uint8_t buffer[BUFFER_SIZE];		/* Codec transfer buffer */	int count;				/* Codec transfer buffer counter */	int nsamples;				/* Cumulative number of codec samples */	uint16_t seq_num;			/* Cumulative packet sequence */	int frame_count;			/* Current frames in buffer*/};struct bluetooth_alsa_config {	char device[18];		/* Address of the remote Device */	int has_device;	uint8_t transport;		/* Requested transport */	int has_transport;	uint16_t rate;	int has_rate;	uint8_t channel_mode;		/* A2DP only */	int has_channel_mode;	uint8_t allocation_method;	/* A2DP only */	int has_allocation_method;	uint8_t subbands;		/* A2DP only */	int has_subbands;	uint8_t block_length;		/* A2DP only */	int has_block_length;	uint8_t bitpool;		/* A2DP only */	int has_bitpool;	int autoconnect;};struct bluetooth_data {	snd_pcm_ioplug_t io;	struct bluetooth_alsa_config alsa_config;	/* ALSA resource file parameters */	volatile snd_pcm_sframes_t hw_ptr;	int transport;					/* chosen transport SCO or AD2P */	int link_mtu;					/* MTU for selected transport channel */	volatile struct pollfd stream;			/* Audio stream filedescriptor */	struct pollfd server;				/* Audio daemon filedescriptor */	uint8_t buffer[BUFFER_SIZE];		/* Encoded transfer buffer */	int count;					/* Transfer buffer counter */	struct bluetooth_a2dp a2dp;			/* A2DP data */	pthread_t hw_thread;				/* Makes virtual hw pointer move */	int pipefd[2];					/* Inter thread communication */	int stopped;	sig_atomic_t reset;             /* Request XRUN handling */};static int audioservice_send(int sk, const bt_audio_msg_header_t *msg);static int audioservice_expect(int sk, bt_audio_msg_header_t *outmsg,				int expected_type);static int bluetooth_start(snd_pcm_ioplug_t *io){	DBG("bluetooth_start %p", io);	return 0;}static int bluetooth_stop(snd_pcm_ioplug_t *io){	DBG("bluetooth_stop %p", io);	return 0;}static void *playback_hw_thread(void *param){	struct bluetooth_data *data = param;	unsigned int prev_periods;	double period_time;	struct timespec start;	struct pollfd fds[2];	int poll_timeout;	data->server.events = POLLIN;	/* note: only errors for data->stream.events */	fds[0] = data->server;	fds[1] = data->stream;	prev_periods = 0;	period_time = 1000000.0 * data->io.period_size / data->io.rate;	if (period_time > (int) (MIN_PERIOD_TIME * 1000))		poll_timeout = (int) (period_time / 1000.0f);	else		poll_timeout = MIN_PERIOD_TIME;	clock_gettime(CLOCK_MONOTONIC, &start);	while (1) {		unsigned int dtime, periods;		struct timespec cur, delta;		int ret;		if (data->stopped)			goto iter_sleep;		if (data->reset) {			DBG("Handle XRUN in hw-thread.");			data->reset = 0;			clock_gettime(CLOCK_MONOTONIC, &start);			prev_periods = 0;		}		clock_gettime(CLOCK_MONOTONIC, &cur);		priv_timespecsub(&cur, &start, &delta);		dtime = delta.tv_sec * 1000000 + delta.tv_nsec / 1000;		periods = 1.0 * dtime / period_time;		if (periods > prev_periods) {			char c = 'w';			int frags = periods - prev_periods, n;			data->hw_ptr += frags *	data->io.period_size;			data->hw_ptr %= data->io.buffer_size;			for (n = 0; n < frags; n++) {				/* Notify user that hardware pointer				 * has moved * */				if (write(data->pipefd[1], &c, 1) < 0)					pthread_testcancel();			}			/* Reset point of reference to avoid too big values			 * that wont fit an unsigned int */			if (delta.tv_sec < UINT_SECS_MAX)				prev_periods = periods;			else {				prev_periods = 0;				clock_gettime(CLOCK_MONOTONIC, &start);			}		}iter_sleep:		/* sleep up to one period interval */		ret = poll(fds, 2, poll_timeout);		if (ret < 0) {			SNDERR("poll error: %s (%d)", strerror(errno), errno);			if (errno != EINTR)				break;		} else if (ret > 0) {			ret = (fds[0].revents) ? 0 : 1;			SNDERR("poll fd %d revents %d", ret, fds[ret].revents);			if (fds[ret].revents & (POLLERR | POLLHUP | POLLNVAL))				break;		}		/* Offer opportunity to be canceled by main thread */		pthread_testcancel();	}	data->hw_thread = 0;	pthread_exit(NULL);}static int bluetooth_playback_start(snd_pcm_ioplug_t *io){	struct bluetooth_data *data = io->private_data;	int err;	DBG("%p", io);	data->stopped = 0;	if (data->hw_thread)		return 0;	err = pthread_create(&data->hw_thread, 0, playback_hw_thread, data);	return -err;}static int bluetooth_playback_stop(snd_pcm_ioplug_t *io){	struct bluetooth_data *data = io->private_data;	DBG("%p", io);	data->stopped = 1;	return 0;}static snd_pcm_sframes_t bluetooth_pointer(snd_pcm_ioplug_t *io){	struct bluetooth_data *data = io->private_data;	return data->hw_ptr;}static void bluetooth_exit(struct bluetooth_data *data){	struct bluetooth_a2dp *a2dp = &data->a2dp;	if (data->server.fd >= 0)		bt_audio_service_close(data->server.fd);	if (data->stream.fd >= 0)		close(data->stream.fd);	if (data->hw_thread) {		pthread_cancel(data->hw_thread);		pthread_join(data->hw_thread, 0);	}	if (a2dp->sbc_initialized)		sbc_finish(&a2dp->sbc);	if (data->pipefd[0] > 0)		close(data->pipefd[0]);	if (data->pipefd[1] > 0)		close(data->pipefd[1]);	free(data);}static int bluetooth_close(snd_pcm_ioplug_t *io){	struct bluetooth_data *data = io->private_data;	DBG("%p", io);	bluetooth_exit(data);	return 0;}static int bluetooth_prepare(snd_pcm_ioplug_t *io){	struct bluetooth_data *data = io->private_data;	char c = 'w';	char buf[BT_AUDIO_IPC_PACKET_SIZE];	struct bt_streamstart_req *start_req = (void*) buf;	bt_audio_rsp_msg_header_t *rsp_hdr = (void*) buf;	struct bt_streamfd_ind *streamfd_ind = (void*) buf;	uint32_t period_count = io->buffer_size / io->period_size;	int opt_name, err;	struct timeval t = { 0, period_count };	DBG("Preparing with io->period_size=%lu io->buffer_size=%lu",					io->period_size, io->buffer_size);	data->reset = 0;	/* As we're gonna receive messages on the server socket, we have to stop the	   hw thread that is polling on it, if any */	if (data->hw_thread) {		pthread_cancel(data->hw_thread);		pthread_join(data->hw_thread, 0);		data->hw_thread = 0;	}	if (io->stream == SND_PCM_STREAM_PLAYBACK)		/* If not null for playback, xmms doesn't display time		 * correctly */		data->hw_ptr = 0;	else		/* ALSA library is really picky on the fact hw_ptr is not null.		 * If it is, capture won't start */		data->hw_ptr = io->period_size;	/* send start */	memset(start_req, 0, BT_AUDIO_IPC_PACKET_SIZE);	start_req->h.msg_type = BT_STREAMSTART_REQ;	err = audioservice_send(data->server.fd, &start_req->h);	if (err < 0)		return err;	err = audioservice_expect(data->server.fd, &rsp_hdr->msg_h,					BT_STREAMSTART_RSP);	if (err < 0)		return err;	if (rsp_hdr->posix_errno != 0) {		SNDERR("BT_START failed : %s(%d)",					strerror(rsp_hdr->posix_errno),					rsp_hdr->posix_errno);		return -rsp_hdr->posix_errno;	}	err = audioservice_expect(data->server.fd, &streamfd_ind->h,					BT_STREAMFD_IND);	if (err < 0)		return err;	if (data->stream.fd >= 0)		close(data->stream.fd);	data->stream.fd = bt_audio_service_get_data_fd(data->server.fd);	if (data->stream.fd < 0) {		return -errno;	}	if (data->transport == BT_CAPABILITIES_TRANSPORT_A2DP) {		opt_name = (io->stream == SND_PCM_STREAM_PLAYBACK) ?						SO_SNDTIMEO : SO_RCVTIMEO;		if (setsockopt(data->stream.fd, SOL_SOCKET, opt_name, &t,							sizeof(t)) < 0)			return -errno;	} else {		opt_name = (io->stream == SND_PCM_STREAM_PLAYBACK) ?						SCO_TXBUFS : SCO_RXBUFS;		if (setsockopt(data->stream.fd, SOL_SCO, opt_name, &period_count,						sizeof(period_count)) == 0)			return 0;		opt_name = (io->stream == SND_PCM_STREAM_PLAYBACK) ?						SO_SNDBUF : SO_RCVBUF;		if (setsockopt(data->stream.fd, SOL_SCO, opt_name, &period_count,						sizeof(period_count)) == 0)			return 0;		/* FIXME : handle error codes */	}	/* wake up any client polling at us */	return write(data->pipefd[1], &c, 1);}static int bluetooth_hsp_hw_params(snd_pcm_ioplug_t *io,					snd_pcm_hw_params_t *params){	struct bluetooth_data *data = io->private_data;	char buf[BT_AUDIO_IPC_PACKET_SIZE];	bt_audio_rsp_msg_header_t *rsp_hdr = (void*) buf;	struct bt_setconfiguration_req *setconf_req = (void*) buf;	struct bt_setconfiguration_rsp *setconf_rsp = (void*) buf;	int err;	DBG("Preparing with io->period_size=%lu io->buffer_size=%lu",					io->period_size, io->buffer_size);	memset(setconf_req, 0, BT_AUDIO_IPC_PACKET_SIZE);	setconf_req->h.msg_type = BT_SETCONFIGURATION_REQ;	strncpy(setconf_req->device, data->alsa_config.device, 18);	setconf_req->transport = BT_CAPABILITIES_TRANSPORT_SCO;	setconf_req->access_mode = (io->stream == SND_PCM_STREAM_PLAYBACK ?			BT_CAPABILITIES_ACCESS_MODE_WRITE :			BT_CAPABILITIES_ACCESS_MODE_READ);	err = audioservice_send(data->server.fd, &setconf_req->h);	if (err < 0)		return err;	err = audioservice_expect(data->server.fd, &rsp_hdr->msg_h,					BT_SETCONFIGURATION_RSP);	if (err < 0)		return err;	if (rsp_hdr->posix_errno != 0) {		SNDERR("BT_SETCONFIGURATION failed : %s(%d)",					strerror(rsp_hdr->posix_errno),					rsp_hdr->posix_errno);		return -rsp_hdr->posix_errno;	}	data->transport = setconf_rsp->transport;	data->link_mtu = setconf_rsp->link_mtu;	return 0;}static uint8_t default_bitpool(uint8_t freq, uint8_t mode){	switch (freq) {	case BT_SBC_SAMPLING_FREQ_16000:	case BT_SBC_SAMPLING_FREQ_32000:		return 53;	case BT_SBC_SAMPLING_FREQ_44100:		switch (mode) {		case BT_A2DP_CHANNEL_MODE_MONO:		case BT_A2DP_CHANNEL_MODE_DUAL_CHANNEL:			return 31;		case BT_A2DP_CHANNEL_MODE_STEREO:		case BT_A2DP_CHANNEL_MODE_JOINT_STEREO:			return 53;		default:			DBG("Invalid channel mode %u", mode);			return 53;		}	case BT_SBC_SAMPLING_FREQ_48000:		switch (mode) {		case BT_A2DP_CHANNEL_MODE_MONO:		case BT_A2DP_CHANNEL_MODE_DUAL_CHANNEL:			return 29;		case BT_A2DP_CHANNEL_MODE_STEREO:		case BT_A2DP_CHANNEL_MODE_JOINT_STEREO:			return 51;		default:			DBG("Invalid channel mode %u", mode);			return 51;		}	default:		DBG("Invalid sampling freq %u", freq);		return 53;	}}static int bluetooth_a2dp_init(struct bluetooth_data *data,				snd_pcm_hw_params_t *params){	struct bluetooth_alsa_config *cfg = &data->alsa_config;	sbc_capabilities_t *cap = &data->a2dp.sbc_capabilities;	unsigned int max_bitpool, min_bitpool, rate, channels;	int dir;	snd_pcm_hw_params_get_rate(params, &rate, &dir);	snd_pcm_hw_params_get_channels(params, &channels);	switch (rate) {	case 48000:		cap->frequency = BT_SBC_SAMPLING_FREQ_48000;		break;	case 44100:		cap->frequency = BT_SBC_SAMPLING_FREQ_44100;		break;	case 32000:		cap->frequency = BT_SBC_SAMPLING_FREQ_32000;		break;	case 16000:		cap->frequency = BT_SBC_SAMPLING_FREQ_16000;		break;	default:		DBG("Rate %d not supported", rate);		return -1;	}	if (cfg->has_channel_mode)		cap->channel_mode = cfg->channel_mode;	else if (channels == 2) {		if (cap->channel_mode & BT_A2DP_CHANNEL_MODE_JOINT_STEREO)			cap->channel_mode = BT_A2DP_CHANNEL_MODE_JOINT_STEREO;		else if (cap->channel_mode & BT_A2DP_CHANNEL_MODE_STEREO)			cap->channel_mode = BT_A2DP_CHANNEL_MODE_STEREO;		else if (cap->channel_mode & BT_A2DP_CHANNEL_MODE_DUAL_CHANNEL)			cap->channel_mode = BT_A2DP_CHANNEL_MODE_DUAL_CHANNEL;	} else {		if (cap->channel_mode & BT_A2DP_CHANNEL_MODE_MONO)			cap->channel_mode = BT_A2DP_CHANNEL_MODE_MONO;	}	if (!cap->channel_mode) {

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
亚洲午夜精品网| 欧美日韩久久一区| 国产精品入口麻豆九色| 成人高清视频在线观看| 亚洲人成在线播放网站岛国 | 日韩一级在线观看| 美女爽到高潮91| 久久久精品蜜桃| 99视频精品免费视频| 亚洲激情图片小说视频| 欧美色区777第一页| 麻豆国产精品官网| 日本一二三不卡| 欧美三级一区二区| 久久99精品国产麻豆婷婷| 久久精品夜色噜噜亚洲a∨| 99精品偷自拍| 日韩精品欧美精品| 日本一区二区免费在线观看视频 | 久久成人麻豆午夜电影| 国产校园另类小说区| www.在线成人| 午夜精品123| 国产精品视频一二| 91精品国产免费久久综合| 国产精品一线二线三线| 亚洲黄一区二区三区| 日韩久久久久久| 色偷偷一区二区三区| 久久爱www久久做| 亚洲综合在线五月| 久久免费视频一区| 欧美精品在线观看播放| 成人激情综合网站| 蜜桃传媒麻豆第一区在线观看| 中文一区在线播放| 91精品国产欧美日韩| 成人v精品蜜桃久久一区| 日韩国产高清影视| 亚洲精品欧美激情| 国产亚洲成年网址在线观看| 7777精品伊人久久久大香线蕉完整版| 春色校园综合激情亚洲| 麻豆精品国产传媒mv男同| 夜夜嗨av一区二区三区四季av| 久久久久久久国产精品影院| 欧美日韩成人在线| 91毛片在线观看| 成人av影视在线观看| 激情综合色综合久久| 五月综合激情网| 一区二区三区四区不卡在线 | 日韩电影在线一区| 亚洲高清视频在线| 亚洲欧美视频在线观看| 久久久精品2019中文字幕之3| 欧美一级高清片| 欧美中文一区二区三区| 99久久99久久精品免费看蜜桃| 国产制服丝袜一区| 秋霞影院一区二区| 亚洲va国产va欧美va观看| 亚洲素人一区二区| 久久不见久久见中文字幕免费| 一区二区三区毛片| 亚洲欧美日本在线| 亚洲特黄一级片| 日日摸夜夜添夜夜添精品视频| 精品免费一区二区三区| 成人午夜视频网站| 亚洲高清免费在线| 亚洲日本韩国一区| 日韩一区二区精品| 欧美三级韩国三级日本三斤| 国产精品性做久久久久久| 国产精品区一区二区三| 欧美少妇bbb| 青青草97国产精品免费观看无弹窗版| 91首页免费视频| 男人操女人的视频在线观看欧美| 精品国产免费人成电影在线观看四季| 精品1区2区3区| 日韩一区二区在线观看视频播放| 91精品国产综合久久精品性色| 91女厕偷拍女厕偷拍高清| 精品一区二区在线免费观看| 亚洲丝袜精品丝袜在线| 精品99一区二区三区| 欧美四级电影在线观看| 国产福利一区二区三区在线视频| 亚洲欧美另类小说| 久久久噜噜噜久久中文字幕色伊伊| 99久久国产综合色|国产精品| 老司机免费视频一区二区| 婷婷国产v国产偷v亚洲高清| 亚洲欧美怡红院| 国产精品免费免费| 欧美国产精品劲爆| 国产精品久久久久一区二区三区共| 欧美男生操女生| 在线播放欧美女士性生活| 欧美日韩黄视频| 欧美午夜电影一区| 91精品久久久久久久91蜜桃| 欧美三级视频在线| 欧美专区日韩专区| 精品久久久久av影院 | 欧美mv日韩mv国产| 色噜噜狠狠成人中文综合 | 91精品蜜臀在线一区尤物| 亚洲午夜在线观看视频在线| 久久精品一区二区三区不卡| 色噜噜狠狠一区二区三区果冻| 欧美日韩另类一区| 精品国产91乱码一区二区三区| 久久久www免费人成精品| 亚洲另类在线视频| 久久99久久久欧美国产| 成人激情黄色小说| 日韩午夜激情视频| 国产欧美精品一区二区三区四区| 亚洲一区二区三区四区在线观看 | 中文字幕视频一区二区三区久| 亚洲欧洲成人精品av97| 午夜激情久久久| 欧美丝袜丝nylons| 欧美va天堂va视频va在线| 久久久精品国产99久久精品芒果| 国产精品色一区二区三区| 一区二区成人在线| 91精品国产91久久久久久最新毛片| 国产夫妻精品视频| 免费视频一区二区| 亚洲综合久久久| 国产亚洲精品资源在线26u| 欧美视频精品在线| 91理论电影在线观看| 激情小说欧美图片| 蜜桃久久av一区| 日本在线播放一区二区三区| 国产精品国模大尺度视频| 日本美女视频一区二区| 欧美精品一级二级三级| 日韩av不卡在线观看| 麻豆精品一二三| 亚洲欧美自拍偷拍| 久久男人中文字幕资源站| 91精品久久久久久久99蜜桃 | 有坂深雪av一区二区精品| 精品国产免费人成电影在线观看四季| 91国产成人在线| 欧美亚洲国产一区二区三区| 成人不卡免费av| 9i看片成人免费高清| 99精品黄色片免费大全| 高潮精品一区videoshd| 国产一区二区导航在线播放| 老司机精品视频在线| 国产精品正在播放| av一区二区不卡| 欧美性videosxxxxx| 日韩欧美一级二级三级| 国产亚洲精品中文字幕| 国产精品传媒在线| 国产亚洲欧美色| 亚洲国产精品精华液2区45| 国产精品天天摸av网| 一区二区三区美女| 免费成人深夜小野草| 国产高清不卡一区| 欧美日韩成人综合在线一区二区 | 在线免费观看不卡av| 日韩欧美亚洲一区二区| 国产精品视频在线看| 亚洲第一成人在线| 成人永久aaa| 日韩欧美电影一区| 一区二区三区在线视频播放| 美腿丝袜亚洲色图| 欧美私模裸体表演在线观看| 欧美一卡2卡3卡4卡| 亚洲色图色小说| 老司机免费视频一区二区| 在线中文字幕不卡| 亚洲丝袜自拍清纯另类| 国产一区二区福利| 91美女片黄在线| 国产一区二区日韩精品| 亚洲成人免费在线观看| 亚洲你懂的在线视频| 国产肉丝袜一区二区| 欧美一区二区网站| 在线播放日韩导航| 日韩一级欧美一级| 亚洲国产高清在线| 欧日韩精品视频| 亚洲成人手机在线| 久久久久久久久久美女| 91猫先生在线| 另类人妖一区二区av|