I happened to use Simplescalar before. Actually, Simplescalar have already implemented the true LRU algorithm.
The following comment clearly describes the function update_way_list.
/* insert BLK into the order way chain in SET at location WHERE */
static void
update_way_list(struct cache_set_t *set, /* set contained way chain */
struct cache_blk_t *blk, /* block to insert */
enum list_loc_t where) /* insert location */
The code that you cited is from the "cache miss" case when the cache is accessed:
switch (cp->policy) {
case LRU:
case FIFO:
repl = cp->sets[set].way_tail;
update_way_list(&cp->sets[set], repl, Head);
break;
}
Here the last way of the set is chose as victim and it is moved to the Head of the set.
Later the replaced block data is written back and then the victim is replaced by the new data block.
The most important part that distinguishes LRU and FIFO comes from the "cache hit" case:
/* if LRU replacement and this is not the first element of list, reorder */
if (blk->way_prev && cp->policy == LRU)
{
/* move this block to head of the way (MRU) list */
update_way_list(&cp->sets[set], blk, Head);
}
Therefore, the ways in a set are following the decreasing order of age: The head of the set is the MRU (Most Recently Used) block while the tail is the LRU.
This is exactly the true LRU algorithm: When there is a cache hit, the hit block is promoted to the MRU way while preserving the order of others. When there is a cache miss, the LRU block is chosen as victim and the new block is placed in the MRU way. If we remove the previous "cache hit" code, no access history is recorded and the ways in a set are following the access order, thus providing FIFO behavior. If we remove the line
update_way_list(&cp->sets[set], repl, Head);
in the previous "cache miss" code, then the new block will be placed in the LRU way, thus providing LIP (LRU Insertion Policy) behavior.