AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Zurück Delphi-PRAXiS Sprachen und Entwicklungsumgebungen FreePascal TMREW-Synchronisierer BeginRead() reentrant bei BeginWrite() eines anderen Threads?
Thema durchsuchen
Ansicht
Themen-Optionen

TMREW-Synchronisierer BeginRead() reentrant bei BeginWrite() eines anderen Threads?

Ein Thema von MStoll · begonnen am 30. Mär 2014 · letzter Beitrag vom 31. Mär 2014
Antwort Antwort
MStoll

Registriert seit: 15. Nov 2005
131 Beiträge
 
Turbo Delphi für Win32
 
#1

TMREW-Synchronisierer BeginRead() reentrant bei BeginWrite() eines anderen Threads?

  Alt 30. Mär 2014, 17:21
Hallo Leute,

ich habe folgende Situation mit einem TMultiReadExclusiveWrite-Synchronisierer (MREW):
Delphi-Quellcode:
procedure TMyClass.A;
begin
  MREW.BeginRead();
  try
   //...
  finally
    MREW.EndRead();
  end;
end;

procedure TMyClass.B;
begin
  MREW.BeginRead();
  try
   // ...
   // rufe irgendwann evtl. mal über ein paar Ecken TMyClass.A (oder sich selbst rekursiv) auf
   // ...
  finally
    MREW.EndRead();
  end;
end;

procedure TMyClass.C;
begin
  MREW.BeginWrite();
  try
   //...
  finally
    MREW.EndWrite();
  end;
end;
Ich habe jetzt 2 Threads, nennen wir sie T1 und T2. T1 ruft öfter mal TMyClass.B auf und T2 öfter mal TMyClass.C. Ich nehme an, dass folgendes passiert:
- T1 ruft B auf, ist aber noch nicht über die oben erwähnten paar Ecken bei A angekommen
- T2 ruft C auf.
- T1 kommt bei A (oder auf einer tieferen Rekursionsebene wieder bei B) an.

Nun befindet sich ja T1 in dem geschützten Read-Block von procedure B, was beduetet das T2 bei BeginWrite() in Methode C warten muss. Nun kommt T1 bei A (oder nochmal bei B) an und wartet bei BeginRead() in A bzw. B, weil vorher eine BeginWrite()-Anfrage eingereiht ist, die aber nicht weiter kommen kann, weil ja T1 den MREW noch lesend in Beschlag hat. Quasi also folgende Reihenfolge:

(T1) MREW.BeginRead();
(T2) MREW.BeginWrite();
(T1) MREW.BeginRead(); --> Deadlock

Ist das so gewollt? D.h. dass zwei Threads, die ansonsten nix miteinander zu tun haben müssen, sich an einem MREW aufhängen können? Das verwundert mich deswegen, weil die Standard-Erklärungen für einen Deadlock normalerweise eine inkonsistente Akquirierung von mindestens 2 Locks oder einen nicht-reentranten Synchronisierer erfordert.

Ich benutze den MREW statt einer CriticalSection, weil ich tendenziell wesentlich mehr lesende als schreibende Aufrufe habe.

Ich bitte einfach mal um ein paar Kommentare dazu: z.B. was ihr in solchen Situationen macht, ob man nicht doch besser eine CriticalSection benutzen sollte oder wie man einen MREW tatsächlich reentrant bekommt.

Viele Grüße
"Man soll nie mehr essen als mit Gewalt reingeht!" (n.n.)
  Mit Zitat antworten Zitat
Benutzerbild von Uwe Raabe
Uwe Raabe

Registriert seit: 20. Jan 2006
Ort: Lübbecke
11.453 Beiträge
 
Delphi 12 Athens
 
#2

AW: TMREW-Synchronisierer BeginRead() reentrant bei BeginWrite() eines anderen Thread

  Alt 30. Mär 2014, 23:14
Eigentlich sollte ein Thread mit einem offenen BeginRead ohne Probleme ein zweites Mal durch das BeginRead laufen, auch wenn ein anderer Thread im BeginWrite wartet. Zumindest unter XE3 (gerade offen) lese ich das aus den Sourcen. Zu Turbo Delphi, wie in deinem Profil angegeben, kann ich leider nichts sagen.

Es ist auch nicht schwer - allenfalls etwas länglich - einen entsprechenden, einfachen Testfall herzustellen. Ich habe dazu nur im Moment keine Zeit.
Uwe Raabe
Certified Delphi Master Developer
Embarcadero MVP
Blog: The Art of Delphi Programming
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu
Online

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

AW: TMREW-Synchronisierer BeginRead() reentrant bei BeginWrite() eines anderen Thread

  Alt 30. Mär 2014, 23:53
Wenn ein BeginRead offen ist, dann kann der selbe Thread nochmal ein BeginRead aufrufen, genauso wie andere Threads das können.

Auch könnte ein Thread ein BeginWrite aufrufen, selbst wenn vorher ein BeginRead in diesem Thread startete.
Wobei es da auf die Implementation ankommt.
Ein BeginWrite wartet mindestens, bis alle BeginReads aus anderen Threads beendet wurden.

Aber wenn man z.B. in zwei Thread ein BeginRead ausführt und daraufhin direkt ein BeginWrite, dann könnte/wird man in einen Deadlock laufen, da beide Threads auf das Ende der
Sperren des anderen Threads warten.
Neuste Erkenntnis:
Seit Pos einen dritten Parameter hat,
wird PoSex im Delphi viel seltener praktiziert.
  Mit Zitat antworten Zitat
MStoll

Registriert seit: 15. Nov 2005
131 Beiträge
 
Turbo Delphi für Win32
 
#4

AW: TMREW-Synchronisierer BeginRead() reentrant bei BeginWrite() eines anderen Thread

  Alt 31. Mär 2014, 00:06
Sorry, hatte nicht explizit erwähnt, dass ich hier mit FreePascal arbeite.

Laut Kommentaren aus dem Quelltext priorisiert der Synchronisierer hier BeginWrite(), d.h. alle Reader sollen noch ihr Ding zu Ende bringen und alle weiteren BeginRead()-Aufrufe werden hintenangestellt, auch die eines aktuellen Readers.

Das ist jedenfalls heimtückisch, wenn ein solcher Synchronisierer nicht immer reentrant ist.

Weitere Tests haben das bestätigt. Ich weiß nicht, wie das bei Delphi genau aussieht von der Implementierung des Synchronisierers.

(Edit)
@himitsu: das mit dem BeginWrite() direkt nach BeginRead() geht zumindest hier bei FreePascal auch nicht, das habe ich auch einmal explizit getestet. Das führt unweigerlich zu einem Deadlock. Kann natürlich sein, dass das bei Delphi anders ist.
"Man soll nie mehr essen als mit Gewalt reingeht!" (n.n.)
  Mit Zitat antworten Zitat
Benutzerbild von Uwe Raabe
Uwe Raabe

Registriert seit: 20. Jan 2006
Ort: Lübbecke
11.453 Beiträge
 
Delphi 12 Athens
 
#5

AW: TMREW-Synchronisierer BeginRead() reentrant bei BeginWrite() eines anderen Thread

  Alt 31. Mär 2014, 00:23
Aber wenn man z.B. in zwei Thread ein BeginRead ausführt und daraufhin direkt ein BeginWrite, dann könnte/wird man in einen Deadlock laufen, da beide Threads auf das Ende der
Sperren des anderen Threads warten.
Das sehe ich anders! Das BeginWrite entfernt zunächst den eigenen ReadLock, bevor es auf die Freigabe der anderen ReadLocks wartet. Das erlaubt dem Thread mit dem ersten BeginRead mit seinem BeginWrite durchzukommen. Dieses CodeSite-Log zeigt, daß es funktioniert:
Miniaturansicht angehängter Grafiken
mrew.png  
Uwe Raabe
Certified Delphi Master Developer
Embarcadero MVP
Blog: The Art of Delphi Programming
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu
Online

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

AW: TMREW-Synchronisierer BeginRead() reentrant bei BeginWrite() eines anderen Thread

  Alt 31. Mär 2014, 01:35
Vielliecht wurde da mal was geändert?

Es ist zwar schon viele Jahre her, daber (vermutlich) ist damals mal ein Projekt, an soneiner Stelle, öfters mal hängen geblieben.
Jedenfalls nach einem Umbau und vorherrigem Freigeben des BeginRead, blieb es dann nicht mehr hängen.

Stellt der nach dem EndWrite das BeginRead eigentlich wieder her?
Neuste Erkenntnis:
Seit Pos einen dritten Parameter hat,
wird PoSex im Delphi viel seltener praktiziert.

Geändert von himitsu (31. Mär 2014 um 12:47 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Uwe Raabe
Uwe Raabe

Registriert seit: 20. Jan 2006
Ort: Lübbecke
11.453 Beiträge
 
Delphi 12 Athens
 
#7

AW: TMREW-Synchronisierer BeginRead() reentrant bei BeginWrite() eines anderen Thread

  Alt 31. Mär 2014, 08:40
Vielliecht wurde da mal was geändert?
Der Test war jetzt mit XE3.

Stellt der nach dem EndWrite das BeginWrite eigentlich wieder her?
Das BeginRead, ja!
Uwe Raabe
Certified Delphi Master Developer
Embarcadero MVP
Blog: The Art of Delphi Programming
  Mit Zitat antworten Zitat
MStoll

Registriert seit: 15. Nov 2005
131 Beiträge
 
Turbo Delphi für Win32
 
#8

AW: TMREW-Synchronisierer BeginRead() reentrant bei BeginWrite() eines anderen Thread

  Alt 31. Mär 2014, 13:30
Anscheinend gibt's da dann deutliche Unterschiede zwischen Free Pascal und den neueren Delphi-Versionen. Ich muss dringend mal schauen, wie Turbo Delphi das macht. Nicht, dass es da generelle, hinterlistige Inkompatibilitäten gibt.

Ich habe mir in der Zwischenzeit mal so weitergeholfen, dass ich eine Subklasse zu TMREW gebaut habe, die (durch eine eigene CriticalSection geschützt) über die Reader Buch führt und für jeden zählt, wie oft er reinkommt. Nur beim ersten Reinkommen und dem dazugehörigen Rauskommen eines Threads werden die entsprechenden BeginRead()- bzw. EndRead()-Aufrufe aus der TMREW-Klasse aufgerufen, anderenfalls die Zähler entsprechend erhöht bzw. verkleinert.

Da ich nur ein paar wenige Threads habe, die darauf zugreifen, geht das bis jetzt ganz gut.
"Man soll nie mehr essen als mit Gewalt reingeht!" (n.n.)
  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 16:07 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