AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Zurück Delphi-PRAXiS Programmierung allgemein Programmieren allgemein [ASM] Lange Strings als Function - Wert zurückgeben
Thema durchsuchen
Ansicht
Themen-Optionen

[ASM] Lange Strings als Function - Wert zurückgeben

Ein Thema von Chemiker · begonnen am 18. Jan 2008 · letzter Beitrag vom 21. Jan 2008
Antwort Antwort
Seite 1 von 2  1 2      
Benutzerbild von Chemiker
Chemiker

Registriert seit: 14. Aug 2005
1.859 Beiträge
 
Delphi 11 Alexandria
 
#1

[ASM] Lange Strings als Function - Wert zurückgeben

  Alt 18. Jan 2008, 20:24
Hallo,

in der Dephi – Hilfe steht das man lange Strings im EAX – Register wieder zurückgibt.

Zitat:
Funktionsergebnisse
Assembler-Funktionen liefern ihre Ergebnisse folgendermaßen zurück:
• Ordinale Werte werden in AL (8-Bit-Werte), AX (16-Bit-Werte) oder EAX (32-Bit- Werte) zurückgeliefert.
• Reelle Werte werden in ST(0) über den Register-Stack des Coprozessors zurückgegeben. (Currency-Werte werden mit dem Faktor 10000 skaliert.)
• Zeiger einschließlich langer Strings werden in EAX zurückgeliefert.
• Kurze Strings und Varianten werden an die temporären Adresse zurückgegeben, auf die @Result zeigt.
z.B.:
Delphi-Quellcode:
function strRueckgabe (s: PChar) : String;
ASM
… Mach was mit s in ASM
Functionsergebnis in EAX
END;
Mir ist jetzt nur nicht klar, wie ich den langen String in EAX bekomme (Also der Pointer), muss ich dafür in der Function selber den Speicherplatz reservieren, oder hat ein anderer Register die Anfangsadresse?

Bis bald Chemiker
wer gesund ist hat 1000 wünsche wer krank ist nur einen.
  Mit Zitat antworten Zitat
Apollonius

Registriert seit: 16. Apr 2007
2.325 Beiträge
 
Turbo Delphi für Win32
 
#2

Re: [ASM] Lange Strings als Function - Wert zurückgeben

  Alt 18. Jan 2008, 20:27
Schau dir einfach mal an, was der Delphi-Compiler normalerweise bastelt. Für dich dürfte die Funktion System.@NewAnsiString interessant sein (ich habe ziemlich lange gebraucht, bis ich begriffen habe, dass man die ganzen "Compiler-Magic-Funktion" aus System immer voll qualifizieren muss).
Dein PChar liegt ja am Anfang auch in eax.
Wer erweist der Welt einen Dienst und findet ein gutes Synonym für "Pointer"?
"An interface pointer is a pointer to a pointer. This pointer points to an array of pointers, each of which points to an interface function."
  Mit Zitat antworten Zitat
Benutzerbild von 3_of_8
3_of_8

Registriert seit: 22. Mär 2005
Ort: Dingolfing
4.129 Beiträge
 
Turbo Delphi für Win32
 
#3

Re: [ASM] Lange Strings als Function - Wert zurückgeben

  Alt 18. Jan 2008, 20:28
Dafür gibt es String-Funktionen. Schreib einfach mal irgendwas mit Strings in Delphi, mach nen Breakpoint rein und schau dir im CPU-Fenster an, wie der Compiler es macht.
Manuel Eberl
„The trouble with having an open mind, of course, is that people will insist on coming along and trying to put things in it.“
- Terry Pratchett
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

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

Re: [ASM] Lange Strings als Function - Wert zurückgeben

  Alt 18. Jan 2008, 21:22
jupp, schau mal in die System.pas, da gibt's die nötigen StringFunktionen, oder im CPU-Fenster nachsehn was Delphi gern aufruft.

wenn es unbedingt sein muß, dann kannst du auch einen kleinen Umweg gehn
Delphi-Quellcode:
Function strRueckgabe(S: PChar): String;
  Var Temp: Array[0..1023] of Char;

  Begin
    ASM
      MOV EAX, &S
      MOV EDX, &Temp

      // in EAX steht nun der EingabeString-S
      // und in EDX stünde der Zeiger zu einem Puffer für den Rückgabewert
    END;
    Result := Temp;
  End;
Neuste Erkenntnis:
Seit Pos einen dritten Parameter hat,
wird PoSex im Delphi viel seltener praktiziert.
  Mit Zitat antworten Zitat
Benutzerbild von Chemiker
Chemiker

Registriert seit: 14. Aug 2005
1.859 Beiträge
 
Delphi 11 Alexandria
 
#5

Re: [ASM] Lange Strings als Function - Wert zurückgeben

  Alt 19. Jan 2008, 01:27
Hallo,

danke für die Antworten.

Trotzdem verstehe ich immer noch nicht wie es funktioniert.

Mal ein Beispiel für die Rückgabe eines integer – Wertes.

Delphi-Quellcode:
function ASM_strLaenge(s: PChar): integer;
ASM
      PUSH EBX // EBX sichern
      MOV EDX, EAX // Adresse in EDX
      XOR EAX, EAX // EAX auf 0 setzen
      MOV EBX, EAX // EBX aof 0 setzen
      JMP @@Vergleich
@@Weiter:
      INC EDX // Adresse um 1 Erhöhen
      INC EAX // Anzahl um 1 erhöhen
@@Vergleich:
      MOV BL, [EDX] // Zeichen laden nach BL
      CMP byte ptr [EDX], $00 // Zeichen auf #0 testen
      JNZ @@Weiter // Wenn nicht String Ende dann weiter
      POP EBX // EBX vom Stack holen
END;
So das ist kein Problem.

Aber, wenn ich diese zugegebener Weise nutzlose Funktion, in ASM schreiben möchte, wie sieht denn dann der Code aus.

Delphi-Quellcode:
function ASM_StringTest(s: String):String;
begin
   Result:= s;
end;

Bis bald Chemiker
wer gesund ist hat 1000 wünsche wer krank ist nur einen.
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

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

Re: [ASM] Lange Strings als Function - Wert zurückgeben

  Alt 19. Jan 2008, 02:02
dein
Delphi-Quellcode:
function ASM_StringTest(s: String):String;
begin
   Result:= s;
end;
sieht in ASM virtuell so aus
Delphi-Quellcode:
Function ASM_StringTest(S: String; In Result: String): String;
// das "In Result: String" wird sozusagen vom Compiler hinzugefügt, um
// Stringvariablen und andere dynamische Arrays optimaler verwalten zu können.
ASM
  // EAX = Refferenz auf S
  // EDX = Refferenz auf Result (vorinitialisiert)

  // EDX bearbeiten, z.B. S zuweisen

  MOV EAX, ...{EDX}  // und zum Schluß die Refferenz in EAX zurückgeben
End;


Bei Varianten mit festen Speichern, wie in Beitrag #4, braucht man sich dagegen um Speicherresservierung garkeine Sorgen zu machen ... S als PChar interpretieren (bei 'nem reinem Lesezugriff ist das möglich), der Puffer braucht nicht verwaltet zu werden und um den Rest kümmert sich Delphi.
Neuste Erkenntnis:
Seit Pos einen dritten Parameter hat,
wird PoSex im Delphi viel seltener praktiziert.
  Mit Zitat antworten Zitat
Benutzerbild von Chemiker
Chemiker

Registriert seit: 14. Aug 2005
1.859 Beiträge
 
Delphi 11 Alexandria
 
#7

Re: [ASM] Lange Strings als Function - Wert zurückgeben

  Alt 20. Jan 2008, 18:09
Hallo,

habe die Rückgabe eines Strings in einer Function jetzt so gelöst. Kann einer sich den Code mal ansehen, ob er so ok ist. Das Umwandeln in Kleinbuchstabe vom String ist nur für Testzwecke um zu sehen ob das ganze Ordentlich funktioniert.

Delphi-Quellcode:
function ASM_StringTest(s: String): String;
var
  StrPChar: PChar; // String Adresse von s merken
ASM
        PUSH EBX // EBX --> Stack
        MOV StrPChar, EAX
//******************************************************************************
// String - Länge von s bestimmen
// Parameter: EAX <-- Adresse von s
//******************************************************************************
        MOV EBX, EAX // Adresse s --> EBX
        XOR EAX, EAX // EAX:= 0 // Vorbreitung zu zählen
        JMP @@Vergleich
@@Weiter1:
        INC EBX // EBX:= s[EBX]
        INC EAX // EAX:= EAX + 1 // Zähler um 1 erhöhen
@@Vergleich:
                                  // Zeichen was an EBX in s steht vergleichen
        CMP byte ptr [EBX], $00 // if EBX <> #0 then
                                  // Weiter1;
        JNZ @@Weiter1 // endif
//******************************************************************************
// In EAX ist die String - Länge von s
// String - Länge ENDE
//******************************************************************************
        MOV EBX, EAX // s String- Länge --> EBX
//******************************************************************************
// Vorbereitung um den Speicher für den Rückgabe - String vorzubereiten
// Zustand der Reg.: EAX = String - Länge von s
// EDX = Adresse vom Rückgabe - String
// Für den Aufruf der Procedure: _LStrSetLength in der System-Unit müssen die
// beiden Inhalte der Reg. getauscht werden.
// procedure _LStrSetLength (Adresse, String-Länge)
// _LStrSetLength (EAX, EDX) (
        XCHG EDX, EAX // Adresse vom Rückgabe-String in EAX
                                   // Länge die Reserviert werden soll in EDX
        CALL System.@LStrSetLength // _LStrSetLength (EAX, EDX);
                                   // _LStrSetLength (Adresse, Länge)
