<acronym id="s8ci2"><small id="s8ci2"></small></acronym>
<rt id="s8ci2"></rt><rt id="s8ci2"><optgroup id="s8ci2"></optgroup></rt>
<acronym id="s8ci2"></acronym>
<acronym id="s8ci2"><center id="s8ci2"></center></acronym>
0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

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

3天內不再提示

鴻蒙內核源碼分析:關于內存涉及的C7,C2,C13三個寄存器

鴻蒙系統HarmonyOS ? 來源:oschina. ? 作者:深入研究鴻蒙 ? 2020-10-29 10:41 ? 次閱讀

ARM-CP15協處理器

ARM處理器使用協處理器15(CP15)的寄存器來控制cache、TCM和存儲器管理。CP15的寄存器只能被MRC和MCR(Move to Coprocessor from ARM Register )指令訪問,包含16個32位的寄存器,其編號為0~15。本篇重點講解其中的 C7,C2,C13三個寄存器。

拆解一段匯編代碼

上來看段匯編,讀懂內核源碼不會點匯編是不行的 , 但不用發怵,沒那么恐怖,由淺入深, 內核其實挺好玩的。見于 arm.h,里面全是這些玩意。

#define DSB __asm__ volatile("dsb" ::: "memory")
#define ISB __asm__ volatile("isb" ::: "memory")
#define DMB __asm__ volatile("dmb" ::: "memory")

