Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Sonstige Fragen zu Delphi (https://www.delphipraxis.net/19-sonstige-fragen-zu-delphi/)
-   -   Wie TLS Callback in Delphi (https://www.delphipraxis.net/155467-wie-tls-callback-delphi.html)

itsme 25. Okt 2010 10:15

Wie TLS Callback in Delphi
 
Hallo zusammen... :wink:

Ich habe eine vlt. etwas ungewöhnliche Frage, nämlich wie kann ich eine TLS Callback Funktion in Delphi implementieren?

Längere Suche im Netz hat lediglich Lösungen in ASM und C++ ans Licht gebracht, wobei in C++ die Funktion offensichtlich programmiertechnisch, vor die Main() Funktion gepackt wird.
Aber wie sieht das in Delphi aus?

Wozu brauche ich sowas?
Nun mich stört, dass bei Mediaprogrammen immer die bass.dll separat dazu gelegt werden muss, daher habe ich die bass.dll als Resource mit in die EXE gepackt, um sie zur Laufzeit aus der Resource temporär zu erzeugen...

Code:
begin
  Application.Initialize;

  sAppPath := IncludeTrailingPathDelimiter(ExtractFileDir(Application.ExeName));
  if not FileExists(sAppPath + 'Bass.dll') then
  begin
     fName := sAppPath + 'Bass.dll';
     rStream := TResourceStream.Create(hInstance, 'Bass', RT_RCDATA);
    try
      fStream := TFileStream.Create(fName, fmCreate);
      try
        fStream.CopyFrom(rStream, 0);
      finally
        fStream.Free;
      end;
    finally
      rStream.Free;
    end;
  end;

  Application.CreateForm(TForm1, Form1);
  Application.Run;
end.
Das klappt auch alles wunderbar, leider wird offensichtlich so früh versucht die DLL zu laden, dass jeder programmierte Aufruf einer Funktion dazu führt, dass wg. fehlender bass.dll das Programm erst gar nicht gestartet wird.
Selbst ein BASS_Free() in OnClose, welches ja erst zu Programmende benötigt, sprich aufgerufen wird, führt zu o. a. Fehler!

Daher meine Überlegung, die temporäre Erzeugung der bass.dll per TLS Callback, also vor Erreichen des Program Entry Point, zu gestalten, um dieser Fehlermeldung zu entgehen und ein (hoffentlich) funktionierendes Programm zu erhalten.

Jeder hilfreiche Tipp ist herzlich willkommen... :-D

DeddyH 25. Okt 2010 10:17

AW: Wie TLS Callback in Delphi
 
Wird die DLL statisch gelinkt? Dann wundert mich das Verhalten nicht, da sie zum Programmstart ja noch gar nicht vorhanden ist. Oder hab ich Dich falsch verstanden?

himitsu 25. Okt 2010 10:24

AW: Wie TLS Callback in Delphi
 
Wenn die DLL-Funktionen statisch eingebunden sind (external), dann werden alle Funktionsaufrufe und die zugehörigen DLLs vom ExeLoader der Anwendung (bzw. von Windows) zuerst geladen und erst danach wird der Code der Anwendung ausgeführt.

Ist ja auch richtig so, denn so werden erst alle Abhängigkeiten initialisiert, welche vom auszuführenden Code benötigt werden.

Wenn du die DLL erst zur Laufzeit initilasisieren willst, dann mußt du die DLL-Funktionen dynamisch laden (MSDN-Library durchsuchenGetProcAddress), ebenso, wie du es mit der DLL machst.

Die Andere Alternative wäre ein eigener Loader, welcher noch vor dem eigentlichen Programmcode die DLL läd, aber das dynamische Laden sollte einfacher/besser sein.

itsme 25. Okt 2010 10:28

AW: Wie TLS Callback in Delphi
 
Es geht weder statisch, noch dynamisch...

Ein wenig herumexperimentieren mit Breakpoints hat gezeigt, das die Fehlermeldung nicht nur vor LoadLibrary auftaucht, sondern selbst ein BP auf Application.Initialize wird est gar nicht erreicht!

Und letzterer liegt ja in jedem Fall, noch vor Erzeugung der temporären bass.dll :?

himitsu 25. Okt 2010 10:38

AW: Wie TLS Callback in Delphi
 
Es geht hier nicht um ein statisches oder dynamisches Laden der DLL, diese lädst du ja definitiv dynamisch (LoadLibrary).

Wie verwendest du denn diese DLL?
Ohne weitere Kenntnis dessen kann man hier nicht weiter helfen.

Es dürfen auf keinen Fall irgendwelche statischen Referenzen auf irgendwelche Resourcen aus dieser DLL im Programm vorhanden sein, welche von einem Loader vor dem Programmcode versucht werden zu laden, also bevor du die DLL nicht geladen hast.
Zitat:

Zitat von z.B. sowas darfnirgendwo vorkommen
procedure test; external ...


itsme 25. Okt 2010 11:11

AW: Wie TLS Callback in Delphi
 
Ok ich poste mal ein wenig mehr Code, da ich das ganze ohnehin nur in einem Testprojekt ausprobiert habe.
Wobei ich einfach mal das dynamische Laden weggelassen habe, da kompilieren auch so fehlerfrei funktioniert, was mit uses BASS zu tun hat.
Ich bin mir derzeit echt nicht im Klaren darüber, ob ich die bass.pas mit den Deklarationen benutzen darf (muss), oder ob ich bei LoadLibrary und GetProcAddress, diese besser weglasse...
Mag sein, dass hier ein Fehler meinerseits vorliegt, ich werde das aber noch ausprobieren.

Code:
uses
Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, Bass;

procedure TForm1.OnActivate(Sender: TObject);
begin
//  BASS_Init(-1, 44100, 0, Handle, nil);
end;

procedure TForm1.OnClose(Sender: TObject; var Action: TCloseAction);
var
  fName: string;

begin
//  BASS_Free();
  fName := 'Bass.dll';
  if FileExists(fName) then
    if DeleteFile(fName) then
      Application.Terminate;
end;
Sobald ich eine der auskommentierten Funktionen in Betrieb nehme, kommt die Fehlermeldung mit der fehlenden bass.dll, ja auch, wenn ich über die entsprechenden API-Funktionen dynamisch linke. :)

Nichtsdestotrotz würde ich gerne auf meine Eingangsfrage zurückkommen.
Selbst wenn es durch eure Hilfe mit der bass.dll klappen sollte, würde ich trotzdem gerne wissen, wie man TLS Callback in Delphi implementiert.

omata 25. Okt 2010 11:25

AW: Wie TLS Callback in Delphi
 
Ich bin mir nicht ganz sicher, ob ich verstehe was du da eigentlich wissen willst. Aber ich bin der Meinung, genau das mache ich hier

itsme 25. Okt 2010 12:01

AW: Wie TLS Callback in Delphi
 
@omata

Ich habe nun natürlich nicht in der Kürze der Zeit die gesamten Attachments herunterladen können, sondern lediglich im Schnelldurchgang den gesamten Thread gelesen...

Leider habe ich weder etwas über temporäres Erstellen der bass.dll aus einer Resource in der EXE finden können, noch wie man dieselbe anschließend problemlos nutzen kann.

Und das Problem TLS Callback ist dort ganz bestimmt nicht behandelt worden.
Irgendwie habe ich das Gefühl, dass dieser Begriff doch für Verunsicherung und evtl. falsche Vorstellungen sorgt.
TLS steht für Thread Local Storage und die hier erwähnte Callback-Funktion ist ein Teil des Programmcodes, welcher bereits vor dem endgültigen Laden des Programms ausgeführt wird.
Letztendlich quasi unsichtbar, was sich viele Protektoren o. ä. Software zu Nutze machen.

Ich will hier gerne eingestehen, ein Hobbyprogrammierer älteren Semesters zu sein, der öfter eigene Unzulänglichkeiten kompensieren muss, daher den Delphi-Debugger nicht besonders prickelnd findet, ferner sein Hobby über die Assemblerprogrammierung begonnen hat und daher gerne mit externen Debuggern a là OllyDbg arbeitet.
Dort kann man z.B. gut einstellen "Break on TLS Callback" und wird feststellen, dass dieser Breakpoint schon lange vor Erreichen des OEP, also des Anfangs des Programms erreicht wird, was letztendlich meinen Gedankengang dahin geführt hat, hier bereits eine DLL zu erzeugen, welche ich dann nach Programmstart tatsächlich und uneingeschränkt nutzen kann... :)