//******************************************************************************
// In EAX ist die Adresse vom Rückgabe - Sring
//* ****************************************************************************

                                  // Prüfen ob der String ='' ist
                                  // if StrLaenge = 0 then exit;
      MOV ECX, EBX // ECX := StrLaenge;
      TEST ECX, ECX // Flags setzen
      JZ @@Schluss // Wenn die String - Länge = 0 dann ist

//******************************************************************************
// Kopieren eines Strings der in KleinBuchstaben umgewandelt wird und in den
// Rückgabe String kopiert wird.
// nur für Testzwecke ob der Rückgabe - String richtig zurückgegeben wird.
//******************************************************************************

      PUSH EAX // Adresse vom Rückgabe - String --> Stack

      PUSH ESI // ESI --> Stack
      PUSH EDI // EDS --> Stack
      MOV ESI, StrPChar
      MOV EDI, [EAX]
@@CopySchleife:
      LODSb
      CMP AL, 'A'
      JB @@Buchstabe
      CMP AL, 'Z'
      JA @@Buchstabe
      ADD AL, 32
@@Buchstabe:
      STOSb
      AND AL, AL
      JNZ @@CopySchleife
      POP EDI // Stack --> EDI
      POP ESI // Stack --> ESI

      POP EAX // Adresse Rückgabe String --> EAX
@@Schluss:
      POP EBX // Stack --> EBX

END;




Bis bald Chemiker
wer gesund ist hat 1000 wünsche wer krank ist nur einen.
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

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

Re: [ASM] Lange Strings als Function - Wert zurückgeben

  Alt 20. Jan 2008, 19:12
Delphi-Quellcode:
// EAX = Zeiger auf String

TEST EAX, EAX
JZ @EmptyString // S = ''
MOV EDX, DWOD PTR [EAX - 4]

// EDX = StringLänge in Byte (ohne #0)
EAX zeigt auf @TAnsiStringInfo.Data
und TAnsiStringInfo[S].ElementCount = Length(S)
Delphi-Quellcode:
TAnsiStringInfo = packed Record
  RefCount: LongInt;
  ElementCount: LongInt;
  Data: packed Array[1..High(Integer) - SDynArrayInfo] of AnsiChar;
End;
und MSDN-Library durchsuchenLStrLen macht intern sowas wie dein "CMP byte ptr [EBX], $00"
PChar_Length(EAX) = LStrLen(EAX)
[edit=Matze]Code auf Wunsch von himitsu angepasst. MfG, Matze[/edit]
Neuste Erkenntnis:
Seit Pos einen dritten Parameter hat,
wird PoSex im Delphi viel seltener praktiziert.
  Mit Zitat antworten Zitat
Benutzerbild von Chemiker
Chemiker

Registriert seit: 14. Aug 2005
1.859 Beiträge
 
Delphi 11 Alexandria
 
#9

Re: [ASM] Lange Strings als Function - Wert zurückgeben

  Alt 21. Jan 2008, 21:07
Hallo himitsu,

das ist ja eine sehr kurze Methode um die Länge von einem AnsiString zu ermitteln. Habe den Record zwar auch beim Analysieren von der System – Unit gesehen, aber nicht getraut zu Verwenden (wegen Seiteneffekte).

Nur so wie Du den Code – Schnipsel geschrieben hast wird immer die Länge des Strings mit Null angezeigt.

XOR EAX, EAX // ergibt immer 0 Habe es etwas modifiziert:

Delphi-Quellcode:
function ASM_StrLaengeKurz(s: String): integer;
ASM
  MOV EDX, EAX
  JZ @@EmptyString // S = ''
  MOV EAX, [EDX - 4]
@@EmptyString:
END;
Danke für Deinen Hinweis.


Bis bald Chemiker
wer gesund ist hat 1000 wünsche wer krank ist nur einen.
  Mit Zitat antworten Zitat
Apollonius

Registriert seit: 16. Apr 2007
2.325 Beiträge
 
Turbo Delphi für Win32
 
#10

Re: [ASM] Lange Strings als Function - Wert zurückgeben

  Alt 21. Jan 2008, 21:22
Dein bedingter Sprung scheint mir sehr gewagt. Mov setzt keine Flags, also ist das Zero-Flag nach dem Zufallsprinzip gesetzt.
Ich würde die Funktion folglich so schreiben:
Delphi-Quellcode:
function ASM_StrLaengeKurz(s: String): integer;
ASM
  TEST EAX, EAX
  JZ @@EmptyString // S = ''
  MOV EAX, [EAX - 4]
@@EmptyString:
END;
Wer erweist der Welt einen Dienst und findet ein gutes Synonym für "Pointer"?
"An interface pointer is a pointer to a pointer. This pointer points to an array of pointers, each of which points to an interface function."
  Mit Zitat antworten Zitat
Antwort Antwort
Seite 1 von 2  1 2      


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 22:25 Uhr.
Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz