ViewVC Help
View File | Revision Log | Show Annotations | Download File | View Changeset | Root Listing
root/public/ibx/trunk/fbintf/client/2.5/FB25Events.pas
Revision: 45
Committed: Tue Dec 6 10:33:46 2016 UTC (7 years, 11 months ago) by tony
Content type: text/x-pascal
File size: 9813 byte(s)
Log Message:
Committing updates for Release R2-0-0

File Contents

# User Rev Content
1 tony 45 (*
2     * Firebird Interface (fbintf). The fbintf components provide a set of
3     * Pascal language bindings for the Firebird API. Although predominantly
4     * a new development they include source code taken from IBX and may be
5     * considered a derived product. This software thus also includes the copyright
6     * notice and license conditions from IBX.
7     *
8     * Except for those parts dervied from IBX, contents of this file are subject
9     * to the Initial Developer's Public License Version 1.0 (the "License"); you
10     * may not use this file except in compliance with the License. You may obtain a
11     * copy of the License here:
12     *
13     * http://www.firebirdsql.org/index.php?op=doc&id=idpl
14     *
15     * Software distributed under the License is distributed on an "AS
16     * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
17     * implied. See the License for the specific language governing rights
18     * and limitations under the License.
19     *
20     * The Initial Developer of the Original Code is Tony Whyman.
21     *
22     * The Original Code is (C) 2016 Tony Whyman, MWA Software
23     * (http://www.mwasoftware.co.uk).
24     *
25     * All Rights Reserved.
26     *
27     * Contributor(s): ______________________________________.
28     *
29     *)
30     {************************************************************************}
31     { }
32     { Borland Delphi Visual Component Library }
33     { InterBase Express core components }
34     { }
35     { Copyright (c) 1998-2000 Inprise Corporation }
36     { }
37     { InterBase Express is based in part on the product }
38     { Free IB Components, written by Gregory H. Deatz for }
39     { Hoagland, Longo, Moran, Dunst & Doukas Company. }
40     { Free IB Components is used under license. }
41     { }
42     { The contents of this file are subject to the InterBase }
43     { Public License Version 1.0 (the "License"); you may not }
44     { use this file except in compliance with the License. You }
45     { may obtain a copy of the License at http://www.Inprise.com/IPL.html }
46     { Software distributed under the License is distributed on }
47     { an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either }
48     { express or implied. See the License for the specific language }
49     { governing rights and limitations under the License. }
50     { The Original Code was created by InterBase Software Corporation }
51     { and its successors. }
52     { Portions created by Inprise Corporation are Copyright (C) Inprise }
53     { Corporation. All Rights Reserved. }
54     { Contributor(s): Jeff Overcash }
55     { }
56     { IBX For Lazarus (Firebird Express) }
57     { Contributor: Tony Whyman, MWA Software http://www.mwasoftware.co.uk }
58     { Portions created by MWA Software are copyright McCallum Whyman }
59     { Associates Ltd 2011 - 2015 }
60     { }
61     {************************************************************************}
62     unit FB25Events;
63    
64     {$IFDEF FPC}
65     {$mode objfpc}{$H+}
66     {$interfaces COM}
67     {$ENDIF}
68    
69     interface
70    
71     uses
72     {$IFDEF WINDOWS}Windows, {$ENDIF}Classes, SysUtils, IB, FB25ClientAPI, FB25Attachment,
73     IBExternals, IBHeader, syncobjs, FBEvents;
74    
75     type
76     TFB25Events = class;
77    
78     { TEventhandlerInterface }
79    
80     TEventhandlerInterface = class
81     private
82     FOwner: TFB25Events;
83     {$IFDEF WINDOWS}
84     {Make direct use of Windows API as TEventObject don't seem to work under
85     Windows!}
86     FEventHandler: THandle;
87     {$ELSE}
88     FEventWaiting: TEventObject;
89     {$ENDIF}
90     public
91     constructor Create(aOwner: TFB25Events);
92     destructor Destroy; override;
93     procedure eventCallbackFunction(length: short; updated: PChar);
94     procedure WaitForEvent;
95     procedure CancelWait;
96     end;
97    
98     { TFB25Events }
99    
100     TFB25Events = class(TFBEvents,IEvents)
101     private
102     FEventID: ISC_LONG;
103     FDBHandle: TISC_DB_HANDLE;
104     FEventHandlerThread: TObject;
105     FAsyncEventCallback: TEventhandlerInterface;
106     protected
107     procedure CancelEvents(Force: boolean = false); override;
108     function GetIEvents: IEvents; override;
109     public
110     constructor Create(DBAttachment: TFB25Attachment; Events: TStrings);
111     destructor Destroy; override;
112    
113     {IEvents}
114     procedure WaitForEvent;
115     procedure AsyncWaitForEvent(EventHandler: TEventHandler);
116     end;
117    
118     implementation
119    
120     uses FBMessages;
121    
122     type
123    
124     { TEventHandlerThread }
125    
126     TEventHandlerThread = class(TThread)
127     private
128     FOwner: TFB25Events;
129     FEventHandler: TEventhandlerInterface;
130     protected
131     procedure Execute; override;
132     public
133     constructor Create(Owner: TFB25Events; EventHandler: TEventhandlerInterface);
134     procedure Terminate;
135     end;
136    
137     { TEventHandlerThread }
138    
139     procedure TEventHandlerThread.Execute;
140     begin
141     while not Terminated do
142     begin
143     FEventHandler.WaitForEvent;
144    
145     if not Terminated then
146     FOwner.EventSignaled;
147     end;
148     end;
149    
150     constructor TEventHandlerThread.Create(Owner: TFB25Events;
151     EventHandler: TEventhandlerInterface);
152     begin
153     inherited Create(true);
154     FOwner := Owner;
155     FEventHandler := EventHandler;
156     FreeOnTerminate := true;
157     Start;
158     end;
159    
160     procedure TEventHandlerThread.Terminate;
161     begin
162     inherited Terminate;
163     FEventHandler.CancelWait;
164     end;
165    
166     {This procedure is used for the event call back - note the cdecl }
167    
168     procedure IBEventCallback( ptr: pointer; length: short; updated: PChar); cdecl;
169     begin
170     if (ptr = nil) or (length = 0) or (updated = nil) then
171     Exit;
172     { Handle events asynchronously in second thread }
173     TEventhandlerInterface(ptr).eventCallbackFunction(length,updated);
174     end;
175    
176     { TEventhandlerInterface }
177    
178     constructor TEventhandlerInterface.Create(aOwner: TFB25Events);
179     var
180     PSa : PSecurityAttributes;
181     {$IFDEF WINDOWS}
182     Sd : TSecurityDescriptor;
183     Sa : TSecurityAttributes;
184     begin
185     InitializeSecurityDescriptor(@Sd,SECURITY_DESCRIPTOR_REVISION);
186     SetSecurityDescriptorDacl(@Sd,true,nil,false);
187     Sa.nLength := SizeOf(Sa);
188     Sa.lpSecurityDescriptor := @Sd;
189     Sa.bInheritHandle := true;
190     PSa := @Sa;
191     {$ELSE}
192     GUID : TGUID;
193     begin
194     PSa:= nil;
195     {$ENDIF}
196     inherited Create;
197     {$IFDEF WINDOWS}
198     FEventHandler := CreateEvent(PSa,false,true,nil);
199     {$ELSE}
200     CreateGuid(GUID);
201     FEventWaiting := TEventObject.Create(PSa,false,false,GUIDToString(GUID));
202     {$ENDIF}
203     FOWner := aOwner;
204     end;
205    
206     destructor TEventhandlerInterface.Destroy;
207     begin
208     {$IFDEF WINDOWS}
209     CloseHandle(FEventHandler);
210     {$ELSE}
211     if assigned(FEventWaiting) then FEventWaiting.Free;
212     {$ENDIF}
213     inherited Destroy;
214     end;
215    
216     procedure TEventhandlerInterface.eventCallbackFunction(length: short;
217     updated: PChar);
218     begin
219     FOwner.FCriticalSection.Enter;
220     try
221     if FOwner.FResultBuffer <> nil then
222     Move(updated[0], FOwner.FResultBuffer[0], length);
223     {$IFDEF WINDOWS}
224     SetEvent(FEventHandler);
225     {$ELSE}
226     FEventWaiting.SetEvent;
227     {$ENDIF}
228     finally
229     FOwner.FCriticalSection.Leave
230     end;
231     end;
232    
233     procedure TEventhandlerInterface.WaitForEvent;
234     begin
235     {$IFDEF WINDOWS}
236     WaitForSingleObject(FEventHandler,INFINITE);
237     {$ELSE}
238     FEventWaiting.WaitFor(INFINITE);
239     {$ENDIF}
240     end;
241    
242     procedure TEventhandlerInterface.CancelWait;
243     begin
244     {$IFDEF WINDOWS}
245     SetEvent(FEventHandler);
246     {$ELSE}
247     FEventWaiting.SetEvent;
248     {$ENDIF}
249     end;
250    
251    
252     { TFB25Events }
253    
254     procedure TFB25Events.CancelEvents(Force: boolean);
255     begin
256     FCriticalSection.Enter;
257     try
258     if not FInWaitState then Exit;
259     with Firebird25ClientAPI do
260     if (Call(isc_Cancel_events( StatusVector, @FDBHandle, @FEventID),false) > 0) and not Force then
261     IBDatabaseError;
262    
263     FInWaitState := false;
264     inherited CancelEvents(Force);
265     finally
266     FCriticalSection.Leave
267     end;
268     end;
269    
270     function TFB25Events.GetIEvents: IEvents;
271     begin
272     Result := self;
273     end;
274    
275     constructor TFB25Events.Create(DBAttachment: TFB25Attachment; Events: TStrings);
276     begin
277     inherited Create(DBAttachment,DBAttachment,Events);
278     FDBHandle := DBAttachment.Handle;
279     FAsyncEventCallback := TEventhandlerInterface.Create(self);
280     FEventHandlerThread := TEventHandlerThread.Create(self,FAsyncEventCallback);
281     end;
282    
283     destructor TFB25Events.Destroy;
284     begin
285     CancelEvents(true);
286     if assigned(FEventHandlerThread) then
287     TEventHandlerThread(FEventHandlerThread).Terminate;
288     if assigned(FAsyncEventCallback) then
289     TEventhandlerInterface(FAsyncEventCallback).Free;
290     inherited Destroy;
291     end;
292    
293     procedure TFB25Events.AsyncWaitForEvent(EventHandler: TEventHandler);
294     var callback: pointer;
295     begin
296     FCriticalSection.Enter;
297     try
298     if FInWaitState then
299     IBError(ibxeInEventWait,[nil]);
300    
301     CreateEventBlock;
302     FEventHandler := EventHandler;
303     callback := @IBEventCallback;
304     with Firebird25ClientAPI do
305     Call(isc_que_events( StatusVector, @FDBHandle, @FEventID, FEventBufferLen,
306     FEventBuffer, TISC_CALLBACK(callback), PVoid(FAsyncEventCallback)));
307     FInWaitState := true;
308     finally
309     FCriticalSection.Leave
310     end;
311     end;
312    
313     procedure TFB25Events.WaitForEvent;
314     begin
315     if FInWaitState then
316     IBError(ibxeInEventWait,[nil]);
317    
318     FInWaitState := true;
319     try
320     CreateEventBlock;
321     with Firebird25ClientAPI do
322     Call(isc_wait_for_event(StatusVector,@FDBHandle, FEventBufferlen,FEventBuffer,FResultBuffer));
323     finally
324     FInWaitState := false;
325     end;
326     end;
327    
328     end.
329