Linux:内核源代码角度看文件和Socket

文章目录

  • 文件和Socket

文件和Socket

在之前写的网络服务,它们的本质其实就是一个进程,而对于每一个打开的文件来说,都要有一个自己对应的文件描述符,其中会默认打开对应的012,作为标准输入标准输出标准错误,这是之前的认识

在这里插入图片描述
如上图所示,进程的PCB当中会包含有对应的struct files_struct结构体,而又会包含对应的struct file结构体,在这个结构体当中可以看到对应的文件的操作方法集和对应的文件缓冲区

而实际上在对应的这个当中会存在一个void* private_data,那这个字段是什么意思呢?在源码的注释中显示的是这是需要一个对应的驱动等等的信息,想要理解它就必须要先看一下Socket结构体了:

当操作系统在创建网络套接字的时候,会创建出一大堆的数据结构,那其中有一个数据结构叫做struct socket

在这里插入图片描述
在这个结构体当中会存在一个file指针,这个socket会指向一个文件,而这个void的指针,也会反过来去指向这个socket结构体(如果这是一个网络文件的话),所以说Linux下一切皆文件,最终网络文件会挂接在这个struct file之下,未来在应用层看待的时候,只需要按照文件描述符,这样就能找到这个文件的socket,而在这个socket结构体当中又存在一个__wait_queue_head,这说明是在进程阻塞等待的时候,实际上就是要把自己的PCB链入到指定的数据结构中,那么现在在网络中也是,当网络里面读取的数据不就绪的时候,那么就会把当前网络文件也会挂接到对应的阻塞队列当中

那在创建套接字的时候,TCP和UDP是如何被创建的?其实在内核当中是使用了一个类似于C语言多态的感觉:

在这里插入图片描述
不管是tcp_sock还是udp_sock,它们的最前面的字段都是叫做是struct sock,而在这个结构体当中就会存在对应的接收缓冲区和发送缓冲区,根据socket的标记位,就可以确定到底是TCP还是UDP,如果是TCP就把这个发送接收缓冲区进行初始化,如果是UDP就不管它

在文件当中会存在文件对应的方法集,如果它指向的是磁盘,那么就代表的是磁盘对应的读写方法,但如果是指向的是一个网络,那么就是网络相关的读写方法,那在socket当中也有对应的网络方法,那这两个方法有什么区别?

在struct file这样的结构体当中,它负责解决的网络方面的方法其实是对上层的方法,因为当前数据是在应用层的,那么数据想要进入文件描述符中一定是要想办法从应用层拷贝到传输层当中,如何拷贝?其实依靠的就是这个文件描述符,然后拷贝到接收缓冲区中,那下一步要做的就是把处于传输层的数据再想办法拷贝到下一层当中,所以从本质上来说,每一层的所谓的方法集,本质上就是把数据进行加工,进行解析和处理,然后把数据再进行一步一步的拷贝,因此TCP它把数据发出去了吗?其实根本没有,它在底层就是把数据做了它自己的加工,然后把数据传递到了下层当中,TCP和UDP压根不会把数据发出去,只是会用对应的网络协议栈的方法来对于数据做加工,然后再把数据传递下去

所以每一个文件描述符就都会对应上面的这么一套系统,每需要进行发送一次消息就要建立这么一套信息,而操作系统当中必定会存在很多这样的报文,那么操作系统必然要对于收到的这些或者要发出去的报文做管理,那管理工作是如何进行管理的?

struct sk_buff_head {
	/* These two members must be first. */
	struct sk_buff	*next;
	struct sk_buff	*prev;

	__u32		qlen;
	spinlock_t	lock;
};

在这个当中看到了锁的存在,如何理解锁?因为在内核当中对于数据的加工,本质上就是一种生产消费者模型,所以是需要有自旋锁这样的锁存在的

在这里插入图片描述

sk_buff该如何理解?用下面这个图来进行表示:

在这里插入图片描述
这四个指针会进行相互配合,能够表示出对应的报头和有效载荷,它本质上就是内存中的一部分空间,所以对应一个数据在网络协议栈进行传输,本质上就是让上面的这个结构在网络协议栈当中进行流动,换句话说,所谓的添加报头还是删除报头,就是让头指针不断移动,向上就是新增报头,向下就是删除报头,从而实现了封装和解包的操作

