This repository has been archived on 2026-04-18. You can view files and clone it, but cannot push or open issues or pull requests.
Virtual-Memory-Demo/Assets/Tests/EditMode/Step4PageTableAndFifoTests.cs

135 lines
5.2 KiB
C#

using System;
using NUnit.Framework;
using VMdemo.Simulation;
namespace VMdemo.Tests.EditMode
{
public class Step4PageTableAndFifoTests
{
[Test]
public void TwoLevelPageTable_SetPresent_ThenTryGetPresentPfn_Hits()
{
var pageTable = new TwoLevelPageTable(3, 2);
pageTable.SetPresent(5UL, 3UL, 9UL, dirty: true);
var hit = pageTable.TryGetPresentPfn(5UL, 3UL, out var pfn);
var exists = pageTable.TryGetEntry(5UL, 3UL, out var entry);
Assert.IsTrue(hit);
Assert.That(pfn, Is.EqualTo(9UL));
Assert.IsTrue(exists);
Assert.That(entry.Present, Is.True);
Assert.That(entry.Dirty, Is.True);
}
[Test]
public void TwoLevelPageTable_MarkNotPresent_ThenMisses()
{
var pageTable = new TwoLevelPageTable(3, 2);
pageTable.SetPresent(1UL, 2UL, 7UL);
var changed = pageTable.MarkNotPresent(1UL, 2UL);
var changedAgain = pageTable.MarkNotPresent(1UL, 2UL);
var hitAfterMark = pageTable.TryGetPresentPfn(1UL, 2UL, out _);
var exists = pageTable.TryGetEntry(1UL, 2UL, out var entry);
Assert.IsTrue(changed);
Assert.IsFalse(changedAgain);
Assert.IsFalse(hitAfterMark);
Assert.IsTrue(exists);
Assert.IsFalse(entry.Present);
}
[Test]
public void PhysicalMemoryManager_AllocateForVpn_EvictsInFifoOrder()
{
var manager = new PhysicalMemoryManager(2UL);
var r1 = manager.AllocateForVpn(1UL);
var r2 = manager.AllocateForVpn(2UL);
var r3 = manager.AllocateForVpn(3UL); // evict 1
var r4 = manager.AllocateForVpn(4UL); // evict 2
Assert.That(r1.AssignedPfn, Is.EqualTo(0UL));
Assert.That(r2.AssignedPfn, Is.EqualTo(1UL));
Assert.That(r3.AssignedPfn, Is.EqualTo(0UL));
Assert.That(r4.AssignedPfn, Is.EqualTo(1UL));
Assert.IsFalse(r1.Eviction.HasEvicted);
Assert.IsFalse(r2.Eviction.HasEvicted);
Assert.IsTrue(r3.Eviction.HasEvicted);
Assert.That(r3.Eviction.EvictedVpn, Is.EqualTo(1UL));
Assert.That(r3.Eviction.EvictedPfn, Is.EqualTo(0UL));
Assert.IsTrue(r4.Eviction.HasEvicted);
Assert.That(r4.Eviction.EvictedVpn, Is.EqualTo(2UL));
Assert.That(r4.Eviction.EvictedPfn, Is.EqualTo(1UL));
var fifoOrder = manager.GetFifoVpnOrder();
Assert.That(fifoOrder.Count, Is.EqualTo(2));
Assert.That(fifoOrder[0], Is.EqualTo(3UL));
Assert.That(fifoOrder[1], Is.EqualTo(4UL));
}
[Test]
public void Resolve_WhenEvictionHappens_SyncsPageTableAndTlb()
{
var pageTable = new TwoLevelPageTable(4, 4);
var memory = new PhysicalMemoryManager(1UL);
var tlb = new TlbCache(4);
const ulong vpn1 = 1UL;
pageTable.GetIndicesFromVpn(vpn1, out var l1_1, out var l2_1);
var first = MemoryAccessResolver.Resolve(vpn1, l1_1, l2_1, pageTable, memory, tlb);
Assert.IsTrue(first.PageFault);
Assert.IsFalse(first.PageTableHit);
Assert.IsFalse(first.Eviction.HasEvicted);
Assert.IsTrue(tlb.Lookup(vpn1, out _));
var second = MemoryAccessResolver.Resolve(vpn1, l1_1, l2_1, pageTable, memory, tlb);
Assert.IsFalse(second.PageFault);
Assert.IsTrue(second.PageTableHit);
const ulong vpn2 = 2UL;
pageTable.GetIndicesFromVpn(vpn2, out var l1_2, out var l2_2);
var third = MemoryAccessResolver.Resolve(vpn2, l1_2, l2_2, pageTable, memory, tlb);
Assert.IsTrue(third.PageFault);
Assert.IsTrue(third.Eviction.HasEvicted);
Assert.That(third.Eviction.EvictedVpn, Is.EqualTo(vpn1));
Assert.IsFalse(tlb.Lookup(vpn1, out _));
Assert.IsTrue(tlb.Lookup(vpn2, out _));
pageTable.GetIndicesFromVpn(vpn1, out var evictedL1, out var evictedL2);
Assert.IsFalse(pageTable.TryGetPresentPfn(evictedL1, evictedL2, out _));
Assert.IsFalse(memory.TryGetResidentPfn(vpn1, out _));
Assert.IsTrue(memory.TryGetResidentPfn(vpn2, out _));
}
[Test]
public void HighVolumeAccess_N10000_RunsWithoutStateCorruption()
{
var pageTable = new TwoLevelPageTable(8, 8);
var memory = new PhysicalMemoryManager(16UL);
var tlb = new TlbCache(8);
var random = new Random(123);
for (var i = 0; i < 10000; i++)
{
var vpn = (ulong)random.Next(0, 512);
pageTable.GetIndicesFromVpn(vpn, out var l1, out var l2);
var result = MemoryAccessResolver.Resolve(vpn, l1, l2, pageTable, memory, tlb);
var fifoOrder = memory.GetFifoVpnOrder();
Assert.That(memory.ResidentCount, Is.LessThanOrEqualTo(16));
Assert.That(fifoOrder.Count, Is.EqualTo(memory.ResidentCount));
if (result.PageTableHit)
{
Assert.IsFalse(result.PageFault);
}
}
}
}
}