AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Thema durchsuchen
Ansicht
Themen-Optionen

asm für meine VeryLongInteger-Unit

Ein Thema von Talia · begonnen am 22. Mär 2011 · letzter Beitrag vom 30. Mär 2011
Antwort Antwort
Benutzerbild von himitsu
himitsu

Registriert seit: 11. Okt 2003
Ort: Elbflorenz
44.326 Beiträge
 
Delphi 12 Athens
 
#1

AW: asm für meine VeryLongInteger-Unit

  Alt 22. Mär 2011, 12:14
PS: Die TList in deinem "Record".
Sicher das du keine Speicherlecks und andere Probleme damit verursachst?

Mein Vorschlag die Speicherverwaltung für derartige Records zu erweitern,
um derartige Objekte "ordentlich" innerhalb dieser Records verwalten zu können,
wurde bisher immernoch hartnäckig von Embarcadero ignoriert.

http://qc.embarcadero.com/wc/qcmain.aspx?d=82524
Ein Therapeut entspricht 1024 Gigapeut.
  Mit Zitat antworten Zitat
Talia

Registriert seit: 6. Feb 2010
11 Beiträge
 
#2

AW: asm für meine VeryLongInteger-Unit

  Alt 22. Mär 2011, 12:36
Hallo,

ich schwenke jetzt wohl doch um auf einen array of Cardinal für die Ziffern. Die TList war nur auf den ersten Blick so praktisch dadurch, daß sie sich selbst um ihre Länge kümmert und .Count liefert. Schade, nun habe ich doch die Probleme mit der blöden SetLengtherei am Hals.

Gruß Talia
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

Registriert seit: 11. Okt 2003
Ort: Elbflorenz
44.326 Beiträge
 
Delphi 12 Athens
 
#3

AW: asm für meine VeryLongInteger-Unit

  Alt 22. Mär 2011, 13:22
Und Achtung, "normale" dynamische Arrays haben keine Referenzkontrolle in den Schreibzugriffen implementiert.
(Strings sind zwar auch "nur" dynamische Arrays, aber diese besitzen soeine Kontrolle)

Delphi-Quellcode:
var
  a, b: array of Integer;

SetLength(a, 10);
a[5] := 20;
b := a;
b[5] := 30;
ShowMessage(IntToStr(a[5])); // a[5] = 30 ???

SetLength(a, 10);
a[5] := 20;
b := a;
SetLength(b, Length(b)); // vor Schreibzugriffen auf ein Array
b[5] := 30;
ShowMessage(IntToStr(a[5])); // juhu, a[5] ist noch 20 :)
Auch gibt es ein immernoch nicht behobenes Compiler-Problem, wenn in dem Record nur ein Array und kein weiteres Feld enthalten ist.

PS: von mir gammeln in der DP einige Prototypen/Vorlagen rum, für derartige Typen.
Ein Therapeut entspricht 1024 Gigapeut.
  Mit Zitat antworten Zitat
Talia

Registriert seit: 6. Feb 2010
11 Beiträge
 
#4

AW: asm für meine VeryLongInteger-Unit

  Alt 24. Mär 2011, 09:13
So, ich habe schon wieder umdisponiert, bin wohl etwas flatterhaft:

Delphi-Quellcode:
const
  MaxDigits = 1039; // (2^32)^1040 = 1,89782E10018 => 10.000 dezi Stellen o.K.

type
  TVLInt = packed record
  strict private
    Digits: array[0..MaxDigits] of Cardinal;
    IsNegative: Boolean;
    Length: Integer;
    ...
  public
    class operator Implicit(const a: Cardinal): TVLInt;
    ...


implementation

class operator TVLInt.Implicit(const a: Cardinal): TVLInt;
asm
  // eax <= a
  mov [edx], eax
  mov edx.IsNegative, 0
  xor ecx, ecx
  test eax, eax // ZF if eax = 0
  jz @@IsZero
  add ecx, 1
@@IsZero:
  mov edx.Length, ecx
  // edx = @Result
