代码拉取完成,页面将自动刷新
//
// Created by zhouheyu on 18-1-16.
//
#include "List.h"
//创建双向链表
pNode CreateList()
{
//分配初始内存
pNode pHead=(pNode)malloc(sizeof(Node));
if(NULL==pHead){
fprintf(stderr,"error happened in CASA/CreateList\n");
fprintf(stderr,"malloc for pHead failed!\n");
exit(-1);
}
pHead->LPN=-1;
pHead->isD=-1;
pHead->Pre=pHead;
pHead->Next=pHead;
return pHead;
}
//删除整个链表,释放内存(这里有点小问题)
void FreeList(pNode pHead)
{
pNode pt=pHead->Pre,ps;
while(pt!=pHead){
ps=pt;
pt=pt->Next;
free(ps);
}
free(pHead);
}
//判断链表是否为空
int IsEmptyList(pNode pHead)
{
pNode pt=pHead->Next;
if(pt==pHead)
{
return 1;
}else
{
return 0;
}
}
//返回链表的长度
int GetListLength(pNode pHead)
{
int length=0;
pNode pt=pHead->Next;
while (pt !=pHead)
{
length++;
pt=pt->Next;
}
return length;
}
//从链表中找到特定的LPN值,并返回节点的指针位置,如果不存在返回NULL
pNode FindLPNinList(pNode pHead,int LPN)
{
pNode ps=NULL,pt=pHead->Next;
int count=0;
while(pt!=pHead)
{
count++;
if(pt->LPN==LPN){
ps=pt;
break;
}
pt=pt->Next;
}
//调试输出语句遍历循环了多少次
// printf("the while count is %d\n",count);
return ps;
}
//向链表中删除节点,删除位置的节点
int DeleteEleList(pNode pHead,int pos)
{
pNode pt=pHead,ps=NULL;
//pos=0就是pHead
if(pos>0&&pos<GetListLength(pHead)+1){
while(1)
{
pos--;
if(pos==0)
break;
pt=pt->Next;
}
//此时的pt是pos的前一个节点
//ps才是要删除的节点位置
ps=pt->Next;
pt->Next=ps->Next;
ps->Next->Pre=pt;
//释放ps指向的节点
free(ps);
return 1;
}else{
printf("delete pos %d is error\n",pos);
return 0;
}
}
//该函数完成指定节点的指针返回,根据指定的位置返回节点的指针
pNode FindIndexNode(pNode pHead,int index)
{
pNode ps=NULL,pt=pHead->Next;
// index是从1开始计数的
int count=1;
while(count<index)
{
if(pt==pHead){
fprintf(stderr,"error happened in FindIndexNode\n");
fprintf(stderr,"index over the double-list-size\n");
return NULL;
}
pt=pt->Next;
count++;
}
return pt;
}
//将命中的数据页移动到MRU位置,输入参数是LRU队列和命中数据页的指针
int MoveToMRU(pNode pHead,pNode Hit)
{
pNode pt=NULL,ps=NULL;
//首先切断原来位置的前后链接关系
pt=Hit->Pre;
pt->Next=Hit->Next;
Hit->Next->Pre=pt;
//将命中的数据页潜入到头部
Hit->Pre=pHead;
Hit->Next=pHead->Next;
//链接关系
pHead->Next->Pre=Hit;
pHead->Next=Hit;
return 0;
}
//这个函数返回的是删除页的状态(是否为脏页),关于删除的页编号通过传值参数DelLPN改变
int DeleteLRU(pNode pHead,int *DelLPN)
{
// 删除指定队列的LRU位置的
int D_flag;
pNode pVictim=NULL,Ps;
pVictim=pHead->Pre;
// 错误测试
if(pVictim==pHead || pVictim==NULL){
fprintf(stderr,"error happened in DeleteLRU:\n");
fprintf(stderr,"List Maybe Empty,can not Delte Node\n");
assert(0);
}
// 删除尾部的节点,注意再函数调用结束修改对应的长度
* DelLPN=pVictim->LPN;
D_flag=pVictim->isD;
// 删除节点
Ps=pVictim->Pre;
Ps->Next=pHead;
pHead->Pre=Ps;
free(pVictim);
return D_flag;
}
//将一个全新的节点添加到队列的MRU位置
int AddNewToMRU(pNode pHead,pNode New)
{
// 需要注意的是新添加的节点的前后节点都没有关联任何位置节点
// Insert
New->Pre=pHead;
New->Next=pHead->Next;
// link
pHead->Next->Pre=New;
pHead->Next=New;
return 0;
}
//查看链表中的节点是否存在干净页节点,如果不存在干净页则返回NULL
pNode IsCleanNodeInList(pNode pHead)
{
pNode ps=NULL,pt=pHead->Pre;
while(pt!=pHead){
if(pt->isD==0){
ps=pt;
return ps;
}
pt=pt->Pre;
}
return ps;
}
//基于二次机会的冷探测机制,找到节点中isCold的节点,并返回该节点
pNode FindColdNodeInList(pNode pHead)
{
pNode Victim=NULL,pt=NULL;
int count=0,L;
L=GetListLength(pHead);
// 基于二次机会的冷探测机制,将isCold的节点置1移动到链表的MRU位置
Victim=pHead->Pre;
while(Victim->isCold!=1) {
pt = Victim;
Victim = Victim->Pre;
pt->isCold = 1;
//之前的Victim移动到Head后面
pt->Pre = pHead;
pt->Next = pHead->Next;
pHead->Next->Pre = pt;
pHead->Next = pt;
//衔接新的尾部
pHead->Pre = Victim;
Victim->Next = pHead;
//debug
count++;
if (count > L + 2) {
fprintf(stderr,"exist error in while\n");
assert(0);
}
}
//test for debug脏页提出一定最后是尾部的LRU
if (Victim != pHead->Pre) {
fprintf(stderr,"this operation exist error\n");
assert(0);
}
return Victim;
}
int DelVictimNodeInList(pNode pHead,pNode Victim)
{
pNode ps;
int DelLPN=-1;
DelLPN=Victim->LPN;
ps=Victim->Pre;
// 删除链接关系
ps->Next=Victim->Next;
Victim->Next->Pre=ps;
// 释放对应的节点Victim的内存
free(Victim);
return DelLPN;
}
//无论热区还是冷区,选择剔除的时候都是优先置换干净页,之后基于二次机会遍历选择脏页
//函数返回的是需要剔除页的节点指针
pNode FindVictimNode_CleanFirst(pNode pHead)
{
pNode Victim=NULL;
//debug test
if(IsEmptyList(pHead)!=0){
fprintf(stderr,"error happened in FindVictimNode_CleanFirst ");
fprintf(stderr,"the list is empty!\n");
assert(0);
}
//debug test
//先遍历寻找队列中尾部的干净页优先提出
Victim=IsCleanNodeInList(pHead);
if(Victim==NULL){
// 不存在干净页则选择脏页,基于二次机会找到脏页
Victim=FindColdNodeInList(pHead);
}
return Victim;
}
//YMB
int GetCListLength(pNode pHead)
{
int length=0;
pNode pt=pHead->Next;
while (pt!=pHead)
{
if(pt->isD==-1)
{
length++;
}
pt=pt->Next;
}
return length;
}
int GetDListLength(pNode pHead)
{
int length=0;
pNode pt=pHead->Next;
while (pt!=pHead)
{
if(pt->isD!=-1)
{
length++;
}
pt=pt->Next;
}
return length;
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。