if TestOperConst(Trd[1],listTriad,iOp1)
and TestOperConst(Trd[2],listTriad,iOp2) then
begin { тогда вычисляем значение операции, }
Ops[1].ConstVal:=
CalcTriad(Trd.TrdType,iOp1,iOp2);
{ запоминаем его в триаде «CONST», которую
записываем в список вместо прежней триады }
listTriad.Items[i]:= TTriad.Create(TRD_CONST,Ops);
{Если на прежнюю триаду была ссылка, сохраняем ее}
listTriad[i].IsLinked:= Trd.IsLinked;
Trd.Free; { Уничтожаем прежнюю триаду }
end;
end;
end;
end;
constructor TDepInfo.Create(iInfo: longint);
{ Создание информационной структуры для чисел зависимости }
begin
inherited Create; {Вызываем конструктор базового класса}
iDep:= iInfo; { Запоминаем число зависимости }
end;
procedure TDepInfo.SetInfo(iIdx: integer; iInfo: longint);
{ Функция записи числа зависимости }
begin iDep:= iInfo; end;
function TDepInfo.GetInfo(iIdx: integer): longint;
{ Функция чтения числа зависимости }
begin Result:= iDep; end;
function CalcDepOp(listTriad: TTriadList;
Op: TOperand): longint;
{Функция вычисления числа зависимости для операнда триады}
begin
Result:= 0;
case Op.OpType of { Выборка по типу операнда }
OP_VAR: { Если это переменная – смотрим ее информационную
структуру, и если она есть, берем число зависимости }
if Op.VarLink.Info <> nil then Result:=
Op.VarLink.Info.Info[0];
OP_LINK: { Если это ссылка на триаду,
то берем число зависимости триады }
Result:= listTriad[Op.TriadNum].Info;
end{case};
end;
function CalcDep(listTriad: TTriadList;
Trd: TTriad): longint;
{ Функция вычисления числа зависимости триады }
var iDepTmp: longint;
begin
Result:= CalcDepOp(listTriad,Trd[1]);
iDepTmp:= CalcDepOp(listTriad,Trd[2]);
{ Число зависимости триады есть число на единицу большее,
чем максимальное из чисел зависимости ее операндов }
if iDepTmp > Result then Result:= iDepTmp+1
else Inc(Result);
Trd.Info:= Result;
end;
procedure OptimizeSame(listTriad: TTriadList);
{ Процедура оптимизации путем исключения лишних операций }
var
i,j,iStart,iCnt,iNum: integer;
Ops: TOpArray;
Trd: TTriad;
begin { Начало линейного участка – начало списка триад }
iStart:= 0;
ClearTreeInfo; { Очищаем информационные структуры
таблицы идентификаторов }
Ops[1].OpType:= OP_LINK; { Заполняем операнды }
Ops[2].OpType:= OP_CONST; { для триады типа «SAME» }
Ops[2].ConstVal:= 0;
iCnt:= listTriad.Count-1;
for i:=0 to iCnt do { Для всех триад списка }
begin { выполняем алгоритм }
Trd:= listTriad[i];
if Trd.IsLinked then {Если триада помечена ссылкой, }
begin { то линейный участок кода закончен – очищаем }
ClearTreeInfo; { информационные структуры идентификаторов и }
iStart:= i; { запоминаем начало линейного участка }
end;
for j:=1 to 2 do { Если любой операнд триады ссылается
if Trd[j].OpType = OP_LINK then { на триаду «SAME», }
begin { то переставляем ссылку на предыдущую, }
iNum:= Trd[j].TriadNum;{ совпадающую с ней триаду }
if listTriad[iNum].TrdType = TRD_SAME then
Trd.Links[j]:= listTriad[iNum].Links[1];
end;
if Trd.TrdType = TRD_ASSIGN then { Если триада типа }
begin { «присвоение» – запоминаем число зависимости
связанной с нею переменной }
Trd[1].VarLink.Info:= TDepInfo.Create(i+1);
end
else { Если триада – одна из линейных операций }
if Trd.TrdType in TriadLineSet then
begin { Вычисляем число зависимости триады }
CalcDep(listTriad,Trd);
for j:=iStart to i-1 do { На всем линейном участке }
begin { ищем совпадающую триаду с таким же }
if Trd.IsEqual(listTriad[j]) { числом зависимости }
and (Trd.Info = listTriad[j].Info) then
begin { Если триада найдена, запоминаем ссылку }
Ops[1].TriadNum:= j;
{ запоминаем ее в триаде типа «SAME», которую
записываем в список вместо прежней триады }
listTriad.Items[i]:=
TTriad.Create(TRD_SAME,Ops);
listTriad[i].IsLinked:= Trd.IsLinked; { Если на
прежнюю триаду была ссылка, сохраняем ее }
Trd.Free; { Уничтожаем прежнюю триаду }
Break; { Прерываем поиск }
end;
end;
end{if};
end{for};
end;
end.
Модуль создания списка триад на основе дерева разбора
Листинг П3.12. Создание списка триад на основе дерева разбора
unit TrdMake; {!!! Зависит от входного языка!!!}
interface
{ Модуль, обеспечивающий создание списка триад на основе
структуры синтаксического разбора }
uses LexElem, Triads, SyntSymb;
function MakeTriadList(symbTop: TSymbol;
listTriad: TTriadList): TLexem;
{ Функция создания списка триад начиная от корневого
символа дерева синтаксического разбора.
Функция возвращает nil при успешном выполнении, иначе
она возвращает ссылку на лексему, где произошла ошибка }
implementation
uses LexType, TrdType;
Читать дальше
Конец ознакомительного отрывка
Купить книгу