himitsu 25. Okt 2010 12:07

AW: Wie TLS Callback in Delphi
 
Noch einen/zwei Tipp(s):

- die DLL nicht in das Programmverzeichnis entpacken, sondern lieber ins temporäre Verzeichnis von Windows (MSDN-Library durchsuchenGetTempPath).

- statt dem FileStream direkt über die WinAPI gehn, also WriteFile und MSDN-Library durchsuchenCreateFile.
bei Letzerem gibst du dann noch FILE_ATTRIBUTE_TEMPORARY mit an.

Grund:
Nicht immer hat man Schreibrechte im Programmordner und nicht jeder User mag es, wenn man da rumschreibt, bzw. man nutzt einen schreibgeschützten Datenträger (CD/DVD).
FILE_ATTRIBUTE_TEMPORARY sagt Windows, daß es eine temporäre Datei ist, welche dann automatisch gelöscht wird, sobald alle Handles auf diese Datei geschlossen sind.
Auch wenn dein Programm abstürzt und somit das OnClose nicht mehr ausgeführt würde

itsme 25. Okt 2010 12:31

AW: Wie TLS Callback in Delphi
 
@himitsu

Ok, ich will und werde deine Tipps gerne berücksichtigen, wobei die User eigentlich aus mir und meiner Familie bestehen, wo es nicht so die große Rolle spielen würde... :)

