1 Star 2 Fork 1

ZhouHeyu/Mix-SSD

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
disksim_diskcache.c 52.92 KB
一键复制 编辑 原始数据 按行查看 历史
颜明博 提交于 2018-06-25 19:31 . first commit
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698
/*
* DiskSim Storage Subsystem Simulation Environment (Version 3.0)
* Revision Authors: John Bucy, Greg Ganger
* Contributors: John Griffin, Jiri Schindler, Steve Schlosser
*
* Copyright (c) of Carnegie Mellon University, 2001, 2002, 2003.
*
* This software is being provided by the copyright holders under the
* following license. By obtaining, using and/or copying this software,
* you agree that you have read, understood, and will comply with the
* following terms and conditions:
*
* Permission to reproduce, use, and prepare derivative works of this
* software is granted provided the copyright and "No Warranty" statements
* are included with all reproductions and derivative works and associated
* documentation. This software may also be redistributed without charge
* provided that the copyright and "No Warranty" statements are included
* in all redistributions.
*
* NO WARRANTY. THIS SOFTWARE IS FURNISHED ON AN "AS IS" BASIS.
* CARNEGIE MELLON UNIVERSITY MAKES NO WARRANTIES OF ANY KIND, EITHER
* EXPRESSED OR IMPLIED AS TO THE MATTER INCLUDING, BUT NOT LIMITED
* TO: WARRANTY OF FITNESS FOR PURPOSE OR MERCHANTABILITY, EXCLUSIVITY
* OF RESULTS OR RESULTS OBTAINED FROM USE OF THIS SOFTWARE. CARNEGIE
* MELLON UNIVERSITY DOES NOT MAKE ANY WARRANTY OF ANY KIND WITH RESPECT
* TO FREEDOM FROM PATENT, TRADEMARK, OR COPYRIGHT INFRINGEMENT.
* COPYRIGHT HOLDERS WILL BEAR NO LIABILITY FOR ANY USE OF THIS SOFTWARE
* OR DOCUMENTATION.
*
*/
/*
* DiskSim Storage Subsystem Simulation Environment (Version 2.0)
* Revision Authors: Greg Ganger
* Contributors: Ross Cohen, John Griffin, Steve Schlosser
*
* Copyright (c) of Carnegie Mellon University, 1999.
*
* Permission to reproduce, use, and prepare derivative works of
* this software for internal use is granted provided the copyright
* and "No Warranty" statements are included with all reproductions
* and derivative works. This software may also be redistributed
* without charge provided that the copyright and "No Warranty"
* statements are included in all redistributions.
*
* NO WARRANTY. THIS SOFTWARE IS FURNISHED ON AN "AS IS" BASIS.
* CARNEGIE MELLON UNIVERSITY MAKES NO WARRANTIES OF ANY KIND, EITHER
* EXPRESSED OR IMPLIED AS TO THE MATTER INCLUDING, BUT NOT LIMITED
* TO: WARRANTY OF FITNESS FOR PURPOSE OR MERCHANTABILITY, EXCLUSIVITY
* OF RESULTS OR RESULTS OBTAINED FROM USE OF THIS SOFTWARE. CARNEGIE
* MELLON UNIVERSITY DOES NOT MAKE ANY WARRANTY OF ANY KIND WITH RESPECT
* TO FREEDOM FROM PATENT, TRADEMARK, OR COPYRIGHT INFRINGEMENT.
*/
/*
* DiskSim Storage Subsystem Simulation Environment
* Authors: Greg Ganger, Bruce Worthington, Yale Patt
*
* Copyright (C) 1993, 1995, 1997 The Regents of the University of Michigan
*
* This software is being provided by the copyright holders under the
* following license. By obtaining, using and/or copying this software,
* you agree that you have read, understood, and will comply with the
* following terms and conditions:
*
* Permission to use, copy, modify, distribute, and sell this software
* and its documentation for any purpose and without fee or royalty is
* hereby granted, provided that the full text of this NOTICE appears on
* ALL copies of the software and documentation or portions thereof,
* including modifications, that you make.
*
* THIS SOFTWARE IS PROVIDED "AS IS," AND COPYRIGHT HOLDERS MAKE NO
* REPRESENTATIONS OR WARRANTIES, EXPRESS OR IMPLIED. BY WAY OF EXAMPLE,
* BUT NOT LIMITATION, COPYRIGHT HOLDERS MAKE NO REPRESENTATIONS OR
* WARRANTIES OF MERCHANTABILITY OR FITNESS FOR ANY PARTICULAR PURPOSE OR
* THAT THE USE OF THE SOFTWARE OR DOCUMENTATION WILL NOT INFRINGE ANY
* THIRD PARTY PATENTS, COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS. COPYRIGHT
* HOLDERS WILL BEAR NO LIABILITY FOR ANY USE OF THIS SOFTWARE OR
* DOCUMENTATION.
*
* This software is provided AS IS, WITHOUT REPRESENTATION FROM THE
* UNIVERSITY OF MICHIGAN AS TO ITS FITNESS FOR ANY PURPOSE, AND
* WITHOUT WARRANTY BY THE UNIVERSITY OF MICHIGAN OF ANY KIND, EITHER
* EXPRESSED OR IMPLIED, INCLUDING WITHOUT LIMITATION THE IMPLIED
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE REGENTS
* OF THE UNIVERSITY OF MICHIGAN SHALL NOT BE LIABLE FOR ANY DAMAGES,
* INCLUDING SPECIAL , INDIRECT, INCIDENTAL, OR CONSEQUENTIAL DAMAGES,
* WITH RESPECT TO ANY CLAIM ARISING OUT OF OR IN CONNECTION WITH THE
* USE OF OR IN CONNECTION WITH THE USE OF THE SOFTWARE, EVEN IF IT HAS
* BEEN OR IS HEREAFTER ADVISED OF THE POSSIBILITY OF SUCH DAMAGES
*
* The names and trademarks of copyright holders or authors may NOT be
* used in advertising or publicity pertaining to the software without
* specific, written prior permission. Title to copyright in this software
* and any associated documentation will at all times remain with copyright
* holders.
*/
#include "disksim_global.h"
#include "disksim_iosim.h"
#include "disksim_stat.h"
#include "disksim_disk.h"
/* This variable is now in the diskinfo structure, but the real question */
/* is why it exists at all... */
//int LRU_at_seg_list_head = 0;
#if 0
static int disk_buffer_segment_wrap_needed (segment *seg, int endblkno)
{
return(seg->startblkno < (endblkno - seg->size));
}
#endif
void disk_buffer_segment_wrap (segment *seg, int endblkno)
{
seg->startblkno = max(seg->startblkno, (endblkno - seg->size));
}
/* Determines whether the iorequest passed in overlaps in any way with the given
* segment in the cache -rcohen
*/
int disk_buffer_overlap (segment *seg, ioreq_event *curr)
{
int tmp;
if ((curr->blkno >= seg->startblkno) && (curr->blkno < seg->endblkno)) {
return(TRUE);
}
tmp = curr->blkno + curr->bcount;
if ((tmp > seg->startblkno) && (tmp <= seg->endblkno)) {
return(TRUE);
}
if ((curr->blkno <= seg->startblkno) && (tmp >= seg->endblkno)) {
return(TRUE);
}
return(FALSE);
}
void disk_buffer_remove_from_seg (diskreq *currdiskreq)
{
segment *seg = currdiskreq->seg;
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, "%12.6f %8p Entering disk_buffer_remove_from_seg\n",simtime,currdiskreq);
fflush(outputfile);
}
ASSERT(seg != NULL);
if (seg->diskreqlist == currdiskreq) {
seg->diskreqlist = currdiskreq->seg_next;
} else {
diskreq *tmpdiskreq = seg->diskreqlist;
while (tmpdiskreq) {
if (tmpdiskreq->seg_next == currdiskreq) {
tmpdiskreq->seg_next = currdiskreq->seg_next;
break;
}
tmpdiskreq = tmpdiskreq->seg_next;
}
ASSERT(tmpdiskreq != NULL);
}
currdiskreq->seg = NULL;
ASSERT1((seg->diskreqlist || seg->state != BUFFER_DIRTY),"seg->state",seg->state);
}
/* return real or effective owner of segment or NULL. For effective, a
READ segment OWNER may be overridden by a READ with an outblkno that
is less than the (effective) OWNER's outblkno
*/
diskreq* disk_buffer_seg_owner (segment *seg, int effective)
{
diskreq *seg_owner = seg->diskreqlist;
diskreq *tmpdiskreq = seg->diskreqlist;
while (seg_owner) {
if (seg_owner->flags & SEG_OWNED) {
break;
}
seg_owner = seg_owner->seg_next;
}
if (effective && seg->access && (seg->access->flags & READ)) {
while (tmpdiskreq) {
if ((seg->recyclereq != tmpdiskreq) &&
(!tmpdiskreq->ioreqlist ||
(tmpdiskreq->ioreqlist->flags & READ))) {
if (!seg_owner || tmpdiskreq->outblkno < seg_owner->outblkno) {
seg_owner = tmpdiskreq;
/*
fprintf(stderr,"%.6f effective seg_owner found\n",simtime);fflush(stderr);
*/
}
}
tmpdiskreq = tmpdiskreq->seg_next;
}
}
return(seg_owner);
}
/* A recyclable/reusable segment has a single diskreq. That diskreq must be a
read which has all of its remaining data in the segment or a write which
only needs to send completion up the line.
*/
int disk_buffer_reusable_segment_check (disk *currdisk, segment *currseg)
{
diskreq *currdiskreq = currseg->diskreqlist;
ioreq_event *currioreq;
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, "%12.6f Entering disk_buffer_reusable_segment_check\n",simtime);
fflush(outputfile);
}
if (currdisk->acctime < 0.0 && !currseg->recyclereq && currdiskreq &&
!currdiskreq->seg_next && currdiskreq->ioreqlist &&
!(currdiskreq->flags & COMPLETION_RECEIVED)) {
if (currdiskreq->ioreqlist->flags & READ) {
currioreq = currdiskreq->ioreqlist;
while (currioreq->next) {
currioreq = currioreq->next;
}
if ((currdiskreq->outblkno >= currseg->startblkno) &&
(currdiskreq->outblkno < currseg->endblkno) &&
((currioreq->blkno + currioreq->bcount) > currseg->startblkno) &&
((currioreq->blkno + currioreq->bcount) <= currseg->endblkno)) {
if (!(currdiskreq->flags & SEG_OWNED)) {
disk_buffer_attempt_seg_ownership(currdisk,currdiskreq);
}
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, " segment %8p is reusable\n",currseg);
fflush(outputfile);
}
return(TRUE);
}
} else { /* WRITE */
currioreq = currdiskreq->ioreqlist;
while (currioreq->next) {
currioreq = currioreq->next;
}
if (currdiskreq->inblkno >= (currioreq->blkno + currioreq->bcount)) {
if (!(currdiskreq->flags & SEG_OWNED)) {
disk_buffer_attempt_seg_ownership(currdisk,currdiskreq);
}
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, " segment %8p is reusable\n",currseg);
fflush(outputfile);
}
return(TRUE);
}
}
}
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, " segment %8p is NOT reusable\n",currseg);
fflush(outputfile);
}
return(FALSE);
}
static int disk_buffer_recyclable_segment_check (disk *currdisk, segment *currseg, int isread)
{
diskreq *currdiskreq = currseg->diskreqlist;
ioreq_event *currioreq;
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, "%12.6f Entering disk_buffer_recyclable_segment_check\n",simtime);
fflush(outputfile);
}
if (currdisk->acctime < 0.0 && !currseg->recyclereq && currdiskreq &&
!currdiskreq->seg_next && currdiskreq->ioreqlist &&
!(currdiskreq->flags & COMPLETION_RECEIVED)) {
if (currdiskreq->ioreqlist->flags & READ) {
if (isread ||
(!currdisk->dedicatedwriteseg &&
(currdisk->numdirty < currdisk->numwritesegs))) {
currioreq = currdiskreq->ioreqlist;
while (currioreq->next) {
currioreq = currioreq->next;
}
if ((currdiskreq->outblkno >= currseg->startblkno) &&
(currdiskreq->outblkno < currseg->endblkno) &&
((currioreq->blkno + currioreq->bcount) > currseg->startblkno) &&
((currioreq->blkno + currioreq->bcount) <= currseg->endblkno)) {
if (!(currdiskreq->flags & SEG_OWNED)) {
disk_buffer_attempt_seg_ownership(currdisk,currdiskreq);
}
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, " segment %8p is recyclable\n",currseg);
fflush(outputfile);
}
return(TRUE);
}
}
} else { /* WRITE */
if (!isread || !currdisk->dedicatedwriteseg) {
currioreq = currdiskreq->ioreqlist;
while (currioreq->next) {
currioreq = currioreq->next;
}
if (currdiskreq->inblkno >= (currioreq->blkno + currioreq->bcount)) {
if (!(currdiskreq->flags & SEG_OWNED)) {
disk_buffer_attempt_seg_ownership(currdisk,currdiskreq);
}
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, " segment %8p is recyclable\n",currseg);
fflush(outputfile);
}
return(TRUE);
}
}
}
}
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, " segment %8p is NOT recyclable\n",currseg);
fflush(outputfile);
}
return(FALSE);
}
segment* disk_buffer_recyclable_segment (disk *currdisk, int isread)
{
segment *currseg = currdisk->seglist;
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, "%12.6f Entering disk_buffer_recyclable_segment\n",simtime);
fprintf (outputfile, " numdirty = %d\n", currdisk->numdirty);
fflush(outputfile);
}
if (currdisk->acctime >= 0.0) {
return(NULL);
}
while (currseg) {
if (disk_buffer_recyclable_segment_check(currdisk, currseg, isread)) {
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, " recyclable segment found\n");
fflush(outputfile);
}
return(currseg);
}
currseg = currseg->next;
}
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf(outputfile, " No recyclable segment found\n");
fflush(outputfile);
}
return(NULL);
}
/* if BUFFER_READING, checks to see if cache can and should "fit in" blkno.
if BUFFER_WRITING, checks to see if blkno is in the cache and can be
written to media.
*/
int disk_buffer_block_available (disk *currdisk, segment *seg, int blkno)
{
diskreq *seg_owner;
if (blkno < 0) {
return(FALSE);
}
if (seg->state == BUFFER_READING) {
if ((blkno < seg->endblkno) && (blkno >= seg->startblkno)) {
return(FALSE);
}
if ((blkno < seg->startblkno) &&
((!currdisk->readanyfreeblocks) ||
(seg->endblkno != seg->startblkno))) {
return(FALSE);
}
seg_owner = disk_buffer_seg_owner(seg,TRUE);
if (seg_owner && !(seg_owner->flags & COMPLETION_SENT) &&
((blkno - seg_owner->outblkno) >= seg->size)) {
return(FALSE);
}
} else if (seg->state == BUFFER_WRITING) {
if ((blkno < currdisk->effectivehda->inblkno) ||
(blkno >= seg->endblkno)) {
return(FALSE);
}
if ((currdisk->effectivehda->ioreqlist) &&
(currdisk->extra_write_disconnect)) {
if (currdisk->effectivehda->ioreqlist->bcount < 45) {
if ((seg->endblkno - seg->startblkno) <
min(currdisk->effectivehda->ioreqlist->bcount, 10)) {
return(FALSE);
}
} else {
if ((seg->endblkno - seg->startblkno) < 15) {
return(FALSE);
}
}
}
} else {
fprintf(stderr, "Not actively using disk at disk_buffer_block_available\n");
exit(1);
}
return(TRUE);
}
/* This function may be out of date. Goal is/was to assist in understanding */
/* how sequentiality/locality is affected by disk array data organizations */
/* and disk striping in particular. */
void disk_interferestats (disk *currdisk, ioreq_event *curr)
{
if (!device_printinterferestats) {
return;
}
if ((curr->cause != currdisk->lastgen) && (curr->flags & (SEQ|LOCAL))) {
if (curr->flags & SEQ) {
currdisk->stat.interfere[0]++;
} else {
currdisk->stat.interfere[1]++;
}
}
currdisk->lastgen = curr->cause;
}
static void disk_buffer_stats (disk *currdisk, ioreq_event *curr, segment *seg, int hittype)
{
double size;
if (!device_printbufferstats) {
return;
}
/* Add stats for read hits on write data? */
switch (hittype) {
case BUFFER_NOMATCH:
if (curr->flags & READ) {
currdisk->stat.readmisses++;
} else {
currdisk->stat.writemisses++;
}
break;
case BUFFER_WHOLE:
currdisk->stat.fullreadhits++;
break;
case BUFFER_APPEND:
currdisk->stat.appendhits++;
break;
case BUFFER_PREPEND:
currdisk->stat.prependhits++;
break;
case BUFFER_PARTIAL:
if (seg->startblkno <= curr->blkno) {
size = (double) (seg->endblkno - curr->blkno);
} else {
/*
size = (double) (curr->blkno + curr->bcount - seg->startblkno);
*/
fprintf(stderr, "Tail hits not currently allowed.\n");
exit(1);
}
if (seg->state == BUFFER_READING) {
/*
fprintf (outputfile, "Ongoing hit: blkno %d bcount %d segstart %d segstop %d read %d\n", curr->blkno, curr->bcount, seg->startblkno, seg->endblkno, seg->access->type);
*/
currdisk->stat.readinghits++;
currdisk->stat.runreadingsize += size;
currdisk->stat.remreadingsize += curr->bcount - size;
} else {
/*
fprintf (outputfile, "Partial hit: blkno %d bcount %d segstart %d segstop %d read %d\n", curr->blkno, curr->bcount, seg->startblkno, seg->endblkno, ((curr->flags & READ) && (seg->state == BUFFER_CLEAN)));
*/
currdisk->stat.parthits++;
currdisk->stat.runpartsize += size;
currdisk->stat.rempartsize += curr->bcount - size;
}
break;
default:
fprintf(stderr, "Invalid hittype in disk_buffer_stats - blkno %d, bcount %d, state %d\n", curr->blkno, curr->bcount, seg->state);
exit(1);
}
return;
}
/* segment selection priority for read requests:
11 BUFFER_WHOLE, BUFFER_DIRTY (write data is freshest, less chance of
data scrolling off the segment quickly)
10 BUFFER_WRITING (write data is freshest)
9 BUFFER_READING (possible prefetch-stream hit)
8 BUFFER_CLEAN
7 BUFFER_PARTIAL, BUFFER_DIRTY (write data is freshest)
6 BUFFER_WRITING (write data is freshest)
5 BUFFER_READING (possible prefetch-stream hit, includes
case of current prefetch block is first
block in the new request)
4 BUFFER_CLEAN
3 BUFFER_NOMATCH, BUFFER_EMPTY (doesn't trash other data)
2 BUFFER_CLEAN
1 BUFFER_READING (must be last place for preempt check)
not usable:
BUFFER_NOMATCH, BUFFER_DIRTY
BUFFER_WRITING
BUFFER_CLEAN (if pending requests)
BUFFER_READING (if not pre-emptable or pending requests)
recycled segs
dedicatedwriteseg
not usable if !enablecache
BUFFER_DIRTY
BUFFER_WRITING
BUFFER_READING (if non-stoppable)
not usable if no read hits on write data:
BUFFER_WHOLE, BUFFER_DIRTY
BUFFER_WRITING
BUFFER_CLEAN (if previous request was a write)
BUFFER_PARTIAL, BUFFER_DIRTY
BUFFER_WRITING
BUFFER_CLEAN (if previous request was a write)
NO seg usable if overlapping dirty data exists which cannot be "hit"
-- return BUFFER_COLLISION hittype if seg in question is not "reusable".
NO seg usable if two segments are found with overlapping dirty data
-- return BUFFER_COLLISION hittype.
*/
/* this routine sets currdiskreq->seg and currdiskreq->hittype */
/* as per above table, it finds the best segment in the cache matching this
* request. If it doesn't find one, it sets seg to NULL and hittype to
* BUFFER_NOMATCH */
static segment *
disk_buffer_select_read_segment(disk *currdisk, diskreq *currdiskreq)
{
segment *seg;
ioreq_event *first_ioreq = currdiskreq->ioreqlist;
ioreq_event *tmpioreq;
int best_value = 0;
int curr_value;
int curr_hittype = 0;
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, "%12.6f %8p Entering disk_buffer_select_read_segment\n",simtime,currdiskreq);
fflush(outputfile);
}
currdiskreq->seg = NULL;
currdiskreq->hittype = BUFFER_NOMATCH;
seg = currdisk->seglist;
while (seg) {
curr_value = -1;
if (currdiskreq->hittype == BUFFER_COLLISION) { }
else if (seg->recyclereq) { }
else if ((currdisk->dedicatedwriteseg)
&& (seg == currdisk->dedicatedwriteseg))
{
/* check for collision with dirty data */
if ((seg->state == BUFFER_DIRTY)
|| (seg->state == BUFFER_WRITING))
{
tmpioreq = currdiskreq->ioreqlist;
while (tmpioreq) {
if (disk_buffer_overlap(seg,tmpioreq)) {
if (!disk_buffer_reusable_segment_check(currdisk, seg)) {
currdiskreq->hittype = BUFFER_COLLISION;
currdiskreq->seg = NULL;
}
break;
}
tmpioreq = tmpioreq->next;
}
}
}
else if (seg->state == BUFFER_EMPTY) {
curr_value = 3;
curr_hittype = BUFFER_NOMATCH;
}
else if ((first_ioreq->blkno < seg->startblkno)
|| (first_ioreq->blkno >= seg->endblkno))
{
curr_hittype = BUFFER_NOMATCH;
if ((seg->state == BUFFER_CLEAN) && !(seg->diskreqlist)) {
curr_value = 2;
}
else if (seg->state == BUFFER_READING) {
if ((seg->endblkno == first_ioreq->blkno) &&
currdisk->almostreadhits) {
/* special case: block currently being prefetched is the
* first block of this request. We count this as a
* partial read hit.
*/
curr_hittype = BUFFER_PARTIAL;
curr_value = 5;
}
else {
if (!seg->diskreqlist->ioreqlist) {
curr_value = 1;
}
}
}
else if ((first_ioreq->blkno < seg->startblkno)
&& ((seg->state == BUFFER_DIRTY)
|| (seg->state == BUFFER_WRITING)))
{
/* check for collision with dirty data */
tmpioreq = currdiskreq->ioreqlist;
while (tmpioreq) {
if (disk_buffer_overlap(seg,tmpioreq)) {
if (!disk_buffer_reusable_segment_check(currdisk, seg)) {
currdiskreq->hittype = BUFFER_COLLISION;
currdiskreq->seg = NULL;
}
break;
}
tmpioreq = tmpioreq->next;
}
}
}
else {
switch (seg->state) {
case BUFFER_DIRTY:
if (currdisk->enablecache) {
if ((currdisk->readhitsonwritedata)
&& (best_value < 10)) {
curr_value = 11;
}
else {
if (!disk_buffer_reusable_segment_check(currdisk, seg)) {
currdiskreq->hittype = BUFFER_COLLISION;
currdiskreq->seg = NULL;
}
}
}
break;
case BUFFER_WRITING:
if (currdisk->enablecache) {
if ((currdisk->readhitsonwritedata)
&& (best_value < 10))
{
curr_value = 10;
}
else {
if (!disk_buffer_reusable_segment_check(currdisk, seg)) {
currdiskreq->hittype = BUFFER_COLLISION;
currdiskreq->seg = NULL;
}
}
}
break;
case BUFFER_READING:
if (currdisk->enablecache) {
curr_value = 9;
}
break;
case BUFFER_CLEAN:
if (currdisk->readhitsonwritedata
|| (seg->access && (seg->access->flags & READ)))
{
curr_value = 8;
}
break;
default:
ddbg_assert(0);
}
if (seg->endblkno >= (first_ioreq->blkno + first_ioreq->bcount)) {
curr_hittype = BUFFER_WHOLE;
}
else {
curr_hittype = BUFFER_PARTIAL;
curr_value -= 4;
}
}
if (curr_value > best_value) {
currdiskreq->seg = seg;
currdiskreq->hittype = curr_hittype;
best_value = curr_value;
}
seg = seg->next;
}
/* If BUFFER_NOMATCH && BUFFER_READING, perform preemption check.
* Note that currdiskreq->seg should be set before the call.
*/
if ((best_value == 1)
&& !disk_buffer_stopable_access(currdisk,currdiskreq))
{
currdiskreq->seg = NULL;
}
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, " segment = %8p, hittype = %d\n",currdiskreq->seg,currdiskreq->hittype);
fflush(outputfile);
}
return currdiskreq->seg;
}
/* segment selection priority for write requests:
7 BUFFER_APPEND, BUFFER_WRITING (keep the stream going)
6 BUFFER_DIRTY
5 BUFFER_PREPEND, BUFFER_DIRTY
4 BUFFER_NOMATCH, BUFFER_CLEAN, overlapping (will be emptied anyways)
3 BUFFER_EMPTY (don't trash other data)
2 BUFFER_CLEAN
1 BUFFER_NOMATCH, BUFFER_READING (must be last place for preempt check)
not usable:
BUFFER_APPEND (if no combining writes)
BUFFER_PREPEND (if no combining writes)
BUFFER_NOMATCH, BUFFER_DIRTY
BUFFER_WRITING
BUFFER_READING (if not pre-emptable or pending requests)
BUFFER_CLEAN (if pending requests)
recycled segs
NO seg usable if overlapping data exists which cannot be "cleaned"
-- return BUFFER_COLLISION hittype if seg is question isn't "reusable".
*/
/* this routine sets currdiskreq->seg and currdiskreq->hittype */
static segment* disk_buffer_select_write_segment(disk *currdisk, diskreq *currdiskreq)
{
segment *seg;
diskreq *tmp_diskreq;
diskreq *holddiskreq;
ioreq_event *first_ioreq;
ioreq_event *last_ioreq;
ioreq_event *tmpioreq;
int best_value = 0;
int curr_value;
int curr_hittype;
/*
int reusable_dirty_segment = FALSE;
*/
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, "%12.6f %8p Entering disk_buffer_select_write_segment\n",simtime,currdiskreq);
fprintf (outputfile, " numdirty = %d\n", currdisk->numdirty);
fflush(outputfile);
}
currdiskreq->seg = NULL;
currdiskreq->hittype = BUFFER_NOMATCH;
seg = currdisk->seglist;
while (seg) {
curr_value = -1;
curr_hittype = BUFFER_NOMATCH;
if (currdiskreq->hittype == BUFFER_COLLISION) {
} else if (seg->recyclereq) {
} else if ((currdisk->dedicatedwriteseg) &&
(seg != currdisk->dedicatedwriteseg)) {
/* check for collision with uncleanable data */
if (((seg->state == BUFFER_CLEAN) &&
(seg->diskreqlist)) ||
((seg->state == BUFFER_READING) &&
((seg->diskreqlist->ioreqlist) ||
(seg->diskreqlist->seg_next) ||
!disk_buffer_stopable_access(currdisk,currdiskreq)))) {
tmpioreq = currdiskreq->ioreqlist;
while (tmpioreq) {
if (disk_buffer_overlap(seg,tmpioreq)) {
if (!disk_buffer_reusable_segment_check(currdisk, seg)) {
currdiskreq->hittype = BUFFER_COLLISION;
currdiskreq->seg = NULL;
}
break;
}
tmpioreq = tmpioreq->next;
}
}
} else if (seg->state == BUFFER_EMPTY) {
if (currdisk->numdirty < currdisk->numwritesegs) {
/*
|| reusable_dirty_segment) {
*/
curr_value = 3;
}
} else if (seg->state == BUFFER_READING) {
if (!seg->diskreqlist->ioreqlist && !seg->diskreqlist->seg_next) {
if (currdisk->numdirty < currdisk->numwritesegs) {
/*
|| reusable_dirty_segment) {
*/
curr_value = 1;
}
} else {
tmpioreq = currdiskreq->ioreqlist;
while (tmpioreq) {
if (disk_buffer_overlap(seg,tmpioreq)) {
if (!disk_buffer_reusable_segment_check(currdisk, seg)) {
currdiskreq->hittype = BUFFER_COLLISION;
currdiskreq->seg = NULL;
}
break;
}
tmpioreq = tmpioreq->next;
}
}
} else if (seg->state == BUFFER_CLEAN) {
if (!seg->diskreqlist) {
if (currdisk->numdirty < currdisk->numwritesegs) {
/*
|| reusable_dirty_segment) {
*/
curr_value = 2;
first_ioreq = currdiskreq->ioreqlist;
while (first_ioreq) {
if (disk_buffer_overlap(seg,first_ioreq)) {
curr_value = 4;
break;
}
first_ioreq = first_ioreq->next;
}
}
} else {
/* check for collision with uncleanable data */
tmpioreq = currdiskreq->ioreqlist;
while (tmpioreq) {
if (disk_buffer_overlap(seg,tmpioreq)) {
if (!disk_buffer_reusable_segment_check(currdisk, seg)) {
currdiskreq->hittype = BUFFER_COLLISION;
currdiskreq->seg = NULL;
}
break;
}
tmpioreq = tmpioreq->next;
}
}
} else if ((!currdisk->writecomb) ||
(seg->outstate == BUFFER_TRANSFERING)) {
/* no PREPEND's or APPEND's allowed */
/* check for collision with uncleanable data */
tmpioreq = currdiskreq->ioreqlist;
while (tmpioreq) {
if (disk_buffer_overlap(seg,tmpioreq)) {
if (!disk_buffer_reusable_segment_check(currdisk, seg)) {
currdiskreq->hittype = BUFFER_COLLISION;
currdiskreq->seg = NULL;
}
break;
}
tmpioreq = tmpioreq->next;
}
/*
if ((currdisk->numdirty >= currdisk->numwritesegs) &&
!reusable_dirty_segment &&
disk_buffer_reusable_segment_check(currdisk, seg)) {
reusable_dirty_segment = TRUE;
currdiskreq->seg = NULL;
currdiskreq->hittype = BUFFER_NOMATCH;
seg = currdisk->seglist;
best_value = 0;
continue;
}
*/
} else {
first_ioreq = currdiskreq->ioreqlist;
tmp_diskreq = holddiskreq = seg->diskreqlist;
while (tmp_diskreq->seg_next) {
tmp_diskreq = tmp_diskreq->seg_next;
if (!(tmp_diskreq->ioreqlist->flags & READ)) {
holddiskreq = tmp_diskreq;
}
}
last_ioreq = holddiskreq->ioreqlist;
while (last_ioreq && last_ioreq->next) {
last_ioreq = last_ioreq->next;
}
if ((first_ioreq->blkno == (last_ioreq->blkno + last_ioreq->bcount)) &&
(first_ioreq->blkno == seg->endblkno) &&
(seg->size > (first_ioreq->blkno - (disk_buffer_seg_owner(seg,FALSE))->inblkno))) {
curr_hittype = BUFFER_APPEND;
curr_value = ((seg->state == BUFFER_WRITING) ? 7 : 6);
} else if (seg->state == BUFFER_DIRTY) {
int total_bcount = currdiskreq->ioreqlist->bcount;
last_ioreq = currdiskreq->ioreqlist;
while (last_ioreq->next) {
last_ioreq = last_ioreq->next;
total_bcount += last_ioreq->bcount;
}
holddiskreq = seg->diskreqlist;
while (holddiskreq->ioreqlist->flags & READ) {
holddiskreq = holddiskreq->seg_next;
}
first_ioreq = holddiskreq->ioreqlist;
if ((first_ioreq->blkno == (last_ioreq->blkno + last_ioreq->bcount)) &&
(first_ioreq->blkno == seg->startblkno) &&
((seg->startblkno == seg->endblkno) || (seg->hold_bcount == 0)) &&
(seg->size >= ((seg->endblkno - seg->startblkno) + total_bcount)) &&
(!currdisk->extradisc_diskreq ||
(currdisk->extradisc_diskreq->seg != seg))) {
curr_hittype = BUFFER_PREPEND;
curr_value = 5;
}
}
if (curr_hittype == BUFFER_NOMATCH) {
/* check for collision with uncleanable data */
tmpioreq = currdiskreq->ioreqlist;
while (tmpioreq) {
if (disk_buffer_overlap(seg,tmpioreq)) {
if (!disk_buffer_reusable_segment_check(currdisk, seg)) {
currdiskreq->hittype = BUFFER_COLLISION;
currdiskreq->seg = NULL;
}
break;
}
tmpioreq = tmpioreq->next;
}
/*
if ((currdisk->numdirty >= currdisk->numwritesegs) &&
!reusable_dirty_segment &&
disk_buffer_reusable_segment_check(currdisk, seg)) {
reusable_dirty_segment = TRUE;
currdiskreq->seg = NULL;
currdiskreq->hittype = BUFFER_NOMATCH;
seg = currdisk->seglist;
best_value = 0;
continue;
}
*/
}
}
if (curr_value > best_value) {
currdiskreq->seg = seg;
currdiskreq->hittype = curr_hittype;
best_value = curr_value;
}
seg = seg->next;
}
/* If BUFFER_NOMATCH && BUFFER_READING, perform preemption check.
Note that currdiskreq->seg should be set before the call.
*/
if ((best_value == 1) &&
!disk_buffer_stopable_access(currdisk,currdiskreq)) {
/* check for collision with uncleanable data */
tmpioreq = currdiskreq->ioreqlist;
while (tmpioreq) {
if (disk_buffer_overlap(currdiskreq->seg,tmpioreq)) {
if (!disk_buffer_reusable_segment_check(currdisk, seg)) {
currdiskreq->hittype = BUFFER_COLLISION;
}
break;
}
tmpioreq = tmpioreq->next;
}
currdiskreq->seg = NULL;
}
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, " segment = %8p, hittype = %d\n",currdiskreq->seg,currdiskreq->hittype);
fflush(outputfile);
}
return(currdiskreq->seg);
}
segment * disk_buffer_select_segment (disk *currdisk, diskreq *currdiskreq,
int set_extradisc)
{
segment *seg;
if (currdisk->const_acctime) {
currdiskreq->seg = NULL;
currdiskreq->hittype = BUFFER_NOMATCH;
return(NULL);
}
if(currdiskreq->ioreqlist->flags & READ) {
seg = disk_buffer_select_read_segment(currdisk, currdiskreq);
if (currdisk->dedicatedwriteseg && (seg == currdisk->dedicatedwriteseg)) {
fprintf(stderr, "Read request about to use the write segment\n");
exit(1);
}
}
else { /* WRITE */
seg = disk_buffer_select_write_segment(currdisk, currdiskreq);
}
/* if suffering extra write disconnects and no extradisc_diskreq exists */
/* and this is a write following a non-sequential write, set the */
/* EXTRA_WRITE_DISCONNECT diskreq flag. */
if (set_extradisc
&& currdisk->extra_write_disconnect
&& !currdisk->neverdisconnect
&& !currdisk->extradisc_diskreq
&& !(currdiskreq->ioreqlist->flags & READ)
&& seg
&& ((LRU_at_seg_list_head && !seg->prev)
|| (!LRU_at_seg_list_head && !seg->next))
&& seg->access
&& !(seg->access->flags & READ)
&& (currdiskreq->ioreqlist->blkno != seg->endblkno))
{
currdiskreq->flags |= EXTRA_WRITE_DISCONNECT;
}
return(seg);
}
/* check for BUFFER hit (and no COLLISIONS's) */
static int disk_buffer_check_read_segments (disk *currdisk, ioreq_event *currioreq, int* buffer_reading)
{
segment *seg;
int return_hittype = BUFFER_NOMATCH;
int read_hit_on_dirty_data = FALSE;
if ((disk_printhack > 1) && (simtime >= disk_printhacktime)) {
fprintf (outputfile, "%12.6f %8p Entering disk_buffer_check_read_segments\n",simtime,currioreq);
fflush(outputfile);
}
if (!currdisk->enablecache) {
return(BUFFER_NOMATCH);
}
seg = currdisk->seglist;
while (seg) {
if (seg->recyclereq) {
} else if (seg->state == BUFFER_EMPTY) {
} else if ((currdisk->dedicatedwriteseg) &&
(seg == currdisk->dedicatedwriteseg)) {
/* check for collision with dirty data */
if ((seg->state == BUFFER_DIRTY) || (seg->state == BUFFER_WRITING)) {
if (disk_buffer_overlap(seg,currioreq) &&
!disk_buffer_reusable_segment_check(currdisk, seg)) {
return(BUFFER_COLLISION);
}
}
} else if ((currioreq->blkno < seg->startblkno) ||
(currioreq->blkno >= seg->endblkno)) {
if (seg->state == BUFFER_CLEAN) {
} else if (seg->state == BUFFER_READING) {
if ((seg->endblkno == currioreq->blkno) &&
currdisk->almostreadhits) {
/* special case: block currently being prefetched is
the first block of this request. We count this as
a partial read hit.
*/
if (return_hittype == BUFFER_NOMATCH) {
*buffer_reading = TRUE;
return_hittype = BUFFER_PARTIAL;
}
}
} else if (currioreq->blkno < seg->startblkno) {
/* check for collision with dirty data */
if (disk_buffer_overlap(seg,currioreq) &&
!disk_buffer_reusable_segment_check(currdisk, seg)) {
return(BUFFER_COLLISION);
}
}
} else {
switch (seg->state) {
case BUFFER_DIRTY:
case BUFFER_WRITING:
if (read_hit_on_dirty_data ||
(!currdisk->readhitsonwritedata &&
!disk_buffer_reusable_segment_check(currdisk, seg))) {
return(BUFFER_COLLISION);
}
read_hit_on_dirty_data = TRUE;
break;
case BUFFER_READING:
break;
case BUFFER_CLEAN:
if (!currdisk->readhitsonwritedata &&
seg->access && !(seg->access->flags & READ)) {
continue;
}
break;
default:
ddbg_assert(0);
}
if (seg->endblkno >= (currioreq->blkno + currioreq->bcount)) {
return_hittype = BUFFER_WHOLE;
*buffer_reading = (seg->state == BUFFER_READING);
} else if (return_hittype != BUFFER_WHOLE) {
return_hittype = BUFFER_PARTIAL;
*buffer_reading = (seg->state == BUFFER_READING);
}
}
seg = seg->next;
}
if ((disk_printhack > 1) && (simtime >= disk_printhacktime) &&
(return_hittype != BUFFER_NOMATCH)) {
fprintf (outputfile, " HITable segment found\n");
fflush(outputfile);
}
return(return_hittype);
}
/* check for directly APPENDable WRITING segment (and no COLLISION's) */
static int disk_buffer_check_write_segments (disk *currdisk, ioreq_event *currioreq)
{
segment *seg;
int return_hittype = BUFFER_NOMATCH;
diskreq *tmp_diskreq;
diskreq *holddiskreq;
ioreq_event *last_ioreq;
if ((disk_printhack > 1) && (simtime >= disk_printhacktime)) {
fprintf (outputfile, "%12.6f %8p Entering disk_buffer_check_write_segments\n",simtime,currioreq);
fflush(outputfile);
}
if (!currdisk->writecomb) {
return(BUFFER_NOMATCH);
}
seg = currdisk->seglist;
while (seg) {
if (seg->recyclereq) {
}
else if (seg->state == BUFFER_EMPTY) {
}
else if (seg->state == BUFFER_READING) {
return(BUFFER_NOMATCH);
}
else if ((currdisk->dedicatedwriteseg) &&
(seg != currdisk->dedicatedwriteseg)) {
/* check for collision with uncleanable data */
if ((seg->state == BUFFER_CLEAN) && seg->diskreqlist) {
if (disk_buffer_overlap(seg,currioreq) &&
!disk_buffer_reusable_segment_check(currdisk, seg)) {
return(BUFFER_COLLISION);
}
}
}
else if ((seg->state == BUFFER_CLEAN) ||
(seg->state == BUFFER_DIRTY))
{
/* check for collision with uncleanable data */
if (seg->diskreqlist) {
if (disk_buffer_overlap(seg,currioreq) &&
!disk_buffer_reusable_segment_check(currdisk, seg)) {
return(BUFFER_COLLISION);
}
}
}
else if (seg->outstate == BUFFER_TRANSFERING) {
/* check for collision with uncleanable data */
if (disk_buffer_overlap(seg,currioreq) &&
!disk_buffer_reusable_segment_check(currdisk, seg)) {
return(BUFFER_COLLISION);
}
}
else {
tmp_diskreq = holddiskreq = seg->diskreqlist;
while (tmp_diskreq->seg_next) {
tmp_diskreq = tmp_diskreq->seg_next;
if (!(tmp_diskreq->ioreqlist->flags & READ)) {
holddiskreq = tmp_diskreq;
}
}
last_ioreq = holddiskreq->ioreqlist;
while (last_ioreq && last_ioreq->next) {
last_ioreq = last_ioreq->next;
}
if ((currioreq->blkno == (last_ioreq->blkno + last_ioreq->bcount)) &&
(currioreq->blkno == seg->endblkno) &&
(seg->size > (currioreq->blkno - (disk_buffer_seg_owner(seg,FALSE))->inblkno))) {
return_hittype = BUFFER_APPEND;
}
else {
/* check for collision with uncleanable data */
if (disk_buffer_overlap(seg,currioreq) &&
!disk_buffer_reusable_segment_check(currdisk, seg)) {
return(BUFFER_COLLISION);
}
}
}
seg = seg->next;
}
if ((disk_printhack > 1) && (simtime >= disk_printhacktime) &&
(return_hittype == BUFFER_APPEND)) {
fprintf (outputfile, " APPENDable segment found\n");
fflush(outputfile);
}
return(return_hittype);
}
/* Checks for cache/buffer hits */
int disk_buffer_check_segments (disk *currdisk, ioreq_event *currioreq,
int* buffer_reading)
{
if (currioreq->flags & READ) {
return(disk_buffer_check_read_segments(currdisk, currioreq, buffer_reading));
} else {
return(disk_buffer_check_write_segments(currdisk, currioreq));
}
}
void disk_buffer_set_segment (disk *currdisk, diskreq *currdiskreq)
{
segment *seg = currdiskreq->seg;
segment *tmp_seg;
diskreq *tmp_diskreq;
ioreq_event *tmp_ioreq;
int is_read = (currdiskreq->ioreqlist->flags & READ);
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, "%12.6f %8p Entering disk_buffer_set_segment\n",simtime,currdiskreq);
fflush(outputfile);
}
if (!seg) {
fprintf(stderr, "diskreq has NULL segment in disk_buffer_set_segment\n");
exit(1);
}
/* check to make sure segment/diskreq combination is valid */
if (is_read && (seg == currdisk->dedicatedwriteseg)) {
fprintf(stderr, "Read request about to use the write segment\n");
exit(1);
}
if (seg->recyclereq && (currdiskreq != seg->recyclereq)) {
fprintf(stderr, "non-recyclereq detected for recycled segment in disk_buffer_set_segment\n");
exit(1);
}
switch (seg->state) {
case BUFFER_EMPTY:
if (currdiskreq == seg->recyclereq) {
fprintf(stderr, "seg->state of BUFFER_EMPTY for recyclereq in disk_buffer_set_segment\n");
exit(1);
}
break;
case BUFFER_CLEAN:
if (currdiskreq == seg->recyclereq) {
fprintf(stderr, "seg->state of BUFFER_CLEAN for recyclereq in disk_buffer_set_segment\n");
exit(1);
}
if (is_read && currdiskreq->hittype != BUFFER_NOMATCH && !currdisk->readhitsonwritedata && seg->access && !(seg->access->flags & READ)) {
fprintf(stderr, "Unallowed read hit on write data disk_buffer_set_segment\n");
exit(1);
}
break;
case BUFFER_DIRTY:
if (currdiskreq == seg->recyclereq) {
fprintf(stderr, "seg->state of BUFFER_DIRTY for recyclereq in disk_buffer_set_segment\n");
exit(1);
}
if (currdiskreq->hittype == BUFFER_NOMATCH) {
fprintf(stderr, "Attempt to overwrite dirty data in disk_buffer_set_segment\n");
exit(1);
} else if (is_read && !currdisk->readhitsonwritedata) {
fprintf(stderr, "Unallowed read hit on write data disk_buffer_set_segment\n");
exit(1);
}
if (!is_read) {
currdisk->stat.writecombs++;
}
break;
case BUFFER_READING:
if (currdiskreq == seg->recyclereq) {
if (!(currdiskreq->ioreqlist->flags & READ)) {
fprintf(stderr, "seg->state of BUFFER_READING for write recyclereq in disk_buffer_set_segment\n");
exit(1);
}
} else if (!is_read || (currdiskreq->hittype == BUFFER_NOMATCH)) {
if (!disk_buffer_stopable_access(currdisk,currdiskreq)) {
fprintf(stderr, "Attempt to re-target active read segment in disk_buffer_set_segment\n");
exit(1);
}
}
break;
case BUFFER_WRITING:
if (currdiskreq == seg->recyclereq) {
if (currdiskreq->ioreqlist->flags & READ) {
fprintf(stderr, "seg->state of BUFFER_WRITING for read recyclereq in disk_buffer_set_segment\n");
exit(1);
}
} else {
if ((currdiskreq->hittype == BUFFER_NOMATCH) ||
(currdiskreq->hittype == BUFFER_PREPEND)) {
fprintf(stderr, "Attempt to re-target active write segment in disk_buffer_set_segment\n");
exit(1);
}
if (!is_read) {
currdisk->stat.writecombs++;
}
}
break;
default:
fprintf(stderr, "Invalid segment state in disk_buffer_set_segment - blkno %d, bcount %d, state %d\n", currdiskreq->ioreqlist->blkno, currdiskreq->ioreqlist->bcount, seg->state);
exit(1);
}
/* end of sanity checks, now do the work */
disk_buffer_stats(currdisk,currdiskreq->ioreqlist,seg,currdiskreq->hittype);
if (LRU_at_seg_list_head) {
/* place segment at beginning of disk's segment list (for LRU) */
if (seg->prev) {
seg->prev->next = seg->next;
if (seg->next) {
seg->next->prev = seg->prev;
}
seg->next = currdisk->seglist;
currdisk->seglist->prev = seg;
seg->prev = NULL;
currdisk->seglist = seg;
}
} else {
/* place segment at end of disk's segment list (for LRU) */
if (seg->next) {
if (seg->prev) {
seg->prev->next = seg->next;
} else {
currdisk->seglist = seg->next;
}
seg->next->prev = seg->prev;
tmp_seg = currdisk->seglist;
while (tmp_seg->next) {
tmp_seg = tmp_seg->next;
}
tmp_seg->next = seg;
seg->prev = tmp_seg;
seg->next = NULL;
}
}
/* clear out clean, non-active, overlapping segments if write */
if (!is_read) {
tmp_seg = currdisk->seglist;
while (tmp_seg) {
if (tmp_seg != seg) {
tmp_ioreq= currdiskreq->ioreqlist;
while (tmp_ioreq) {
if ((tmp_seg->state != BUFFER_EMPTY) &&
disk_buffer_overlap(tmp_seg,tmp_ioreq)) {
if ((tmp_seg->state == BUFFER_CLEAN) &&
(tmp_seg->diskreqlist == NULL)) {
tmp_seg->state = BUFFER_EMPTY;
/*
} else {
fprintf(stderr, "Request overlaps with non-clean segment in disk_buffer_set_segment\n");
exit(1);
*/
}
}
tmp_ioreq = tmp_ioreq->next;
}
}
tmp_seg = tmp_seg->next;
}
}
/* add diskreq to seg's diskreq list, sorted in ascending blkno order */
if (!seg->diskreqlist) {
if (seg->access == NULL) {
seg->access = ioreq_copy(currdiskreq->ioreqlist);
seg->access->type = NULL_EVENT;
}
seg->diskreqlist = currdiskreq;
currdiskreq->seg_next = NULL;
} else { /* non-empty list */
diskreq *prev_diskreq = 0;
tmp_diskreq = seg->diskreqlist;
while (tmp_diskreq) {
if (tmp_diskreq->ioreqlist &&
(currdiskreq->ioreqlist->blkno < tmp_diskreq->ioreqlist->blkno)) {
currdiskreq->seg_next = tmp_diskreq;
if (seg->diskreqlist == tmp_diskreq) {
seg->diskreqlist = currdiskreq;
} else {
prev_diskreq->seg_next = currdiskreq;
}
break;
}
prev_diskreq = tmp_diskreq;
tmp_diskreq = tmp_diskreq->seg_next;
}
if (!tmp_diskreq) {
prev_diskreq->seg_next = currdiskreq;
currdiskreq->seg_next = NULL;
}
}
/* set watermark */
/* I'm still figuring out this watermark business -rcohen */
tmp_ioreq = currdiskreq->ioreqlist;
if (is_read) {
if (currdisk->reqwater) {
currdiskreq->watermark = max(1, (int) ((double) min(seg->size, tmp_ioreq->bcount) * currdisk->readwater));
} else {
currdiskreq->watermark = (int) (((double) seg->size * currdisk->readwater) + (double) 0.9999999999);
}
} else { /* WRITE */
if (currdisk->reqwater) {
currdiskreq->watermark = max(1, (int) ((double) min(seg->size, tmp_ioreq->bcount) * currdisk->writewater));
} else {
currdiskreq->watermark = (int) (((double) seg->size * currdisk->writewater) + (double) 0.9999999999);
}
}
}
int disk_buffer_attempt_seg_ownership (disk *currdisk, diskreq *currdiskreq)
{
segment *seg = currdiskreq->seg;
diskreq *tmp_diskreq;
ioreq_event *currioreq = currdiskreq->ioreqlist;
int currdiskreq_found = FALSE;
int write_found = FALSE;
int write_incomplete = 0;
ioreq_event *tmpioreq;
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, "%12.6f %8p Entering disk_buffer_attempt_seg_ownership\n",simtime,currdiskreq);
fflush(outputfile);
}
if (!seg) {
fprintf(stderr, "diskreq has NULL segment in disk_buffer_attempt_seg_ownership\n");
exit(1);
}
if (currdiskreq->flags & SEG_OWNED) {
fprintf(stderr, "diskreq already owns seg in disk_buffer_attempt_seg_ownership\n");
exit(1);
}
tmp_diskreq = seg->diskreqlist;
while (tmp_diskreq) {
if (tmp_diskreq->flags & SEG_OWNED) {
if (currdiskreq->hittype == BUFFER_PREPEND) {
if (seg->state != BUFFER_DIRTY) {
fprintf(stderr, "PREPEND of active request detected in disk_buffer_attempt_seg_ownership\n");
exit(1);
}
/* release ownership to new request and set up hold areas */
tmp_diskreq->flags &= ~SEG_OWNED;
tmp_diskreq->hittype = BUFFER_APPEND;
if (seg->endblkno != seg->startblkno) {
if (seg->hold_bcount != 0) {
fprintf(stderr, "PREPEND of segment already 'holding' data detected in disk_buffer_attempt_seg_ownership\n");
exit(1);
}
seg->hold_blkno = seg->startblkno;
seg->hold_bcount = seg->endblkno - seg->startblkno;
}
} else {
break;
}
}
if (tmp_diskreq == currdiskreq) {
currdiskreq_found = TRUE;
} else if (!currdiskreq_found && tmp_diskreq->ioreqlist &&
!(tmp_diskreq->ioreqlist->flags & READ)) {
write_found = TRUE;
tmpioreq = tmp_diskreq->ioreqlist;
while (tmpioreq->next) {
tmpioreq = tmpioreq->next;
}
if (tmp_diskreq->inblkno < (tmpioreq->blkno + tmpioreq->bcount)) {
write_incomplete = TRUE;
}
}
tmp_diskreq = tmp_diskreq->seg_next;
}
if (!tmp_diskreq) {
ASSERT(!write_found || !write_incomplete);
currdiskreq->flags |= SEG_OWNED;
if (currioreq &&
(!currdisk->effectivehda ||
(currdisk->effectivehda == currdiskreq) ||
(currdisk->effectivehda->seg != seg))) {
seg->access->flags = currioreq->flags;
}
/* set state, startblkno, endblkno, and readahead fields */
if (!currioreq || (currioreq->flags & READ)) {
/* check to see if the cache segment needs to be reset */
#if 0 /* GROK: removed on 10/10/99, because it causes problems, but */
/* we are not sure that removing this won't break something */
/* else down the road... */
if ((currdiskreq->hittype != BUFFER_NOMATCH) &&
(seg->startblkno > currdiskreq->outblkno)) {
currdiskreq->hittype = BUFFER_NOMATCH;
}
#endif
if (currdiskreq->hittype == BUFFER_NOMATCH) {
ASSERT(currioreq != NULL);
ASSERT1(((seg->access->type == NULL_EVENT) ||
(seg->recyclereq == currdiskreq)),"seg->access->type",seg->access->type);
if (currdiskreq != seg->recyclereq) {
seg->state = BUFFER_CLEAN;
seg->access->blkno = currdiskreq->outblkno;
}
seg->startblkno = seg->endblkno = currdiskreq->outblkno;
seg->minreadaheadblkno = seg->maxreadaheadblkno = -1;
seg->hold_bcount = 0;
}
if (currioreq) {
while (currioreq->next) {
currioreq = currioreq->next;
}
seg->minreadaheadblkno = max(seg->minreadaheadblkno, min((currioreq->blkno + currioreq->bcount + currdisk->minreadahead), currdisk->model->dm_sectors));
seg->maxreadaheadblkno = max(seg->maxreadaheadblkno, min(disk_buffer_get_max_readahead(currdisk,seg,currioreq), currdisk->model->dm_sectors));
if ((seg->endblkno >= (currioreq->blkno + currioreq->bcount)) &&
(currdisk->effectivehda == currdiskreq)) {
seg->access->flags |= BUFFER_BACKGROUND;
}
}
} else { /* WRITE */
if (!currdisk->effectivehda ||
(currdisk->effectivehda == currdiskreq) ||
(currdisk->effectivehda->seg != seg)) {
if ((currdiskreq != seg->recyclereq) &&
(seg->access->type == NULL_EVENT)){
seg->access->blkno = currdiskreq->inblkno;
}
if (currdiskreq->flags & COMPLETION_RECEIVED) {
seg->access->flags |= BUFFER_BACKGROUND;
}
}
if (currdiskreq->hittype == BUFFER_NOMATCH) {
seg->minreadaheadblkno = seg->maxreadaheadblkno = -1;
seg->hold_bcount = 0;
seg->startblkno = seg->endblkno = currdiskreq->inblkno;
if (currdiskreq != seg->recyclereq) {
seg->state = BUFFER_DIRTY;
currdisk->numdirty++;
if ((disk_printhack > 1) && (simtime >= disk_printhacktime)) {
fprintf (outputfile, " numdirty++ = %d\n",currdisk->numdirty);
fflush(outputfile);
}
ASSERT1(((currdisk->numdirty >= 0) && (currdisk->numdirty <= currdisk->numwritesegs)),"numdirty",currdisk->numdirty);
}
} else if (currdiskreq->hittype == BUFFER_PREPEND) {
seg->startblkno = seg->endblkno = currdiskreq->inblkno;
}
}
}
if (disk_printhack && (simtime >= disk_printhacktime)) {
fprintf (outputfile, " Ownership = %d\n",(currdiskreq->flags & SEG_OWNED));
fflush(outputfile);
}
return(currdiskreq->flags & SEG_OWNED);
}
int disk_buffer_get_max_readahead (disk *currdisk, segment *seg, ioreq_event *curr)
{
int endreq;
if (!(curr->flags & READ)) {
fprintf(stderr, "No read-ahead for write accesses, in disk_buffer_get_max_readahead\n");
exit(1);
}
endreq = curr->blkno + curr->bcount;
if (currdisk->contread == BUFFER_NO_READ_AHEAD) {
return(endreq);
} else if (currdisk->contread == BUFFER_DEC_PREFETCH_SCHEME) {
int startlbn;
int endlbn;
struct dm_pbn pbn;
// band* bandptr;
curr->blkno += curr->bcount;
// bandptr =
// was MAP_FULL
currdisk->model->layout->dm_translate_ltop(currdisk->model,
curr->blkno,
MAP_FULL,
&pbn,
0); // remapsector
currdisk->model->layout->dm_get_track_boundaries(currdisk->model,
&pbn,
&startlbn,
&endlbn,
0); // remapsector
// new track_boundaries semantics
endlbn++;
curr->blkno -= curr->bcount;
if (seg->startblkno < startlbn) {
return (endlbn + currdisk->model->layout->dm_get_sectors_lbn(currdisk->model, curr->blkno));
} else {
return(endlbn);
}
}
if (currdisk->maxreadahead) {
return(endreq + currdisk->maxreadahead);
}
if (currdisk->keeprequestdata > 0) {
if ((curr->bcount >= seg->size)
|| ((seg->size - curr->bcount) < currdisk->minreadahead))
{
return(endreq + currdisk->minreadahead);
}
return(curr->blkno + seg->size);
}
return(endreq + seg->size);
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
C
1
https://gitee.com/ZhouHeyu/Flashsim-origin.git
git@gitee.com:ZhouHeyu/Flashsim-origin.git
ZhouHeyu
Flashsim-origin
Mix-SSD
master

搜索帮助