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

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

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

mmap原理詳解

科技綠洲 ? 來源:Linux開發(fā)架構(gòu)之路 ? 作者:Linux開發(fā)架構(gòu)之路 ? 2023-11-09 14:59 ? 次閱讀
  1. 一句話概括mmap

mmap的作用,在應(yīng)用這一層,是讓你把文件的某一段,當作內(nèi)存一樣來訪問。將文件映射到物理內(nèi)存,將進程虛擬空間映射到那塊內(nèi)存。

這樣,進程不僅能像訪問內(nèi)存一樣讀寫文件,多個進程映射同一文件,還能保證虛擬空間映射到同一塊物理內(nèi)存,達到內(nèi)存共享的作用。

  1. 虛擬內(nèi)存?虛擬空間?

其實是一個概念,前一篇對于這個詞沒有確切的定義,現(xiàn)在定義一下:

虛擬空間就是進程看到的所有地址組成的空間,虛擬空間是某個進程對分配給它的所有物理地址(已經(jīng)分配的和將會分配的)的重新映射。

而虛擬內(nèi)存,為啥叫虛擬內(nèi)存,是因為它就不是真正的內(nèi)存,是假的,因為它是由地址組成的空間,所以在這里,使用虛擬空間這個詞更加確切和易懂。(不過虛擬內(nèi)存這個詞也不算錯)

2.1 虛擬空間原理

2.1.1物理內(nèi)存

首先,物理地址實際上也不是連續(xù)的,通常是包含作為主存的DRAM和IO寄存器

圖片

以前的CPU(如X86)是為IO劃分單獨的地址空間,所以不能用直接訪問內(nèi)存的方式(如指針)IO,只能用專門的方法(in/read/out/write)諸如此類。

現(xiàn)在的CPU利用PCI總線將IO寄存器映射到物理內(nèi)存,所以出現(xiàn)了基于內(nèi)存訪問的IO。

還有一點補充的,就如同進程空間有一塊內(nèi)核空間一樣,物理內(nèi)存也會有極小一部分是不能訪問的,為內(nèi)核所用。

2.1.2三個總線

這里再補充下三個總線的知識,即:地址總線、數(shù)據(jù)總線、控制總線

  • 地址總線,用來傳輸?shù)刂?/li>
  • 數(shù)據(jù)總線,用來傳輸數(shù)據(jù)
  • 控制總線,用來傳輸命令

比如CPU通過控制總線發(fā)送讀取命令,同時用地址總線發(fā)送要讀取的數(shù)據(jù)虛地址,經(jīng)過MMU后到內(nèi)存

內(nèi)存通過數(shù)據(jù)總線將數(shù)據(jù)傳輸給CPU。

虛擬地址的空間和指令集的地址長度有關(guān),不一定和物理地址長度一致,比如現(xiàn)在的64位處理器,從VA角度看來,可以訪問64位的地址,但地址總線長度只有48位,所以你可以訪問一個位于2^52這個位置的地址。

2.1.3虛擬內(nèi)存地址轉(zhuǎn)換(虛地址轉(zhuǎn)實地址)

上面已經(jīng)明確了虛擬內(nèi)存是虛擬空間,即地址的集合這一概念?;诖耍瑏碚f說原理。

如果還記得操作系統(tǒng)課程里面提到的虛地址,那么這個虛地址就是虛擬空間的地址了,虛地址通過轉(zhuǎn)換得到實地址,轉(zhuǎn)換方式課程內(nèi)也講得很清楚,虛地址頭部包含了頁號(段地址和段大小,看存儲模式:頁存儲、段存儲,段頁式),剩下部分是偏移量,經(jīng)過MMU轉(zhuǎn)換成實地址。

圖片

存儲方式

圖片

如圖則是頁式存儲動態(tài)地址變換的方式

虛擬地址頭部為頁號通過查詢頁表得到物理頁號,假設(shè)一頁時1K,那么頁號*偏移量就得到物理地址

圖片

如圖所示,段式存儲

虛擬地址頭部為段號,段表中找到段基地址加上偏移量得到實地址

圖片

段頁式結(jié)合兩者,如圖所示。

  1. mmap映射

至此,如果對虛擬空間已經(jīng)了解了,那么接下來,作為coder,應(yīng)該自動把虛擬空間無視掉,因為Linux的目的也是要讓更多額進程能享用內(nèi)存,又不讓進程做麻煩的事情,是將虛擬空間和MMU都透明化,讓進程(和coder)只需要管對內(nèi)存怎樣使用。

所以現(xiàn)在開始不再強調(diào)虛擬空間了。

mmap就是將文件映射到內(nèi)存上,進程直接對內(nèi)存進行讀寫,然后就會反映到磁盤上。

圖片

  • 虛擬空間獲取到一段連續(xù)的地址
  • 在沒有讀寫的時候,這個地址指向不存在的地方(所以,上圖中起始地址和終止地址是還沒分配給 進程的)
  • 好了,根據(jù)偏移量,進程要讀文件數(shù)據(jù)了,數(shù)據(jù)占在兩個頁當中(物理內(nèi)存著色部分)
  • 這時,進程開始使用內(nèi)存了,所以O(shè)S給這兩個頁分配了內(nèi)存(即缺頁異常)(其余部分還是沒有分配)
  • 然后剛分配的頁內(nèi)是空的,所以再將相同偏移量的文件數(shù)據(jù)拷貝到物理內(nèi)存對應(yīng)頁上。
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 寄存器
    +關(guān)注

    關(guān)注

    31

    文章

    5271

    瀏覽量

    119654
  • 內(nèi)存
    +關(guān)注

    關(guān)注

    8

    文章

    2947

    瀏覽量

    73731
  • 文件
    +關(guān)注

    關(guān)注

    1

    文章

    555

    瀏覽量

    24639
