| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111 | { %CPU=i386,x86_64 }program blea;{$IF not defined(CPUX86) and not defined(CPUX86_64)}  {$FATAL This test program requires an Intel x86 or x64 processor }{$ENDIF}{$MODE OBJFPC}{$ASMMODE Intel}{$DEFINE DETECTCPU}uses  SysUtils {$ifdef DETECTCPU}, CPU {$endif};type  TBenchmarkProc = function(const Input, X, Y: LongWord): LongWord;function Checksum_PAS(const Input, X, Y: LongWord): LongWord;var  Counter: LongWord;begin  Result := Input;  Counter := Y;  while (Counter > 0) do    begin      Result := Result + X + $87654321;      Result := Result xor Counter;      Dec(Counter);    end;end;function Checksum_ADD(const Input, X, Y: LongWord): LongWord; assembler; nostackframe;asm@Loop1:  ADD Input, $87654321  ADD Input, X  XOR Input, Y  DEC Y  JNZ @Loop1  MOV Result, Inputend;function Checksum_LEA(const Input, X, Y: LongWord): LongWord; assembler; nostackframe;asm@Loop2:  LEA Input, [Input + X - 2023406815] {+$87654321 in decimal}  XOR Input, Y  DEC Y  JNZ @Loop2  MOV Result, Inputend;function Benchmark(const name: string; proc: TBenchmarkProc; Z, X: LongWord): LongWord;const  internal_reps = 1000;var  start: TDateTime;  time: double;  reps: cardinal;begin  Result := Z;  reps := 0;  Write(name, ': ');  start := Now;  repeat    inc(reps);    Result := proc(Result, X, internal_reps);  until (reps >= 100000);  time := ((((Now - start) * SecsPerDay) * 1e9) / internal_reps) / reps;  WriteLn(time:0:(2 * ord(time < 10)), ' ns/call');end;var  Results: array[0..2] of LongWord;  FailureCode: Integer;begin{$ifdef CPUX86_64}  Write('64-bit');{$else CPUX86_64}  Write('32-bit');{$endif CPUX86_64}{$ifdef DETECTCPU}  if CPUBrandString <> '' then    begin      WriteLn(' CPU = ', CPUBrandString);      WriteLn('-------------', StringOfChar('-', length(CPUBrandString)));    end;{$else}  WriteLn;{$endif}  Results[0] := Benchmark('   Pascal control case', @Checksum_PAS, 5000000, 1000);  Results[1] := Benchmark(' Using LEA instruction', @Checksum_LEA, 5000000, 1000);  Results[2] := Benchmark('Using ADD instructions', @Checksum_ADD, 5000000, 1000);  FailureCode := 0;  if (Results[0] <> Results[1]) then    begin      WriteLn('ERROR: Checksum_LEA doesn''t match control case');      FailureCode := FailureCode or 1;    end;  if (Results[0] <> Results[2]) then    begin      WriteLn('ERROR: Checksum_ADD doesn''t match control case');      FailureCode := FailureCode or 2    end;  if FailureCode <> 0 then    Halt(FailureCode);end.
 |