0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

通過mmap實(shí)現(xiàn)零拷貝技術(shù)

Linux閱碼場(chǎng) ? 來源:Linux內(nèi)核遠(yuǎn)航者 ? 作者:Linux內(nèi)核遠(yuǎn)航者 ? 2022-06-28 17:33 ? 次閱讀

1.開場(chǎng)白

  • 環(huán)境: 處理器架構(gòu):arm64 內(nèi)核源碼:linux-5.11 ubuntu版本:20.04.1 代碼閱讀工具:vim+ctags+cscope

我們知道,linux系統(tǒng)中用戶空間和內(nèi)核空間是隔離的,用戶空間程序不能隨意的訪問內(nèi)核空間數(shù)據(jù),只能通過中斷或者異常的方式進(jìn)入內(nèi)核態(tài),一般情況下,我們使用copy_to_user和copy_from_user等內(nèi)核api來實(shí)現(xiàn)用戶空間和內(nèi)核空間的數(shù)據(jù)拷貝,但是像顯存這樣的設(shè)備如果也采用這樣的方式就顯的效率非常底下,因?yàn)橛脩艚?jīng)常需要在屏幕上進(jìn)行繪制,要消除這種復(fù)制的操作就需要應(yīng)用程序直接能夠訪問顯存,但是顯存被映射到內(nèi)核空間,應(yīng)用程序是沒有訪問權(quán)限的,如果顯存也能同時(shí)映射到用戶空間那就不需要拷貝操作了,于是字符設(shè)備中提供了mmap接口,可以將內(nèi)核空間映射的那塊物理內(nèi)存再次映射到用戶空間,這樣用戶空間就可以直接訪問不需要任何拷貝操作,這就是我們今天要說的0拷貝技術(shù)。

下面是正常情況下用戶空間和內(nèi)核空間數(shù)據(jù)訪問圖示:

d0997780-f677-11ec-ba43-dac502259ad0.png

2. 體驗(yàn)一下

首先我們通過一個(gè)例子來感受一下:

驅(qū)動(dòng)代碼:

注:驅(qū)動(dòng)代碼中使用misc框架來實(shí)現(xiàn)字符設(shè)備,misc框架會(huì)處理如創(chuàng)建字符設(shè)備,創(chuàng)建設(shè)備等通用的字符設(shè)備處理,我們只需要關(guān)心我們的實(shí)際的邏輯即可(內(nèi)核中大量使用misc設(shè)備框架來使用字符設(shè)備操作集如ioctl接口,像實(shí)現(xiàn)系統(tǒng)虛擬化kvm模塊,實(shí)現(xiàn)安卓進(jìn)程間通信的binder模塊等)。

0copy_demo.c

#include
#include
#include
#include
#include


#defineMISC_DEV_MINOR5

staticchar*kbuff;


