?? tm.cpp
字號:
/***** 該代碼文件所包含頭文件 *****/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
/* 宏定義常量TRUE為1 */
#ifndef TRUE
#define TRUE 1
#endif
/* 宏定義常量FALSE為0 */
#ifndef FALSE
#define FALSE 0
#endif
/***************** 常量 *******************/
/* 為大型程序擴展,指令存儲區大小,定義為1024 */
#define IADDR_SIZE 1024
/* 為大型程序擴展,數據存儲區大小,定義為1024 */
#define DADDR_SIZE 1024
/* 寄存器數量,定義為8 */
#define NO_REGS 8
/* PC寄存器,定義為7 */
#define PC_REG 7
/* 目標代碼行大小,定義為121 */
#define LINESIZE 121
/* 字大小,定義為20 */
#define WORDSIZE 20
/**************** 類型 *******************/
/* 指令尋址模式類型 */
typedef enum
{
opclRR, /* 寄存器尋址模式類型,操作數使用寄存器r,s,t */
opclRM, /* 寄存器-內存尋址模式類型,操作數使用寄存器r,內存地址d+s */
opclRA /* 寄存器-立即數尋址模式類型,操作數使用寄存器r,立即數值d+s */
} OPCLASS;
/* 操作指令標識碼類型 */
typedef enum opcode{
/************* 寄存器尋址模式指令標識碼 ***************/
opHALT, /* 停止指令:結束程序執行,忽略操作數 */
opIN, /* 輸入指令:將外部變量讀入寄存器,使用第r個寄存器,忽略s,t參數 */
opOUT, /* 輸出指令:將寄存器的值輸出,使用第r個寄存器,忽略s,t參數 */
opADD, /* 加法指令:寄存器r的值賦為寄存器s的值與寄存器t的值的和 */
opSUB, /* 減法指令:寄存器r的值賦為寄存器s的值與寄存器t的值的差 */
opMUL, /* 乘法指令:寄存器r的值賦為寄存器s的值與寄存器t的值的積 */
opDIV, /* 除法指令:寄存器r的值賦為寄存器s的值與寄存器t的值的商 */
/* 寄存器尋址模式指令限制標志, *
* 操作標識碼枚舉值小于opRRLim的指令均為寄存器尋址模式指令 */
opRRLim,
/************** 寄存器-內存尋址模式指令標識碼 ****************/
opLD, /* 載入指令:寄存器r的值賦為地址為d+reg(s)的內存單元的值 */
opST, /* 設置指令:將地址為d+reg(s)的內存單元值賦為寄存器r的值 */
/* 寄存器-內存尋址模式指令限制標志 *
* 操作標識碼枚舉值小于opRMLim且大于opRRLim的均為寄存器-內存尋址模式指令 */
opRMLim,
/************* 寄存器-立即數尋址模式指令標識碼 ****************/
opLDA, /* 載入指令:將寄存器r的值賦為立即數d與寄存器s的值的和 */
opLDC, /* 載入指令:將寄存器r的值賦為立即數d,參數s被忽略 */
opJLT, /* 如果寄存器r的值小于0,將第7個寄存器的值賦為d+reg(s) *
* 第7個寄存器為pc程序計數寄存器 */
opJLE, /* 如果寄存器r的值小于等于0,將pc寄存器的值賦為d+reg(s) */
opJGT, /* 如果寄存器r的值大于0,將pc寄存器的值賦為d+reg(s) */
opJGE, /* 如果寄存器r的值大于等于0,將pc寄存器的值賦為d+reg(s) */
opJEQ, /* 如果寄存器r的值等于0,將pc寄存器的值賦為d+reg(s) */
opJNE, /* 如果寄存器r的值不等于0,將pc寄存器的值賦為d+reg(s) */
/* 寄存器-立即數尋址模式指令限制標志
* 操作標識碼枚舉值小于opRALim且大于opRMLim的均為寄存器-立即數尋址模式指令 */
opRALim
} OPCODE;
/************ 指令單步執行結果類型 ************/
typedef enum {
srOKAY, /* 正常 */
srHALT, /* 停止 */
srIMEM_ERR, /* 指令存儲錯 */
srDMEM_ERR, /* 數據存儲錯 */
srZERODIVIDE /* 除數為零錯 */
} STEPRESULT;
/* 指令結構類型:操作碼,操作數1,操作數2,操作數3 */
typedef struct {
int iop ;
int iarg1 ;
int iarg2 ;
int iarg3 ;
} INSTRUCTION;
/******** 變量 ********/
int iloc = 0 ; /* 指令存儲計數指針,初始為0 */
int dloc = 0 ; /* 數據存儲計數指針,初始為0 */
int traceflag = FALSE; /* 指令執行追蹤標志,初始為FALSE */
int icountflag = FALSE; /* 指令執行計數標志,初始為FALSE */
/* iMem用于指令存儲,為1024長的指令結構數組 */
INSTRUCTION iMem [IADDR_SIZE];
/* dMem用于數據存儲,為1024長的整數類型數組 */
int dMem [DADDR_SIZE];
/* reg用于寄存器存儲,為8長的整數類型數組 */
int reg [NO_REGS];
/* 指令操作碼表,對應尋址模式分為三類 */
char * opCodeTab[]
= {"HALT","IN","OUT","ADD","SUB","MUL","DIV","????",
/* 寄存器尋址模式指令類型 */
"LD","ST","????",
/* 寄存器-內存尋址模式指令類型 */
"LDA","LDC","JLT","JLE","JGT","JGE","JEQ","JNE","????"
/* 寄存器-立即數尋址模式指令類型 */
};
/** 單步執行結果狀態表 **/
char * stepResultTab[]
= {"OK","Halted","Instruction Memory Fault",
"Data Memory Fault","Division by 0"
};
char pgmName[20]; /* 用于存儲程序文件名 */
FILE *pgm ; /* 程序文件指針 */
char in_Line[LINESIZE] ; /* 用于存儲一行代碼,為121長的字符數組 */
int lineLen ; /* in_Line中行結尾字符位置 */
int inCol ; /* 用于指出在in_Line中的當前字符位置 */
int num ; /* 用于存儲當前所得數值 */
char word[WORDSIZE] ; /* 用于存儲當前的字,為20長的字符數組 */
char ch ; /* 當前代碼行中當前位置上的字符 */
int done ;
/****************************************************/
/* 函數名 opClass */
/* 功 能 指令尋址模式分類函數 */
/* 說 明 該函數對給定的指令操作碼枚舉值c進行分類 */
/* 返回指令所屬尋址模式 */
/****************************************************/
int opClass( int c )
{
/* 如果枚舉值c小于opRRLim,則指令為寄存器尋址模式指令類型 */
if ( c <= opRRLim) return ( opclRR );
/* 如果枚舉值c小于opRMLim,則指令為寄存器-內存尋址模式指令類型 */
else if ( c <= opRMLim) return ( opclRM );
/* 為寄存器-立即數尋址模式指令類型 */
else return ( opclRA );
}
/********************************************************/
/* 函數名 writeInstruction */
/* 功 能 指令輸出函數 */
/* 說 明 該函數將指令存儲區中指令以指定格式輸出到屏幕 */
/********************************************************/
void writeInstruction ( int loc )
{
/* loc為所要輸出的指令在指令存儲區中地址,輸出到屏幕 */
printf( "%5d: ", loc) ;
/* 輸出指令地址loc在0-1023有效的指令存儲區地址范圍之內 */
if ( (loc >= 0) && (loc < IADDR_SIZE) )
{
/* 輸出地址為loc上的指令操作碼值iMem[loc].iop和第一操作數iMem[loc].iarg1 */
printf("%6s%3d,", opCodeTab[iMem[loc].iop], iMem[loc].iarg1);
/* 根據指令的尋址模式分類處理 */
switch ( opClass(iMem[loc].iop) )
{
/* 輸出指令為寄存器尋址模式指令,以給定形式輸出操作數2,操作數3 */
case opclRR: printf("%1d,%1d", iMem[loc].iarg2, iMem[loc].iarg3);
break;
/* 輸出指令為寄存器-立即數尋址模式指令,和寄存器-內存尋址模式指令 *
* 以給定形式輸出操作數2,操作數3 */
case opclRM:
case opclRA: printf("%3d(%1d)", iMem[loc].iarg2, iMem[loc].iarg3);
break;
}
/* 向屏幕輸出換行符 */
printf ("\n") ;
}
} /* writeInstruction */
/****************************************************/
/* 函數名 getCh */
/* 功 能 字符獲取函數 */
/* 說 明 如果當前行中字符未讀完,則函數返回當前字符 */
/* 否則,函數返回空格字符 */
/****************************************************/
void getCh (void)
{
/* 在當前代碼行in_Line中,當前字符列數inCol未超過代碼行實際長度lineLen *
* 取得當前行中當前位置的字符,送入ch */
if (++inCol < lineLen)
ch = in_Line[inCol] ;
/* 如果inCol超出當前代碼行長度范圍,則ch賦為空格 */
else ch = ' ' ;
} /* getCh */
/********************************************************/
/* 函數名 nonBlank */
/* 功 能 非空字符獲取函數 */
/* 說 明 如果成功從當前行中取得非空字符,函數返回TRUE */
/* 否則,函數返回FALSE */
/********************************************************/
int nonBlank (void)
{
/* 在當前代碼行in_Line中,當前字符位置inCol中為空格字符 *
* 在當前代碼行in_Line中,當前字符位置inCol下移,略過空格 */
while ((inCol < lineLen)
&& (in_Line[inCol] == ' ') )
inCol++ ;
/* 在當前代碼行in_Line中,遇到非空字符 */
if (inCol < lineLen)
{
/* 取當前字符位置inCol中的字符送入ch, *
* 函數返回TRUE(已定義為1),ch中得到非空字符 */
ch = in_Line[inCol] ;
return TRUE ; }
/* 當前代碼行已經讀完,將當前字符ch 賦為空格, *
* 函數返回FALSE(已定義為0),ch中為空格字符 */
else
{ ch = ' ' ;
return FALSE ; }
} /* nonBlank */
/****************************************************************/
/* 函數名 getNum */
/* 功 能 數值獲取函數 */
/* 說 明 將代碼行中連續出現的有加減運算的數term合并計數, */
/* 所的數值送入為num.如果成功得到數值,則函數返回TRUE; */
/* 否則,函數返回FALSE */
/****************************************************************/
int getNum (void)
{ int sign; /* 符號因子 */
int term; /* 用于記錄當前錄入的局部數值 */
int temp = FALSE; /* 記錄函數返回值,初始為假 */
num = 0 ; /* 用于記錄所有加減運算后的最終數值結果 */
do
{ sign = 1; /* 符號因子初始為1 */
/* 調用函數nonBlank()略過當前位置的空格后, *
* 所得到的當前非空字符ch為+或-.(+/-的連續出現處理) */
while ( nonBlank() && ((ch == '+') || (ch == '-')) )
{ temp = FALSE ;
/* 當前字符ch為"-"時,符號因子sign設為-1 */
if (ch == '-') sign = - sign ;
/* 取當前代碼行中下一字符到當前字符ch中 */
getCh();
}
term = 0 ; /* 當前錄入的局部數值初始為0 */
nonBlank(); /* 略過當前位置上的空格 */
/* 當前字符ch為數字,局部數值的循環處理 */
while (isdigit(ch))
{ temp = TRUE ; /* 函數返回值設為TRUE,成功得到數字 */
/* 將字符序列轉化為數值形式,進行進位累加 */
term = term * 10 + ( ch - '0' ) ;
getCh(); /* 取當前代碼行中下一字符到當前字符ch中 */
}
/* 將局部數值帶符號累加,得到最終數值num */
num = num + (term * sign) ;
} while ( (nonBlank()) && ((ch == '+') || (ch == '-')) ) ;
return temp;
} /* getNum */
/****************************************************/
/* 函數名 getWord */
/* 功 能 單詞獲取函數 */
/* 說 明 函數從當前代碼行中獲取單詞.如果得到字符, */
/* 則函數返回TRUE;否則,函數返回FALSE */
/****************************************************/
int getWord (void)
{
int temp = FALSE; /* 函數返回值初始為FALSE */
int length = 0; /* 單詞長度初始為0 */
/* 在當前代碼行中成功獲取非空字符ch */
if (nonBlank ())
{
/* 當前非空字符ch為字母或數字 */
while (isalnum(ch))
{
/* 當前單詞word未超過規定字長WORDSIZE-1(為單詞結束字符留一空位) *
* 將當前字符ch讀入到單詞末尾 */
if (length < WORDSIZE-1) word [length++] = ch ;
getCh() ; /* 取當前代碼行中下一字符 */
}
/* 給當前單詞word加入結束字符 */
word[length] = '\0';
/* 設置函數返回值,當讀入字word非空的時候為TRUE */
temp = (length != 0);
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -