首页 > 其他分享 >零拷贝

零拷贝

时间:2023-10-21 23:00:12浏览次数:35  
标签:DMA 内核 缓冲区 拷贝 拷贝到 CPU

目录

零拷贝

传统拷贝

read:

  1. 当使用read调用来读取数据,此时会将用户态转化为内核态
  2. CPU对DMA控制器发起一个调用命令
  3. DMA将数据从硬盘拷贝到内核缓冲区中,拷贝完后通知CPU数据准备好了
  4. CPU将数据从内核缓冲区中拷贝到用户缓冲区中
  5. 在4过程中需要将内核态切换为用户态
  6. 拷贝完后,read调用结束,并唤醒已阻塞的进程

write:

  1. 当用户线程使用write调用来写入数据,此时会将用户态转化为内核态
  2. 将用户缓冲区中的数据拷贝到socket缓冲区
  3. CPU对DMA控制器发起一个调用命令
  4. DMA控制将数据从socket缓冲区拷贝到网卡设备上(拷贝到网卡,消费者才能消费到数据)
  5. 拷贝完成后,发送中断信号通知CPU数据已拷贝完成,CPU收到信号后,将内核态切换为用户态,系统调用返回,并唤醒阻塞写的进程

mmap(内存地址映射) + write


mmap:内存地址映射,将内核空间的PageCache映射到用户缓冲区

  • 发现传统拷贝过程中,将数据拷贝到用户空间里并没有做任何修改,所以其实没必要拷贝到这里
  • 因此这里采用内存映射的方式,将内核缓冲区映射到用户缓冲区(使用的是虚拟内存,不会占用物理内存),就可以减少一次CPU拷贝过程
  1. 用户进程发起mmp系统调用,此时用户态切换为内核态
  2. CPU对DMA控制器发起一个调用命令
  3. DMA将数据从硬盘拷贝到内核缓冲区中
  4. 拷贝完成后,mmap方法返回调用,此时内核态切换为用户态
  5. 用户线程使用write调用来写入数据,此时会将用户态转化为内核态
  6. CPU直接把数据从内核缓冲区拷贝到socket缓冲区
  7. DMA控制将数据从socket缓冲区拷贝到网卡设备上
  8. 拷贝完成后,发送中断信号通知CPU数据已拷贝完成,CPU收到信号后,将内核态切换为用户态,系统调用返回,并唤醒阻塞写的进程
  • 总结:mmap读取数据快的原因就是,因为建立了pageCache到用户进程的虚拟地址映射,避免了把数据从pageCache拷贝到用户进程的过程,从而减少了一次CPU拷贝

  • 注意,依然是两次系统调用,四次上下文切换,但只有三次数据拷贝,其中一次CPU拷贝,两次 DMA 拷贝

kafka生态系统组成

sendfile

与mmap不同在于:

  1. 数据是完全不经过用户空间,它是直接把数据从内核缓冲区拷贝到socket缓冲区
  2. 减少了一次系统调用,sendfile系统调用它是包括了把数据从磁盘拷贝到内核态再拷贝到网卡设备这整个过程,而mmap系统调用不包括把数据从内核态拷贝到网卡设备,这一步需要使用write系统调用

总结:包括一次系统调用,三次数据拷贝,其中一次CPU拷贝、两次DMA拷贝

kafka底层代码调用的是:

这个方法使用了sendfile系统调用

sendfile + DMA gather copy

Linux2.4内核版本开始,对sendfile进行了改进

  • 引入了gather操作
  • 可以实现了无CPU拷贝

与sendfile区别:

  1. 它是把内核缓冲区中的数据的数据描述信息(包括:内存地址、地址偏移量等)读取到socket缓冲区中,然后就会发生一次DMA gather copy,DMA会根据数据的内存地址和偏移量等信息把数据批量的从内核缓冲区读取到网卡设备上,读取完毕后,发送信号,sendfile系统调用返回,上下文切换到用户态
  2. 省去了一次CPU拷贝过程

真正的零拷贝:指没有CPU拷贝的过程

splice + DMA copy

与sendfile的区别:

  1. 在内核缓冲区与socket缓冲区之间搭建了一个环形的管道,将内核缓冲区绑定到管道的写端,把socket缓冲区绑定到管道的读端,所以数据将通过这个管道被读取到socket缓冲区中,然后通过DMA将数据拷贝到网卡设备上,拷贝完成后,splice函数返回,上下文切换到用户态
  2. 省去了一次CPU拷贝过程

