Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

334 lines
9.1 KiB

  1. /********************************************************************/
  2. /* Copyright(c) 1995 Microsoft Corporation */
  3. /********************************************************************/
  4. //***
  5. //
  6. // Filename: evdsptch.c
  7. //
  8. // Description: This module contains the event dispatcher for the
  9. // DDM's procedure-driven state machine
  10. //
  11. // Author: Stefan Solomon (stefans) June 9, 1992.
  12. //
  13. //***
  14. #include "ddm.h"
  15. #include "handlers.h"
  16. #include "objects.h"
  17. #include "timer.h"
  18. #include "util.h"
  19. #include <raserror.h>
  20. #include <ddmif.h>
  21. #include <sechost.h>
  22. #include <stdlib.h>
  23. #include "rasmanif.h"
  24. //***
  25. //
  26. // Function: EventDispatcher
  27. //
  28. // Descr: Waits for events to be signaled and invokes the proper
  29. // event handler. Returns when DDM has terminated.
  30. //
  31. //***
  32. DWORD
  33. EventDispatcher(
  34. IN LPVOID arg
  35. )
  36. {
  37. EVENT_HANDLER * pEventHandler;
  38. DWORD dwSignaledEvent;
  39. //
  40. // Indicate that this thread is running
  41. //
  42. InterlockedIncrement( gblDDMConfigInfo.lpdwNumThreadsRunning );
  43. RegNotifyChangeKeyValue( gblDDMConfigInfo.hkeyParameters,
  44. TRUE,
  45. REG_NOTIFY_CHANGE_LAST_SET,
  46. gblSupervisorEvents[DDM_EVENT_CHANGE_NOTIFICATION],
  47. TRUE );
  48. RegNotifyChangeKeyValue( gblDDMConfigInfo.hkeyAccounting,
  49. TRUE,
  50. REG_NOTIFY_CHANGE_LAST_SET,
  51. gblSupervisorEvents[DDM_EVENT_CHANGE_NOTIFICATION1],
  52. TRUE );
  53. RegNotifyChangeKeyValue( gblDDMConfigInfo.hkeyAuthentication,
  54. TRUE,
  55. REG_NOTIFY_CHANGE_LAST_SET,
  56. gblSupervisorEvents[DDM_EVENT_CHANGE_NOTIFICATION2],
  57. TRUE );
  58. while( TRUE )
  59. {
  60. dwSignaledEvent = WaitForMultipleObjectsEx(
  61. NUM_DDM_EVENTS
  62. + ( gblDeviceTable.NumDeviceBuckets * 3 ),
  63. gblSupervisorEvents,
  64. FALSE,
  65. INFINITE,
  66. TRUE);
  67. if ( ( dwSignaledEvent == 0xFFFFFFFF ) ||
  68. ( dwSignaledEvent == WAIT_TIMEOUT ) )
  69. {
  70. DDMTRACE2("WaitForMultipleObjectsEx returned %d, GetLastError=%d",
  71. dwSignaledEvent, GetLastError() );
  72. continue;
  73. }
  74. //
  75. // DDM has terminated so return
  76. //
  77. if ( dwSignaledEvent == DDM_EVENT_SVC_TERMINATED )
  78. {
  79. LPDWORD lpdwNumThreadsRunning =
  80. gblDDMConfigInfo.lpdwNumThreadsRunning;
  81. //
  82. // If we were running and we are now shutting down, clean up
  83. // gracefully
  84. //
  85. if ( gblDDMConfigInfo.pServiceStatus->dwCurrentState
  86. == SERVICE_STOP_PENDING )
  87. {
  88. DDMCleanUp();
  89. }
  90. //
  91. // Decrease the count for this thread
  92. //
  93. InterlockedDecrement( lpdwNumThreadsRunning );
  94. return( NO_ERROR );
  95. }
  96. //
  97. // invoke the handler associated with the signaled event
  98. //
  99. if ( dwSignaledEvent < NUM_DDM_EVENTS )
  100. {
  101. //
  102. // Some DDM event
  103. //
  104. gblEventHandlerTable[dwSignaledEvent].EventHandler();
  105. }
  106. else if ( dwSignaledEvent < ( NUM_DDM_EVENTS
  107. + gblDeviceTable.NumDeviceBuckets ) )
  108. {
  109. //
  110. // The event was a RASMAN event
  111. //
  112. RmEventHandler( dwSignaledEvent );
  113. }
  114. else if ( dwSignaledEvent < ( NUM_DDM_EVENTS
  115. + gblDeviceTable.NumDeviceBuckets * 2 ) )
  116. {
  117. //
  118. // A frame was received on a port
  119. //
  120. RmRecvFrameEventHandler( dwSignaledEvent );
  121. }
  122. else if ( dwSignaledEvent != WAIT_IO_COMPLETION )
  123. {
  124. //
  125. // We got a disconnect on a dialed out port
  126. //
  127. RasApiDisconnectHandler( dwSignaledEvent );
  128. }
  129. }
  130. return( NO_ERROR );
  131. }
  132. //***
  133. //
  134. // Function: SecurityDllEventHandler
  135. //
  136. // Descr: This will handle all events from the 3rd party security Dll.
  137. // Either the security dialog with the client completed
  138. // successfully, in which case we continue on with the connection,
  139. // or we log the error and bring down the line.
  140. //
  141. //***
  142. VOID
  143. SecurityDllEventHandler(
  144. VOID
  145. )
  146. {
  147. LPWSTR auditstrp[3];
  148. SECURITY_MESSAGE message;
  149. PDEVICE_OBJECT pDevObj;
  150. DWORD dwBucketIndex;
  151. WCHAR wchUserName[UNLEN+1];
  152. //
  153. // loop to get all messages
  154. //
  155. while( ServerReceiveMessage( MESSAGEQ_ID_SECURITY, (BYTE *) &message ) )
  156. {
  157. EnterCriticalSection( &(gblDeviceTable.CriticalSection) );
  158. //
  159. // identify the message recipient
  160. //
  161. if ( ( pDevObj = DeviceObjGetPointer( message.hPort ) ) == NULL )
  162. {
  163. LeaveCriticalSection( &(gblDeviceTable.CriticalSection) );
  164. return;
  165. }
  166. //
  167. // action on the message type
  168. //
  169. switch( message.dwMsgId )
  170. {
  171. case SECURITYMSG_SUCCESS:
  172. //
  173. // Stop timer for 3rd party security
  174. //
  175. TimerQRemove( (HANDLE)pDevObj->hPort, SvSecurityTimeout );
  176. RasFreeBuffer( pDevObj->pRasmanSendBuffer );
  177. pDevObj->pRasmanSendBuffer = NULL;
  178. //
  179. // copy the user name
  180. //
  181. MultiByteToWideChar( CP_ACP,
  182. 0,
  183. message.UserName,
  184. -1,
  185. pDevObj->wchUserName,
  186. UNLEN+1 );
  187. //
  188. // copy the domain name
  189. //
  190. MultiByteToWideChar( CP_ACP,
  191. 0,
  192. message.Domain,
  193. -1,
  194. pDevObj->wchDomainName,
  195. DNLEN+1 );
  196. pDevObj->SecurityState = DEV_OBJ_SECURITY_DIALOG_INACTIVE;
  197. //
  198. // Change RASMAN state to CONNECTED from LISTENCOMPLETE and signal
  199. // RmEventHandler
  200. //
  201. RasPortConnectComplete(pDevObj->hPort);
  202. dwBucketIndex = DeviceObjHashPortToBucket( pDevObj->hPort );
  203. SetEvent( gblSupervisorEvents[NUM_DDM_EVENTS+dwBucketIndex] );
  204. DDM_PRINT(
  205. gblDDMConfigInfo.dwTraceId,
  206. TRACE_FSM,
  207. "SecurityDllEventHandler: Security DLL success \n" );
  208. break;
  209. case SECURITYMSG_FAILURE:
  210. //
  211. // Log the fact that the use failed to pass 3rd party security.
  212. //
  213. MultiByteToWideChar( CP_ACP,
  214. 0,
  215. message.UserName,
  216. -1,
  217. wchUserName,
  218. UNLEN+1 );
  219. auditstrp[0] = wchUserName;
  220. auditstrp[1] = pDevObj->wchPortName;
  221. DDMLogError( ROUTERLOG_SEC_AUTH_FAILURE, 2, auditstrp, NO_ERROR );
  222. //
  223. // Hang up the line
  224. //
  225. DDM_PRINT(
  226. gblDDMConfigInfo.dwTraceId,
  227. TRACE_FSM,
  228. "SecurityDllEventHandler:Security DLL failure %s\n",
  229. message.UserName );
  230. if ( pDevObj->SecurityState == DEV_OBJ_SECURITY_DIALOG_ACTIVE )
  231. {
  232. DevStartClosing(pDevObj);
  233. }
  234. else if ( pDevObj->SecurityState==DEV_OBJ_SECURITY_DIALOG_STOPPING )
  235. {
  236. pDevObj->SecurityState = DEV_OBJ_SECURITY_DIALOG_INACTIVE;
  237. DevCloseComplete(pDevObj);
  238. }
  239. break;
  240. case SECURITYMSG_ERROR:
  241. auditstrp[0] = pDevObj->wchPortName;
  242. DDMLogErrorString( ROUTERLOG_SEC_AUTH_INTERNAL_ERROR, 1, auditstrp,
  243. message.dwError, 1);
  244. DDM_PRINT(
  245. gblDDMConfigInfo.dwTraceId,
  246. TRACE_FSM,
  247. "SecurityDllEventHandler:Security DLL failure %x\n",
  248. message.dwError );
  249. if ( pDevObj->SecurityState == DEV_OBJ_SECURITY_DIALOG_ACTIVE )
  250. {
  251. DevStartClosing(pDevObj);
  252. }
  253. else if ( pDevObj->SecurityState==DEV_OBJ_SECURITY_DIALOG_STOPPING )
  254. {
  255. pDevObj->SecurityState = DEV_OBJ_SECURITY_DIALOG_INACTIVE;
  256. DevCloseComplete(pDevObj);
  257. }
  258. break;
  259. default:
  260. RTASSERT(FALSE);
  261. break;
  262. }
  263. LeaveCriticalSection( &(gblDeviceTable.CriticalSection) );
  264. }
  265. }