STATIC INLINE VOID OsArmWriteBpiallis(UINT32 val)
{
    __asm__ volatile("mcr p15, 0, %0, c7,c1,6" ::"r"(val));
    __asm__ volatile("isb" ::: "memory");
}
指 令 說 明 語法格式
mcr 將ARM處理器的寄存器中的數據寫到CP15中的寄存器中 mcr{} p15, , , , , {
mrc 將CP15中的寄存器中的數據讀到ARM處理器的寄存器中 mcr{} p15, , , , , {

cond:為指令執行的條件碼。當cond忽略時指令為無條件執行。
Opcode_1:協處理器的特定操作碼. 對于CP15寄存器來說,opcode1=0
Rd:作為源寄存器的ARM寄存器,其值將被傳送到協處理器寄存器中,或者將協處理器寄存器的值傳送到該寄存器里面 ,通常為R0
CRn:作為目標寄存器的協處理器寄存器,其編號是C~C15。
CRm:協處理器中附加的目標寄存器或源操作數寄存器。如果不需要設置附加信息,將CRm設置為c0,否則結果未知
Opcode_2:可選的協處理器特定操作碼。(用來區分同一個編號的不同物理寄存器,當不需要提供附加信息時,指定為0

這句匯編的指令字面意思是: 將ARM寄存器R0的數據寫到CP15中編號為7的寄存器中,值由外面傳進來。

例如 OsArmWriteBpiallis(0)做了4個動作

1.把0值寫入R0寄存器,注意這個寄存器是ARM即CPU的寄存器,::"r"(val)意思代表向GCC編譯器聲明,會修改R0寄存器的值,改之前提前打好招呼,都是紳士文明人。其實編譯器的功能是非常強大的,不僅僅是大家普遍認為的只是編譯代碼的工具而已。

2.volatile的意思還是告訴編譯器,不要去優化這段代碼,原封不動的生成目標指令。

3."isb" ::: "memory" 還是告訴編譯器內存的內容可能被更改了,需要無效所有Cache,并訪問實際的內容,而不是Cache!

4.再把R0的值寫入到C7中,C7是CP15協處理器的寄存器。C7寄存器是負責什么的?對照下面的表。

CP15有哪些寄存器

寄存器編號 基本作用 在MMU中的作用 在PU中的作用
0 ID編碼(只讀) ID編碼和cache類型
1 控制位(可讀寫) 各種控制位
2 存儲保護和控制 地址轉換表基地址 Cachability的控制位
3 存儲保護和控制 域訪問控制位 Bufferablity控制位
4 存儲保護和控制 保留 保留
5 存儲保護和控制 內存失效狀態 訪問權限控制位
6 存儲保護和控制 內存失效地址 保護區域控制
7 高速緩存和寫緩存 高速緩存和寫緩存控制
8 存儲保護和控制 TLB控制 保留
9 高速緩存和寫緩存 高速緩存鎖定
10 存儲保護和控制 TLB鎖定 保留
11 保留
12 保留
13 進程標識符 進程標識符
14 保留
15 因不同設計而異 因不同設計而異 因不同設計而異

這句話真正的意思是:關閉高速緩存和寫緩存控制!,其他部分寄存器下面會講,先有個大概印象。

mmu從哪里獲取 page table 的信息?答案是: TTB

TTB寄存器(Translation table base)

參考上表可知TTB寄存器是CP15協處理器的C2寄存器,存頁表的基地址,即一級映射描述符表的基地址。圍繞著TTB鴻蒙提供了以下讀取函數。簡單說就是內核從外面不斷的修改和讀取寄存器值,而MMU只會直接通過硬件讀取這個寄存器的值,以達到MMU獲取不一樣的頁表進行進程虛擬地址和物理地址的轉換。還記得嗎?每個進程的頁表都是獨立的!

那么什么情況下會修改里面的值呢?換頁表意味著mmu在進行上下文的切換!還是直接看代碼吧。

mmu上下文

只被這一個函數調用。毫無疑問LOS_ArchMmuContextSwitch是關鍵函數。

typedef struct ArchMmu {
    LosMux              mtx;            /**< arch mmu page table entry modification mutex lock */
    VADDR_T             *virtTtb;       /**< translation table base virtual addr */
    PADDR_T             physTtb;        /**< translation table base phys addr */
    UINT32              asid;           /**< TLB asid */
    LOS_DL_LIST         ptList;         /**< page table vm page list */
} LosArchMmu;

// mmu 上下文切換
VOID LOS_ArchMmuContextSwitch(LosArchMmu *archMmu)
{
    UINT32 ttbr;
    UINT32 ttbcr = OsArmReadTtbcr();//讀取TTB寄存器的狀態值
    if (archMmu) {
        ttbr = MMU_TTBRx_FLAGS | (archMmu->physTtb);//進程TTB物理地址值
        /* enable TTBR0 */
        ttbcr &= ~MMU_DESCRIPTOR_TTBCR_PD0;//使能TTBR0
    } else {
        ttbr = 0;
        /* disable TTBR0 */
        ttbcr |= MMU_DESCRIPTOR_TTBCR_PD0;
    }

    /* from armv7a arm B3.10.4, we should do synchronization changes of ASID and TTBR. */
    OsArmWriteContextidr(LOS_GetKVmSpace()->archMmu.asid);//這里先把asid切到內核空間的ID
    ISB;
    OsArmWriteTtbr0(ttbr);//通過r0寄存器將進程頁面基址寫入TTB
    ISB;
    OsArmWriteTtbcr(ttbcr);//寫入TTB狀態位
    ISB;
    if (archMmu) {
        OsArmWriteContextidr(archMmu->asid);//通過R0寄存器寫入進程標識符至C13寄存器
        ISB;
    }
}
// c13 asid(Adress Space ID)進程標識符
STATIC INLINE VOID OsArmWriteContextidr(UINT32 val)
{
    __asm__ volatile("mcr p15, 0, %0, c13,c0,1" ::"r"(val));
    __asm__ volatile("isb" ::: "memory");
}

再看下那些地方會調用LOS_ArchMmuContextSwitch,下圖一目了然。

有四個地方會切換mmu上下文

第一:通過調度算法,被選中的進程的空間改變了,自然映射頁表就跟著變了,需要切換mmu上下文,還是直接看代碼。代碼不是很多,就都貼出來了,都加了注釋,不記得調度算法的可去系列篇中看鴻蒙內核源碼分析(調度機制篇),里面有詳細的闡述。

//調度算法-進程切換
STATIC VOID OsSchedSwitchProcess(LosProcessCB *runProcess, LosProcessCB *newProcess)
{
    if (runProcess == newProcess) {
        return;
    }

#if (LOSCFG_KERNEL_SMP == YES)
    runProcess->processStatus = OS_PROCESS_RUNTASK_COUNT_DEC(runProcess->processStatus);
    newProcess->processStatus = OS_PROCESS_RUNTASK_COUNT_ADD(newProcess->processStatus);

    LOS_ASSERT(!(OS_PROCESS_GET_RUNTASK_COUNT(newProcess->processStatus) > LOSCFG_KERNEL_CORE_NUM));
    if (OS_PROCESS_GET_RUNTASK_COUNT(runProcess->processStatus) == 0) {//獲取當前進程的任務數量
#endif
        runProcess->processStatus &= ~OS_PROCESS_STATUS_RUNNING;
        if ((runProcess->threadNumber > 1) && !(runProcess->processStatus & OS_PROCESS_STATUS_READY)) {
            runProcess->processStatus |= OS_PROCESS_STATUS_PEND;
        }
#if (LOSCFG_KERNEL_SMP == YES)
    }
#endif
    LOS_ASSERT(!(newProcess->processStatus & OS_PROCESS_STATUS_PEND));//斷言進程不是阻塞狀態
    newProcess->processStatus |= OS_PROCESS_STATUS_RUNNING;//設置進程狀態為運行狀態

    if (OsProcessIsUserMode(newProcess)) {//用戶模式下切換進程mmu上下文
        LOS_ArchMmuContextSwitch(&newProcess->vmSpace->archMmu);//新進程->虛擬空間中的->Mmu部分入參
    }

#ifdef LOSCFG_KERNEL_CPUP
    OsProcessCycleEndStart(newProcess->processID, OS_PROCESS_GET_RUNTASK_COUNT(runProcess->processStatus) + 1);
#endif /* LOSCFG_KERNEL_CPUP */

    OsCurrProcessSet(newProcess);//將進程置為 g_runProcess

    if ((newProcess->timeSlice == 0) && (newProcess->policy == LOS_SCHED_RR)) {//為用完時間片或初始進程分配時間片
        newProcess->timeSlice = OS_PROCESS_SCHED_RR_INTERVAL;//重新分配時間片,默認 20ms
    }
}

這里再啰嗦一句,系列篇中已經說了兩個上下文切換了,一個是這里的因進程切換引起的mmu上下文切換,還一個是因task切換引起的CPU的上下文切換,還能想起來嗎?

第二:是加載ELF文件的時候會切換mmu,一個嶄新的進程誕生了,具體將在 鴻蒙內核源碼分析(啟動加載篇) 會細講,敬請關注系列篇動態。

其余是虛擬空間回收和刷新空間的時候,這個就自己看代碼去吧。

mmu是如何快速的通過虛擬地址找到物理地址的呢?答案是:TLB,注意上面還有個TTB,一個是寄存器, 一個是cache,別搞混了。

TLB(translation lookaside buffer)

TLB是硬件上的一個cache,因為頁表一般都很大,并且存放在內存中,所以處理器引入MMU后,讀取指令、數據需要訪問兩次內存:首先通過查詢頁表得到物理地址,然后訪問該物理地址讀取指令、數據。為了減少因為MMU導致的處理器性能下降,引入了TLB,可翻譯為“地址轉換后援緩沖器”,也可簡稱為“快表”。簡單地說,TLB就是頁表的Cache,其中存儲了當前最可能被訪問到的頁表項,其內容是部分頁表項的一個副本。只有在TLB無法完成地址翻譯任務時,才會到內存中查詢頁表,這樣就減少了頁表查詢導致的處理器性能下降。詳細看

照著圖說吧,步驟是這樣的。

1.圖中的page table的基地址就是上面TTB寄存器值,整個page table非常大,有多大接下來會講,所以只能存在內存里,TTB中只是存一個開始位置而已。

2. 虛擬地址是程序的地址邏輯地址,也就是喂給CPU的地址,必須經過MMU的轉換后變成物理內存才能取到真正的指令和數據。

3.TLB是page table的迷你版,MMU先從TLB里找物理頁,找不到了再從page table中找,從page table中找到后會放入TLB中,注意這一步非常非常的關鍵。因為page table是屬于進程的會有很多個,而TLB只有一個,不放入就會出現多個進程的page table都映射到了同一個物理頁框而不自知。一個物理頁同時只能被一個page table所映射。但除了TLB的唯一性外,要做到不錯亂還需要了一個東西,就是進程在映射層面的唯一標識符 -asid。

asid寄存器

asid(Adress Space ID) 進程標識符,屬于CP15協處理器的C13號寄存器,ASID可用來唯一標識進程,并為進程提供地址空間保護。當TLB試圖解析虛擬頁號時,它確保當前運行進程的ASID與虛擬頁相關的ASID相匹配。如果不匹配,那么就作為TLB失效。除了提供地址空間保護外,ASID允許TLB同時包含多個進程的條目。如果TLB不支持獨立的ASID,每次選擇一個頁表時(例如,上下文切換時),TLB就必須被沖刷(flushed)或刪除,以確保下一個進程不會使用錯誤的地址轉換。

TLB頁表中有一個bit來指明當前的entry是global(nG=0,所有process都可以訪問)還是non-global(nG=1,only本process允許訪問)。如果是global類型,則TLB中不會tag ASID;如果是non-global類型,則TLB會tag上ASID,且MMU在TLB中查詢時需要判斷這個ASID和當前進程的ASID是否一致,只有一致才證明這條entry當前process有權限訪問。

看到了嗎?如果每次mmu上下文切換時,把TLB全部刷新已保證TLB中全是新進程的映射表,固然是可以,但效率太低了?。?!進程的切換其實是秒級亞秒級的,地址的虛實轉換是何等的頻繁啊,怎么會這么現實呢,真實的情況是TLB中有很多很多其他進程占用的物理內存的記錄還在,當然他們對物理內存的使用權也還在。所以當應用程序 new了10M內存以為是屬于自己的時候,其實在內核層面根本就不屬于你,還是別人在用,只有你用了1M的那一瞬間真正1M物理內存才屬于你,而且當你的進程被其他進程切換后,很大可能你用的那1M也已經不在物理內存中了,已經被置換到硬盤上了。明白了嗎?只關注應用開發的同學當然可以說這關我鳥事,給我的感覺有就行了,但想熟悉內核的同學就必須要明白,這是每分每秒都在發生的事情。

最后一個函數留給大家,asid是如何分配的?

/* allocate and free asid */
status_t OsAllocAsid(UINT32 *asid)
{
    UINT32 flags;
    LOS_SpinLockSave(&g_cpuAsidLock, &flags);
    UINT32 firstZeroBit = LOS_BitmapFfz(g_asidPool, 1UL << MMU_ARM_ASID_BITS);
    if (firstZeroBit >= 0 && firstZeroBit < (1UL << MMU_ARM_ASID_BITS)) {
        LOS_BitmapSetNBits(g_asidPool, firstZeroBit, 1);
        *asid = firstZeroBit;
        LOS_SpinUnlockRestore(&g_cpuAsidLock, flags);
        return LOS_OK;
    }

    LOS_SpinUnlockRestore(&g_cpuAsidLock, flags);
    return firstZeroBit;
}

編輯:hfy

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • ARM
    ARM
    +關注

    關注

    134

    文章

    8720

    瀏覽量

    362843
  • 寄存器
    +關注

    關注

    30

    文章

    5164

    瀏覽量

    118135
  • 存儲器
    +關注

    關注

    38

    文章

    7212

    瀏覽量

    162391
  • 協處理器
    +關注

    關注

    0

    文章

    71

    瀏覽量

    18022
  • 鴻蒙系統
    +關注

    關注

    183

    文章

    2622

    瀏覽量

    65412
收藏 人收藏

    評論

    相關推薦

    寄存器內存的區別

    在計算機體系結構中,寄存器內存是兩個至關重要的組成部分。它們各自承擔著不同的角色,共同確保計算機系統的正常運行。本文將對寄存器內存進行詳細的介紹,包括它們的定義、功能以及二者之間的
    的頭像 發表于 05-12 17:11 ?257次閱讀

    干貨滿滿:ARM的內核寄存器講解

    內核寄存器與外設寄存器內核寄存器與外設寄存器是完全不同的概念。
    發表于 04-17 11:47 ?481次閱讀
    干貨滿滿:ARM的<b class='flag-5'>內核</b><b class='flag-5'>寄存器</b>講解

    STM32H7使用硬件I2C進行讀寫操作失敗的原因?

    當Size為0時,在發出設備地址,寄存器地址后,重置I2C外設寄存器并退出函數,依然無法實現跟F4單片機同樣的效果,通過邏輯分析儀查看,發現總線一直被占用,后續讀取無法正常進行。 請問
    發表于 03-11 08:12

    STM8S103如何用硬件I2C定時重復讀取從機的一狀態寄存器?

    沒接觸過MCU,為了解碼制作苦學了幾天STM8單片機I2C,網上查了很多資料,現在通過硬件I2C可以讀寫從機了, 寫可以寫很多寄存器,但是讀只能讀一
    發表于 03-08 07:40

    鴻蒙內核源碼分析:MMU 協處理器

    ? Cortex?-A Series Version: 4.0 Programmer’s Guide》。 在這些 C15 寄存器中和 MMU 關系較大的有 C2、C7、C17 寄存器,
    的頭像 發表于 02-20 14:28 ?249次閱讀
    <b class='flag-5'>鴻蒙</b>輕<b class='flag-5'>內核</b><b class='flag-5'>源碼</b><b class='flag-5'>分析</b>:MMU 協處理器

    arm三個寄存器在gdb調試時的作用

    arm三個寄存器在gdb調試時作用? ARM是一種廣泛使用的微處理器架構,它廣泛應用于移動設備、嵌入式系統和其他高性能計算設備。當我們在使用gdb(GNU調試器)調試ARM程序時,了解和理解三個重要
    的頭像 發表于 01-31 10:44 ?339次閱讀

    如何配置TC38x I2C中斷?

    我們的項目有 MCAL,但是 Loader 只是一 while 循環,不要使用操作系統模塊。 我們已經通過輪詢模式實現了 I2C 通信,然后我們想切換到中斷模式。 我們如何 CAN 通過寄存器
    發表于 01-29 07:42

    I2C模塊在初始化后,illd I2C中的寫操作函數IfxI2c_I2c_write會因為該寄存器BS位為1而直接返回的原因?

    我的I2C模塊在初始化后,寄存器BUSSTAT.B.BS為1,處于忙碌狀態。此時illdI2C中的寫操作函數 IfxI2c_I2c_write(IfxI
    發表于 01-22 06:09

    MAX17853采樣電壓大于1400mv的時候,C13C14的電壓采樣不到怎么解決?

    當MAX17853的14節電壓采樣接滿的時候,單節采樣電壓大于1400mv的時候,C13C14的電壓采樣不到,隨著采樣電壓的增加,最多可出現C11-C14四節電池電壓采樣不到,請問這種出現這種情況的原因是什么?怎么解決? 附件
    發表于 01-03 11:10

    請問能用I2C讀取ADXL345連續的幾個寄存器嗎?

    在實際的操作過程中發現,不能使用I2c讀取ADXL345連續的幾個寄存器(0x32-0x37),看到論壇里有說明SPI可以,我在這里想問問專業的工程師I2C可以做到嗎?雖然可以使用dataready
    發表于 01-02 08:05

    ADXL345的三個軸的偏置寄存器中的值有什么意義呢?

    ADXL345的三個軸的偏置寄存器中的值有什么意義呢?為什么一定要有偏置才能工作呢?而且偏置最大才2g。如果我設置成滿偏置,那么靜止的時候讀出來的數據寄存器的值是多少呢?
    發表于 01-02 07:17

    請問ADE7880無功功率寄存器是否有映射寄存器地址?

    ADE7880: 關于無功功率的寄存器AFVAR,BFVAR,CFVAR,查閱手冊,其說明是只能通過HSDC接口進行讀取,但是我翻閱論壇之前的帖子,有說明其有映射寄存器地址,但是手冊我無法發現,請幫我解答,謝謝。 ADE7880
    發表于 12-26 07:12

    ADV7393 I2C寄存器地址無應答是怎么回事?

    我們在Xilinx ZYNQPL側配置I2C,電平為1.8V,SCL在30Khz,器件地址(0x54)有應答,但是有時寄存器地址或者數據不給應答,時序如圖1所示。 插圖1 我們還使用了PS側
    發表于 12-05 06:03

    基于 FPGA 的模擬 I2C協議系統設計

    。 表 3 數據傳輸寄存器內容 7)數據接收寄存器數據接收寄存器用于保存通過 I2C 總線接收到的最后一
    發表于 08-14 18:21

    NANO120中I2C驅動寄存器寫入不了怎么解決?

    0); NVIC_EnableIRQ(I2C0_IRQn); DEBUGMSG(\"I2C0 initial over!\\n\"); } 然后在讀寫操作24C02中,對I2C的控
    發表于 06-27 09:23
    亚洲欧美日韩精品久久_久久精品AⅤ无码中文_日本中文字幕有码在线播放_亚洲视频高清不卡在线观看
    <acronym id="s8ci2"><small id="s8ci2"></small></acronym>
    <rt id="s8ci2"></rt><rt id="s8ci2"><optgroup id="s8ci2"></optgroup></rt>
    <acronym id="s8ci2"></acronym>
    <acronym id="s8ci2"><center id="s8ci2"></center></acronym>