查看: 1340|回复: 0

[分享] 无缓存下对Flash的AHB读访问竟然是这样!

[复制链接]
  • TA的每日心情
    开心
    2021-12-8 10:59
  • 签到天数: 305 天

    [LV.8]以坛为家I

    7

    主题

    1147

    帖子

    1

    金牌会员

    Rank: 6Rank: 6

    积分
    13662
    最后登录
    2024-1-4
    发表于 2021-5-10 17:10:43 | 显示全部楼层 |阅读模式
    本帖最后由 dutu-1 于 2021-5-10 17:17 编辑

    大家好,我是痞子衡,是正经搞技术的痞子。今天痞子衡给大家介绍的是实抓Flash信号波形来看i.MXRT的FlexSPI外设下AHB读访问情形

    上一篇文章 《i.MXRT中FlexSPI外设对AHB Burst Read特性的支持》 里痞子衡介绍了FlexSPI外设在不开启Prefetch功能下响应AHB master的访问请求完全受AHB总线Burst Read特性决定,这是FlexSPI外设最基础的对Flash访问支持功能,研究这个其实是很有意义的,这可以反映出XiP下最原始的代码执行效率。

    我们知道在实际项目中,XiP应用程序常常是在L1 Cache和Prefetch加持下运行的,代码执行效率会得到大大提升,但无论是怎样的缓存策略,极限情况下(比如大数据块搬移,长跳转指令)最终还是拼得FlexSPI最基础的读访问支持。今天痞子衡就从抓Flash信号波形角度带大家真切感受下这最基础的AHB读访问情形(为更清晰地分析结果,本次主要涉及数据总线AHB访问,暂不涉及指令总线AHB访问):

    一、实验准备

    痞子衡用i.MXRT1050-EVKB来做这个AHB读访问实验,这块板子上的Flash被痞子衡更换过,目前的型号是华邦W25Q64JWS-IQ。我们基于 \SDK_2.9.1_EVKB-IMXRT1050\boards\evkbimxrt1050\demo_apps\led_blinky\iar 例程(记得切换到 flexspi_nor_debug build)来简单修改一下,把启动头FDCB修改如下,设置Flash工作于30MHz Fast Read Quad I/O SDR模式,调成30MHz低速是为了方便后续用示波器抓Flash信号去分析。

    1. const flexspi_nor_config_t qspiflash_config = {
    2.     .memConfig =
    3.         {
    4.             .tag              = FLEXSPI_CFG_BLK_TAG,
    5.             .version          = FLEXSPI_CFG_BLK_VERSION,
    6.             .readSampleClkSrc = kFlexSPIReadSampleClk_LoopbackFromDqsPad,
    7.             .csHoldTime       = 3u,
    8.             .csSetupTime      = 3u,
    9.             .controllerMiscOption = 0x10,
    10.             .deviceType       = kFlexSpiDeviceType_SerialNOR,
    11.             .sflashPadType    = kSerialFlash_4Pads,
    12.             // Flash工作于30MHz
    13.             .serialClkFreq    = kFlexSpiSerialClk_30MHz,
    14.             .sflashA1Size     = 8u * 1024u * 1024u,
    15.             .lookupTable =
    16.                 {
    17.                     // Quad I/O Fast Read SDR LUTs
    18.                     [4*CMD_LUT_SEQ_IDX_READ + 0] = FLEXSPI_LUT_SEQ(CMD_SDR,   FLEXSPI_1PAD, 0xEB, RADDR_SDR, FLEXSPI_4PAD, 0x18),
    19.                     [4*CMD_LUT_SEQ_IDX_READ + 1] = FLEXSPI_LUT_SEQ(MODE8_SDR, FLEXSPI_4PAD, 0xF0, DUMMY_SDR, FLEXSPI_4PAD, 0x04),
    20.                     [4*CMD_LUT_SEQ_IDX_READ + 2] = FLEXSPI_LUT_SEQ(READ_SDR,  FLEXSPI_4PAD, 0x04, STOP,      FLEXSPI_1PAD, 0x00),
    21.                     [4*CMD_LUT_SEQ_IDX_READ + 3] = 0,
    22.                 },
    23.         },
    24.     .pageSize           = 256u,
    25.     .sectorSize         = 4u * 1024u,
    26.     .blockSize          = 64u * 1024u,
    27.     .isUniformBlockSize = false,
    28. };
    复制代码

    下图是华邦W25Q64JWS-IQ芯片的Fast Read Quad I/O SDR传输时序图,Dummy Cycle连同MODE8_SDR序列一共6个SCK周期,此外还有个特别注意点,MODE8_SDR序列参数值需要被设成0xFx,我们上面修改的FDCB启动头是符合要求的。

    1.png

    现在让我们把示波器拿出来,四路探头分别连到板载Flash器件的CE#、SCK、SI_IO0、SO_IO1引脚(IO2、IO3因探头有限就不抓取了,IO[1:0]足够我们分析时序了),然后将 led_blinky 工程下载进Flash运行便可以观测结果了。

    2.png

    二、实验代码

    因为我们下载的是一个XIP工程,代码的执行本身也会触发Flash中的指令读取,这会影响我们在示波器上观测AHB读数据测试结果,所以我们可以在main()函数里把SysTick初始化去掉(不要中断),并且调用如下ramfunc型函数 test_ahb_read() 来做测试(痞子衡直接利用了IAR软件的特性),这样代码跑起来后,Flash上发生的读访问均来自我们想要测试的AHB读数据操作(这也意味着ICache是否开启对本系列测试结果没有影响,但不管怎么,我们统一关掉):

    Note: DCache和Prefetch必须要全部关闭,否则哪怕测试代码里对同一个地方循环读取,但在Flash引脚上根本看不到周期性信号波形,因为系统做了缓存,后续的读取操作可能直接发生在缓存区里(32KB DCache, 1KB AHB RX prefetch buffer)了。

    1. #define AHB_ADDR_START (0x60002400)

    2. #if (defined(__ICCARM__))
    3. #pragma optimize = none
    4. __ramfunc
    5. #endif
    6. void test_ahb_read(void)
    7. {
    8.     /* Disable L1 I-Cache*/
    9.     SCB_DisableICache();

    10.     /* Disable L1 D-Cache*/
    11.     SCB_DisableDCache();

    12.     /* Disable FlexSPI AHB read prefetch */
    13.     FLEXSPI->AHBCR &= ~(FLEXSPI_AHBCR_PREFETCHEN_MASK | FLEXSPI_AHBCR_CACHABLEEN_MASK);
    14.    
    15.     while (1)
    16.     {
    17.         SDK_DelayAtLeastUs(10, SystemCoreClock);
    18.         for (uint32_t i = 1; i <= 8; i++)
    19.         {   
    20.             SDK_DelayAtLeastUs(2, SystemCoreClock);
    21.             memcpy((void *)0x20200000, (void *)AHB_ADDR_START, i);
    22.         }
    23.     }
    24. }
    复制代码

    因为我们用了memcpy来做Flash数据拷贝,memcpy功能实际上是IAR软件自带库 ABImemcpy.a 里面的 __aeabi_memcpy、__aeabi_memcpy4、__aeabi_memcpy8 等函数实现的,因此我们还需要在工程链接文件里将 ABImemcpy.o 链接到RAM区;并且我们还用了SDK_DelayAtLeastUs()来分隔每次memcpy()波形结果,还需要将这个函数里调用的相关代码放到RAM区(fsl_common.c里)。

    1. initialize by copy { readwrite,
    2.                      section .textrw,
    3.                      // 确保 memcpy() 相关代码全在RAM里
    4.                      object ABImemcpy.o,
    5.                      // 确保 SDK_DelayAtLeastUs() 相关代码全在RAM里
    6.                      object fsl_common.o,
    7.                      object I64DivZer.o,
    8.                      object I64DivMod.o
    9.                      };
    10. do not initialize  { section .noinit };
    复制代码

    一切准备就绪后具体测试就是设置不同的AHB_ADDR_START值(这里主要是考虑地址对齐)来观测Flash信号的实际波形。此外为了便于分辨IO[1:0]上的数据,我们最好定义一块特殊const数据区,根据Flash传输时序图,其中数据Byte[4]和Byte[0]是在IO0线上传输、Byte[5]和Byte[1]是在IO1线上传输的,这4bit共有16种不同值组合,我们将这16种不同值放在ahbRdBlock[16]数组中,并将其链接在 0x60002400 - 0x6000240f 地址空间里。

    1. // 在工程源文件中
    2. const uint8_t ahbRdBlock[16] @ ".ahbRdBuffer" = {0x00, 0x01, 0x02, 0x03,
    3.                                                  0x10, 0x11, 0x12, 0x13,
    4.                                                  0x20, 0x21, 0x22, 0x23,
    5.                                                  0x30, 0x31, 0x32, 0x33};
    6. // 在工程链接文件中
    7. keep{ section .ahbRdBuffer };
    8. place at address mem:0x60002400 { readonly section .ahbRdBuffer };
    复制代码
    三、实验结果3.1 AHB_ADDR_START = 0x6002400 即八字节对齐

    我们先来看AHB_ADDR_START = 0x6002400时抓取一次完整for循环结果的波形(见下图),可以看到在八字节对齐的地址下使用memcpy拷贝1/2/4/8字节,均仅产生一次CS信号有效周期(拉低),在这CS有效期间完成全部所需数据的读取。但是拷贝3/5/6/7字节时,会拆分出多个CS有效周期。

    3.png

    当使用memcpy拷贝3/5/6字节时,会拆分出2个CS有效周期(见下图),这里第一个CS周期看起来似乎是多余的,为什么是这种结果,这需要深入研究AHB机制(痞子衡会另写文章分析);

    • 当拷贝3字节时,第一个CS周期实际读取了前2个字节 [0x60002400, 0x60002401],第二个CS周期读取了全部3字节 [0x60002400, 0x60002402]。
    • 当拷贝5字节时,第一个CS周期实际读取了前4个字节 [0x60002400, 0x60002403],第二个CS周期读取了全部5字节 [0x60002400, 0x60002404]。
    • 当拷贝6字节时,第一个CS周期实际读取了前4个字节 [0x60002400, 0x60002403],第二个CS周期读取了全部6字节 [0x60002400, 0x60002405]。

    4.png

    当使用memcpy拷贝7字节时,会拆分出3个CS有效周期(见下图),这里前两个CS周期看起来似乎都是多余的;

    • 当拷贝7字节时,第一个CS周期实际读取了前4个字节 [0x60002400, 0x60002403],第二个CS周期实际读取了前6个字节 [0x60002400, 0x60002405],第三个CS周期读取了全部7字节 [0x60002400, 0x60002406]。

    5.png

    3.2 AHB_ADDR_START = 0x6002404 即四字节对齐

    AHB_ADDR_START = 0x6002404时抓取一次完整for循环结果的波形(见下图),可以看到在四字节对齐的地址下使用memcpy拷贝1/2/4字节,均仅产生一次CS信号有效周期(拉低),在这CS有效期间完成全部所需数据的读取。

    但是拷贝3/5/6/7/8字节时,会拆分出多个CS有效周期。不过其中拷贝5/6/8字节,是合理的拆分,并没有冗余读取。

    6.png

    3.3 AHB_ADDR_START = 0x6002401 即奇地址

    AHB_ADDR_START = 0x6002401时抓取一次完整for循环结果的波形(见下图),这种情况下CS拆分特别严重,几乎都存在冗余读取。

    7.png

    3.4 AHB_ADDR_START = 0x6002402 即偶地址

    AHB_ADDR_START = 0x6002402时抓取一次完整for循环结果的波形(见下图),这种情况下CS拆分特别严重,几乎都存在冗余读取。

    8.png

    3.5 AHB_ADDR_START = 0x6002403

    AHB_ADDR_START = 0x6002403时抓取一次完整for循环结果的波形(见下图),这种情况下CS拆分特别严重,几乎都存在冗余读取。

    9.png

    至此,实抓Flash信号波形来看i.MXRT的FlexSPI外设下AHB读访问情形痞子衡便介绍完毕了,掌声在哪里~~~

    文章出处:痞子衡嵌入式


    哎...今天够累的,签到来了~
    回复

    使用道具 举报

    您需要登录后才可以回帖 注册/登录

    本版积分规则

    关闭

    站长推荐上一条 /4 下一条

    Archiver|手机版|小黑屋|恩智浦技术社区

    GMT+8, 2024-5-7 02:25 , Processed in 0.112425 second(s), 20 queries , MemCache On.

    Powered by Discuz! X3.4

    Copyright © 2001-2024, Tencent Cloud.

    快速回复 返回顶部 返回列表