staticssize_tmisc_dev_read(structfile*filep,char__user*buf,size_tcount,loff_t*offset)
{

intret;

size_tlen=(count>PAGE_SIZE?PAGE_SIZE:count);

pr_info("######%s:%dkbuff:%s######
",__func__,__LINE__,kbuff);

ret=copy_to_user(buf,kbuff,len);//這里使用copy_to_user來進(jìn)程內(nèi)核空間到用戶空間拷貝

returnlen-ret;
}

staticssize_tmisc_dev_write(structfile*filep,constchar__user*buf,size_tcount,loff_t*offset)
{
pr_info("######%s:%d######
",__func__,__LINE__);
return0;
}

staticintmisc_dev_mmap(structfile*filep,structvm_area_struct*vma)
{
intret;
unsignedlongstart;

start=vma->vm_start;

ret=remap_pfn_range(vma,start,virt_to_phys(kbuff)>>PAGE_SHIFT,
PAGE_SIZE,vma->vm_page_prot);//使用remap_pfn_range來映射物理頁面到進(jìn)程的虛擬內(nèi)存中virt_to_phys(kbuff)>>PAGE_SHIFT作用是將內(nèi)核的虛擬地址轉(zhuǎn)化為實(shí)際的物理地址頁幀號(hào)創(chuàng)建頁表的權(quán)限為通過mmap傳遞的vma->vm_page_prot映射大小為1頁

returnret;
}

staticlongmisc_dev_ioctl(structfile*filep,unsignedintcmd,unsignedlongargs)
{
pr_info("######%s:%d######
",__func__,__LINE__);
return0;
}



staticintmisc_dev_open(structinode*inodep,structfile*filep)
{
pr_info("######%s:%d######
",__func__,__LINE__);
return0;
}

staticintmisc_dev_release(structinode*inodep,structfile*filep)
{
pr_info("######%s:%d######
",__func__,__LINE__);
return0;
}


staticstructfile_operationsmisc_dev_fops={
.open=misc_dev_open,
.release=misc_dev_release,
.read=misc_dev_read,
.write=misc_dev_write,
.unlocked_ioctl=misc_dev_ioctl,
.mmap=misc_dev_mmap,
};

staticstructmiscdevicemisc_dev={
MISC_DEV_MINOR,
"misc_dev",
&misc_dev_fops,
};

staticint__initmisc_demo_init(void)
{
misc_register(&misc_dev);//注冊(cè)misc設(shè)備(讓misc來幫我們處理創(chuàng)建字符設(shè)備的通用代碼,這樣我們就不需要在去做這些和我們的實(shí)際邏輯無關(guān)的代碼處理了)


kbuff=(char*)__get_free_page(GFP_KERNEL);//申請(qǐng)一個(gè)物理頁面(返回對(duì)應(yīng)的內(nèi)核虛擬地址,內(nèi)核初始化的時(shí)候會(huì)做線性映射,將整個(gè)ddr內(nèi)存映射到線性映射區(qū),所以我們不需要做頁表映射)
if(NULL==kbuff)
return-ENOMEM;

pr_info("######%s:%d######
",__func__,__LINE__);
return0;
}

staticvoid__exitmisc_demo_exit(void)
{
free_page((unsignedlong)kbuff);

misc_deregister(&misc_dev);
pr_info("######%s:%d######
",__func__,__LINE__);
}

module_init(misc_demo_init);
module_exit(misc_demo_exit);
MODULE_LICENSE("GPL");

應(yīng)用代碼:test.c

#include
#include
#include
#include
#include
#include
#include



intmain(intargc,char**argv)
{

intfd;
char*ptr;
charbuff[32];

fd=open("/dev/misc_dev",O_RDWR);//打開字符設(shè)備
if(fd"failtoopen");
return-1;
}

ptr=mmap(NULL,4096,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0);//映射字符設(shè)備到進(jìn)程的地址空間權(quán)限為可讀可寫映射為共享大小為一個(gè)頁面
if(ptr==MAP_FAILED){
perror("failtommap");
return-1;
}


memcpy(ptr,"helloworld!!!", 15);//寫mmap映射的內(nèi)存直接操作,不需要進(jìn)行特權(quán)級(jí)別的陷入!


if(read(fd,buff,15)==-1){//讀接口來讀取映射的內(nèi)存,這里會(huì)進(jìn)行內(nèi)核空間到用戶空間的數(shù)據(jù)拷貝(需要調(diào)用系統(tǒng)調(diào)用在內(nèi)核空間進(jìn)行拷貝,然后才能訪問)
perror("failtoread");
return-1;
}
puts(buff);

pause();
return0;
}

Makefile文件:

exportARCH=arm64
exportCROSS_COMPILE=aarch64-linux-gnu-

KERNEL_DIR?=~/kernel/linux-5.11
obj-m:=0copy_demo.o

modules:
$(MAKE)-C$(KERNEL_DIR)M=$(PWD)modules

app:
aarch64-linux-gnu-gcctest.c-otest
cptest$(KERNEL_DIR)/kmodules

clean:
$(MAKE)-C$(KERNEL_DIR)M=$(PWD)clean

install:
cp*.ko$(KERNEL_DIR)/kmodules

編譯驅(qū)動(dòng)代碼和應(yīng)用代碼,然后拷貝到qemu中運(yùn)行:

編譯驅(qū)動(dòng)模塊代碼:
$makemodules

編譯并拷貝應(yīng)用:
$makeapp

拷貝驅(qū)動(dòng)模塊到qemu:
$makeinstall

加載驅(qū)動(dòng)代碼:
#insmod0copy_demo.ko
[23328.532194]######misc_demo_init:91######

查看生成的設(shè)備節(jié)點(diǎn):
#ls-l/dev/misc_dev
crw-rw----10010,5Apr719:26/dev/misc_dev

后臺(tái)運(yùn)行應(yīng)用程序:
#./test&
#[23415.280501]######misc_dev_open:56######
[23415.281052]######misc_dev_read:20kbuff:helloworld!!!######
helloworld!!!

查看test的pid:
#pidoftest
1768


查看內(nèi)存映射:
#cat/proc/1768/maps
aaaabc5a0000-aaaabc5a1000r-xp0000000000:198666193/mnt/test
aaaabc5b0000-aaaabc5b1000r--p0000000000:198666193/mnt/test
aaaabc5b1000-aaaabc5b2000rw-p0000100000:198666193/mnt/test
aaaacf033000-aaaacf054000rw-p0000000000:000[heap]
ffff8a911000-ffff8aa52000r-xp00000000fe:00152/lib/libc-2.27.so
ffff8aa52000-ffff8aa61000---p00141000fe:00152/lib/libc-2.27.so
ffff8aa61000-ffff8aa65000r--p00140000fe:00152/lib/libc-2.27.so
ffff8aa65000-ffff8aa67000rw-p00144000fe:00152/lib/libc-2.27.so
ffff8aa67000-ffff8aa6b000rw-p0000000000:000
ffff8aa6b000-ffff8aa88000r-xp00000000fe:00129/lib/ld-2.27.so
ffff8aa91000-ffff8aa92000rw-s0000000000:05152/dev/misc_dev//映射設(shè)備文件到用戶空間
ffff8aa92000-ffff8aa94000rw-p0000000000:000
ffff8aa94000-ffff8aa96000r--p0000000000:000[vvar]
ffff8aa96000-ffff8aa97000r-xp0000000000:000[vdso]
ffff8aa97000-ffff8aa98000r--p0001c000fe:00129/lib/ld-2.27.so
ffff8aa98000-ffff8aa9a000rw-p0001d000fe:00129/lib/ld-2.27.so
ffffecb5a000-ffffecb7b000rw-p0000000000:000[stack]

執(zhí)行了以上步驟可以發(fā)現(xiàn)最終內(nèi)核中出現(xiàn)了我在應(yīng)用程序中寫入的“hello world!!!“ 字符串,應(yīng)用程序也能成功讀取到(當(dāng)然本文講解的0拷貝實(shí)現(xiàn)的驅(qū)動(dòng)接口是mmap,而我們讀取使用的是read接口,里面我們用copy_to_user來實(shí)現(xiàn)的,當(dāng)然我們可以直接操作mmap映射的內(nèi)存不需要任何拷貝操作)。

查看應(yīng)用程序的內(nèi)存映射發(fā)現(xiàn),/dev/misc_dev設(shè)備被映射到了ffff8aa91000-ffff8aa92000這段用戶空間地址范圍,而且權(quán)限為rw-s(可讀可寫共享)。

寫到這里可能大家還是有點(diǎn)不明白那我來解釋下:

1.用戶空間不能直接訪問內(nèi)核空間數(shù)據(jù)(不能直接讀寫),一旦訪問發(fā)生缺頁異常,產(chǎn)生段錯(cuò)誤,必須通過read這樣的接口來訪問,而read這樣的接口會(huì)通過系統(tǒng)調(diào)用的方式寫入到內(nèi)核態(tài),然后通過copy_to_user這樣的內(nèi)核api來拷貝內(nèi)核空間數(shù)據(jù)到用戶空間之后才能正常訪問。

2.通過mmap這種方式之后,用戶進(jìn)程可以直接訪問這塊內(nèi)存,memcpy訪問的也只不過是用戶空間地址,由于訪問的時(shí)候已經(jīng)分配好了物理頁面和建立好了物理頁到虛擬頁的映射,所有不會(huì)發(fā)生缺頁異常,也不會(huì)發(fā)生用戶態(tài)到內(nèi)核態(tài)的陷入動(dòng)作。

3.用戶態(tài)進(jìn)程正常訪問內(nèi)核態(tài)數(shù)據(jù)需要首先通過系統(tǒng)調(diào)用等方式陷入內(nèi)核,進(jìn)行數(shù)據(jù)拷貝,然后再次回到用戶態(tài),用戶態(tài)和內(nèi)核態(tài)直接的進(jìn)出需要進(jìn)行上下文切換,需要2次上下文切換,需要一定的開銷,而mmap映射好之后以后訪問都不需要進(jìn)行上下文切換。

4.mmap映射這種方法由于物理頁面通過頁面共享更加節(jié)省內(nèi)存,而用戶態(tài)和內(nèi)核態(tài)內(nèi)存拷貝需要兩份物理頁面。

3.實(shí)現(xiàn)原理

我們發(fā)現(xiàn)通過mmap映射之后,我們?cè)趹?yīng)用程序中可以直接讀寫這段內(nèi)存,不需要任何用戶空間和內(nèi)核空間的拷貝動(dòng)作,大大提高了內(nèi)存訪問效率,那么就是是如何實(shí)現(xiàn)的呢?下面我們來揭開它神秘的面紗:

實(shí)現(xiàn)0拷貝功不可沒的是mmap接口中的remap_pfn_range內(nèi)核api,它將內(nèi)核空間映射的物理內(nèi)存重新映射到了用戶空間,下面我們來看這個(gè)函數(shù)的實(shí)現(xiàn):remap_pfn_range函數(shù)參數(shù)如下:

intremap_pfn_range(structvm_area_struct*vma,unsignedlongaddr,
|unsignedlongpfn,unsignedlongsize,pgprot_tprot)

vma為需要映射的進(jìn)程的vma(進(jìn)程調(diào)用mmap的時(shí)候內(nèi)核會(huì)找到一個(gè)合適的vma), addr為vma中的一個(gè)起始映射地址(這是用戶空間的一個(gè)虛擬地址),pfn為頁幀號(hào)(在驅(qū)動(dòng)的mmap接口中會(huì)將內(nèi)核空間的地址轉(zhuǎn)化為物理地址的頁幀號(hào)),size為需要映射的大小,prot為映射的權(quán)限(一般取mmap時(shí)傳遞的權(quán)限如rw)

remap_pfn_range實(shí)現(xiàn)主要如下代碼段:

remap_pfn_range
...
pgd=pgd_offset(mm,addr);
flush_cache_range(vma,addr,end);
do{
next=pgd_addr_end(addr,end);
err=remap_p4d_range(mm,pgd,addr,next,
pfn+(addr>>PAGE_SHIFT),prot);
if(err)
break;
}while(pgd++,addr=next,addr!=end);

解釋下:remap_pfn_range函數(shù)會(huì)查找進(jìn)程的頁表,然后填寫頁表,會(huì)將映射的物理頁幀號(hào)和訪問權(quán)限填寫到進(jìn)程的對(duì)應(yīng)頁表中,這會(huì)遍歷進(jìn)程的各級(jí)頁表找到最終的頁表項(xiàng)然后進(jìn)行填寫,具體過程自行查看代碼。

我們需要注意的是:

1.一般情況下,用戶程序調(diào)用mmap只是申請(qǐng)?zhí)摂M內(nèi)存(即是獲得一塊沒有使用用戶空間內(nèi)存,使用vma描述),實(shí)際的物理頁表都是通過進(jìn)程訪問的時(shí)候缺頁異常的方式來申請(qǐng)的,但是本場(chǎng)景中是物理頁面已經(jīng)申請(qǐng)好了,進(jìn)程訪問時(shí)不會(huì)再發(fā)生缺頁異常,不會(huì)申請(qǐng)物理頁面。

2.同樣,物理頁面到用戶空間虛擬頁面的映射也在調(diào)用mmap的時(shí)候,驅(qū)動(dòng)調(diào)用mmap接口的remap_pfn_range映射好了,也不需要在訪問的時(shí)候發(fā)生缺頁異常來建立映射。所以,只要用戶進(jìn)程通過mmap映射之后就可以正常訪問,訪問過程中不會(huì)發(fā)生缺頁異常,映射虛擬頁對(duì)應(yīng)的物理頁面已經(jīng)在驅(qū)動(dòng)中申請(qǐng)好映射好。

下面給出mmap映射原理的圖示:

d0adc1d6-f677-11ec-ba43-dac502259ad0.png


4.應(yīng)用場(chǎng)景

最后,我們來看下使用framebuffer的lcd對(duì)0拷貝的使用情況

fbmem_init//drivers/video/fbdev/core/fbmem.c
->register_chrdev(FB_MAJOR,"fb",&fb_fops)//注冊(cè)framebuffer字符設(shè)備

->structfile_operationsfb_fops={
->.mmap=fb_mmap
->fb_mmap//framebuffer的實(shí)現(xiàn)
->vm_iomap_memory
->io_remap_pfn_range
->remap_pfn_range

->fb_class=class_create(THIS_MODULE,"graphics")//創(chuàng)建設(shè)備類

lcd驅(qū)動(dòng)代碼中會(huì)設(shè)置好最終注冊(cè)framebuffer:

xxxfb_probe
->register_framebuffer
->do_register_framebuffer
->fb_info->dev=device_create(fb_class,fb_info->device,
|MKDEV(FB_MAJOR,i),NULL,"fb%d",i);//創(chuàng)建設(shè)備會(huì)出現(xiàn)/dev/fdx設(shè)備節(jié)點(diǎn)

可以看到當(dāng)系統(tǒng)支持framebuffer設(shè)備時(shí),在fbmem_init中會(huì)創(chuàng)建framebuffer設(shè)備類關(guān)聯(lián)字符設(shè)備操作集fb_fops,lcd的驅(qū)動(dòng)代碼中會(huì)調(diào)用register_framebuffer創(chuàng)建framebuffer設(shè)備(就會(huì)創(chuàng)建出了/dev/fdx 設(shè)備節(jié)點(diǎn)),應(yīng)用程序就可以通過mmap來映射framebuffer設(shè)備到用戶空間,然后進(jìn)行屏幕繪制操作,不需要任何數(shù)據(jù)拷貝。

5.總結(jié)

可以看的出,通過mmap實(shí)現(xiàn)0拷貝非常簡(jiǎn)單,只需要在驅(qū)動(dòng)的mmap接口中調(diào)用remap_pfn_range來將內(nèi)核空間映射的那塊物理頁再次映射到用戶空間即可,這就實(shí)現(xiàn)了用戶空間和內(nèi)核空間的數(shù)據(jù)共享,這和用戶進(jìn)程之間的共享內(nèi)存機(jī)制非常相似,都需要操作進(jìn)程的頁表將這段物理內(nèi)存映射到進(jìn)程虛擬地址空間。

原文標(biāo)題:5.總結(jié)

文章出處:【微信公眾號(hào):Linux閱碼場(chǎng)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

審核編輯:湯梓紅
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 處理器
    +關(guān)注

    關(guān)注

    68

    文章

    18927

    瀏覽量

    227234
  • 內(nèi)核
    +關(guān)注

    關(guān)注

    3

    文章

    1336

    瀏覽量

    40084
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11123

    瀏覽量

    207919
  • 內(nèi)存映射
    +關(guān)注

    關(guān)注

    0

    文章

    14

    瀏覽量

    7388

原文標(biāo)題:5.總結(jié)

文章出處:【微信號(hào):LinuxDev,微信公眾號(hào):Linux閱碼場(chǎng)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    無線室內(nèi)定位系統(tǒng)是通過什么技術(shù)實(shí)現(xiàn)的呢?

    無線室內(nèi)定位系統(tǒng)作為現(xiàn)代科技的重要組成部分,廣泛應(yīng)用于智能家居、企業(yè)管理、醫(yī)療護(hù)理等多個(gè)領(lǐng)域。這些系統(tǒng)通過不同的技術(shù)手段實(shí)現(xiàn)高精度的室內(nèi)定位,為用戶帶來便捷與高效。那么,無線室內(nèi)定位系統(tǒng)究竟是
    的頭像 發(fā)表于 09-10 09:28 ?175次閱讀
    無線室內(nèi)定位系統(tǒng)是<b class='flag-5'>通過</b>什么<b class='flag-5'>技術(shù)</b><b class='flag-5'>實(shí)現(xiàn)</b>的呢?

    拆解mmap內(nèi)存映射的本質(zhì)!

    mmap 內(nèi)存映射里所謂的內(nèi)存其實(shí)指的是虛擬內(nèi)存,在調(diào)用 mmap 進(jìn)行匿名映射的時(shí)候(比如進(jìn)行堆內(nèi)存的分配),是將進(jìn)程虛擬內(nèi)存空間中的某一段虛擬內(nèi)存區(qū)域與物理內(nèi)存中的匿名內(nèi)存頁進(jìn)行映射,當(dāng)調(diào)用
    的頭像 發(fā)表于 01-24 14:30 ?914次閱讀
    拆解<b class='flag-5'>mmap</b>內(nèi)存映射的本質(zhì)!

    如何評(píng)價(jià)跑汽車的NAC技術(shù)?NAC技術(shù)是怎么出現(xiàn)的?

    日前,跑汽車推出了一項(xiàng)讓駕駛員體驗(yàn)“科目二”駕駛樂趣的技術(shù),這項(xiàng)名為NAC的駕駛技術(shù)可以實(shí)現(xiàn)系統(tǒng)控制油門剎車,駕駛員控制方向。
    的頭像 發(fā)表于 01-16 16:53 ?1172次閱讀
    如何評(píng)價(jià)<b class='flag-5'>零</b>跑汽車的NAC<b class='flag-5'>技術(shù)</b>?NAC<b class='flag-5'>技術(shù)</b>是怎么出現(xiàn)的?

    如何通過Matlab進(jìn)行極點(diǎn)求解?

     對(duì)于包含變量的傳遞函數(shù),如果其是二階的,利用Matlab求極點(diǎn)表達(dá)式是容易實(shí)現(xiàn)的。但對(duì)于更高階的系統(tǒng)而言,想通過Matlab來求得解析解是極其困難的。
    的頭像 發(fā)表于 11-30 12:23 ?3375次閱讀
    如何<b class='flag-5'>通過</b>Matlab進(jìn)行<b class='flag-5'>零</b>極點(diǎn)求解?

    機(jī)器人拷貝數(shù)據(jù)傳輸編程開發(fā)

    編程開發(fā) 為了方便大家使用,TogetherROS針對(duì)拷貝功能進(jìn)行了封裝,風(fēng)格類似ROS2中話題通信的接口,還是話題通信一樣的流程,我們只需要修改幾個(gè)函數(shù)就可以實(shí)現(xiàn)啦。 運(yùn)行例程 $ source
    的頭像 發(fā)表于 11-27 16:55 ?378次閱讀
    機(jī)器人<b class='flag-5'>零</b><b class='flag-5'>拷貝</b>數(shù)據(jù)傳輸編程開發(fā)

    如何進(jìn)行拷貝性能測(cè)試

    TogetherROS?·Bot拷貝性能測(cè)試 我們使用TogetherROS?·Bot系統(tǒng)內(nèi)部集成的性能測(cè)試工具——performance_test,來評(píng)估下開啟拷貝前后的性能差異
    的頭像 發(fā)表于 11-27 16:51 ?341次閱讀
    如何進(jìn)行<b class='flag-5'>零</b><b class='flag-5'>拷貝</b>性能測(cè)試

    什么是拷貝技術(shù)

    的應(yīng)用中,經(jīng)常會(huì)發(fā)生這種問題,導(dǎo)致CPU都在做數(shù)據(jù)拷貝,沒有時(shí)間處理其他的應(yīng)用功能了,直接的感覺就是處理卡頓。 拷貝技術(shù) 針對(duì)這種問題,
    的頭像 發(fā)表于 11-27 16:20 ?377次閱讀
    什么是<b class='flag-5'>零</b><b class='flag-5'>拷貝</b><b class='flag-5'>技術(shù)</b>

    通過輻射發(fā)射測(cè)試:如何避免采用復(fù)雜的EMI抑制技術(shù)實(shí)現(xiàn)緊湊、高性價(jià)比的隔離設(shè)計(jì)

    電子發(fā)燒友網(wǎng)站提供《通過輻射發(fā)射測(cè)試:如何避免采用復(fù)雜的EMI抑制技術(shù)實(shí)現(xiàn)緊湊、高性價(jià)比的隔離設(shè)計(jì).pdf》資料免費(fèi)下載
    發(fā)表于 11-22 10:32 ?0次下載
    <b class='flag-5'>通過</b>輻射發(fā)射測(cè)試:如何避免采用復(fù)雜的EMI抑制<b class='flag-5'>技術(shù)</b>以<b class='flag-5'>實(shí)現(xiàn)</b>緊湊、高性價(jià)比的隔離設(shè)計(jì)

    mmap原理詳解

    一句話概括mmap mmap的作用,在應(yīng)用這一層,是讓你把文件的某一段,當(dāng)作內(nèi)存一樣來訪問。將文件映射到物理內(nèi)存,將進(jìn)程虛擬空間映射到那塊內(nèi)存。 這樣,進(jìn)程不僅能像訪問內(nèi)存一樣讀寫文件,多個(gè)進(jìn)程映射
    的頭像 發(fā)表于 11-09 14:59 ?563次閱讀
    <b class='flag-5'>mmap</b>原理詳解

    傳統(tǒng)IO演化至拷貝的過程

    拷貝(Zero-Copy)用于在數(shù)據(jù)讀寫過程中減少不需要的CPU拷貝,CPU就那么幾個(gè),減少它的負(fù)擔(dān)自然可以提高處理效率。數(shù)據(jù)傳輸有本地的文件拷貝
    的頭像 發(fā)表于 11-09 09:09 ?357次閱讀
    傳統(tǒng)IO演化至<b class='flag-5'>零</b><b class='flag-5'>拷貝</b>的過程

    Python中淺拷貝與深拷貝的操作

    【例子】淺拷貝與深拷貝中 list1 = [ 123 , 456 , 789 , 213 ]list2 = list1list3 = list1[:] print (list2) # [ 123
    的頭像 發(fā)表于 11-02 10:58 ?315次閱讀

    433超強(qiáng)拷貝滾動(dòng)碼遙控器解決方案

    新的遙控器。這個(gè)時(shí)候就可以用上拷貝滾動(dòng)碼遙控器了,只需要清除對(duì)碼,拷貝對(duì)碼就可以, 使用非常方便。 并且由于滾動(dòng)碼遙控器的復(fù)雜性,它們通常比固定碼遙控器更貴。如果需要增添遙控器,也是可以使用我們的拷貝滾動(dòng)碼遙控器
    的頭像 發(fā)表于 10-27 19:35 ?2394次閱讀
    433超強(qiáng)<b class='flag-5'>拷貝</b>滾動(dòng)碼遙控器解決方案

    超強(qiáng)拷貝滾動(dòng)碼遙控器,兼容性好,反應(yīng)靈敏,配對(duì)簡(jiǎn)單

    拷貝滾動(dòng)碼遙控器是一種能夠復(fù)制并拷貝原遙控器數(shù)據(jù)的遙控器。本方案采用 性能超強(qiáng)的 ARM-32 位單片機(jī)XL32F002A ,可拷貝目前市場(chǎng)上絕大多數(shù)常見品牌遙控器,并替代原遙控器使用。 可用于替代
    的頭像 發(fā)表于 10-20 01:00 ?1802次閱讀
    超強(qiáng)<b class='flag-5'>拷貝</b>滾動(dòng)碼遙控器,兼容性好,反應(yīng)靈敏,配對(duì)簡(jiǎn)單

    什么是深拷貝和淺拷貝

    ("com.ys.test.Person").newInstance(); ③、通過 Constructor 類的 newInstance 方法 這和第二種方法類都是通過反射來實(shí)現(xiàn)通過
    的頭像 發(fā)表于 10-13 11:29 ?563次閱讀
    什么是深<b class='flag-5'>拷貝</b>和淺<b class='flag-5'>拷貝</b>

    關(guān)于STM32的字庫拷貝函數(shù)

    關(guān)于STM32的字庫拷貝函數(shù)(超實(shí)用)
    發(fā)表于 09-26 08:12