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.

475 lines
11 KiB

  1. /*++
  2. Copyright (c) 2001 Microsoft Corporation
  3. Module Name:
  4. tunf.c
  5. Abstract:
  6. utility routines to handle opening and closing the tunmp device.
  7. Environment:
  8. Kernel mode only.
  9. Revision History:
  10. alid 10/22/2001 modified for tunmp
  11. --*/
  12. #include "precomp.h"
  13. #define __FILENUMBER 'FNUT'
  14. NTSTATUS
  15. TunFOpen(
  16. IN PDEVICE_OBJECT DeviceObject,
  17. IN PIRP pIrp
  18. )
  19. /*++
  20. Routine Description:
  21. Hanndles IRP_MJ_CREATE. Here we set the device status in use, assigns a pointer
  22. from the file object to the adapter object using pIrpSp->FileObject->FsContext,
  23. allocates packet and buffer pools, and returns a success status
  24. Arguments:
  25. pDeviceObject - Pointer to the device object.
  26. pIrp - Pointer to the request packet.
  27. Return Value:
  28. Status is returned.
  29. --*/
  30. {
  31. PIO_STACK_LOCATION pIrpSp;
  32. NDIS_STATUS NdisStatus;
  33. NTSTATUS NtStatus = STATUS_SUCCESS;
  34. UINT i;
  35. BOOLEAN Found = FALSE;
  36. PLIST_ENTRY pListEntry;
  37. PTUN_ADAPTER pAdapter = NULL;
  38. BOOLEAN DerefAdapter = FALSE;
  39. DEBUGP(DL_INFO, ("Open: DeviceObject %p\n", DeviceObject));
  40. do
  41. {
  42. pIrpSp = IoGetCurrentIrpStackLocation(pIrp);
  43. TUN_ACQUIRE_LOCK(&TunGlobalLock);
  44. for (pListEntry = TunAdapterList.Flink;
  45. pListEntry != &TunAdapterList;
  46. pListEntry = pListEntry->Flink)
  47. {
  48. pAdapter = CONTAINING_RECORD(pListEntry,
  49. TUN_ADAPTER,
  50. Link);
  51. if(pAdapter->DeviceObject == DeviceObject)
  52. {
  53. Found = TRUE;
  54. TUN_REF_ADAPTER(pAdapter);
  55. DerefAdapter = TRUE;
  56. break;
  57. }
  58. }
  59. TUN_RELEASE_LOCK(&TunGlobalLock);
  60. if (Found == FALSE)
  61. {
  62. NtStatus = STATUS_NO_SUCH_DEVICE;
  63. break;
  64. }
  65. TUN_ACQUIRE_LOCK(&pAdapter->Lock);
  66. if (TUN_TEST_FLAG(pAdapter, TUN_ADAPTER_OPEN))
  67. {
  68. //
  69. // adapter is already open by another device. fail
  70. //
  71. TUN_RELEASE_LOCK(&pAdapter->Lock);
  72. NtStatus = STATUS_INVALID_DEVICE_STATE;
  73. break;
  74. }
  75. TUN_SET_FLAG(pAdapter, TUN_ADAPTER_OPEN | TUN_ADAPTER_CANT_HALT);
  76. TUN_RELEASE_LOCK(&pAdapter->Lock);
  77. //Assign a pointer to the adapter object from the file object
  78. pIrpSp->FileObject->FsContext = (PVOID)pAdapter;
  79. //Get the device connected to the network by cable plugging in
  80. NdisMIndicateStatus(pAdapter->MiniportHandle,
  81. NDIS_STATUS_MEDIA_CONNECT,
  82. NULL,
  83. 0);
  84. NdisMIndicateStatusComplete(pAdapter->MiniportHandle);
  85. DerefAdapter = FALSE;
  86. } while (FALSE);
  87. //
  88. //Complete the IRP
  89. //
  90. pIrp->IoStatus.Information = 0;
  91. pIrp->IoStatus.Status = NtStatus;
  92. IoCompleteRequest(pIrp, IO_NO_INCREMENT);
  93. if (DerefAdapter)
  94. {
  95. TUN_DEREF_ADAPTER(pAdapter);
  96. }
  97. return (NtStatus);
  98. }
  99. //************************************************************************
  100. NTSTATUS
  101. TunFClose(
  102. IN PDEVICE_OBJECT pDeviceObject,
  103. IN PIRP pIrp
  104. )
  105. /*++
  106. Routine Description:
  107. Handles IRP_MJ_CLOSE. Here we change the device state into available (not in
  108. use), free the the file object's pointer to the adapter object, free the
  109. allocated packet/buffer pools, and set the success status.
  110. Arguments:
  111. pDeviceObject - Pointer to the device object.
  112. pIrp - Pointer to the request packet.
  113. Return Value:
  114. Status is returned.
  115. --*/
  116. {
  117. //1 this may be moved to clean_up
  118. NTSTATUS NtStatus;
  119. PIO_STACK_LOCATION pIrpSp;
  120. PTUN_ADAPTER pAdapter;
  121. PLIST_ENTRY pRcvPacketEntry;
  122. PNDIS_PACKET pRcvPacket;
  123. pIrpSp = IoGetCurrentIrpStackLocation(pIrp);
  124. pAdapter = (PTUN_ADAPTER)pIrpSp->FileObject->FsContext;
  125. DEBUGP(DL_INFO, ("Close: FileObject %p\n",
  126. IoGetCurrentIrpStackLocation(pIrp)->FileObject));
  127. //If no adapter/device object is associated with the this file object
  128. if (pAdapter == NULL)
  129. {
  130. NtStatus = STATUS_IO_DEVICE_ERROR;
  131. }
  132. else
  133. {
  134. TUN_ACQUIRE_LOCK(&pAdapter->Lock);
  135. TUN_CLEAR_FLAG(pAdapter, TUN_ADAPTER_OPEN);
  136. TUN_RELEASE_LOCK(&pAdapter->Lock);
  137. NdisMIndicateStatus(pAdapter->MiniportHandle,
  138. NDIS_STATUS_MEDIA_DISCONNECT,
  139. NULL,
  140. 0);
  141. NdisMIndicateStatusComplete(pAdapter->MiniportHandle);
  142. //Let the adapter object free
  143. pIrpSp->FileObject->FsContext = NULL;
  144. TUN_ACQUIRE_LOCK(&pAdapter->Lock);
  145. //
  146. //Emtpy the received packets queue, and return the packets to NDIS
  147. //with success status
  148. //
  149. while(!IsListEmpty(&pAdapter->RecvPktQueue))
  150. {
  151. //
  152. //Remove the first queued received packet from the entry list
  153. //
  154. pRcvPacketEntry = pAdapter->RecvPktQueue.Flink;
  155. RemoveEntryList(pRcvPacketEntry);
  156. ExInterlockedDecrementLong(&pAdapter->RecvPktCount, &pAdapter->Lock);
  157. //Get the packet from
  158. pRcvPacket = CONTAINING_RECORD(pRcvPacketEntry,
  159. NDIS_PACKET,
  160. MiniportReserved[0]);
  161. TUN_RELEASE_LOCK(&pAdapter->Lock);
  162. //Indicate NDIS about comletion of the packet
  163. NdisMSendComplete(pAdapter->MiniportHandle,
  164. pRcvPacket,
  165. NDIS_STATUS_FAILURE);
  166. TUN_ACQUIRE_LOCK(&pAdapter->Lock);
  167. }
  168. TUN_RELEASE_LOCK(&pAdapter->Lock);
  169. NtStatus = STATUS_SUCCESS;
  170. }
  171. //1 who should do the testing of tunmp
  172. //
  173. //Complete the IRP
  174. //
  175. pIrp->IoStatus.Information = 0;
  176. pIrp->IoStatus.Status = NtStatus;
  177. IoCompleteRequest(pIrp, IO_NO_INCREMENT);
  178. TUN_ACQUIRE_LOCK(&pAdapter->Lock);
  179. TUN_CLEAR_FLAG(pAdapter, TUN_ADAPTER_CANT_HALT);
  180. if (pAdapter->HaltEvent != NULL)
  181. {
  182. NdisSetEvent(pAdapter->HaltEvent);
  183. }
  184. TUN_RELEASE_LOCK(&pAdapter->Lock);
  185. KdPrint(("\nTunFClose: Exit\n"));
  186. return (NtStatus);
  187. }
  188. //************************************************************************
  189. NTSTATUS
  190. TunFCleanup(
  191. IN PDEVICE_OBJECT pDeviceObject,
  192. IN PIRP pIrp
  193. )
  194. /*++
  195. Routine Description:
  196. Handles IRP_MJ_CLEANUP. Here we reset the driver's cancel entry point to NULL
  197. in every IRP currently in the driver's internal queue of read IRPs, cancel
  198. all the queued IRPs, and return a success status.
  199. Arguments:
  200. pDeviceObject - Pointer to the device object.
  201. pIrp - Pointer to the request packet.
  202. Return Value:
  203. Status is returned.
  204. --*/
  205. {
  206. PIO_STACK_LOCATION pIrpSp;
  207. NTSTATUS NtStatus;
  208. PTUN_ADAPTER pAdapter;
  209. pIrpSp = IoGetCurrentIrpStackLocation(pIrp);
  210. pAdapter = (PTUN_ADAPTER)pIrpSp->FileObject->FsContext;
  211. DEBUGP(DL_INFO, ("Cleanup: FileObject %p, pAdapter %p\n",
  212. pIrpSp->FileObject, pAdapter));
  213. if (pAdapter != NULL)
  214. {
  215. TUN_STRUCT_ASSERT(pAdapter, mc);
  216. //
  217. // Mark this endpoint.
  218. //
  219. TUN_ACQUIRE_LOCK(&pAdapter->Lock);
  220. TUN_CLEAR_FLAG(pAdapter, TUN_ADAPTER_OPEN);
  221. pAdapter->pFileObject = NULL;
  222. TUN_RELEASE_LOCK(&pAdapter->Lock);
  223. TunCancelPendingReads(pAdapter);
  224. }
  225. NtStatus = STATUS_SUCCESS;
  226. //
  227. //Complete the IRP
  228. //
  229. pIrp->IoStatus.Information = 0;
  230. pIrp->IoStatus.Status = NtStatus;
  231. IoCompleteRequest(pIrp, IO_NO_INCREMENT);
  232. DEBUGP(DL_INFO, ("Cleanup: OpenContext %p\n", pAdapter));
  233. return (NtStatus);
  234. }
  235. //************************************************************************
  236. NTSTATUS
  237. TunFIoControl(
  238. IN PDEVICE_OBJECT pDeviceObject,
  239. IN PIRP pIrp
  240. )
  241. /*++
  242. Routine Description:
  243. This is the dispatch routine for handling device IOCTL requests.
  244. Arguments:
  245. pDeviceObject - Pointer to the device object.
  246. pIrp - Pointer to the request packet.
  247. Return Value:
  248. Status is returned.
  249. --*/
  250. {
  251. PIO_STACK_LOCATION pIrpSp;
  252. PTUN_ADAPTER pAdapter;
  253. NTSTATUS NtStatus = STATUS_SUCCESS;
  254. ULONG BytesReturned = 0;
  255. PUCHAR OutputBuffer = NULL;
  256. pIrpSp = IoGetCurrentIrpStackLocation(pIrp);
  257. pAdapter = (PTUN_ADAPTER)pIrpSp->FileObject->FsContext;
  258. // pIrp->IoStatus.Information = 0;
  259. //if no adapter/device object is associated with this file object
  260. if (pAdapter == NULL)
  261. {
  262. pIrp->IoStatus.Status = STATUS_IO_DEVICE_ERROR;
  263. IoCompleteRequest(pIrp, IO_NO_INCREMENT);
  264. return NtStatus;
  265. }
  266. //1 check for valid adapter
  267. pIrp->IoStatus.Information = 0;
  268. OutputBuffer = (PUCHAR)pIrp->AssociatedIrp.SystemBuffer;
  269. switch (pIrpSp->Parameters.DeviceIoControl.IoControlCode)
  270. {
  271. case IOCTL_TUN_GET_MEDIUM_TYPE:
  272. if (pIrpSp->Parameters.DeviceIoControl.OutputBufferLength < sizeof(NDIS_MEDIUM))
  273. {
  274. NtStatus = STATUS_BUFFER_TOO_SMALL;
  275. }
  276. else
  277. {
  278. *((PNDIS_MEDIUM)OutputBuffer) = pAdapter->Medium;
  279. BytesReturned = sizeof(ULONG);
  280. }
  281. break;
  282. case IOCTL_TUN_GET_MTU:
  283. if (pIrpSp->Parameters.DeviceIoControl.OutputBufferLength < sizeof(ULONG))
  284. {
  285. NtStatus = STATUS_BUFFER_TOO_SMALL;
  286. }
  287. else
  288. {
  289. *((PULONG)OutputBuffer) = pAdapter->MediumMaxPacketLen;
  290. BytesReturned = sizeof(ULONG);
  291. }
  292. break;
  293. case IOCTL_TUN_GET_PACKET_FILTER:
  294. if (pIrpSp->Parameters.DeviceIoControl.OutputBufferLength < sizeof(ULONG))
  295. {
  296. NtStatus = STATUS_BUFFER_TOO_SMALL;
  297. }
  298. else
  299. {
  300. *((PULONG)OutputBuffer) = pAdapter->PacketFilter;
  301. BytesReturned = sizeof(ULONG);
  302. }
  303. break;
  304. case IOCTL_TUN_GET_MINIPORT_NAME:
  305. if (pIrpSp->Parameters.DeviceIoControl.OutputBufferLength < pAdapter->MiniportName.Length + sizeof(USHORT))
  306. {
  307. NtStatus = STATUS_BUFFER_TOO_SMALL;
  308. }
  309. else
  310. {
  311. *((PUSHORT)OutputBuffer) = pAdapter->MiniportName.Length;
  312. TUN_COPY_MEM(OutputBuffer + sizeof(USHORT),
  313. (PUCHAR)pAdapter->MiniportName.Buffer,
  314. pAdapter->MiniportName.Length);
  315. BytesReturned = pAdapter->MiniportName.Length + sizeof(USHORT);
  316. }
  317. break;
  318. default:
  319. NtStatus = STATUS_NOT_SUPPORTED;
  320. break;
  321. }
  322. pIrp->IoStatus.Information = BytesReturned;
  323. pIrpSp->Parameters.DeviceIoControl.OutputBufferLength = BytesReturned;
  324. pIrp->IoStatus.Status = NtStatus;
  325. IoCompleteRequest(pIrp, IO_NO_INCREMENT);
  326. return (NtStatus);
  327. }