zoukankan      html  css  js  c++  java
  • ARMv8 Linux内核源代码分析:__flush_dcache_all()

    1.1

     

    /*

     *  __flush_dcache_all()

    *  Flush the wholeD-cache.

     * Corrupted registers: x0-x7, x9-x11

     */

    ENTRY(__flush_dcache_all)

    //保证之前的訪存指令的顺序

        dsb sy           

          //读cache level id register

        mrs x0, clidr_el1           // read clidr

          //取bits[26:24](Level of Coherency for the cache hierarchy.)

    //须要遵循cache一致性的cache层级(比如有3级cache,但2级须要做一致性)

        and x3, x0, #0x7000000      // extract loc from clidr

          //逻辑右移23位,把bits[26:24]放到bits[2:0]

        lsr x3, x3, #23         // left align loc bit field

          //假设须要做cache一致性的层级为0,则不须要flush,跳转到finished标记处。

        cbz x3, finished            // if loc is 0, then no need toclean

          //x10存放cache级,从level0 cache開始做flush

          //下面三个循环loop3set/wayx9),

    //loop2indexx7),loop1cachelevel(x10)

        mov x10, #0             // start clean at cache level 0

    loop1:

    //x10+2后右移一位正好等于1,再加上x10本身正好等于3

          //每运行一次loop1,x2+3*运行次数,目的在于把x0(clidr_el1)右移3位,

    //取下一个cache的ctype type fields字段,clidr_el1的格式见《ARMv8 ARM》

        add x2, x10, x10, lsr #1        /

          //x0逻辑右移x2位,给x1,提取cache类型放到x1中,x0中存放:clidr_el1

        lsr x1, x0, x2         

          //掩掉高位,仅仅取当前cache类型

        and x1, x1, #7 

          /* 推断当前cache是什么类型:

    * 000  No cache.

    * 001  Instruction cache only.

    * 010  Data cache only.

    * 011  Separate instruction and data caches.

    * 100  Unified cache.

    */

          //小于2说明data cache不存在或者仅仅有icache,

    //跳转skip运行,大于等于2继续运行

        cmp x1, #2             

        b.lt   skip               

         

    /*

     * Save/disable and restore interrupts.

     * .macro save_and_disable_irqs, olddaif

     * mrs olddaif,daif                                                                                                                                                      

     * disable_irq

     * .endm

    */

          //保存daif到x9寄存器中,关闭中断

        save_and_disable_irqs x9        // make CSSELR and CCSIDR access atomic

          //选择当前cache级进行操作,csselr_el1寄存器bit[3:1]选择要操作的cache级

          //第一次运行时x10=0,选择level 0级cache

        msr csselr_el1,x10        

          //isb用于同步新的cssr和csidr寄存器

        isb                

          //由于运行了“msr csselr_el1,x10”,所以要又一次读取ccsidr_el1

        mrs x1, ccsidr_el1          // read the new ccsidr

        /*

    * .macro  restore_irqs, olddaif                                                                                                                                          

         * msr daif, olddaif

        . * endm

            */

        restore_irqs x9

          //x1存储ccsidr_el1内容,低三位是(Log2(Number of bytes in cache line)) – 4

          //4x2=(Log2(Numberof bytes in cache line))

        and x2, x1, #7          // extract the length of the cachelines

        add x2, x2, #4          // add 4 (line length offset)

        mov x4, #0x3ff

          //逻辑右移3位,提取bits[12:3](Associativityof cache) – 1,

          //x4存储cacheway

        and x4, x4, x1, lsr #3     // find maximum number on the way size

          //计算x4前面0的个数,存到x5

        clz x5, x4              // find bit position of way sizeincrement

          //提取bits[27:13]位:(Number of sets in cache) - 1

        mov x7, #0x7fff

          //x7中存储cache中的set

        and x7, x7, x1, lsr #13     // extract max number of the index size

    loop2:

          //把x4值备份

        mov x9, x4              // create working copy of max waysize

    loop3:

          //把须要操作哪个way存储到x6

        lsl x6, x9, x5

          //确定操作哪一级的哪个way(x10指定操作哪一级cache)

        orr x11, x10, x6            // factor way and cache number intox11

          //确定操作哪个set

        lsl x6, x7, x2

        orr x11, x11, x6            // factor index number into x11

          //x11中存储了哪一级cache(10),哪一路cache(x9),哪个set(x7)

        dc  cisw, x11           // clean & invalidate by set/way

          //way数-1

        subs   x9, x9, #1          // decrementthe way

        b.ge   loop3

        subs   x7, x7, #1          // decrementthe index

        b.ge   loop2

    skip:

        add x10, x10, #2            // increment cache number,

    //为什么加2不是1?见loop1标号处解释

        cmp x3, x10

        b.gt   loop1

    finished:

        mov x10, #0             // swith back to cache level 0

        msr csselr_el1, x10         // select current cache level incsselr

        dsb sy

        isb

        ret

    ENDPROC(__flush_dcache_all)

  • 相关阅读:
    【白嫖】0002 11款 Procreate 皮肤纹理笔刷
    30款 Procreate 书法笔刷 你值得拥有
    PaddleHub(3)
    PaddleHub(2)
    关于大学学习的小思考
    PaddleHub(1)
    OO TA——一次非同寻常的体验
    Vue实现增加删除检索数据
    闲言碎语
    析 合 树 详 解
  • 原文地址:https://www.cnblogs.com/blfshiye/p/4286171.html
Copyright © 2011-2022 走看看