Алексей Молчанов - Системное программное обеспечение. Лабораторный практикум
- Название:Системное программное обеспечение. Лабораторный практикум
- Автор:
- Жанр:
- Издательство:Array Издательство «Питер»
- Год:2005
- Город:Санкт-Петербург
- ISBN:978-5-469-00391-4
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Алексей Молчанов - Системное программное обеспечение. Лабораторный практикум краткое содержание
Книга ориентирована на студентов, обучающихся в технических вузах по специальностям, связанным с вычислительной техникой. Но она будет также полезна всем, чья деятельность так или иначе касается разработки программного обеспечения.
Системное программное обеспечение. Лабораторный практикум - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
begin
Result:= ; Exit;
end;
if flagOpt then { Если оптимизация команд включена }
begin
if sReg = 0 then { Если требуемое значение = 0, }
begin{его можно получить из –1 и 1 с помощью INC и DEC}
if sVal = -1 then Result:= #9'inc'#9'eax'
else
if sVal = 1 then Result:= #9'dec'#9'eax'
else Result:= #9'xor'#9'eax,eax'
end {иначе – с помощью XOR}
else
if sReg = 1 then { Если требуемое значение = 1, }
begin{его можно получить из –1 и 0 с помощью NEG и INC}
if sVal = -1 then Result:= #9'neg'#9'eax'
else
if sVal = 0 then Result:= #9'inc'#9'eax'
else
Result:= #9'xor'#9'eax,eax'#13#10#9'inc'#9'eax';
end {иначе – двумя командами: XOR и INC }
else
if sReg = -1 then { Если требуемое значение = -1, }
begin{его можно получить из 1 и 0 с помощью NEG и DEC}
if sVal = 1 then Result:= #9'neg'#9'eax'
else
if sVal = 0 then Result:= #9'dec'#9'eax'
else
Result:= #9'xor'#9'eax,eax'#13#10#9'dec'#9'eax';
end {иначе – двумя командами: XOR и DEC }
{ Иначе заполняем eax командой MOV }
else Result:= Format(#9'mov'#9'eax,%s',[sReg]);
end { Если оптимизация команд выключена,
всегда заполняем eax командой MOV }
else Result:= Format(#9'mov'#9'eax,%s',[sReg]);
end;
function MakeOpcode(i: integer;{номер текущей триады}
listTriad: TTriadList;{список триад}
const sOp,sReg,{код операции и операнд}
sPrev,{предыдущая команда}
sVal{предыдущая величина в eax}: string;
flagOpt: Boolean{флаг оптимизации}): string;
{ Функция, генерящая код линейных операций над eax }
var Triad: TTriad;{текущая триада}
begin { Запоминаем текущую триаду }
Triad:= listTriad[i];
if flagOpt then { Если оптимизация команд включена }
begin
if sReg = 0 then { Если операнд = 0 }
begin
case Triad.TrdType of
TRD_AND: { Для команды AND результат всегда = 0 }
Result:= MakeMove(0 ,sPrev,sVal,flagOpt);
{ Для OR, "+" и «-» ничего не надо делать }
TRD_OR,TRD_ADD,TRD_SUB: Result:= #9#9;
{ Иначе генерируем код выполняемой операции }
else Result:= Format(#9 %s'#9'eax,%s',[sOp,sReg]);
end{case};
end
else
if sReg = 1 then { Если операнд = 1 }
begin
case Triad.TrdType of
TRD_OR: { Для команды OR результат всегда = 1 }
Result:= MakeMove(1 ,sPrev,sVal,flagOpt);
{ Для AND ничего не надо делать }
TRD_AND: Result:= #9#9;
{ Для "+" генерируем операцию INC }
TRD_ADD: Result:= #9'inc'#9'eax';
{ Для «-» генерируем операцию DEC }
TRD_SUB: Result:= #9'dec'#9'eax';
{ Иначе генерируем код выполняемой операции }
else Result:= Format(#9 %s'#9'eax,%s',[sOp,sReg]);
end{case};
end
else
if sReg = -1 then { Если операнд = -1 }
begin
case Triad.TrdType of
{ Для "+" генерируем операцию DEC }
TRD_ADD: Result:= #9'dec'#9'eax';
{ Для «-» генерируем операцию INC }
TRD_SUB: Result:= #9'inc'#9'eax';
{ Иначе генерируем код выполняемой операции }
else Result:= Format(#9 %s'#9'eax,%s',[sOp,sReg]);
end{case};
end { Иначе генерируем код выполняемой операции }
else Result:= Format(#9 %s'#9'eax,%s',[sOp,sReg]);
end { Если оптимизация команд выключена,
всегда генерируем код выполняемой операции }
else Result:= Format(#9 %s'#9'eax,%s',[sOp,sReg]);
{ Добавляем к результату информацию о триаде
в качестве комментария }
Result:= Result + Format(#9 { %s },
[Triad.MakeString(i)]);
end;
function MakeAsmCode(
listTriad: TTriadList;{входной список триад}
listCode: TStrings;{список строк результирующего кода}
flagOpt: Boolean{флаг оптимизации}): integer;
{ Функция построения ассемблерного кода по списку триад }
var i,iCnt: integer;{счетчик и переменная цикла}
sR: string;{строка для имени регистра}
sPrev,sVal: string;
{строки для хранения предыдущей команды и значения eax}
procedure TakePrevAsm;
{ Процедура, выделяющая предыдущую команду и значение eax
из списка результирующих команд }
var j: integer;
begin
j:= listCode.Count;
if j > 0 then
begin
sPrev:= listCode[j-1];
sVal:= StrPas(PChar(listCode.Objects[j-1]));
end
else
begin
sPrev:= ; sVal:= ;
end;
end;
procedure MakeOper1(const sOp,{код операции}
sAddOp: string;{код дополнительной операции}
iOp: integer{номер операнда в триаде});
{ Функция генерации кода для унарных операций }
var sReg{строка для имени регистра}: string;
begin
TakePrevAsm; {Берем предыдущую команду и значение из eax}
{ Запоминаем имя операнда }
sReg:= GetOpName(i,listTriad,iOp);
if sReg <> then { Если имя пустое, операнд уже есть в
регистре eax от выполнения предыдущей триады,}
begin { иначе его нужно занести в eax }
{ Вызываем функцию генерации кода занесения операнда }
sReg:= MakeMove(sReg,sPrev,sVal,flagOpt);
if sReg <> then listCode.Add(sReg);
end; { Генерируем непосредственно код операции }
listCode.Add(Format(#9 %s'#9'eax'#9 { %s },
[sOp,listTriad[i].MakeString(i)]));
if sAddOp <> then { Если есть дополнительная операция,
генерируем ее код }
listCode.Add(Format(#9 %s'#9'eax,1,[sAddOp]));
if listTriad[i].Info <> 0 then { Если триада связана с
begin { регистром, запоминаем результат в этом регистре }
sReg:= GetRegName(listTriad[i].Info);
{ При этом запоминаем, что сейчас находится в eax }
listCode.AddObject(Format(#9'mov'#9 %s,eax',[sReg]),
TObject(PChar(sReg)));
end;
end;
procedure MakeOper2(const sOp,{код операции}
sAddOp: string{код дополнительная операции});
{ Функция генерации кода для бинарных арифметических
и логических операций }
var sReg1,sReg2{строки для имен регистров}: string;
begin
TakePrevAsm; {Берем предыдущую команду и значение из eax}
{ Запоминаем имена первого и второго операндов }
sReg1:= GetOpName(i,listTriad,1);
sReg2:= GetOpName(i,listTriad,2);
{ Если имя первого операнда пустое, значит, он уже
есть в регистре eax от выполнения предыдущей триады -
вызываем функцию генерации кода для второго операнда }
if (sReg1 = ) or (sReg1 = sVal) then
listCode.Add(MakeOpCode(i,listTriad,sOp,sReg2,
sPrev,sVal,flagOpt))
else { Если имя второго операнда пустое, значит он уже
есть в регистре eax от выполнения предыдущей триады -
вызываем функцию генерации кода для первого операнда }
if (sReg2 = ) or (sReg2 = sVal) then
begin
listCode.Add(MakeOpCode(i,listTriad,sOp,sReg1,
sPrev,sVal,flagOpt));
{ Если есть дополнительная операция, генерируем ее код
(когда операция несимметричная – например "-") }
if sAddOp <> then
listCode.Add(Format(#9 %s'#9'eax',[sAddOp]));
end
else { Если оба операнда не пустые, то надо:
– сначала загрузить в eax первый операнд;
– сгенерировать код для обработки второго операнда.}
begin
sReg1:= MakeMove(sReg1,sPrev,sVal,flagOpt);
if sReg1 <> then listCode.Add(sReg1);
listCode.Add(MakeOpCode(i,listTriad,sOp,sReg2,
sPrev,sVal,flagOpt));
end;
if listTriad[i].Info <> 0 then { Если триада связана с
begin { регистром, запоминаем результат в этом регистре }
sReg1:= GetRegName(listTriad[i].Info);
{ При этом запоминаем, что сейчас находится в eax }
listCode.AddObject(Format(#9'mov'#9 %s,eax',[sReg1]),
TObject(PChar(sReg1)));
end;
end;
procedure MakeCompare(const sOp: string
{флаг операции сравнения});
{ Функция генерации кода для операций сравнения }
var sReg1,sReg2{строки для имен регистров}: string;
Читать дальшеИнтервал:
Закладка: