手写TCP/IP协议栈——TCP数据接收

一、理论部分

使用接收缓冲区来缓存对方发来的数据

二、代码实现部分

在tcp控制块添加接收缓冲区字段

cpp 复制代码
struct _xtcp_t
{
	xtcp_state_t state;
	uint16_t local_port, remote_port;
	xipaddr_t remote_ip;
	uint32_t unack_seq, next_seq;
	uint32_t ack;
	uint16_t remote_mss;
	uint16_t remote_win;
	xtcp_handler_t handler;
	xtcp_buf_t tx_buf, rx_buf;
};

在tcp_alloc函数中初始化rx_buf

cpp 复制代码
static xtcp_t* tcp_alloc(void)
{
	xtcp_t* tcp, * end;
	for (tcp = tcp_socket, end = tcp_socket + XTCP_CFG_MAX_TCP; tcp < end; tcp++)
	{
		if (tcp->state == XTCP_STATE_FREE)
		{
			tcp->state = XTCP_STATE_CLOSED;
			tcp->local_port = 0;
			tcp->remote_port = 0;
			tcp->remote_ip.addr = 0;
			tcp->handler = (xtcp_handler_t)0;
			tcp->remote_win = XTCP_MSS_DEFAULT;
			tcp->remote_mss = XTCP_MSS_DEFAULT;
			tcp->unack_seq = tcp->next_seq = tcp_get_init_seq();
			tcp->ack = 0;
			tcp_buf_init(&tcp->tx_buf);
			tcp_buf_init(&tcp->rx_buf);
			return tcp;
		}
	}

	return (xtcp_t*)0;
}

在http_handler当中测试TCP对于数据的收发功能

cpp 复制代码
static uint8_t rx_buffer[1024];

static xnet_err_t http_handler(xtcp_t* tcp, xtcp_conn_state_t state)
{
	static char* num = "0123456789ABCDEF";

	if (state == XTCP_CONN_CONNECTED)
	{
		printf("http connected.\n");
	}
	else if (state == XTCP_CONN_CLOSED)
	{
		printf("http closed.\n");
	}
	else if (state == XTCP_CONN_DATA_RECV)
	{
		uint8_t* data = rx_buffer;

		uint16_t read_size = xtcp_read(tcp, rx_buffer, sizeof(rx_buffer));
		while (read_size)
		{
			uint16_t curr_size = xtcp_write(tcp, data, read_size);
			data += curr_size;
			read_size -= curr_size;
		}
	}

	return XNET_ERR_OK;
}

xtcp_read函数实现:

cpp 复制代码
uint16_t xtcp_read(xtcp_t* tcp, uint8_t* data, uint16_t size)
{
	return tcp_buf_read(&tcp->rx_buf, data, size);
}

tcp_buf_read函数实现:

cpp 复制代码
static uint16_t tcp_buf_read(xtcp_buf_t* tcp_buf, uint8_t* to, uint16_t size)
{
	int i;

	size = min(size, tcp_buf->data_count);
	for (i = 0; i < size; i++)
	{
		*to++ = tcp_buf->data[tcp_buf->tail++];
		if (tcp_buf->tail >= XTCP_CFG_RTX_BUF_SIZE)
		{
			tcp_buf->tail = 0;
		}
	}
	tcp_buf->data_count -= size;

	return size;
}

在xtcp_in当中添加TCP数据接收部分的处理

1)首先通过tcp_recv函数将packet->data当中的数据读取到接收缓冲区rx_buf

2)收到数据,回复ack,并处理接收的数据

cpp 复制代码
else if (read_size)
{
	tcp_send(tcp, XTCP_FLAG_ACK);
	tcp->handler(tcp, XTCP_CONN_DATA_RECV);
}

完整处理TCP数据接收的代码:

cpp 复制代码
case XTCP_STATE_ESTABLISHED:
	{
		if (tcp_hdr->hdr_flags.flags & (XTCP_FLAG_ACK | XTCP_FLAG_FIN))
		{
			if (tcp_hdr->hdr_flags.flags & XTCP_FLAG_ACK)
			{
				if ((tcp_hdr->ack > tcp->unack_seq) && (tcp_hdr->ack <= tcp->next_seq))
				{
					uint16_t curr_ack_size = tcp_hdr->ack - tcp->unack_seq;
					tcp_buf_add_acked_count(&tcp->tx_buf, curr_ack_size);
					tcp->unack_seq += curr_ack_size;
				}
			}

			read_size = tcp_recv(tcp, (uint8_t*)tcp_hdr->hdr_flags.flags, packet->data, packet->size);

			if (tcp_hdr->hdr_flags.flags & XTCP_FLAG_FIN)
			{
				tcp->state = XTCP_STATE_LAST_ACK;
				tcp->ack++;
				tcp_send(tcp, XTCP_FLAG_FIN | XTCP_FLAG_ACK);
			}
			else if (read_size)
			{
				tcp_send(tcp, XTCP_FLAG_ACK);
				tcp->handler(tcp, XTCP_CONN_DATA_RECV);
			}
			else if (tcp_buf_wait_send_count(&tcp->tx_buf))
			{
				tcp_send(tcp, XTCP_FLAG_ACK);
			}
			
		}
		break;
	}

其中tcp_recv函数实现:

cpp 复制代码
static uint16_t tcp_recv(xtcp_t* tcp, uint8_t flags, uint8_t* from, uint16_t size)
{
	uint16_t read_size = tcp_buf_write(&tcp->rx_buf, from, size);
	tcp->ack += read_size;

	return read_size;
}

最后在tcp_send函数当中,把发送窗口的大小动态调整一下:

cpp 复制代码
tcp_hdr->window = swap_order16(tcp_buf_free_count(&tcp->rx_buf));

三、测试部分

ok,今天的你就到此为止吧,明天还要接着🐺啊!

相关推荐
万象.1 天前
Linux传输层TCP,UDP相关内容
linux·tcp/ip·udp
万象.1 天前
Linux数据链路层通信原理及报文格式
linux·网络·网络协议
雅欣鱼子酱1 天前
Type-C供电PD协议取电Sink芯片ECP5702,可二端头分开供电调整亮度,适用于LED灯带户外防水超亮灯条方案
c语言·开发语言
victory04311 天前
Agent 面试知识树 + 高频追问答案库
网络·面试·职场和发展
Yungoal1 天前
B/S和C/S架构在服务端接收请求
c语言·开发语言·架构
XuanTao771 天前
【分享】Lightroom高级版⭕Ai图片剪辑 天空修补
数码相机·计算机网络·网络安全·软件工程·软件构建
小心我捶你啊1 天前
提升爬虫稳定性的关键,Python爬虫代理IP解析与轮换策略
爬虫·python·tcp/ip
谪星·阿凯1 天前
SQL注入漏洞进阶篇:从盲注到WAF绕过的全面解析
数据库·sql·计算机网络
Blurpath住宅代理1 天前
原生IP vs 广播IP:如何从网络层判断代理IP的“真实性”?
网络·静态ip·动态ip·代理·代理ip·住宅ip·住宅代理
港股研究社1 天前
腾讯音乐的多元增长新路径:音乐IP经济
大数据·人工智能·tcp/ip