收藏 人收藏

    評論

    相關(guān)推薦

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

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

    編譯例程partition_mmap,報錯no such vaddr range怎么解決?

    內(nèi)存映射問題:編譯例程partition_mmap,報錯no such vaddr range怎么解決?
    發(fā)表于 09-26 07:03

    Linux的mmap文件內(nèi)存映射機制

    Linux的mmap文件內(nèi)存映射機制在講述文件映射的概念時, 不可避免的要牽涉到虛存(SVR 4的VM). 實際上, 文件映射是虛存的中心概念, 文件映射一方面給用戶提供了一組措施, 好似用戶將文件
    發(fā)表于 03-08 09:54

    字符設(shè)備驅(qū)動另一種寫法—mmap方法操作LED

    。經(jīng)過自己的研究之后,我發(fā)現(xiàn)還有另外一種寫法,直接在應(yīng)用層操作,省去了內(nèi)核中的地址映射部分,使得用戶可以在應(yīng)用層直接操作LED。 mmap方法是把設(shè)備物理地址直接映射到用戶空間的一種系統(tǒng)調(diào)用方法,他使得
    發(fā)表于 01-02 17:38

    dma_alloc_coherent申請內(nèi)存的訪問速度,請問有什么辦法能加快訪問mmap的DMA內(nèi)存?

    使用dma_alloc_coherent申請了一塊內(nèi)存,然后使用mmap映射到用戶空間。然后,我用千兆網(wǎng)卡(CPSW驅(qū)動)進行發(fā)送(UDP方式),測量到的速度僅有12.5MB/s。 我又另外做了一個測試
    發(fā)表于 06-04 07:47

    mmap()函數(shù)映射到內(nèi)存中出現(xiàn)bus error的錯誤

    在2440開發(fā)板上將一副BMP圖片顯示到LCD上(不用GUI),我的做法是將BMP圖片用mmap()函數(shù)映射到內(nèi)存中,在將其讀到Framebuffer設(shè)備中顯示(frambuffer有驅(qū)動),映射
    發(fā)表于 02-25 12:42

    在arm里怎樣實現(xiàn)mmap編寫驅(qū)動和應(yīng)用共享內(nèi)存呢

    ② 確定屬性:是否使用 cache、buffer③ 建立映射關(guān)系在file_operation里面建立mmap進行mmap的函數(shù)編寫這樣在驅(qū)動程序的內(nèi)存空間就被建立了映射,用應(yīng)用程序訪問讀取,都行應(yīng)用程序
    發(fā)表于 05-17 09:59

    rt-smart qemu-vexpress-a9平臺mmap錯誤是什么原因?

    mmap系統(tǒng)調(diào)用傳進去的pgoffset有問題是什么原因?
    發(fā)表于 11-01 11:06

    linux_mmap_access_performance

    linux 內(nèi)存訪問提升性能的一片論文,需要理解kernel的mmap方式,比較適合優(yōu)化驅(qū)動
    發(fā)表于 02-23 15:48 ?14次下載

    mmap系統(tǒng)調(diào)用和vmalloc獲取地址空間

    mmap()系統(tǒng)調(diào)用是在用戶進程與內(nèi)核之間共享內(nèi)存區(qū)域的常用方法。我們最近有個程序,需要應(yīng)用進程能夠讀取內(nèi)核驅(qū)動獲取的數(shù)據(jù),經(jīng)過簡單的調(diào)研,決定采用mmap方式。
    的頭像 發(fā)表于 02-02 16:13 ?4255次閱讀

    mmap作為Linux內(nèi)存管理的關(guān)鍵之一

    mmap將一個文件或者其它對象映射進內(nèi)存。文件被映射到多個頁上,如果文件的大小不是所有頁的大小之和,最后一個頁不被使用的空間將會清零。munmap執(zhí)行相反的操作,刪除特定地址區(qū)域的對象映射。
    發(fā)表于 04-28 17:16 ?591次閱讀
    <b class='flag-5'>mmap</b>作為Linux內(nèi)存管理的關(guān)鍵之一

    linux drivers中的mmap實現(xiàn)

    將設(shè)備驅(qū)動內(nèi)核空間的內(nèi)存映射到用戶空間里,可以通過用戶空間中的mmap系統(tǒng)調(diào)用代替系統(tǒng)調(diào)用write和read。目的是提高讀寫效率。
    發(fā)表于 05-15 10:31 ?1583次閱讀

    Linux的mmap文件內(nèi)存映射機制

    講述mmap和munmap系統(tǒng)調(diào)用了. mmap調(diào)用實際上就是一個內(nèi)存對象vma的創(chuàng)建過程, mmap的調(diào)用格式是:  void * mmap(void *start, size_t
    發(fā)表于 04-02 14:35 ?409次閱讀

    一文詳細了解mmap內(nèi)存映射

    mmap是一種內(nèi)存映射的方法,這一功能可以用在文件的處理上,即將一個文件或者其它對象映射到進程的地址空間,實現(xiàn)文件磁盤地址和進程虛擬地址空間中一段虛擬地址的一一對映關(guān)系。在編程時可以使某個磁盤文件
    的頭像 發(fā)表于 05-05 15:32 ?2304次閱讀

    內(nèi)核mmap_sem鎖的危害和相關(guān)優(yōu)化

    mmap_sem鎖是進程為了保護自身虛擬地址空間不受多線程并發(fā)訪問影響而設(shè)計的。
    的頭像 發(fā)表于 02-07 16:01 ?645次閱讀