所以,封装和解包只需要进行移动指针

网络协议栈是一个层状结构,有传输层网络层等等,所以所谓的协议栈本质上就是用特定数据结构表述的协议,并且也是和特定协议匹配的方法集,这个如何理解?

在传输层之前,所有的报文都是在一起的,当传输到传输层之后,就会根据不同的报文类型,把报文传递到对应的文件描述符所对应的socket下,这样就能做到对于报文做管理的工作,而所谓的方法就是用指针移动来进行添加报头,就实现了加协议和去协议这样的操作效果

所以未来,应用层想要获得报文,操作系统内部就要想办法通过上面那张图所示的这么一系列方法,来从对应的接收缓冲区中把数据拿上来就可以了,这样的思想很符合所谓的先描述再组织,用sk_buff来进行描述,用队列来进行组织,这样就实现了先描述再组织的操作

那现在对于封装和解包的理解应该上升到这样的程度:所谓的封装和解包,其实就是在传输层,网络层,数据链路层进行不断的封装,并且都是在对应的缓冲区当中进行操作就可以,这样避免了频繁的拷贝,只是使用头指针的移动就可以了

至此,就把文件系统和网络系统关联起来了

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/504579.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

数据结构——lesson13排序之计数排序

💞💞 前言 hello hello~ ,这里是大耳朵土土垚~💖💖 ,欢迎大家点赞🥳🥳关注💥💥收藏🌹🌹🌹 💥个人主页&#x…

如何简化多个 if 的判断结构

多少算太多? 有些人认为数字就是一,你应该总是用至少一个三元运算符来代替任何单个 if 语句。我并不这样认为,但我想强调一些摆脱常见的 if/else 意大利面条代码的方法。 我相信很多开发人员很容易陷入 if/else 陷阱,不是因为其…

ThreadLocal的基本使用

一、ThreadLocal的介绍 ThreadLocal 是 Java 中的一个类,它提供了线程局部变量的功能。线程局部变量是指每个线程拥有自己独立的变量副本,这些变量在不同的线程中互不影响。ThreadLocal 提供了一种在多线程环境下,每个线程都可以独立访问自己…

PS从入门到精通视频各类教程整理全集,包含素材、作业等(4)

PS从入门到精通视频各类教程整理全集,包含素材、作业等 最新PS以及插件合集,可在我以往文章中找到 由于阿里云盘有分享次受限制和文件大小限制,今天先分享到这里,后续持续更新 PS人物数码照片处理技法视频教程 https://www.…

武汉星起航:一站式跨境电商服务引领者,专业高效助力客户出海

武汉星起航电子商务有限公司,坐落于华中地区的商业核心地带——湖北武汉,自公司成立以来,便以提供一站式跨境电商服务为核心发展,致力于为广大客户提供专业、高效、全面的出海解决方案。凭借5对1服务体系、ERP软件授权、中转仓服务…

二、分布式事务

目录 二、分布式事务2.1 什么是分布式事务2.2 分布式事务产生的背景2.3 分布式事务产生的场景2.4 分布式事务理论4.1 CAP理论4.2 Base理论 5、分布式事务的解决方案 二、分布式事务 2.1 什么是分布式事务 一组操作会产⽣多个数据库session会话 此时就会出现分布式事务 2.2 分…

游戏软件出现d3dcompiler_47.dll缺失怎么修复,亲测的六种有效方法推荐

D3DCompiler47.dll是DirectX SDK中的一个重要组件,它提供了将HLSL(High-Level Shading Language)着色器编译为可执行代码的功能。通过使用D3DCompiler47.dll,开发人员可以将复杂的着色器代码转换为可以在GPU上高效运行的机器代码&…

黑马点评项目笔记 II

基于Stream的消息队列 stream是一种数据类型,可以实现一个功能非常完善的消息队列 key:队列名称 nomkstream:如果队列不存在是否自动创建,默认创建 maxlen/minid:设置消息队列的最大消息数量 *|ID 唯一id:…

Vue系列-el挂载

<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>el:挂载点</title> </head> <body&g…

作业 二维数组-定位问题