Ich habe mich ja bereits in in meinem vorherigen post als Hobbyprogrammierer geoutet, hoffe daher hier nicht unnötigerweise Resourcen von den Profis verschwendet zu haben.
Falls doch, bitte ich um Entschuldigung und Nachsicht. :oops:

Aber mit mittlerweile (leider) fast 60, möchte man den Dingen eben restlos auf den Grund gehen, auch wenn es zugegebenermaßen vollkommen uninteressant ist, ob ich auf den Rechner vom Junior nur Test.exe, oder aber Test.exe UND Bass.dll kopiere... :wink:

omata 25. Okt 2010 13:37

AW: Wie TLS Callback in Delphi
 
Sorry, das ich was gesagt habe. Viel Glück bei der Umsetzung.

itsme 25. Okt 2010 13:51

AW: Wie TLS Callback in Delphi
 
@omata

Nein, kein Grund sich zu entschuldigen, ich bin schließlich mit meinem offensichtlich obskuren Problem an die Community herangetreten... :oops:

Wenn man meine Zugehörigkeit und die Anzahl der posts in Relation setzt, wird man feststellen, dass ich so wenig wie möglich stören wollte, versucht hatte meine Probs möglichst selbst zu lösen, trotzdem kommt man irgendwann an den Punkt, wo es selbst einfach nicht weiter geht. :twisted:

So what? :stupid: vlt. kann man ja für 29.99$ irgendwo components erwerben, welche alle meine persönlichen Mühen nicht nur ersetzen, sondern durch ihre Einfachheit, auch ad adsurdum führen!

Jedenfalls möchte ich mich hier nochmals ausdrücklich für jede gutgemeinte Antwort bedanken und hoffe in Zukunft auch mal mit meinen bescheidenen Kenntnissen irgendwo ein kleines Lichtlein zünden zu können... :lol:

nachti1505 25. Okt 2010 16:53

AW: Wie TLS Callback in Delphi
 
Zitat:

Zitat von itsme (Beitrag 1057668)
Ich bin mir derzeit echt nicht im Klaren darüber, ob ich die bass.pas mit den Deklarationen benutzen darf (muss), oder ob ich bei LoadLibrary und GetProcAddress, diese besser weglasse...

Ich weiß nicht sooooo genau, was die Bass.pas macht, aber wenn es eine normale Header-Übersetzung ist, kann es durchaus sein dass diese die DLL statisch einbindet... von daher würde ich sie nicht einbinden, wenn du eh LoadLibrary etc. nutzt.

DeddyH 25. Okt 2010 18:35

AW: Wie TLS Callback in Delphi
 
Letzten Endes ist genau das des Pudels Kern, behaupte ich einfach mal rotzfrech :mrgreen:

omata 25. Okt 2010 19:31

AW: Wie TLS Callback in Delphi
 
Liste der Anhänge anzeigen (Anzahl: 1)
Ich habe es ja schon geschrieben und auch einen Link auf das gegeben, was ich hier nochmal zusammengestellt habe. Das ist aber bestimmt sowieso genau das falsche. Also sorry für diesen sinnlosen Beitrag.

itsme 27. Okt 2010 10:09

AW: Wie TLS Callback in Delphi
 
Zunächst vielen Dank für alle guten Ratschläge, welche mir die richtige Richtung gezeigt haben.
Und nein, es gab hier weder falsche, noch sinnlose posts.
Der Miniplayer von omata war schon in etwa, was ich gesucht hatte, allerdings habe ich ihn leider zu spät gelesen, aber fürs Feintuning war er noch allemal nützlich.
Nachfolgend der Code, den ich nun nach meinen Vorstellungen entwickelt hatte, wobei ich auf jegliches Error-Handling verzichtet habe, um ihn für mein post möglichst kurz zu halten...
Ich denke die Profis hier werden den Code in der Luft zerreißen, aber damit muss ein Hobby-Täter wie ich, wohl oder übel leben. :lol:

