Leaked source code of windows server 2003
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.

291 lines
7.9 KiB

  1. /*++
  2. Copyright (c) 1992 Microsoft Corporation
  3. Module Name:
  4. rdwr.c
  5. Abstract:
  6. This module contains routines for read and write for NTDOS. These
  7. routines saves the switch to user mode. The BOP is handled in the
  8. kernel for performance reasons. These routines are called only for
  9. files. Local DOS devices and named pipe operations never come here.
  10. Author:
  11. Sudeep Bharati (Sudeepb) 04-Mar-1993
  12. Revision History:
  13. --*/
  14. #include "vdmp.h"
  15. VOID
  16. NTFastDOSIO (
  17. PKTRAP_FRAME TrapFrame,
  18. ULONG IoType
  19. );
  20. #ifdef ALLOC_PRAGMA
  21. #pragma alloc_text(PAGE,NTFastDOSIO)
  22. #endif
  23. #define EFLAGS_CF 0x1
  24. #define EFLAGS_ZF 0x40
  25. #define GETFILEPOINTER(hi,lo) (((ULONG)hi << 16) + (ULONG)lo)
  26. #define GETHANDLE(hi,lo) (HANDLE)(((ULONG)hi << 16) + (ULONG)lo)
  27. #define GETBUFFER(hi,lo) (((ULONG)hi << 4) + lo)
  28. #define SVC_DEMFASTREAD 0x42
  29. #define SVC_DEMFASTWRITE 0x43
  30. #define CONSOLE_HANDLE_SIGNATURE 0x00000003
  31. #define CONSOLE_HANDLE(HANDLE) (((ULONG)(HANDLE) & CONSOLE_HANDLE_SIGNATURE) == CONSOLE_HANDLE_SIGNATURE)
  32. #define STD_INPUT_HANDLE (ULONG)-10
  33. #define STD_OUTPUT_HANDLE (ULONG)-11
  34. #define STD_ERROR_HANDLE (ULONG)-12
  35. VOID
  36. NTFastDOSIO (
  37. PKTRAP_FRAME TrapFrame,
  38. ULONG IoType
  39. )
  40. {
  41. HANDLE hFile;
  42. PVOID lpBuf;
  43. ULONG ulBX,ulSI;
  44. LARGE_INTEGER Large;
  45. PIO_STATUS_BLOCK IoStatusBlock;
  46. PFILE_POSITION_INFORMATION CurrentPosition;
  47. NTSTATUS Status;
  48. ULONG CountToIO;
  49. PFILE_END_OF_FILE_INFORMATION EndOfFile;
  50. PVDM_TIB VdmTib;
  51. KIRQL OldIrql;
  52. PAGED_CODE();
  53. //
  54. // Clear CF flag and assume success
  55. //
  56. TrapFrame->EFlags &= ~EFLAGS_CF;
  57. //
  58. // Validate the request
  59. //
  60. if (IoType != SVC_DEMFASTREAD && IoType != SVC_DEMFASTWRITE) {
  61. TrapFrame->EFlags |= EFLAGS_CF;
  62. return;
  63. }
  64. //
  65. // Signal softpc that we are doing disk io for idle detection.
  66. //
  67. try {
  68. *FIXED_NTVDMSTATE_LINEAR_PC_AT |= VDM_IDLEACTIVITY;
  69. }
  70. except (EXCEPTION_EXECUTE_HANDLER) {
  71. ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
  72. TrapFrame->EFlags |= EFLAGS_CF;
  73. return;
  74. }
  75. Status = VdmpGetVdmTib(&VdmTib);
  76. if (!NT_SUCCESS(Status)) { // vdmtib is bad
  77. ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
  78. TrapFrame->EFlags |= EFLAGS_CF;
  79. return;
  80. }
  81. IoStatusBlock = (PIO_STATUS_BLOCK) &VdmTib->TempArea1;
  82. CurrentPosition = (PFILE_POSITION_INFORMATION) &VdmTib->TempArea2;
  83. EndOfFile = (PFILE_END_OF_FILE_INFORMATION) CurrentPosition;
  84. try {
  85. ProbeForWrite (IoStatusBlock, sizeof (IO_STATUS_BLOCK), sizeof (UCHAR));
  86. ProbeForWrite (CurrentPosition, sizeof (FILE_END_OF_FILE_INFORMATION), sizeof (UCHAR));
  87. }
  88. except (EXCEPTION_EXECUTE_HANDLER) {
  89. ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
  90. TrapFrame->EFlags |= EFLAGS_CF;
  91. return;
  92. }
  93. // Get the NT handle
  94. hFile = GETHANDLE((TrapFrame->Eax & 0x0000ffff),(TrapFrame->Ebp & 0x0000ffff));
  95. // advance ip past the bop instruction
  96. // clear carry flag, assuming success
  97. TrapFrame->Eip += 4;
  98. ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
  99. TrapFrame->EFlags &= ~EFLAGS_CF;
  100. if (CONSOLE_HANDLE(hFile) ||
  101. hFile == (HANDLE) STD_INPUT_HANDLE ||
  102. hFile == (HANDLE) STD_OUTPUT_HANDLE ||
  103. hFile == (HANDLE) STD_ERROR_HANDLE )
  104. {
  105. TrapFrame->EFlags |= EFLAGS_CF;
  106. return;
  107. }
  108. // Get the IO buffer
  109. lpBuf = (PVOID) GETBUFFER(TrapFrame->V86Ds, (TrapFrame->Edx & 0x0000ffff));
  110. // Get the Count
  111. CountToIO = TrapFrame->Ecx & 0x0000ffff;
  112. // Get Seek Parameters
  113. ulBX = TrapFrame->Ebx & 0x0000ffff;
  114. ulSI = TrapFrame->Esi & 0x0000ffff;
  115. //
  116. // Lower Irql to PASSIVE_LEVEL for io system
  117. //
  118. OldIrql = KeGetCurrentIrql();
  119. KeLowerIrql (PASSIVE_LEVEL);
  120. //
  121. // Check if we need to seek
  122. //
  123. if (!(TrapFrame->EFlags & EFLAGS_ZF)) {
  124. Large = RtlConvertUlongToLargeInteger(GETFILEPOINTER(ulBX,ulSI));
  125. try {
  126. CurrentPosition->CurrentByteOffset = Large;
  127. }
  128. except(EXCEPTION_EXECUTE_HANDLER) {
  129. goto ErrorExit; // we have caught an exception, error exit
  130. }
  131. Status = NtSetInformationFile (hFile,
  132. IoStatusBlock,
  133. CurrentPosition,
  134. sizeof(FILE_POSITION_INFORMATION),
  135. FilePositionInformation);
  136. if (!NT_SUCCESS(Status)) {
  137. goto ErrorExit;
  138. }
  139. try {
  140. if (CurrentPosition->CurrentByteOffset.LowPart == -1) {
  141. goto ErrorExit;
  142. }
  143. } except (EXCEPTION_EXECUTE_HANDLER) {
  144. goto ErrorExit;
  145. }
  146. }
  147. if (IoType == SVC_DEMFASTREAD) {
  148. Status = NtReadFile (hFile,
  149. NULL,
  150. NULL,
  151. NULL,
  152. IoStatusBlock,
  153. (PVOID)lpBuf,
  154. CountToIO,
  155. NULL,
  156. NULL);
  157. }
  158. else {
  159. if (CountToIO == 0) {
  160. Status = NtQueryInformationFile (hFile,
  161. IoStatusBlock,
  162. CurrentPosition,
  163. sizeof(FILE_POSITION_INFORMATION),
  164. FilePositionInformation);
  165. if (!NT_SUCCESS(Status)) {
  166. goto ErrorExit;
  167. }
  168. try {
  169. EndOfFile->EndOfFile = CurrentPosition->CurrentByteOffset;
  170. }
  171. except(EXCEPTION_EXECUTE_HANDLER) {
  172. goto ErrorExit; // we have caught an exception, error exit
  173. }
  174. Status = NtSetInformationFile (hFile,
  175. IoStatusBlock,
  176. EndOfFile,
  177. sizeof(FILE_END_OF_FILE_INFORMATION),
  178. FileEndOfFileInformation);
  179. if (NT_SUCCESS(Status)) {
  180. KeRaiseIrql(OldIrql, &OldIrql);
  181. return;
  182. }
  183. goto ErrorExit;
  184. }
  185. Status = NtWriteFile (hFile,
  186. NULL,
  187. NULL,
  188. NULL,
  189. IoStatusBlock,
  190. (PVOID)lpBuf,
  191. CountToIO,
  192. NULL,
  193. NULL);
  194. }
  195. if (Status == STATUS_PENDING) {
  196. //
  197. // Operation must complete before return & IoStatusBlock destroyed
  198. //
  199. Status = NtWaitForSingleObject (hFile, FALSE, NULL);
  200. if ( NT_SUCCESS(Status)) {
  201. try {
  202. Status = IoStatusBlock->Status;
  203. } except (EXCEPTION_EXECUTE_HANDLER) {
  204. NOTHING;
  205. }
  206. }
  207. }
  208. KeRaiseIrql(OldIrql, &OldIrql);
  209. if ( NT_SUCCESS(Status) ) {
  210. TrapFrame->Eax &= 0xffff0000;
  211. try {
  212. TrapFrame->Eax |= (USHORT) IoStatusBlock->Information;
  213. } except (EXCEPTION_EXECUTE_HANDLER) {
  214. NOTHING;
  215. }
  216. }
  217. else if (IoType == SVC_DEMFASTREAD && Status == STATUS_END_OF_FILE) {
  218. TrapFrame->Eax &= 0xffff0000;
  219. }
  220. else {
  221. ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
  222. TrapFrame->EFlags |= EFLAGS_CF;
  223. }
  224. return;
  225. ErrorExit:
  226. KeRaiseIrql(OldIrql, &OldIrql);
  227. ASSERT (KeGetCurrentIrql () >= APC_LEVEL);
  228. TrapFrame->EFlags |= EFLAGS_CF;
  229. return;
  230. }