图形相似度 描述 给出两幅相同大小的黑白图像&#xff08;用0-1矩阵&#xff09;表示&#xff0c;求它们的相似度。 说明&#xff1a;若两幅图像在相同位置上的像素点颜色相同&#xff0c;则称它们在该位置具有相同的像素点。 两幅图像的相似度定义为相同像素点数占总像素点数…

P87 4.1 C++ FOR 与Delphi FOR 的区别

输出x, sin(x), cos(x), tan(x)的值。已知X0&#xff0c;10&#xff0c; 20&#xff0c;180。 我用Delphi编写了程序&#xff1a; 第10行出现 给FOR 循环变量赋值i错误。 C中是可以的&#xff0c; 详见&#xff1a;delphi循环的一个小知识_assignment to for-loop variable…

安装JupyterLab的集成环境

Python集成环境安装 不要半途而废&#xff0c;不要作业太多就抛下你手中的笔&#xff0c;拿起你旁边的手机&#xff0c;你觉得这样很有意义吗&#xff1f;一个小时一道题都没做&#xff0c;盯着手机屏幕它能给你一个未来吗&#xff1f;少分心就能多做一道题&#xff0c;多学样本…

Java多线程:定位死锁

检测死锁可以使用jconsole工具&#xff0c;或使用jps定位进程id&#xff0c;再用jstack定位死锁 方案1&#xff1a; 1. 先用jps查看所有的java进程id 2. jstack 进程id定位死锁 3. 查看死锁结果 方案2:从jdk的安装路径中找到bin目录, 点击jconsole

Kafka入门到实战-第五弹

Kafka入门到实战 Kafka常见操作官网地址Kafka概述Kafka的基础操作更新计划 Kafka常见操作 官网地址 声明: 由于操作系统, 版本更新等原因, 文章所列内容不一定100%复现, 还要以官方信息为准 https://kafka.apache.org/Kafka概述 Apache Kafka 是一个开源的分布式事件流平台&…

1.5编写一个程序,输入梯形的上底,下底和高,输出梯形的面积。

1、编写一个程序,输入梯形的上底,下底和高,输出梯形的面积。 package com.kangning.web.controller.system;import java.util.Scanner;/*** 编写一个程序,输入梯形的上底,下底和高,输出梯形的面积。*/ public class CountArea {public static void main(String[] args) …

知乎:多云架构下大模型训练,如何保障存储稳定性?

知乎&#xff0c;中文互联网领域领先的问答社区和原创内容平台&#xff0c;2011 年 1 月正式上线&#xff0c;月活跃用户超过 1 亿。平台的搜索和推荐服务得益于先进的 AI 算法&#xff0c;数百名算法工程师基于数据平台和机器学习平台进行海量数据处理和算法训练任务。 为了提…

java入门学习Day01

本篇文章主要是学会如何使用IDEA&#xff0c;和运行第一个java文件。 java环境安装&#xff1a;Windows下Java环境配置教程_windows java环境配置-CSDN博客 IDEA安装&#xff1a;IDEA 2023.2.5 最新激活码,注册码&#xff08;亲测好用&#xff09; - 异常教程 以上两个链接…

函数栈帧的创建与销毁(最详细的一集)上

前言 1.我们在进行c语言代码编程的时候&#xff0c;常常会把独立的一个功能抽象为函数&#xff0c;利用函数去实现各种的功能&#xff0c;那么&#xff0c;函数是如何调用的&#xff1f;函数的返回值是怎么返回的&#xff1f;参数又是如何传参的&#xff1f;所有这些问题都会跟…

【NLP练习】Pytorch文本分类入门

Pytorch文本分类入门 &#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客 &#x1f356; 原作者&#xff1a;K同学啊 | 接辅导、项目定制一、前期准备 1. 环境安装 确保已经安装torchtext与portalocker库 2. 加载数据 #加载数据 import torch import t…

【滑动窗口】Leetcode 找到字符串中所有字母异位词

题目解析 438. 找到字符串中所有字母异位词 算法讲解 寻找目标串的异位词&#xff0c;我们使用固定长度的滑动窗口&#xff0c;首先我们判断窗口左右的字符是否存在于目标串中&#xff0c;如果不存在就让窗口滑动&#xff1b;存在的话&#xff0c;我们就把字符丢进Hash中&a…