socket TCP 从0实现音频传输 ALSA 播放
RTP标准是采用 UDP 发送,有不少现成的开源库,但不在本文讨论的范围内。
UDP 用户数据报,不提供流程,安全传输的功能,但速度快,能提供多播,广播,没有序列号 SEQ ,有 MTU 限制,1500。
TCP 传输控制协议,提供流控,SEQ ,重传功能,没有数据长度限制,可以发几 M 。
但在使用中还是有很多地方需要注意,否则声音不好听,断断续续或是延时严重。
虽然 TCP 在使用上没有 MTU 限制,但是在真实的2个PC 之音使用 TCP 发送数据,也是被切片的,每次包不能超过 1448 字节,本机对本机,数据包没有 MTU 限制,因为走的是 LOOPBACK ,装个 wireshark 一看就懂了。
为什么是 1448 ,MTU 是1500 减去 包头 20 TCP 头 20 时间戳 12 。
ALSA 的播放有2种打开模式,阻塞 非阻塞 snd_pcm_nonblock(playback_handle, 1);
发送接收流控问题,TCP 发送的数据如果大于接收的速度,就会被缓存起来,一直到接收完成以后。
下面只贴出关键代码,来看这一个问题。
send.c 通过 读取本地的一个 wav 的文件,直接读取 一块内容直接发送到本地端口。
这里隐藏了,2个问题
1,用 TCP 来发送,RTP是使用 UDP 发送,而 UDP 不提供 SEQ ,也就是说,接收到的数据顺序会乱,所以,RTP 在使用 UDP 承载时都会自己加一个头信息
自行维护,SEQ ,时间戳,数据校验
2,UDP 是一发就很快返回,不需要等待接收方的 ACK ,所以用 TCP 发送会慢一点。
play.c
struct sockaddr_in client_addr;
socklen_t client_addr_len;
client_fd = accept(server_fd, (struct sockaddr *)&client_addr, &client_addr_len); if( > client_fd)
{
printf("client connect error");
continue;
}
printf("client connect addr:%s \n", inet_ntoa(client_addr.sin_addr));
while()
{
int read_len;
int ret;
unsigned char buffer[];
read_len = recv(client_fd, buffer, , );
if( >= read_len) break;
ret = snd_pcm_writei(playback_handle, buffer, read_len/);
if(-EPIPE == ret)
{
snd_pcm_prepare(playback_handle);
}
printf("read_len:%d \n", read_len);
}
close(client_fd);
send.c
audio_p = audio_buf;
send_size = ;
while((audio_p - audio_buf) <= stat.st_size)
{
if(- != client_fd) send(client_fd, audio_p, send_size, );
audio_p += send_size;
}
以上2个程序,可以运行,也可以播放WAV ,存在的问题是,send 一下子把所有数据都发过去了,当按 ctrl+c 退出时,play 还有数据,还能播放一段时间。
理想的情况是,就像打电话一样,挂断后马上停止播放。
send 中是直接发送的 WAV 中的 PCM 数据,所以要添加合适的 delay 来模拟真实的音频直播。
delay 时长的计算:4096*1000000/(44100*16*2/8)=23219 ,每次发送的长度是 4096 ,采样率44k 2通道 16位,算出来,休眠时长是 23219us 。
如果直接在 while 发送 数据包时直接添加 usleep(23219) ,就会发现,播放出来非常难听,这是因为 TCP 发送还是需要时间的,这也是RTP 用 UDP 的原因,UDP 发送非常快。
所以,这里需要另开一个线程,专门做 发送 TCP 数据包的功能。
同理,play 的里面也需要把 TCP 接收的单独放到线程中。
原理讲明白了,代码就不贴了。
delay 时长的另一种计算方法:
ffmpeg -i baby.wav
Input #, wav, from 'baby.wav':
Metadata:
artist : Justin Bieber
date :
genre : Dance
title : Baby
album : My World 2.0
track :
encoder : Lavf58.20.100
Duration: ::34.21, bitrate: kb/s
Stream #:: Audio: pcm_s16le ([][][][] / 0x0001), Hz, stereo, s16, kb/s ls -l
-rw-r--r-- root root 1月 : baby.wav (*+)*= ms ________ = _____
x */=23.1974894086877 ms
通过文件大小和播放时长计算,4096长度的需要播放多少时间,这里为了简单没有去掉 WAV 文件头。
使用 getimeofday 来统计播放需要的时长
printf("chunk_size:%ld \n", chunk_size);
audio_p = audio_buf;
while((audio_p - audio_buf) <= stat.st_size)
{
struct timeval t1, t2;
gettimeofday(&t1, NULL);
int ret = snd_pcm_writei(playback_handle, audio_p, chunk_size);
if(-EPIPE == ret)
{
snd_pcm_prepare(playback_handle);
}
gettimeofday(&t2, NULL);
printf("ms:%ld\n", (t2.tv_sec - t1.tv_sec)* + (t2.tv_usec-t1.tv_usec)/);
audio_p += chunk_size * ; //16位 双声道
}
打出来的结果全是 21 20 。如果 chunk_size 设为 1024 那么打出来的结果很混乱 41 21 0 20 。因为 alsa 的 period_size 不正好是4096 ,所以播放用时不同。
socket TCP 从0实现音频传输 ALSA 播放的更多相关文章
- 分享一个分布式消息总线,基于.NET Socket Tcp的发布-订阅框架,附代码下载
一.分布式消息总线 在很多MIS项目之中都有这样的需求,需要一个及时.高效的的通知机制,即比如当使用者A完成了任务X,就需要立即告知使用者B任务X已经完成,在通常的情况下,开发人中都是在使用者B所使用 ...
- 基于.NET Socket Tcp的发布-订阅框架
基于.NET Socket Tcp的发布-订阅框架 一.分布式消息总线 在很多MIS项目之中都有这样的需求,需要一个及时.高效的的通知机制,即比如当使用者A完成了任务X,就需要立即告知使用者B任务X已 ...
- C++写Socket——TCP篇(0)建立连接及双方传输数据
满山的红叶--飘落之时-- 最近接触了点关于用C++写socket的东西,这里总结下. 这里主要是关于TCP的,TCP的特点什么的相关介绍在我另一篇博文里,所以这里直接动手吧. 我们先在windows ...
- 标准C实现基于TCP/IP协议的文件传输
上学期集成程序设计的课堂作业,对于理解TCP/IP实现还是挺有帮助的. TCP/IP编程实现远程文件传输在LUNIX中一般都采用套接字(socket)系统调用. 采用客户/服务器模式,其程序编写步骤如 ...
- http与https与socket tcp/IP与UDP 协议等
网络由下往上分为: 物理层-- 数据链路层-- 网络层-- IP协议 传输层-- ...
- python socket+tcp三次握手四次撒手学习+wireshark抓包
Python代码: server: #!/usr/bin/python # -*- coding: UTF-8 -*- # 文件名:server.py import socket # 导入 socke ...
- 初步探究java中程序退出、GC垃圾回收时,socket tcp连接的行为
初步探究java中程序退出.GC垃圾回收时,socket tcp连接的行为 今天在项目开发中需要用到socket tcp连接相关(作为tcp客户端),在思考中发觉需要理清socket主动.被动关闭时发 ...
- 标准C语言实现基于TCP/IP协议的文件传输
TCP/IP编程实现远程文件传输在LUNIX中一般都采用套接字(socket)系统调用. 采用客户/服务器模式,其程序编写步骤如下: 1.Socket系统调用 为了进行网络I/O,服务器和客户机两 ...
- 27.Socket,TCP,UDP,HTTP基本通信原理
Socket,TCP,UDP,HTTP基本通信原理(摘自百度): TCP.UDP,HTTP 底层通信都是通过 socket 套接字实现 网络上不同的计算机,也可以通信,那么就得使用网络套接字(sock ...
随机推荐
- iOS动画效果合集、飞吧企鹅游戏、换肤方案、画板、文字效果等源码
iOS精选源码 动画知识运用及常见动画效果收集 3D卡片拖拽卡片叠加卡片 iFIERO - FLYING PENGUIN 飞吧企鹅SpriteKit游戏(源码) Swift封装的空数据提醒界面Empt ...
- 树形dp(最小支配集)
http://poj.org/problem?id=3659 #include<iostream> #include<cstring> #include<algorith ...
- labview学习——用户界面模式
根据事件的发出源,事件可以抽象地分为用户界面事件和用户自定义事件.相关的基本知识可以参考有关的书籍,这里不再阐述事件结构的使用方法. 下图所示的结构称为用户界面事件模式,它能够很便捷地响应各种事件并且 ...
- 吴裕雄--天生自然python学习笔记:Python3 多线程
多线程类似于同时执行多个不同程序,多线程运行有如下优点: 使用线程可以把占据长时间的程序中的任务放到后台去处理. 用户界面可以更加吸引人,比如用户点击了一个按钮去触发某些事件的处理,可以弹出一个进度条 ...
- application.properties和application.yml
1.application.properties 文件和 application.yml 文件优先级: 当同时存在时,那application.yml 只是个摆设. 2.yml文件的好处,天然的树状结 ...
- cesium入门示例-HelloWorld
示例准备: 在Cesium ion官网(https://cesium.com/)上注册用户,获取AccessToken,在js代码入口设置Cesium.Ion.defaultAccessToken,即 ...
- 提高你css技能的css开发技巧
好久没整理博客了 进来啰嗦两句 继续抄别人的博客 一.resize实现图片对比 resize的语法如下: resize:none | both | horizontal | vertical 案例 ...
- 基于OpenDDS应用程序开发(3)订阅端实现
连续的三篇博文演示如何基于OpenDDS开发应用程序,将数据从发布端节点发送到订阅端节点,该示例程序由一个发布者发布数据,一个订阅者订阅数据,使用默认的QoS策略和TCP/IP传输方式. 本文是第三篇 ...
- JavaScript if为true的情况
变量如果不为0,null,undefined,false,都会被处理为true.只要变量有非0的值或是某个对象,数组,字符串,都会认为true
- 是AI就躲个飞机-纯Python实现人工智能
你要的答案或许都在这里:小鹏的博客目录 代码下载:Here. 很久以前微信流行过一个小游戏:打飞机,这个游戏简单又无聊.在2017年来临之际,我就实现一个超级弱智的人工智能(AI),这货可以躲避从屏幕 ...