CSAPP fall2015 深入理解计算机系统 Cache lab详解

Cache Lab

cache lab 缓存实验

代码下载

从CSAPP上面下载对应的lab代码

http://csapp.cs.cmu.edu/3e/labs.html

环境准备

需要安装 valgrind。可以参考文章Valgrind centos。

安装好以后执行valgrind --version可以看到版本号。

Cache simulator

  • cache simulator not a cache。我们不是实现一个真正的缓存,只是实现一个模拟器。
  • 不存储内容
  • 不使用block offset
  • 只计算命中数,不命中数和驱逐数(hit count, miss count,eviction count)
  • 缓存模拟器需要在不同的s,b,E下运行。
  • 使用LRU替换策略

Hints

  • 使用二维数组 struct cache_line cache[S][E];
  • S = 2^s
  • cache_line //上面说了不需要block offset,所以可以忽略block的内容
    • Valid bit
    • Tag
    • LRU counter
  • 通过getopt获取命令行输入
    • 返回-1表示没有输入了
    • 通常在循环里面接收参数
    • 需要包含#include <unistd.h>,#include <getopt.h>
    • 通常使用switch来处理不同的输入
    • 考虑如何处理无效输入
    • 更多信息 man 3 getopt
  • fscanf可以指定要读的流(scanf只能读标准输入流),用来读取trace file
    • 参数
      • 一个流的指针
      • 如何解析文件的信息的格式化字符串
      • 其余部分是指向存储解析数据的变量的指针
    • 通常在循环里使用
    • 当命中EOF或者没有匹配到格式化字符串的时候返回-1
    • 更多信息 man fscanf
  • Malloc/free
    • malloc分配数据到heap
    • 记得 free 掉malloc的数据
    • 不要 free 你没有分配的内存

在这里插入图片描述

在这里插入图片描述

要求我们实现csim.c文件,给了一个示例csim-ref文件。

输入./csim-ref -h可以看到我们要实现的东西。

在这里插入图片描述

首先需要接受参数,参数有

  • -h 输出帮助信息
  • -v 可选详细标志,根据示例程序来,就是输出 L 10,1 miss这些信息
  • -s [num] set index bit 数
  • -E [num] 每个set的行数
  • -b [num] block offset bit数
  • -t [file] Trace file文件路径

根据上面的提示可以知道,通过getopt函数来接收参数,并通过switch来处理。读取文件则通过fscanf函数,来读取-t传的文件。

下面是./traces/yi.trace文件的内容

 L 10,1M 20,1L 22,1S 18,1L 110,1L 210,1M 12,1
  • L 代表数据载入
  • S 代表数据存储
  • M 代表数据修改,需要一次载入 + 一次存储
  • 后面的10,20,22这些代表地址
  • 最后的1代表操作内存访问的字节数

完整代码

