{ Корректируем отношение. Если корректировка матрицы
предшествования не используется, то функция должна
вернуть то же самое отношение }
cRule:= CorrectRule(cRule,lexTCur,
listLex[i].LexType,symbStack);
case cRule of
'<, =: { Надо выполнять сдвиг (перенос) }
begin { Помещаем текущую лексему в стек }
symbStack.Push(listLex[i]);
Inc(i); { Увеличиваем счетчик входных лексем }
end;
'>: { Надо выполнять свертку }
if symbStack.MakeTopSymb = nil then
begin { Если не удалось выполнить свертку, }
{ запоминаем текущую лексему как место ошибки }
Result:= TSymbol.CreateLex(listLex[i]);
Break; { Прерываем алгоритм }
end;
else { Отношение не установлено – ошибка разбора }
begin {Запоминаем текущую лексему (место ошибки)}
Result:= TSymbol.CreateLex(listLex[i]);
Break; { Прерываем алгоритм }
end;
end{case};
end{while};
if Result = nil then { Если разбор прошел без ошибок }
begin{Убеждаемся, что в стеке осталось только 2 символа}
if symbStack.Count = 2 then
{ Если да, то верхний символ – результат разбора }
Result:= symbStack[1]
{ Иначе это ошибка – отмечаем место ошибки }
else Result:= TSymbol.CreateLex(listLex[iCnt]);
end;
finally { Уничтожаем временную начальную лексему }
lexStop.Free;
end;
end;
end.
Модуль описания допустимых типов триад
unit TrdType; {!!! Зависит от входного языка!!!}
interface
{ Модуль для описания допустимых типов триад }
const { Имена предопределенных функций и переменных }
NAME_PROG = 'MyCurs';
NAME_INPVAR = 'InpVar';
NAME_RESULT = 'Result';
NAME_FUNCT = 'CompileTest';
NAME_TYPE = 'integer';
type { Типы триад, соответствующие типам допустимых
операций, а также три дополнительных типа триад:
– CONST – для алгоритма свертки объектного кода;
– SAME – для алгоритма исключения лишних операций;
– NOP (No OPerations) – для ссылок на конец списка триад. }
TTriadType = (TRD_IF,TRD_OR,TRD_XOR,TRD_AND,TRD_NOT,
TRD_LT,TRD_GT,TRD_EQ,TRD_NEQ,TRD_ADD,TRD_SUB,TRD_UMIN,
TRD_ASSIGN,TRD_JMP,TRD_CONST,TRD_SAME,TRD_NOP);
{Массив строковых обозначений триад для вывода их на экран}
TTriadStr = array[TTriadType] of string;
const TriadStr: TTriadStr =('if','or','xor','and','not',
'<, >, =, <>, +, -, -,
':=,'jmp','C','same','nop');
{ Множество триад, которые являются линейными операциями }
TriadLineSet: set of TTriadType =
[TRD_OR, TRD_XOR, TRD_AND, TRD_NOT, TRD_ADD, TRD_SUB,
TRD_LT, TRD_GT, TRD_EQ, TRD_NEQ, TRD_UMIN];
implementation
end.
Модуль вычисления значений триад при свертке объектного кода
unit TrdCalc; {!!! Зависит от входного языка!!!}
interface
{ Модуль, вычисляющий значения триад при свертке операций }
uses TrdType;
{ Функция вычисления триады по значениям двух операндов }
function CalcTriad(Triad: TTriadType;
iOp1,iOp2: integer): integer;
implementation
function CalcTriad(Triad: TTriadType;
iOp1,iOp2: integer): integer;
{ Функция вычисления триады по значениям двух операндов }
begin
Result:= 0;
case Triad of
TRD_OR: Result:= (iOp1 or iOp2) and 1;
TRD_XOR: Result:= (iOp1 xor iOp2) and 1;
TRD_AND: Result:= (iOp1 and iOp2) and 1;
TRD_NOT: Result:= (not iOp1) and 1;
TRD_LT: if iOp1 else Result:= 0; TRD_GT: if iOp1>iOp2 then Result:= 1 else Result:= 0; TRD_EQ: if iOp1=iOp2 then Result:= 1 else Result:= 0; TRD_NEQ: if iOp1<>iOp2 then Result:= 1 else Result:= 0; TRD_ADD: Result:= iOp1 + iOp2; TRD_SUB: Result:= iOp1 – iOp2; TRD_UMIN: Result:= – iOp2; end; end; end.
Модуль описания структур данных триад
unit Triads;
interface
{ Модуль, обеспечивающий работу с триадами и их списком }
uses Classes, TblElem, LexElem, TrdType;
type
TTriad = class; { Предварительное описание класса триад }
TOpType = (OP_CONST, OP_VAR, OP_LINK); { Типы операндов:
константа, переменная, ссылка на другую триаду }
TOperand = record { Структура описания операнда в триадах }
case OpType: TOpType of { Тип операнда }
OP_CONST: (ConstVal: integer);{для констант – значение}
OP_VAR: (VarLink: TVarInfo);{ для переменной – ссылка
на элемент таблицы идентификаторов }
OP_LINK: (TriadNum: integer);{ для триады – номер }
end;
TOpArray = array[1..2] of TOperand; {Массив из 2 операндов}
TTriad = class(TObject)
private { Структура данных для описания триады }
TriadType: TTriadType; { Тип триады }
Operands: TOpArray; { Массив операндов }
public
Info: longint; { Дополнительная информация
для оптимизирующих алгоритмов }
IsLinked: Boolean; { Флаг наличия ссылки на эту триаду }
{ Конструктор для создания триады }
constructor Create(Typ: TTriadType; const Ops: TOpArray);
{ Функции для чтения и записи операндов }
function GetOperand(iIdx: integer): TOperand;
procedure SetOperand(iIdx: integer; Op: TOperand);