微处理器模拟器中未处理空引用异常

本文关键字:引用 异常 未处理 模拟器 微处理器 | 更新日期: 2023-09-27 18:26:53

所以我有一个内存类,看起来像这样:

namespace GeminiCore
{
    public class Memory
    {
    public static int[] memory = new int[256];
    public string nextInstruction;
    public static int cacheSize = 8;
    public CPU myCPU;
    public struct frame
    {
        public bool dirtyBit;
        public int isEmpty;
        public int value;
        public int tag;
        public frame(int cacheSize)
        {
            dirtyBit = false;
            isEmpty = 1;
            value = 0;
            tag = 0;
        }
    }
    public int solveMemory(int value, int instr, frame[] myCache)
    {
        Console.WriteLine("I reached the solveMemory!!!");
        int block = value % cacheSize;
        if(instr == 127 || instr == 125 || instr == 124 || instr == 123 
                || instr == 122 || instr == 121|| instr == 120)
        {
            Console.WriteLine("I reached the read section!!!");
            if(myCache[block].isEmpty == 0) //Read hit
                if(myCache[block].tag == value) 
                    return myCache[block].value;
            else
            {
                myCache[block].value = memory[value]; //Read Miss
                myCache[block].tag = value;
                myCache[block].isEmpty = 0;
                Console.WriteLine("Read Miss --- The Cache is as follows: block = " + block + " the value at this block is: " + myCache[block].value + " the tag at this block is: " + myCache[block].tag);
                return myCache[block].value;
            }
        }
        else
        {
            Console.WriteLine("I reached the write section!!!");
            if (myCache[block].isEmpty == 1) //Write Miss
            {
                Console.WriteLine("Write Miss --- The Cache is as follows: block = " + block + " the value at this block is: " + myCache[block].value + " the tag at this block is: " + myCache[block].tag);
                memory[value] = myCPU.ACC;
            }
            else
            {
                if (myCache[block].dirtyBit == false)
                {
                    if (myCache[block].tag != value)
                    {
                        myCache[block].value = myCPU.ACC; //Write Hit
                        myCache[block].dirtyBit = true;
                        myCache[block].tag = value;
                        Console.WriteLine("Write Hit --- The Cache is as follows: block = " + block + " the value at this block is: " + myCache[block].value + " the tag at this block is: " + myCache[block].tag);
                    }
                }
                else
                {
                    memory[myCache[block].tag] = myCache[block].value;
                    myCache[block].value = myCPU.ACC;
                    myCache[block].tag = value;
                    myCache[block].dirtyBit = false;
                    Console.WriteLine("Write Hit --- The Cache is as follows: block = " + block + " the value at this block is: " + myCache[block].value + " the tag at this block is: " + myCache[block].tag);
                }
            }
         }
        return value;
    }
}

}

然后我有了我的CPU类,我只会发布一个错误发生的地方的小片段:

public Memory myMemory;
public static Memory.frame[] myCache = new Memory.frame[Memory.cacheSize];
public void doInstruction()
    {   
        var instr = (finalCodes[i] >> 9) & 127; //Parses op code to find instruction
        var immed = (finalCodes[i] >> 8) & 1; //Parses op code to find immediate value
        var value = (finalCodes[i] & 255); //Parse op code to find value
        foreach (Memory.frame x in myCache)
        {
            Console.WriteLine("Dirtybit: " + x.dirtyBit + " isEmpty: " + x.isEmpty + " tag: " + x.tag + " value: " + x.value);
        }
        switch (instr)
        {
            case (127): //LDA instruction
                if (immed == 1) 
                    ACC = value;
                else if (immed == 0) 
                    //ACC = Memory.memory[value];
                    ACC = myMemory.solveMemory(value, instr, myCache);
                break;
            case (126): //STA instruction
                if (immed == 0)
                {
                    Console.WriteLine("The value is: " + value + " The instruction is: " + instr);
                    foreach (Memory.frame x in myCache)
                    {
                        Console.WriteLine("Dirtybit: " + x.dirtyBit + " isEmpty: " + x.isEmpty + " tag: " + x.tag + " value: " + x.value);
                    }
                    //Memory.memory[value] = ACC;
                    myMemory.solveMemory(value, instr, myCache);
                }

所以我的问题是,当我运行测试时,它接收汇编代码并将其转换为二进制代码,然后运行代码,当我到达第二个命令"sta"时,它应该转到以下行:

 myMemory.solveMemory(value, instr, myCache);

当它到达那个点时,它会给我一个空引用异常。正如你所看到的,我有一些命令行输出,试图看看它的进展。它打印出myCache的内容就在该行之前。然而,它不会到达solveMemory函数中的任何调试语句,甚至不会到达第一个说:的语句

Console.WriteLine("I reached the solveMemory!!!");

我真的不确定是什么导致了这个错误,我已经研究了很长一段时间了。希望你们中的一个人能够找到我到底在哪里搞砸了。谢谢你抽出时间。

微处理器模拟器中未处理空引用异常

public Memory myMemory;

应该是:

public Memory myMemory = new Memory();