| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321 | unit Tests.Rtti.Util;{$ifdef fpc}{$mode objfpc}{$H+}{$endif}interfaceuses  TypInfo, Rtti;{$ifndef fpc}type  CodePointer = Pointer;  PCodePointer = ^CodePointer;  SizeInt = NativeInt;  QWord = UInt64;  TValueHelper = record helper for TValue    class procedure Make<T>(const aValue: T; var aResult: TValue); overload; static;    function AsUnicodeString: UnicodeString;    function AsAnsiString: AnsiString;    function AsChar: Char; inline;    function AsAnsiChar: AnsiChar;    function AsWideChar: WideChar;  end;  TTypeDataHelper = record helper for TTypeData    function SetSize: SizeInt; inline;  end;{$endif}const{$if defined(cpui386) or defined(cpux86_64) or defined(cpum68k)}  DefaultCC = ccReg;{$else}  DefaultCC = ccStdCall;{$endif}function CopyValue({$ifdef fpc}constref{$else}const [ref]{$endif} aValue: TValue): TValue;function EqualValues({$ifdef fpc}constref{$else}const [ref]{$endif} aValue1, aValue2: TValue): Boolean;function TypeKindToStr(aTypeKind: TTypeKind): String; inline;function GetInstValue(aValue: TObject): TValue;function GetPointerValue(aValue: Pointer): TValue;function GetIntValue(aValue: SizeInt): TValue;function GetAnsiString(const aValue: AnsiString): TValue;function GetShortString(const aValue: ShortString): TValue;function GetSingleValue(aValue: Single): TValue;function GetDoubleValue(aValue: Double): TValue;function GetExtendedValue(aValue: Extended): TValue;function GetCompValue(aValue: Comp): TValue;function GetCurrencyValue(aValue: Currency): TValue;{$ifdef fpc}function GetArray(const aArg: array of SizeInt): TValue;{$endif}implementationuses  SysUtils, Math;{$ifndef fpc}class procedure TValueHelper.Make<T>(const aValue: T; var aResult: TValue);begin  TValue.Make(@aValue, PTypeInfo(System.TypeInfo(T)), aResult);end;function TValueHelper.AsUnicodeString: UnicodeString;begin  Result := UnicodeString(AsString);end;function TValueHelper.AsAnsiString: AnsiString;begin  Result := AnsiString(AsString);end;function TValueHelper.AsWideChar: WideChar;begin  if Kind <> tkWideChar then    raise EInvalidCast.Create('Invalid cast');  Result := WideChar(Word(AsOrdinal));end;function TValueHelper.AsAnsiChar: AnsiChar;begin  if Kind <> tkChar then    raise EInvalidCast.Create('Invalid cast');  Result := AnsiChar(Byte(AsOrdinal));end;function TValueHelper.AsChar: Char;begin  Result := AsWideChar;end;function TTypeDataHelper.SetSize: NativeInt;begin  Result := SetTypeOrSize;end;{$endif}function CopyValue({$ifdef fpc}constref{$else}const [ref]{$endif} aValue: TValue): TValue;var  arrptr: Pointer;  len, i: SizeInt;begin  if aValue.Kind = tkDynArray then begin    { we need to decouple the source reference, so we're going to be a bit      cheeky here }    len := aValue.GetArrayLength;    arrptr := Nil;    DynArraySetLength(arrptr, aValue.TypeInfo, 1, @len);    TValue.Make(@arrptr, aValue.TypeInfo, Result);    for i := 0 to len - 1 do      Result.SetArrayElement(i, aValue.GetArrayElement(i));  end else    TValue.Make(aValue.GetReferenceToRawData, aValue.TypeInfo, Result);end;function EqualValues({$ifdef fpc}constref{$else}const [ref]{$endif} aValue1, aValue2: TValue): Boolean;var  td1, td2: PTypeData;  i: SizeInt;begin{$ifdef debug}  Writeln('Empty: ', aValue1.IsEmpty, ' ', aValue2.IsEmpty);  Writeln('Kind: ', TypeKindToStr(aValue1.Kind), ' ', TypeKindToStr(aValue2.Kind));  Writeln('Array: ', aValue1.IsArray, ' ', aValue2.IsArray);{$endif}  if aValue1.IsEmpty and aValue2.IsEmpty then    Result := True  else if aValue1.IsEmpty and not aValue2.IsEmpty then    Result := False  else if not aValue1.IsEmpty and aValue2.IsEmpty then    Result := False  else if aValue1.IsArray and aValue2.IsArray then begin    if aValue1.GetArrayLength = aValue2.GetArrayLength then begin      Result := True;      for i := 0 to aValue1.GetArrayLength - 1 do        if not EqualValues(aValue1.GetArrayElement(i), aValue2.GetArrayElement(i)) then begin{$ifdef debug}          Writeln('Element ', i, ' differs: ', IntToHex(aValue1.GetArrayElement(i).AsOrdinal, 4), ' ', IntToHex(aValue2.GetArrayElement(i).AsOrdinal, 4));{$endif}          Result := False;          Break;        end;    end else      Result := False;  end else if aValue1.Kind = aValue2.Kind then begin    td1 := aValue1.TypeData;    td2 := aValue2.TypeData;    case aValue1.Kind of    {$ifdef fpc}      tkBool:        Result := aValue1.AsBoolean xor not aValue2.AsBoolean;    {$endif}      tkSet:        if td1^.SetSize = td2^.SetSize then          if td1^.SetSize < SizeOf(SizeInt) then            Result := aValue1.AsOrdinal = aValue2.AsOrdinal          else            Result := CompareMem(aValue1.GetReferenceToRawData, aValue2.GetReferenceToRawData, td1^.SetSize)        else          Result := False;      tkEnumeration,      tkChar,      tkWChar,    {$ifdef fpc}      tkUChar,    {$endif}      tkInt64,      tkInteger:        Result := aValue1.AsOrdinal = aValue2.AsOrdinal;    {$ifdef fpc}      tkQWord:        Result := aValue1.AsUInt64 = aValue2.AsUInt64;    {$endif}      tkFloat:        if td1^.FloatType <> td2^.FloatType then          Result := False        else begin          case td1^.FloatType of            ftSingle:              Result := SameValue(Single(aValue1.AsExtended), Single(aValue2.AsExtended));            ftDouble:              Result := SameValue(Double(aValue1.AsExtended), Double(aValue2.AsExtended));            ftExtended:              Result := SameValue(aValue1.AsExtended, aValue2.AsExtended);            ftComp:              Result := aValue1.AsInt64 = aValue2.AsInt64;            ftCurr:              Result := aValue1.AsCurrency = aValue2.AsCurrency;          end;        end;    {$ifdef fpc}      tkSString,    {$else}      tkShortString,    {$endif}      tkUString,    {$ifdef fpc}      tkAString,    {$else}      tkAnsiString,    {$endif}      tkWString:        Result := aValue1.AsString = aValue2.AsString;      tkDynArray,      tkArray:        if aValue1.GetArrayLength = aValue2.GetArrayLength then begin          Result := True;          for i := 0 to aValue1.GetArrayLength - 1 do            if not EqualValues(aValue1.GetArrayElement(i), aValue2.GetArrayElement(i)) then begin              Result := False;              Break;            end;        end else          Result := False;      tkClass,      tkClassRef,      tkInterface,    {$ifdef fpc}      tkInterfaceRaw,    {$endif}      tkPointer:        Result := PPointer(aValue1.GetReferenceToRawData)^ = PPointer(aValue2.GetReferenceToRawData)^;    {$ifdef fpc}      tkProcVar:    {$else}      tkProcedure:    {$endif}        Result := PCodePointer(aValue1.GetReferenceToRawData)^ = PCodePointer(aValue2.GetReferenceToRawData)^;      tkRecord,    {$ifdef fpc}      tkObject,    {$endif}      tkMethod,      tkVariant: begin        if aValue1.DataSize = aValue2.DataSize then          Result := CompareMem(aValue1.GetReferenceToRawData, aValue2.GetReferenceToRawData, aValue1.DataSize)        else          Result := False;      end      else        Result := False;    end;  end else    Result := False;end;function TypeKindToStr(aTypeKind: TTypeKind): String;begin{$ifdef fpc}  Str(aTypeKind, Result);{$else}  Result := GetEnumName(TypeInfo(TTypeKind), Ord(aTypeKind));{$endif}end;function GetInstValue(aValue: TObject): TValue;begin  Result := TValue.{$ifdef fpc}specialize{$endif}From<TObject>(aValue);end;function GetPointerValue(aValue: Pointer): TValue;begin  Result := TValue.{$ifdef fpc}specialize{$endif}From<Pointer>(aValue);end;function GetIntValue(aValue: SizeInt): TValue;begin  Result := TValue.{$ifdef fpc}specialize{$endif}From<SizeInt>(aValue);end;function GetAnsiString(const aValue: AnsiString): TValue;begin  Result := TValue.{$ifdef fpc}specialize{$endif}From<AnsiString>(aValue);end;function GetShortString(const aValue: ShortString): TValue;begin  Result := TValue.{$ifdef fpc}specialize{$endif}From<ShortString>(aValue);end;function GetSingleValue(aValue: Single): TValue;begin  Result := TValue.{$ifdef fpc}specialize{$endif}From<Single>(aValue);end;function GetDoubleValue(aValue: Double): TValue;begin  Result := TValue.{$ifdef fpc}specialize{$endif}From<Double>(aValue);end;function GetExtendedValue(aValue: Extended): TValue;begin  Result := TValue.{$ifdef fpc}specialize{$endif}From<Extended>(aValue);end;function GetCompValue(aValue: Comp): TValue;begin  Result := TValue.{$ifdef fpc}specialize{$endif}From<Comp>(aValue);end;function GetCurrencyValue(aValue: Currency): TValue;begin  Result := TValue.{$ifdef fpc}specialize{$endif}From<Currency>(aValue);end;{$ifdef fpc}function GetArray(const aArg: array of SizeInt): TValue;begin  Result := specialize OpenArrayToDynArrayValue<SizeInt>(aArg);end;{$endif}end.
 |