#include "cachelab.h"
#include <unistd.h>
#include <getopt.h>
#include <stdio.h>
#include <stdlib.h>typedef struct{int valid;int tag;int time_stamp;
} cache_line;int timestamp = 0;// 开始匹配到合适的set 找到命中的cache,如果命中返回1,如果没有命中返回0
int find_hit_cache(cache_line *cache_line,int E,int tag, int*hits) {int isHit = 0;// 循环set中的cache_line 找到是否有匹配tag && validfor(int i = 0; i < E; i++) {if (cache_line[i].valid == 1 && cache_line[i].tag == tag ) {//hitprintf("hit \n");*hits = *hits + 1;isHit = 1;//刷新时间cache_line[i].time_stamp = timestamp;return isHit;}}return isHit;
}// 找到一个空的cache line放进去,找到了就返回1,没有找到就返回0
int find_empty_cache(cache_line *cache_line,int E,int tag) {int have_empty_cache = 0;for (int i = 0; i< E; i++) {if (cache_line[i].valid == 0) {// 空的// 把当前内存放入cachecache_line[i].valid = 1;cache_line[i].tag = tag;cache_line[i].time_stamp = timestamp;// 找到了就不需要替换了have_empty_cache = 1;return have_empty_cache;}}return have_empty_cache;
}// 获取要替换的索引
int get_eviction_index(cache_line *cache_line, int E) {int max_time_stamp = timestamp;int eviction_index = -1;for (int i = 0; i< E; i++) {if (cache_line[i].time_stamp < max_time_stamp) {//找到time_stamp最小的那个max_time_stamp = cache_line[i].time_stamp;eviction_index = i;}}return eviction_index;
}// LRU替换
void LRU(cache_line *cache_line, int E,int tag) {// 获取要替换的索引int eviction_index = get_eviction_index(cache_line, E);// 替换cache_line[eviction_index].valid = 1;cache_line[eviction_index].tag = tag;cache_line[eviction_index].time_stamp = timestamp; 
}// L和S操作,M就调用两次这个
int load_and_store(unsigned address,int b,int s,int u_max,int E,cache_line **cache, int *hits,int *misses,int *evications) {// 获取set index block offset tagint set_index,tag;set_index = (address >> b) & u_max;tag = (address >> b) >> s;// 开始匹配到合适的set 找到命中的cache,如果命中返回1,如果没有命中返回0int isHit = find_hit_cache(cache[set_index], E, tag, hits);if (isHit == 0) {// missprintf("miss \n");*misses = *misses + 1;// 找到一个空的cache line放进去,找到了就返回1,没有找到就返回0int have_empty_cache = find_empty_cache(cache[set_index], E, tag);// 如果没有找到空的cache,就需要LRU替换if (have_empty_cache == 0) {printf("evictions \n");*evications = *evications + 1;//LRU替换LRU(cache[set_index], E, tag);}}// 更新全局时间戳timestamp++;return 0;
}int main(int argc, char** argv)
{// 接受参数 getoptint opt,v,s,E,b,S,B;// 文件FILE        *       pFile;while(-1 != (opt = getopt(argc, argv, "h?v?s:E:b:t:"))){// opt is h,v,s,E,b,t的ASCII码值// 通过switch对不同的参数进行不同的处理switch(opt) {case 'h':printf("./csim: Missing required command line argument \n Usage: ./csim-ref [-hv] -s <num> -E <num> -b <num> -t <file> \n Options: \n -h         Print this help message. \n -v         Optional verbose flag. \n -s <num>   Number of set index bits. \n -E <num>   Number of lines per set. \n -b <num>   Number of block offset bits. \n -t <file>  Trace file. \n\n Examples: \n ./csim -s 4 -E 1 -b 4 -t traces/yi.trace \n ./csim -v -s 8 -E 2 -b 4 -t traces/yi.trace \n");// h参数输出帮助内容break;case 'v':// v参数输出详细信息v = 1;printf("v:%d \n",v);break;case 's':// S is set 2^s 的数量// s is Number of set index bitss = atoi(optarg);S = 1 << s;printf("s:%d, S:%d \n",s,S);break;case 'E':// E is cache line 的数量// Number of lines per setE = atoi(optarg);printf("E:%d \n",E);break;case 'b':// B is block data 的字节// b is Number of block offset bitsb = atoi(optarg);B = 1 << b;printf("b:%d, B:%d \n",b,B);break;case 't':// t is Trace file// 读取文件//t = atoi(optarg);pFile   =       fopen(optarg,"r");printf("t:%s, file:%p \n",optarg,pFile);break;default:printf("非法参数 \n");break;}}if(s == 0 || E == 0 || b == 0) {return 0;}// cache存储cache_line **cache = (cache_line **)malloc(S * sizeof(cache_line *));if (cache == NULL) {printf("内存分配失败 \n");}for(int i = 0; i < S; i++) {cache[i] = (cache_line *)malloc(E * sizeof(cache_line));if(cache[i] == NULL) {printf("内存分配失败,开始回滚 \n");// 在这里需要释放已分配的内存,然后退出for (int j = 0; j < i; ++j) {free(cache[j]);}free(cache);}}int u_max = 1; for(int i = 0; i < s - 1; i++) {u_max = (u_max << 1) | 1;}// 读取文件char        identifier;unsigned    address;int         size;int hits,misses,evictions;
//      Reading lines   like    "       M       20,1"   or      "L      19,3"while(fscanf(pFile," %c %x,%d",&identifier,&address,&size)>0){//      Do      stuff// 开始计算 hits,misses,evictions, hits:0 misses:0 evictions:0//printf("identifier %c, addr:%x, size:%d \n",identifier,address,size);//根据identifier来判断动作L load S store M = 一次L 一次Sif (identifier == 'L' || identifier == 'S'){printf("identifier %c, addr:%x, size:%d \n",identifier,address,size);load_and_store(address,b,s,u_max,E,cache,&hits,&misses,&evictions);} else if (identifier == 'M') {// 一次L 一次Sprintf("identifier %c, addr:%x, size:%d \n",identifier,address,size);load_and_store(address,b,s,u_max,E,cache,&hits,&misses,&evictions);load_and_store(address,b,s,u_max,E,cache,&hits,&misses,&evictions);}}fclose(pFile);  //remember      to      close   file    when    doneprintSummary(hits, misses, evictions);// 释放数组内存for(int i = 0; i< S; i++) {free(cache[i]);}free(cache);return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/434324.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

软考复习之UML设计篇

UML统一建模语言 构件图&#xff1a;描述系统的物理结构&#xff0c;它可以用来显示程序代码如何分解成模块 部署图&#xff1a;描述系统中硬件和软件的物理结构&#xff0c;它描述构成系统架构的软件构件&#xff0c;处理器和设备 用例图&#xff1a;描述系统与外部系统及用…

matlab对负数开立方根得到虚数的解决方案

问题描述&#xff1a;在matlab中&#xff0c;对负数开立方根&#xff0c;不出意外你将得到虚数。 例如 − 27 3 \sqrt[3]{-27} 3−27 ​&#xff0c;我们知道其实数解是-3&#xff0c;但在matlab中的计算结果如下&#xff1a; 问题原因&#xff1a;matlab中的立方根运算是在…

AI嵌入式K210项目-目录

系列文章简介 本系列文章简单介绍K210芯片的使用方法&#xff0c;大致分为两个部分 : C语言裸机开发MicroPython开发 学习第一部分有助于大家对K210芯片的外设接口和内置的各种硬件加速器有更深刻的了解&#xff0c;第二部分有助于大家快速实现功能&#xff1b; 实验所使用的…

Linux系统优化要义

这里不敢说 linux优化奥义&#xff0c;主要是本文比较浅显&#xff0c;适合普通开发相关人员去读 linux作为服务器系统的王者&#xff0c;以稳定性著称&#xff0c;但对于不同的“应用场景”&#xff0c;相关配置还需调整&#xff0c;才能保证业务稳定性。以下是相关总结 IO优…

野火霸道V2学习笔记

野火霸道V2学习笔记 STM32F103学习笔记说明基础配置配置KeilMDK配置串口下载程序美化Keil界面配置VScode 理论知识STM32命名方式例子 置位与清零GPIOGPIO简介GPIO和引脚的区别引脚的分类 GPIO 框图讲解保护二极管推挽输出开漏输出补充: 高阻态与悬空复用功能输出输入模式GPIO框…

SpringBoot之时间数据前端显示格式化

背景 在实际我们通常需要在前端显示对数据操作的时间或者最近的更新时间&#xff0c;如果我们只是简单的使用 LocalDateTime.now()来传入数据不进行任何处理那么我们就会得到非常难看的数据 解决方式&#xff1a; 1). 方式一 在属性上加上注解&#xff0c;对日期进行格式…

架构师的36项修炼-08系统的安全架构设计

本课时讲解系统的安全架构。 本节课主要讲 Web 的攻击与防护、信息的加解密与反垃圾。其中 Web 攻击方式包括 XSS 跨站点脚本攻击、SQL 注入攻击和 CSRF 跨站点请求伪造攻击&#xff1b;防护手段主要有消毒过滤、SQL 参数绑定、验证码和防火墙&#xff1b;加密手段&#xff0c…

Gitlab默认主分支main修改为master

天行健&#xff0c;君子以自强不息&#xff1b;地势坤&#xff0c;君子以厚德载物。 每个人都有惰性&#xff0c;但不断学习是好好生活的根本&#xff0c;共勉&#xff01; 文章均为学习整理笔记&#xff0c;分享记录为主&#xff0c;如有错误请指正&#xff0c;共同学习进步。…

那些年与指针的爱恨情仇(一)---- 指针本质及其相关性质用法

关注小庄 顿顿解馋 (≧∇≦) 引言&#xff1a; 小伙伴们在学习c语言过程中是否因为指针而困扰&#xff0c;指针简直就像是小说女主&#xff0c;它逃咱追&#xff0c;我们插翅难飞…本篇文章让博主为你打理打理指针这个傲娇鬼吧~ 本节我们将认识到指针本质&#xff0c;何为指针和…

【nginx实战】nginx正向代理、反向代理、由反向代理实现的负载均衡、故障转移详解

文章目录 一. 正向代理与反向代理的概念二. Nginx服务器的正向代理服务1. Nginx服务器正向代理服务的配置的3个指令1.1. resolver指令1.2. resolver_timeout指令1.3. proxy_pass指令 2. Nginx服务器正向代理服务的使用 三. Nginx服务器的反向代理服务1. 反向代理的基本指令1.1.…

大创项目推荐 行人重识别(person reid) - 机器视觉 深度学习 opencv python

文章目录 0 前言1 技术背景2 技术介绍3 重识别技术实现3.1 数据集3.2 Person REID3.2.1 算法原理3.2.2 算法流程图 4 实现效果5 部分代码6 最后 0 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; 深度学习行人重识别(person reid)系统 该项目…

C# Graphics对象学习

Graphics对象用于进行绘制&#xff1b; 从哪个对象获取的Graphics&#xff0c;然后进行绘制&#xff0c;就绘制到该对象上&#xff1b; 从位图获取Graphics&#xff0c;然后进行绘制&#xff0c;绘制到该位图上&#xff1b; 从某个控件获取Graphics&#xff0c;然后绘制&…