标签:DMA,内核,缓冲区,拷贝,拷贝到,CPU
From: https://www.cnblogs.com/cheyaoyao/p/17779719.html

相关文章

  • docker cp 命令 - 宿主机与容器互相拷贝文件
    一、从容器拷贝文件到宿主机命令格式:$dockercp<containder-id>:/path/host/path例子:$dockercpc9b7f17d43e9:/opt/hello.txt/home/hello.txt二、从宿主机拷贝文件到容器命令格式:$dockercp/host/path<containder-id>:/path例子:$dockercp/home/hel......
  • Python深浅拷贝
    Python深浅拷贝拷贝/浅拷贝/深拷贝只针对可变数据类型拷贝(赋值)当lt2为lt的拷贝对象时,lt内的可变类型变化,lt2变化;lt内的不可变类型变化,lt2变化简单的赋值lt=[1,2,3]lt2=ltlt.append(4)print(lt)#因为列表是可变类型,所以lt的值变化,lt2的值也会跟着变化print(l......
  • 拷贝对象的开源工具类-FastMapper-TinyMapper-Mapster
    至2023年10月,前两个项目的主要代码分别都有8年和6年历史了。Mapster最近还有修改FastMapperhttps://github.com/FastMapper/FastMapperTinyMapperhttps://github.com/TinyMapper/TinyMapperMapsterhttps://github.com/MapsterMapper/Mapster/Mapster应该只支持net6\net7,三者的......
  • 关于ladsh深拷贝的问题:cloneDeep()
    内容来源:99%的前端都不知道的lodash深拷贝的'BUG'-掘金(juejin.cn)问题代码import{cloneDeep}from'lodash'constpriceList=[1,2]constanimals={priceList}constoption={series:['dog','cat'].map(item=>animals)}......
  • 手写深拷贝
    深拷贝基本实现1深拷贝基本实现2functionisObject(value){3constvalueType=typeofvalue4return(value!==null)&&(valueType==='object'||valueType==='function')5}6functiondeepClone(originValue){7//判断传入的originValu......
  • 手写节流、深拷贝函数实现
    防抖函数基本实现1functiondebounce(fn,delay){2lettimer=null3returnfunction(...args){4if(timer)clearTimeout(timer)5timer=setTimeout(()=>{6fn.apply(this,args)7},delay);8}9}防抖函数完正版......
  • BeanUtils.copyProperties这个方法是深拷贝浅拷贝,还是深复制浅复制?
    1、关于BeanUtils.copyProperties方法的拷贝类型,它是浅拷贝哦。这意味着在拷贝对象时,它只会复制对象的引用而不会复制对象的内容。所以,如果原始对象中的某个属性发生改变,拷贝后的对象中的相应属性也会随之改变。当然,如果你希望进行深拷贝或深复制,需要使用其他方法来实现哦`BeanUt......
  • Python - 深拷贝一个带有指向自身引用的列表,会报错么?紧接着用==比较,会报错么?
    问题描述深拷贝一个带有指向自身引用的列表:列表x中有指向自身的引用,因此x是一个无限嵌套的列表。importcopyx=[1]x.append(x)>>x[1,[...]]y=copy.deepcopy(x)>>y[1,[...]] 深拷贝不报错但是我们发现深度拷贝x到y后,程序并没有出现stackoverf......
  • 浅谈深拷贝和浅拷贝
    深拷贝和浅拷贝深拷贝importcopylist1=[1,2,3,4,[5,6,7]]list2=copy.deepcopy(list1)print(list1)print(list2)[1,2,3,4,[5,6,7]][1,2,3,4,[5,6,7]]list1[0]+=1print(list1)print(list2)[2,2,3,4,[5,6,7]][1,2,3,4,[5,6,7]]list1[4]......
  • 关于C++拷贝控制
    通常来说,对于类内动态分配资源的类需要进行拷贝控制:要在拷贝构造函数、拷贝赋值运算符、析构函数中实现安全高效的操作来管理内存。但是资源管理并不是一个类需要定义自己的拷贝控制成员的唯一原因。C++Primer第5版中给出了一个Message类与Folder类的例子,分别表示电子邮件消息......