![]() |
ASM-Funktion Verständnisfrage und anpassen (asm > Delphicode? bzw asm x86 > x64?)
Aus einem interessanten aber relativ komplizierten Algorithmus habe ich diese Funktion
Delphi-Quellcode:
Das ist im Prinzip ein Random().
function randInt_MT19937(Range: LongInt): LongInt;
asm PUSH EAX CALL genrand_MT19937 POP EDX MUL EDX MOV EAX,EDX end; Der parameter Range wird in EAX geschrieben. Wie würde diese Funktion aber aussehen, wenn als Parameter
Delphi-Quellcode:
dort stände?
Min, Max: LongInt
![]() Hier steht man schreibt den zweiten Parameter in EDX aber der wird schon verwendet. Danach wird genrand_MT19937 aufgerufen. Aber woher weiß die Funktion, was
Delphi-Quellcode:
ist?
Range
Delphi-Quellcode:
Die Unit gibt es hier
function genrand_MT19937: LongInt;
const mag01: array [0 .. 1] of LongInt = (0, MT19937MATRIX_A); var y: LongInt; kk: Integer; begin if mti >= MT19937N{generate MT19937N longints at one time} then begin if mti = (MT19937N + 1) then // if sgenrand_MT19937() has not been called, sgenrand_MT19937(4357); // default initial seed is used for kk := 0 to MT19937N - MT19937M - 1 do begin y := (mt[kk] and MT19937UPPER_MASK) or (mt[kk + 1] and MT19937LOWER_MASK); mt[kk] := mt[kk + MT19937M] xor (y shr 1) xor mag01[y and $00000001]; end; for kk := MT19937N - MT19937M to MT19937N - 2 do begin y := (mt[kk] and MT19937UPPER_MASK) or (mt[kk + 1] and MT19937LOWER_MASK); mt[kk] := mt[kk + (MT19937M - MT19937N)] xor (y shr 1) xor mag01[y and $00000001]; end; y := (mt[MT19937N - 1] and MT19937UPPER_MASK) or (mt[0] and MT19937LOWER_MASK); mt[MT19937N - 1] := mt[MT19937M - 1] xor (y shr 1) xor mag01[y and $00000001]; mti := 0; end; y := mt[mti]; inc(mti); y := y xor (y shr 11); y := y xor (y shl 7) and TEMPERING_MASK_B; y := y xor (y shl 15) and TEMPERING_MASK_C; y := y xor (y shr 18); Result := y; end; ![]() |
AW: ASM-Funktion Verständnisfrage und anpassen
Zitat:
Der ASM-Code liest sich in etwa so:
Delphi-Quellcode:
Wenn du einen Range-Min/Max willst, würde ich einen Wrapper um diese Funktion machen:
PUSH EAX { Parameter 1 = Range auf Stack sichern }
CALL genrand_MT19937 { Random Longint aufrufen, Ergebnis steht in EAX (keine Parameter)} POP EDX { Range vom Stack in EDX holen } MUL EDX { EAX mit EDX multiplizieren } MOV EAX,EDX { Ergebnis in result = EDX zurückgeben }
Delphi-Quellcode:
function RandInt_Range(AMin, AMax: Longint): Longint;
begin Result := randInt_MT19937(AMax - AMin) + AMin; end; |
AW: ASM-Funktion Verständnisfrage und anpassen
Zitat:
Wenn ich randInt_MT19937 mit Hilfe deiner Kommentare in normalen Delphi-Code umwandle, hat das irgendwelche Nachteile? Wenn nicht, würde ich das machen.
Delphi-Quellcode:
Diese Funktion schmeiße ich dann raus
function randInt_MT19937(Range: LongInt): LongInt;
begin Result := Range * genrand_MT19937; end; function randInt_MT19937(Min, Max: LongInt): LongInt; begin Result := randInt_MT19937(Max - Min) + Min; end;
Delphi-Quellcode:
function randFloat_MT19937: Double;
const Minus32: Double = -32.0; asm CALL genrand_MT19937 PUSH 0 PUSH EAX FLD Minus32 FILD qword ptr [ESP] ADD ESP,8 FSCALE FSTP ST(1) end; |
AW: ASM-Funktion Verständnisfrage und anpassen
Zitat:
Zitat:
![]() |
AW: ASM-Funktion Verständnisfrage und anpassen
Delphi-Quellcode:
Ist es normal, dass randInt_MT19937(50, 60) alles zurückgibt, außer eine Zahl zwischen 50 und 60?
function randInt_MT19937(Range: LongInt): LongInt;
begin Result := Range * genrand_MT19937; end; function randInt_MT19937(Min, Max: LongInt): LongInt; begin Result := randInt_MT19937(Max - Min) + Min; end; Die erste Funktion verstehe ich aber auch nicht. Wenn ich randInt_MT19937(100) eingebe, bekomme ich auch alles zurück. Aber es stoppt nicht bei der 50. Negative Zahlen kommen auch manchmal raus. Der ASM-Code macht hingegen das was er soll. Das hier mal Zeile für Zeile übersetzt
Delphi-Quellcode:
Erst kommt Range in EAX. Dann wird das Resultat von genrand_MT19937 in EAX geschrieben. Range in EAX wird überschrieben.
asm
PUSH EAX {Parameter 1 = Range auf Stack sichern} CALL genrand_MT19937 {Random Longint aufrufen, Ergebnis steht in EAX (keine Parameter)} POP EDX {Range vom Stack in EDX holen} MUL EDX {EAX mit EDX multiplizieren} MOV EAX,EDX {Ergebnis in result = EDX zurückgeben} end; Jetzt ist Range plötzlich in EDX (?) und danach wird EAX mit EDX multipliziert. |
AW: ASM-Funktion Verständnisfrage und anpassen (asm > Delphicode? bzw asm x86 > x64?)
Ich muss die Beschreibung des MOV Befehls korrigieren (ist schon 'ne Weile her mit ASM): Die Richtung ist anders herum
Delphi-Quellcode:
Der Delphi-Code sähe dann so aus:
asm
PUSH EAX {Parameter 1 = Range auf Stack sichern} CALL genrand_MT19937 {Random Longint aufrufen, Ergebnis steht in EAX (keine Parameter)} POP EDX {Range vom Stack in EDX holen} MUL EDX {EAX mit EDX multiplizieren, Ergebnis steht in EDX,EAX } MOV EAX,EDX {Die höheren 32-Bits in EDX als result in EAX zurückgeben } end;
Delphi-Quellcode:
Result := Int64Rec(Int64(Range) * genrand_MT19937).Hi; |
AW: ASM-Funktion Verständnisfrage und anpassen (asm > Delphicode? bzw asm x86 > x64?)
Mh ich verstehe es nicht.
Jetzt bekomme ich fast immer negative Zahlen wenn man zum Beispiel 10 als Range angibt und ich verstehe grundsätzlich nicht warum. Bei Range 10 gehen die Ergebnisse ziemlich genau von -5 bis +5. |
AW: ASM-Funktion Verständnisfrage und anpassen (asm > Delphicode? bzw asm x86 > x64?)
Zitat:
Delphi-Quellcode:
ist als
Int64Rec.Hi
Delphi-Quellcode:
deklariert. Da sind negative Werte eher selten zu erwarten.
Cardinal
|
AW: ASM-Funktion Verständnisfrage und anpassen (asm > Delphicode? bzw asm x86 > x64?)
Teste es =) Irgendwas ist hier ganz seltsam. Das verstehe ich ja auch nicht.
Delphi-Quellcode:
function randInt_MT19937(Range: LongInt): LongInt;
begin Result := Int64Rec(Int64(Range) * genrand_MT19937).Hi; end; procedure TForm1.FormCreate(Sender: TObject); var i: Integer; x: string; begin for i := 0 to 30 do x := x + sLineBreak + IntToStr(randInt_MT19937(10)); ShowMessage(x); end;
Code:
[Window Title]
Project1 [Content] 1 1 -4 3 -4 -4 3 -1 -3 3 -1 -3 -2 0 2 -5 //////// -4 2 0 -1 -3 -2 -3 3 -3 5 //////// 2 -3 -4 3 3 [OK] |
AW: ASM-Funktion Verständnisfrage und anpassen (asm > Delphicode? bzw asm x86 > x64?)
Das Problem ist, daß genrand_MT19937 als LongInt deklariert ist, wo es das doch gar nicht sein sollte.
Delphi-Quellcode:
Result := Int64Rec(Int64(Range) * Cardinal(genrand_MT19937)).Hi;
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 07:18 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-2025 by Thomas Breitkreuz