设为首页 收藏本站
查看: 2133|回复: 0

[经验分享] c64xCache

[复制链接]

尚未签到

发表于 2015-11-19 09:29:48 | 显示全部楼层 |阅读模式
  http://hi.baidu.com/yutou410/item/eec7892aa26f2d8c6e2cc33e

  

  1.3   位置规律
       使用Cache是基于时间和空间两种规律,具体说就是CPU读取内存中的某一数据或指令很可能在下一时钟周期或很短的时间内会使用这一指令相邻的指令或数据(空间),有时候当前使用的指令或数据会重复使用(时间),所以CPU不仅将当前指令读进Cache,而且会将相邻指令一并读入,读入的大小取决于Cache的大小和Cacheline的大小.
1.5
       L1P是direct-mapped型,16K,512个line,每个line32byte,正好可以长一个取指包,8条指令。
      Cache缺失有必然缺失,冲突缺失,容量缺失。
      L1D是set-associative型,16K,2way,每way128line,每line64byte,LRU指向最近最少使用的line,没有write miss,也就是说L1D如果没有要写的数据,通过write buffer(write buffer 4个入口,每个入口64bit)直接写到下一级存储器,如果L1D又要写的数据就直接写,并且dirtybit 置1,表示如果要替换该line的数据要先write back该line的数据到下一级存储器,
      DM642有最大可配置为256K的二级Cache,一级cache read miss向二级cache查找,如果没有就有二级cache向外部存储器读,然后传给一级cache,最后送给CPU,如果这期间哪一级line的dirty bit是1,就要向下一级cache进行write back操作。对于不可cache的外部存储器有CPU直接read.如果CPU对于二级cachewrite miss,二级cache首先要从外部存储器读入相应数据,CPU直接在二级cache上修改该数据,不会再将数据cache到L1D,因为L1D是read-allocate,同样如果二级cache相应line的dirty bit是1就要先write back。
   
      两种情况:1 DMA读L2RAM某一line ,首先L2CACHE会检查该line是否被cache在L1D,如果不是,DMA直接读取L2RAMzhogndeshuju ,如果是会将L1D 中相应line的invalidate置位,并且write back数据到L2RAM中。2 DMA写L2RAM某一line,情况差不多。
      对于一般的CPU信号处理加上外围设备的DMA数据读写情况,处理方法是在内存中开辟pingpong缓冲区。inbufa,outbufa,inbufb,outbufb四块内存区,当CPU处理inbufa的数据并将产生的结果放到outbufa,这是外围器件DMA搬移数据到inbufb,另一外围设备将数据从outbufb搬出;同理当CPU处理inbufb并将产生的结果放到outbufb时,外围设备利用DMA分别从inbufa和outbufa搬入数据和搬出数据。一般L2RAMDMA实现代码如下:
      for (i=0; i<(DATASIZE/BUFSIZE)–2; i&#43;=2)
{
/* –––––––––––––––––––––––––––––––––––––––––––––––––––– */
/* InBuffA –> OutBuffA Processing */
/* –––––––––––––––––––––––––––––––––––––––––––––––––––– */
<DMA_transfer(peripheral, InBuffB, BUFSIZE)>
<DMA_transfer(OutBuffB, peripheral, BUFSIZE)>
process(InBuffA, OutBuffA, BUFSIZE);
/* ––––––––––––––––––––––––––––––––––––––––––––––––––––– */
/* InBuffB –> OutBuffB Processing */
/* ––––––––––––––––––––––––––––––––––––––––––––––––––––– */
<DMA_transfer(peripheral, InBuffA, BUFSIZE)>
<DMA_transfer(OutBuffA, peripheral, BUFSIZE)>
process(InBuffB, OutBuffB, BUFSIZE);
}
     上面CACHE的一致性由CPU自动管理,无须程序员设置,而下面要说的片外RAM开辟双缓冲区时要求程序员掌握L2cache 和片外RAM coherence(一致性)以及L2RAM和L1D的一致性,否则程序会出错。
     对于C64x系列,无论什么时候当片外设备DMA写片外RAM开辟的inbuf时,都要使用CACHE_invL2(InBuffB, BUFSIZE, CACHE_WAIT),使L1D当中的inbuffb无效;当片外设备DMA读片外RAM开辟的outbuf时,都要使用CACHE_wbL2(OutBuffB,BUFSIZE, CACHE_WAIT),使L1D当中的相应数据write back 到片外RAM .
      External Memory DMA Double Buffering Code Example
for (i=0; i<(DATASIZE/BUFSIZE)–2; i&#43;=2)
{
/* ––––––––––––––––––––––––––––––––––––––––––––––––––––– */
/* InBuffA –> OutBuffA Processing */
/* ––––––––––––––––––––––––––––––––––––––––––––––––––––– */
CACHE_wbInvL2(InBuffB, BUFSIZE, CACHE_WAIT);
<DMA_transfer(peripheral, InBuffB, BUFSIZE)>
CACHE_wbL2(OutBuffB, BUFSIZE, CACHE_WAIT);
<DMA_transfer(OutBuffB, peripheral, BUFSIZE)>
process(InBuffA, OutBuffA, BUFSIZE);
/* ––––––––––––––––––––––––––––––––––––––––––––––––––––– */
/* InBuffB –> OutBuffB Processing */
/* ––––––––––––––––––––––––––––––––––––––––––––––––––––– */
CACHE_wbInvL2(InBuffA, BUFSIZE, CACHE_WAIT);
<DMA_transfer(peripheral, InBuffA, BUFSIZE)>
CACHE_wbL2(OutBuffA, BUFSIZE, CACHE_WAIT);
<DMA_transfer(OutBuffA, peripheral, BUFSIZE)>
process(InBuffB, OutBuffB, BUFSIZE);
}
     虽然我们可以指定一定大小的buf被write back或者invalidate或者write back&#43;invalidate,但是cache controller是对完整的line操作,这就要求我们在片外RAM开辟内存( 或者数组)作buffer时,尽量使得其大小是CACHE_L2_LINESIZE(128)的整数倍并且CACHE_L2_LINESIZE对齐,为此我们需要:    #pragma DATA_ALIGN(InBuffA, CACHE_L2_LINESIZE)
     #pragma DATA_ALIGN(InBuffB, CACHE_L2_LINESIZE)
     #pragma DATA_ALIGN(OutBuffA,CACHE_L2_LINESIZE)
     #pragma DATA_ALIGN(OutBuffB,CACHE_L2_LINESIZE)

来对齐buffer。
       使用宏#define CACHE_ROUND_TO_LINESIZE(cache,elcnt,elsize) \
                                                                                ((CACHE_#cache#_LINESIZE * \
                                                                                      ((elcnt)*(elsize)/CACHE_#cache#_LINESIZE) &#43; 1) / \
                                                                                            (elsize))
来对其内存,使用如下;
unsigned char InBuffA [CACHE_ROUND_TO_LINESIZE(L2, N, sizeof(unsigned char)];
unsigned char OutBuffA[CACHE_ROUND_TO_LINESIZE(L2, N, sizeof(unsigned char)];
unsigned char InBuffB [CACHE_ROUND_TO_LINESIZE(L2, N, sizeof(unsigned char)];
unsigned char OutBuffB[CACHE_ROUND_TO_LINESIZE(L2, N, sizeof(unsigned char)];
这样我们得到的数组就是内存对齐并且其大小是CACHE_L2_LINESIZE的整数倍,尽管这样做浪费片外RAM。由于L2CACHE最大256K,所以如果生命的数组大小超过256K,就需要多用几次cache write back和invalidate函数。
      如果程序没有配置L2CACHE,也就是说片内256K都作为RAM,那么当利用片外RAM开辟buffer时要考虑的L1D和L1P与片外设备的一致性问题,也有一组相应的write back和invalidate函数对应。

3   基于cache的程序优化
   3.1   应用级优化 (application-level optimization)
      1) 合理设置cache大小,尽量将DMA用到的buffer开在片内RAM上
      2) 将一般性程序代码和数据放到片外RAM,将DSP型代码和数据放到L2RAM。所谓一般性代码是指带有很多条件分支转移的指令,程序执行在空间上有随意性,不利于流水线的形成,外在片外可以发挥L2CACHE 4 way的优势。DSP型代码是指算法型的代码,放在L2RAM,CPU stall 时间少,可以充分发挥DSP速度快的优势。
   3.2 程序级优化(procedural-level optimization)
      1)选择合适的数据类型。能用short就不要用int。
      2)将同一个函数要处理的数据尽量在内存中连续存放。
   3.3避免L1P read miss
      这种情况发生在一个循环体中有两个或以上的函数要执行,要利用#pragma DATA_SECTION伪指令将和CMD文件将其在内存中相邻定位,这样不会发生两个程序对应L1P中相同line所造成的冲突缺失。
     若果循环体中的两个函数大小超过L1P容量,将这两个函数分别放到两个循环体中。这样做会造成中间数据变量的加大。
     3.4避免L1D read miss
      利用#pragma DATA_SECTION伪指令将函数要同时处理的数组在内存中相邻存放。最好再用#pragma DATA_MEM_BANK 将数组内存对齐

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-141006-1-1.html 上篇帖子: Centos部署Xcache提升脚本编译效率 下篇帖子: CentOS 5.x + Apache 2.x + PHP 5.3.x + GD2 + Xcache + vsftpd + MySQL 安装日志
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

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