end;
So läßt es sich leicht rechnen und ich muß mir keine Gedanken um Speicherlecks u. ä. machen. 10.000 dez Ziffern sind für alle meine Anwendungen mhr als genug. Den übertriebenen Speicherverbrauch werde ich wohl verschmerzen.

Gruß Talia
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

Registriert seit: 11. Okt 2003
Ort: Elbflorenz
44.326 Beiträge
 
Delphi 12 Athens
 
#5

AW: asm für meine VeryLongInteger-Unit

  Alt 24. Mär 2011, 09:29
Delphi-Quellcode:
mov [edx], eax
mov edx.IsNegative, 0
Ich bin mir grad nicht so sicher, ob das richtigrum ist.

EAX = Self (der Record selber)
EDX = 1. Parameter, also a

Bei dir sieht es so aus, als wenn du es andersrum machst. :gruebl:
Ein Therapeut entspricht 1024 Gigapeut.
  Mit Zitat antworten Zitat
Talia

Registriert seit: 6. Feb 2010
11 Beiträge
 
#6

AW: asm für meine VeryLongInteger-Unit

  Alt 29. Mär 2011, 14:47
Hallo himitsu,

ja, darüber habe ich mich auch gewundert, aber so funktioniert es wirklich. Im Internet habe ich hierzu 2 Infos gefunden:

1. Bei Methoden: EAX = Self, EDX, ECX

2. Liefert eine Funktion einen array als Ergebnis, wird er behandelt wie eine zusätzliche Var.

Keine Ahnung, warum hier 2. gilt.


Ich habe nochmal an meiner Deklaration rumgefummelt, würde doch lieber einen dyn array verwenden, um für alle Fälle gerüstet zu sein:

Delphi-Quellcode:
type
  TVLInt = record
  strict private
    Digits: array of Cardinal;
    IsNegative: Boolean;
    Length: Integer;
    ...
  public
    class operator Implicit(const a: Cardinal): TVLInt;
    ...
  end;


implementation

class operator TVLInt.Implicit(const a: Cardinal): TVLInt;
begin
  SetLength(Result.Digits,1);
  Result.Digits[0] := a;
  Result.Length := Integer(a > 0);
end;

Dann habe ich mir mit Strg-Alt-c angeguckt, was Delphi beim Anlegen so treibt und folgendes geguttenbergt:

Delphi-Quellcode:
class operator TVLInt.Implicit(const a: Cardinal): TVLInt;
// eax <= a
asm
// MyVLInt.pas.41: begin
  push ebx
  push esi
  mov ebx,edx
  mov esi,eax
// MyVLInt.pas.42: SetLength(Result.Digits,1);
  push $01
  mov eax,ebx
  mov ecx,$00000001
  mov edx,[$00455c9c] // ??? typeInfo??
  call System.@DynArraySetLength
  add esp,$04
// MyVLInt.pas.43: Result.Digits[0] := a;
  mov eax,[ebx]
  mov [eax],esi
// MyVLInt.pas.44: Result.Length := Integer(a > 0);
  test esi,esi
  setnbe al
  and eax,$7f
  mov [ebx+$05],eax
// MyVLInt.pas.71: end;
  pop esi
  pop ebx
  ret
end;

Es gibt hier Schwierigkeiten mit DynArraySetLength (Zugriffsverletzung).

Emba schreibt: DynArraySetLength(var a: Pointer; typeInfo: Pointer; dimCnt: Longint; lengthVec: PLongint);

aber wie man edx richtig bestückt ist mir nicht klar. Ich habe einiges ausprobiert, produziere aber immer Murks.


Gruß Talia
  Mit Zitat antworten Zitat
Talia

Registriert seit: 6. Feb 2010
11 Beiträge
 
#7

AW: asm für meine VeryLongInteger-Unit

  Alt 30. Mär 2011, 11:44
Hallo,

als Notlösung habe ich jetzt eine globale Variable eingeführt und unter initialization TypeInfo zugewiesen. Nicht besonders OOP, funktioniert aber.

Gruß Talia
  Mit Zitat antworten Zitat
Antwort Antwort


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 06:10 Uhr.
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz