2009-08-12 07:03:58 +00:00
|
|
|
/*
|
|
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
|
|
* PROJECT: ReactOS kernel
|
|
|
|
* PURPOSE: Window event handlers
|
|
|
|
* FILE: subsystem/win32/win32k/ntuser/event.c
|
|
|
|
* PROGRAMER: James Tabor (james.tabor@rectos.org)
|
|
|
|
*/
|
2008-04-17 01:48:34 +00:00
|
|
|
|
2010-04-26 13:58:46 +00:00
|
|
|
#include <win32k.h>
|
2008-04-17 01:48:34 +00:00
|
|
|
|
|
|
|
#define NDEBUG
|
|
|
|
#include <debug.h>
|
|
|
|
|
2008-08-02 00:38:01 +00:00
|
|
|
typedef struct _EVENTPACK
|
|
|
|
{
|
|
|
|
PEVENTHOOK pEH;
|
|
|
|
LONG idObject;
|
|
|
|
LONG idChild;
|
|
|
|
} EVENTPACK, *PEVENTPACK;
|
|
|
|
|
2008-07-31 23:48:35 +00:00
|
|
|
static PEVENTTABLE GlobalEvents = NULL;
|
2008-04-17 01:48:34 +00:00
|
|
|
|
|
|
|
/* PRIVATE FUNCTIONS *********************************************************/
|
|
|
|
|
|
|
|
static
|
|
|
|
DWORD
|
|
|
|
FASTCALL
|
|
|
|
GetMaskFromEvent(DWORD Event)
|
|
|
|
{
|
|
|
|
DWORD Ret = 0;
|
|
|
|
|
|
|
|
if ( Event > EVENT_OBJECT_STATECHANGE )
|
|
|
|
{
|
|
|
|
if ( Event == EVENT_OBJECT_LOCATIONCHANGE ) return SRV_EVENT_LOCATIONCHANGE;
|
|
|
|
if ( Event == EVENT_OBJECT_NAMECHANGE ) return SRV_EVENT_NAMECHANGE;
|
|
|
|
if ( Event == EVENT_OBJECT_VALUECHANGE ) return SRV_EVENT_VALUECHANGE;
|
|
|
|
return SRV_EVENT_CREATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( Event == EVENT_OBJECT_STATECHANGE ) return SRV_EVENT_STATECHANGE;
|
|
|
|
|
|
|
|
Ret = SRV_EVENT_RUNNING;
|
|
|
|
|
|
|
|
if ( Event < EVENT_SYSTEM_MENUSTART ) return SRV_EVENT_CREATE;
|
|
|
|
|
|
|
|
if ( Event <= EVENT_SYSTEM_MENUPOPUPEND )
|
|
|
|
{
|
|
|
|
Ret = SRV_EVENT_MENU;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ( Event <= EVENT_CONSOLE_CARET-1 ) return SRV_EVENT_CREATE;
|
|
|
|
if ( Event <= EVENT_CONSOLE_END_APPLICATION ) return SRV_EVENT_END_APPLICATION;
|
|
|
|
if ( Event != EVENT_OBJECT_FOCUS ) return SRV_EVENT_CREATE;
|
|
|
|
}
|
|
|
|
return Ret;
|
|
|
|
}
|
|
|
|
|
2008-07-31 23:48:35 +00:00
|
|
|
static
|
|
|
|
VOID
|
|
|
|
FASTCALL
|
2008-08-02 00:38:01 +00:00
|
|
|
IntSetSrvEventMask( UINT EventMin, UINT EventMax)
|
2008-07-31 23:48:35 +00:00
|
|
|
{
|
2008-08-02 00:38:01 +00:00
|
|
|
UINT event;
|
|
|
|
DPRINT("SetSrvEventMask 1\n");
|
|
|
|
for ( event = EventMin; event <= EventMax; event++)
|
2008-07-31 23:48:35 +00:00
|
|
|
{
|
2008-08-02 00:38:01 +00:00
|
|
|
if ((event >= EVENT_SYSTEM_SOUND && event <= EVENT_SYSTEM_MINIMIZEEND) ||
|
|
|
|
(event >= EVENT_CONSOLE_CARET && event <= EVENT_CONSOLE_END_APPLICATION) ||
|
|
|
|
(event >= EVENT_OBJECT_CREATE && event <= EVENT_OBJECT_ACCELERATORCHANGE))
|
2008-07-31 23:48:35 +00:00
|
|
|
{
|
2009-06-29 03:36:31 +00:00
|
|
|
gpsi->dwInstalledEventHooks |= GetMaskFromEvent(event);
|
2008-07-31 23:48:35 +00:00
|
|
|
}
|
2008-08-02 00:38:01 +00:00
|
|
|
if (event > EVENT_SYSTEM_MINIMIZEEND && event < EVENT_CONSOLE_CARET)
|
2008-07-31 23:48:35 +00:00
|
|
|
{
|
2008-08-02 00:38:01 +00:00
|
|
|
event = EVENT_CONSOLE_CARET-1;
|
2009-06-29 03:36:31 +00:00
|
|
|
gpsi->dwInstalledEventHooks |= GetMaskFromEvent(event);
|
2008-07-31 23:48:35 +00:00
|
|
|
}
|
2008-08-02 00:38:01 +00:00
|
|
|
if (event > EVENT_CONSOLE_END_APPLICATION && event < EVENT_OBJECT_CREATE )
|
2008-07-31 23:48:35 +00:00
|
|
|
{
|
2008-08-02 00:38:01 +00:00
|
|
|
event = EVENT_OBJECT_CREATE-1;
|
2009-06-29 03:36:31 +00:00
|
|
|
gpsi->dwInstalledEventHooks |= GetMaskFromEvent(event);
|
2008-07-31 23:48:35 +00:00
|
|
|
}
|
2008-08-02 00:38:01 +00:00
|
|
|
if (event > EVENT_OBJECT_ACCELERATORCHANGE && event < EVENT_MAX)
|
2008-07-31 23:48:35 +00:00
|
|
|
{
|
2008-08-02 00:38:01 +00:00
|
|
|
event = EVENT_MAX-1;
|
2009-06-29 03:36:31 +00:00
|
|
|
gpsi->dwInstalledEventHooks |= GetMaskFromEvent(event);
|
2008-08-02 00:38:01 +00:00
|
|
|
break;
|
2009-06-29 03:36:31 +00:00
|
|
|
}
|
2008-07-31 23:48:35 +00:00
|
|
|
}
|
2009-06-29 03:36:31 +00:00
|
|
|
if (!gpsi->dwInstalledEventHooks)
|
|
|
|
gpsi->dwInstalledEventHooks |= SRV_EVENT_RUNNING; // Set something.
|
|
|
|
DPRINT("SetSrvEventMask 2 : %x\n", gpsi->dwInstalledEventHooks);
|
2008-04-17 01:48:34 +00:00
|
|
|
}
|
|
|
|
|
2008-07-31 23:48:35 +00:00
|
|
|
static
|
|
|
|
LRESULT
|
|
|
|
FASTCALL
|
|
|
|
IntCallLowLevelEvent( PEVENTHOOK pEH,
|
|
|
|
DWORD event,
|
|
|
|
HWND hwnd,
|
|
|
|
LONG idObject,
|
|
|
|
LONG idChild)
|
|
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
|
|
ULONG_PTR uResult;
|
2008-08-02 00:38:01 +00:00
|
|
|
EVENTPACK EP;
|
|
|
|
|
|
|
|
EP.pEH = pEH;
|
|
|
|
EP.idObject = idObject;
|
|
|
|
EP.idChild = idChild;
|
2008-07-31 23:48:35 +00:00
|
|
|
|
|
|
|
/* FIXME should get timeout from
|
|
|
|
* HKEY_CURRENT_USER\Control Panel\Desktop\LowLevelHooksTimeout */
|
2009-10-06 04:22:11 +00:00
|
|
|
Status = co_MsqSendMessage( pEH->head.pti->MessageQueue,
|
|
|
|
hwnd,
|
|
|
|
event,
|
|
|
|
0,
|
|
|
|
(LPARAM)&EP,
|
|
|
|
5000,
|
|
|
|
TRUE,
|
|
|
|
MSQ_ISEVENT,
|
|
|
|
&uResult);
|
2008-07-31 23:48:35 +00:00
|
|
|
|
|
|
|
return NT_SUCCESS(Status) ? uResult : 0;
|
|
|
|
}
|
|
|
|
|
2008-08-02 00:38:01 +00:00
|
|
|
|
2008-07-31 23:48:35 +00:00
|
|
|
static
|
|
|
|
BOOL
|
|
|
|
FASTCALL
|
|
|
|
IntRemoveEvent(PEVENTHOOK pEH)
|
|
|
|
{
|
|
|
|
if (pEH)
|
|
|
|
{
|
2009-07-26 01:59:08 +00:00
|
|
|
DPRINT("IntRemoveEvent pEH 0x%x\n",pEH);
|
|
|
|
KeEnterCriticalRegion();
|
2008-07-31 23:48:35 +00:00
|
|
|
RemoveEntryList(&pEH->Chain);
|
|
|
|
GlobalEvents->Counts--;
|
2009-06-29 03:36:31 +00:00
|
|
|
if (!GlobalEvents->Counts) gpsi->dwInstalledEventHooks = 0;
|
2010-01-15 13:47:25 +00:00
|
|
|
UserDeleteObject(UserHMGetHandle(pEH), otEvent);
|
2009-07-26 01:59:08 +00:00
|
|
|
KeLeaveCriticalRegion();
|
2008-07-31 23:48:35 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2008-04-17 01:48:34 +00:00
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
|
|
|
|
LRESULT
|
|
|
|
FASTCALL
|
|
|
|
co_EVENT_CallEvents( DWORD event,
|
|
|
|
HWND hwnd,
|
2008-10-23 17:05:40 +00:00
|
|
|
UINT_PTR idObject,
|
|
|
|
LONG_PTR idChild)
|
2008-04-17 01:48:34 +00:00
|
|
|
{
|
2008-08-02 00:38:01 +00:00
|
|
|
PEVENTHOOK pEH;
|
|
|
|
LRESULT Result;
|
|
|
|
PEVENTPACK pEP = (PEVENTPACK)idChild;
|
|
|
|
|
|
|
|
pEH = pEP->pEH;
|
|
|
|
|
2010-01-15 13:47:25 +00:00
|
|
|
Result = co_IntCallEventProc( UserHMGetHandle(pEH),
|
|
|
|
event,
|
|
|
|
hwnd,
|
|
|
|
pEP->idObject,
|
|
|
|
pEP->idChild,
|
|
|
|
(DWORD_PTR)(NtCurrentTeb()->ClientId).UniqueThread,
|
|
|
|
(DWORD)EngGetTickCount(),
|
|
|
|
pEH->Proc);
|
2008-04-17 01:48:34 +00:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2008-08-02 00:38:01 +00:00
|
|
|
VOID
|
|
|
|
FASTCALL
|
|
|
|
IntNotifyWinEvent(
|
|
|
|
DWORD Event,
|
2009-07-26 01:59:08 +00:00
|
|
|
PWND pWnd,
|
2008-08-02 00:38:01 +00:00
|
|
|
LONG idObject,
|
|
|
|
LONG idChild)
|
|
|
|
{
|
|
|
|
PEVENTHOOK pEH;
|
2009-07-26 12:27:15 +00:00
|
|
|
PLIST_ENTRY pLE;
|
2008-08-02 00:38:01 +00:00
|
|
|
LRESULT Result;
|
|
|
|
|
2009-07-26 01:59:08 +00:00
|
|
|
DPRINT("IntNotifyWinEvent GlobalEvents = 0x%x pWnd 0x%x\n",GlobalEvents, pWnd);
|
|
|
|
|
|
|
|
if (!pWnd) return;
|
|
|
|
|
|
|
|
if (pWnd && pWnd->state & WNDS_DESTROYED) return;
|
|
|
|
|
2009-07-01 17:19:16 +00:00
|
|
|
if (!GlobalEvents || !GlobalEvents->Counts) return;
|
2008-08-02 00:38:01 +00:00
|
|
|
|
2009-07-26 12:27:15 +00:00
|
|
|
pLE = GlobalEvents->Events.Flink;
|
|
|
|
pEH = CONTAINING_RECORD(pLE, EVENTHOOK, Chain);
|
2008-08-02 00:38:01 +00:00
|
|
|
do
|
|
|
|
{
|
|
|
|
UserReferenceObject(pEH);
|
|
|
|
// Must be inside the event window.
|
|
|
|
if ( (pEH->eventMin <= Event) && (pEH->eventMax >= Event))
|
|
|
|
{
|
2009-10-06 04:22:11 +00:00
|
|
|
if (pEH->head.pti->pEThread != PsGetCurrentThread())
|
2008-08-02 00:38:01 +00:00
|
|
|
{ // if all process || all thread || other thread same process
|
|
|
|
if (!(pEH->idProcess) || !(pEH->idThread) ||
|
2010-01-05 20:06:33 +00:00
|
|
|
(NtCurrentTeb()->ClientId.UniqueProcess == (PVOID)(DWORD_PTR)pEH->idProcess))
|
2008-08-02 00:38:01 +00:00
|
|
|
{
|
2009-07-26 12:27:15 +00:00
|
|
|
Result = IntCallLowLevelEvent( pEH,
|
|
|
|
Event,
|
|
|
|
UserHMGetHandle(pWnd),
|
|
|
|
idObject,
|
|
|
|
idChild);
|
2008-08-02 00:38:01 +00:00
|
|
|
}
|
|
|
|
}// if ^skip own thread && ((Pid && CPid == Pid && ^skip own process) || all process)
|
|
|
|
else if ( !(pEH->Flags & WINEVENT_SKIPOWNTHREAD) &&
|
|
|
|
( ((pEH->idProcess &&
|
2010-01-05 20:06:33 +00:00
|
|
|
NtCurrentTeb()->ClientId.UniqueProcess == (PVOID)(DWORD_PTR)pEH->idProcess) &&
|
2008-08-02 00:38:01 +00:00
|
|
|
!(pEH->Flags & WINEVENT_SKIPOWNPROCESS)) ||
|
|
|
|
!pEH->idProcess ) )
|
|
|
|
{
|
2009-07-26 12:27:15 +00:00
|
|
|
Result = co_IntCallEventProc( UserHMGetHandle(pEH),
|
|
|
|
Event,
|
|
|
|
UserHMGetHandle(pWnd),
|
|
|
|
idObject,
|
|
|
|
idChild,
|
2008-10-23 17:05:40 +00:00
|
|
|
PtrToUint(NtCurrentTeb()->ClientId.UniqueThread),
|
2009-07-26 12:27:15 +00:00
|
|
|
(DWORD)EngGetTickCount(),
|
|
|
|
pEH->Proc);
|
2008-08-02 00:38:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
UserDereferenceObject(pEH);
|
2009-07-26 12:27:15 +00:00
|
|
|
pLE = pEH->Chain.Flink;
|
|
|
|
pEH = CONTAINING_RECORD(pLE, EVENTHOOK, Chain);
|
|
|
|
} while (pLE != &GlobalEvents->Events);
|
2008-08-02 00:38:01 +00:00
|
|
|
}
|
2008-04-17 01:48:34 +00:00
|
|
|
|
|
|
|
VOID
|
2008-11-29 22:48:58 +00:00
|
|
|
APIENTRY
|
2008-04-17 01:48:34 +00:00
|
|
|
NtUserNotifyWinEvent(
|
|
|
|
DWORD Event,
|
|
|
|
HWND hWnd,
|
|
|
|
LONG idObject,
|
|
|
|
LONG idChild)
|
|
|
|
{
|
2008-08-02 00:38:01 +00:00
|
|
|
PWINDOW_OBJECT Window = NULL;
|
|
|
|
USER_REFERENCE_ENTRY Ref;
|
2008-07-31 23:48:35 +00:00
|
|
|
UserEnterExclusive();
|
2008-08-02 00:38:01 +00:00
|
|
|
|
|
|
|
/* Validate input */
|
|
|
|
if (hWnd && (hWnd != INVALID_HANDLE_VALUE) && !(Window = UserGetWindowObject(hWnd)))
|
|
|
|
{
|
2009-05-23 17:20:30 +00:00
|
|
|
UserLeave();
|
2008-08-02 00:38:01 +00:00
|
|
|
return;
|
2009-06-29 03:36:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (gpsi->dwInstalledEventHooks & GetMaskFromEvent(Event))
|
2008-08-02 00:38:01 +00:00
|
|
|
{
|
|
|
|
UserRefObjectCo(Window, &Ref);
|
2009-07-26 01:59:08 +00:00
|
|
|
IntNotifyWinEvent( Event, Window->Wnd, idObject, idChild);
|
2008-08-02 00:38:01 +00:00
|
|
|
UserDerefObjectCo(Window);
|
|
|
|
}
|
2008-07-31 23:48:35 +00:00
|
|
|
UserLeave();
|
2008-04-17 01:48:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
HWINEVENTHOOK
|
2008-11-29 22:48:58 +00:00
|
|
|
APIENTRY
|
2008-04-17 01:48:34 +00:00
|
|
|
NtUserSetWinEventHook(
|
|
|
|
UINT eventMin,
|
|
|
|
UINT eventMax,
|
|
|
|
HMODULE hmodWinEventProc,
|
|
|
|
PUNICODE_STRING puString,
|
|
|
|
WINEVENTPROC lpfnWinEventProc,
|
|
|
|
DWORD idProcess,
|
|
|
|
DWORD idThread,
|
|
|
|
UINT dwflags)
|
|
|
|
{
|
2008-07-31 23:48:35 +00:00
|
|
|
PEVENTHOOK pEH;
|
|
|
|
HWINEVENTHOOK Ret = NULL;
|
|
|
|
NTSTATUS Status;
|
|
|
|
HANDLE Handle;
|
2008-08-02 00:38:01 +00:00
|
|
|
PETHREAD Thread = NULL;
|
2008-04-17 01:48:34 +00:00
|
|
|
|
2009-07-01 17:19:16 +00:00
|
|
|
DPRINT("NtUserSetWinEventHook hmod 0x%x, pfn 0x%x\n",hmodWinEventProc, lpfnWinEventProc);
|
|
|
|
|
2008-07-31 23:48:35 +00:00
|
|
|
UserEnterExclusive();
|
|
|
|
|
|
|
|
if ( !GlobalEvents )
|
|
|
|
{
|
|
|
|
GlobalEvents = ExAllocatePoolWithTag(PagedPool, sizeof(EVENTTABLE), TAG_HOOK);
|
2008-11-04 23:49:07 +00:00
|
|
|
if (GlobalEvents == NULL)
|
|
|
|
{
|
|
|
|
SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);
|
|
|
|
goto SetEventExit;
|
|
|
|
}
|
2008-07-31 23:48:35 +00:00
|
|
|
GlobalEvents->Counts = 0;
|
|
|
|
InitializeListHead(&GlobalEvents->Events);
|
|
|
|
}
|
|
|
|
|
2008-08-02 00:38:01 +00:00
|
|
|
if (eventMin > eventMax)
|
|
|
|
{
|
|
|
|
SetLastWin32Error(ERROR_INVALID_HOOK_FILTER);
|
|
|
|
goto SetEventExit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!lpfnWinEventProc)
|
|
|
|
{
|
|
|
|
SetLastWin32Error(ERROR_INVALID_FILTER_PROC);
|
|
|
|
goto SetEventExit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((dwflags & WINEVENT_INCONTEXT) && !hmodWinEventProc)
|
|
|
|
{
|
|
|
|
SetLastWin32Error(ERROR_HOOK_NEEDS_HMOD);
|
|
|
|
goto SetEventExit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (idThread)
|
|
|
|
{
|
|
|
|
Status = PsLookupThreadByThreadId((HANDLE)(DWORD_PTR)idThread, &Thread);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
SetLastWin32Error(ERROR_INVALID_THREAD_ID);
|
|
|
|
goto SetEventExit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-15 13:47:25 +00:00
|
|
|
pEH = UserCreateObject(gHandleTable, NULL, &Handle, otEvent, sizeof(EVENTHOOK));
|
2008-07-31 23:48:35 +00:00
|
|
|
if (pEH)
|
|
|
|
{
|
|
|
|
InsertTailList(&GlobalEvents->Events, &pEH->Chain);
|
|
|
|
GlobalEvents->Counts++;
|
|
|
|
|
2009-07-26 12:27:15 +00:00
|
|
|
UserHMGetHandle(pEH) = Handle;
|
2008-08-02 00:38:01 +00:00
|
|
|
if (Thread)
|
2009-10-06 04:22:11 +00:00
|
|
|
pEH->head.pti = Thread->Tcb.Win32Thread;
|
2008-08-02 00:38:01 +00:00
|
|
|
else
|
2009-10-06 04:22:11 +00:00
|
|
|
pEH->head.pti = GetW32ThreadInfo();
|
2008-07-31 23:48:35 +00:00
|
|
|
pEH->eventMin = eventMin;
|
|
|
|
pEH->eventMax = eventMax;
|
|
|
|
pEH->idProcess = idProcess;
|
|
|
|
pEH->idThread = idThread;
|
2009-07-26 12:27:15 +00:00
|
|
|
pEH->Flags = dwflags;
|
2008-07-31 23:48:35 +00:00
|
|
|
|
|
|
|
if (NULL != hmodWinEventProc)
|
|
|
|
{
|
2009-07-01 17:19:16 +00:00
|
|
|
pEH->offPfn = (ULONG_PTR)((char *)lpfnWinEventProc - (char *)hmodWinEventProc);
|
|
|
|
pEH->ihmod = (INT)hmodWinEventProc;
|
|
|
|
pEH->Proc = lpfnWinEventProc;
|
2008-07-31 23:48:35 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
pEH->Proc = lpfnWinEventProc;
|
|
|
|
|
2008-08-02 06:21:55 +00:00
|
|
|
UserDereferenceObject(pEH);
|
|
|
|
|
2008-07-31 23:48:35 +00:00
|
|
|
Ret = Handle;
|
2008-08-02 00:38:01 +00:00
|
|
|
IntSetSrvEventMask( eventMin, eventMax);
|
2008-07-31 23:48:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SetEventExit:
|
2008-08-02 00:38:01 +00:00
|
|
|
if (Thread) ObDereferenceObject(Thread);
|
2008-07-31 23:48:35 +00:00
|
|
|
UserLeave();
|
|
|
|
return Ret;
|
2008-04-17 01:48:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOL
|
2008-11-29 22:48:58 +00:00
|
|
|
APIENTRY
|
2008-04-17 01:48:34 +00:00
|
|
|
NtUserUnhookWinEvent(
|
|
|
|
HWINEVENTHOOK hWinEventHook)
|
|
|
|
{
|
2008-07-31 23:48:35 +00:00
|
|
|
PEVENTHOOK pEH;
|
|
|
|
BOOL Ret = FALSE;
|
2008-04-17 01:48:34 +00:00
|
|
|
|
2008-07-31 23:48:35 +00:00
|
|
|
UserEnterExclusive();
|
|
|
|
|
|
|
|
pEH = (PEVENTHOOK)UserGetObject(gHandleTable, hWinEventHook, otEvent);
|
|
|
|
if (pEH)
|
|
|
|
{
|
|
|
|
Ret = IntRemoveEvent(pEH);
|
|
|
|
}
|
|
|
|
|
|
|
|
UserLeave();
|
|
|
|
return Ret;
|
2008-04-17 01:48:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* EOF */
|