Code:
begin
  Application.Initialize;

  GetTempPath(Length(aTempPath), aTempPath);
  fName := aTempPath;
  StrCat(fName, 'Bass.dll');

  if not FileExists(fName) then
  begin
    nHandle := FindResource(hInstance, 'Bass', RT_RCDATA);
    nPointer := LockResource(LoadResource(hInstance, nHandle));
    nSize := SizeOfResource(hInstance, nHandle);

    hFile := CreateFile(fName, GENERIC_READ or GENERIC_WRITE, FILE_SHARE_READ or FILE_SHARE_WRITE, nil,
             CREATE_ALWAYS, FILE_ATTRIBUTE_TEMPORARY, FILE_FLAG_DELETE_ON_CLOSE);
    WriteFile(hFile, nPointer^, nSize, nCount, nil);
    CloseHandle(hFile);
  end;

type
  QWORD = Int64;
  HSAMPLE = DWORD;
  HCHANNEL = DWORD;
  myFree = function(): bool; stdcall;
  myGetChannel = function(handle: HSAMPLE; onlynew: BOOL): HCHANNEL; stdcall;
  myChannelPlay = function(handle: DWORD; restart: BOOL): BOOL; stdcall;
  myInit = function(dev: LongInt; freq, flags: DWORD; win: HWND; clsid: PGUID): BOOL; stdcall;
  mySampleLoad = function(mem: BOOL; f: Pointer; offset: QWORD; length, max, flags: DWORD): HSAMPLE; stdcall;

var
  Form1: TForm1;
  aTempPath: array[0..255] of char;
  BASS_Free: myFree;
  BASS_Init: myInit;
  BASS_SampleLoad: mySampleLoad;
  BASS_SampleGetChannel: myGetChannel;
  BASS_ChannelPlay: myChannelPlay;
  hLib: THandle;
  hFile: THandle;
  hChan: HCHANNEL;
  hSamp: HSAMPLE;
  fName: PChar;
  nHandle: cardinal;
  nPointer: Pointer;
  nSize: cardinal;
  nCount: cardinal;
  const BASS_UNICODE = $80000000;
  const BASS_SAMPLE_LOOP = 4;

procedure TForm1.OnActivate(Sender: TObject);
begin
  GetTempPath(Length(aTempPath), aTempPath);
  fName := aTempPath;
  StrCat(fName, 'Bass.dll');
  nHandle := FindResource(hInstance, 'Intro', RT_RCDATA);
  nPointer := LockResource(LoadResource(hInstance, nHandle));
  nSize := SizeOfResource(hInstance, nHandle);
  hLib := LoadLibrary(aTempPath);
  @BASS_Free := GetProcAddress(hLib, 'BASS_Free');
  @BASS_Init := GetProcAddress(hLib, 'BASS_Init');
  @BASS_SampleLoad := GetProcAddress(hLib, 'BASS_SampleLoad');
  @BASS_SampleGetChannel := GetProcAddress(hLib, 'BASS_SampleGetChannel');
  @BASS_ChannelPlay := GetProcAddress(hLib, 'BASS_ChannelPlay');
  BASS_Init(-1, 44100, 0, Handle, nil);
  hSamp := BASS_SampleLoad(True, nPointer, 0, nSize, 10,
           BASS_SAMPLE_LOOP {$IFDEF UNICODE} or BASS_UNICODE {$ENDIF});
  hChan := BASS_SampleGetChannel(hSamp, True);
  BASS_ChannelPlay(hChan, True);
end;

procedure TForm1.OnClose(Sender: TObject; var Action: TCloseAction);
begin
  BASS_Free();
  FreeLibrary(hLib);
  if FileExists(fName) then
    DeleteFile(fName);
  Application.Terminate;
end;
LG itsme

Assarbad 27. Okt 2010 10:19

AW: Wie TLS Callback in Delphi
 
Zitat:

Zitat von itsme (Beitrag 1057644)
Längere Suche im Netz hat lediglich Lösungen in ASM und C++ ans Licht gebracht, wobei in C++ die Funktion offensichtlich programmiertechnisch, vor die Main() Funktion gepackt wird.

TLS-Callbacks sind Teil der PE-Datei. Sie werden in der Tat vom Loader (aber nicht wirklich "vor dem Laden des Programms") vor dem Einsprungspunkt in den Code aufgerufen. Das wird und wurde von Malware verwendet um diverse Analysemöglichkeiten zu umgehen ...

Da man das in MSVC über den Linker machen muß, nähme ich mal an, daß es in Delphi ebenfalls über eine Linkeroption oder eine Präprozessordirektive funktioniert.


Alle Zeitangaben in WEZ +1. Es ist jetzt 13:52 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