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.

554 lines
14 KiB

  1. /*++
  2. Copyright (C) Microsoft Corporation, 1993 - 1999
  3. Module Name:
  4. epp.c
  5. Abstract:
  6. This module contains the code to perform all EPP related tasks (including
  7. EPP Software and EPP Hardware modes.)
  8. Author:
  9. Timothy T. Wells (WestTek, L.L.C.) - April 16, 1997
  10. Environment:
  11. Kernel mode
  12. Revision History :
  13. --*/
  14. #include "pch.h"
  15. BOOLEAN
  16. ParIsEppSwWriteSupported(
  17. IN PDEVICE_EXTENSION Extension
  18. );
  19. BOOLEAN
  20. ParIsEppSwReadSupported(
  21. IN PDEVICE_EXTENSION Extension
  22. );
  23. NTSTATUS
  24. ParEnterEppSwMode(
  25. IN PDEVICE_EXTENSION Extension,
  26. IN BOOLEAN DeviceIdRequest
  27. );
  28. VOID
  29. ParTerminateEppSwMode(
  30. IN PDEVICE_EXTENSION Extension
  31. );
  32. NTSTATUS
  33. ParEppSwWrite(
  34. IN PDEVICE_EXTENSION Extension,
  35. IN PVOID Buffer,
  36. IN ULONG BufferSize,
  37. OUT PULONG BytesTransferred
  38. );
  39. NTSTATUS
  40. ParEppSwRead(
  41. IN PDEVICE_EXTENSION Extension,
  42. IN PVOID Buffer,
  43. IN ULONG BufferSize,
  44. OUT PULONG BytesTransferred
  45. );
  46. BOOLEAN
  47. ParIsEppSwWriteSupported(
  48. IN PDEVICE_EXTENSION Extension
  49. )
  50. /*++
  51. Routine Description:
  52. This routine determines whether or not EPP mode is suported
  53. in the write direction by trying negotiate when asked.
  54. Arguments:
  55. Extension - The device extension.
  56. Return Value:
  57. BOOLEAN.
  58. --*/
  59. {
  60. NTSTATUS Status;
  61. // dvdr
  62. ParDump2(PARINFO, ("ParIsEppSwWriteSupported: Entering\n"));
  63. if (!(Extension->HardwareCapabilities & PPT_ECP_PRESENT) &&
  64. !(Extension->HardwareCapabilities & PPT_BYTE_PRESENT)) {
  65. ParDump2(PARINFO, ("ParIsEppSwWriteSupported: Hardware Not Supported Leaving\n"));
  66. // Only use EPP Software in the reverse direction if an ECR is
  67. // present or we know that we can put the data register into Byte mode.
  68. return FALSE;
  69. }
  70. if (Extension->BadProtocolModes & EPP_SW) {
  71. // dvdr
  72. ParDump2(PARINFO, ("ParIsEppSwWriteSupported: Not Supported Leaving\n"));
  73. return FALSE;
  74. }
  75. if (Extension->ProtocolModesSupported & EPP_SW) {
  76. // dvdr
  77. ParDump2(PARINFO, ("ParIsEppSwWriteSupported: Supported Leaving\n"));
  78. return TRUE;
  79. }
  80. // Must use SWEPP Enter and Terminate for this test.
  81. // Internel state machines will fail otherwise. --dvrh
  82. Status = ParEnterEppSwMode (Extension, FALSE);
  83. ParTerminateEppSwMode (Extension);
  84. if (NT_SUCCESS(Status)) {
  85. ParDump2(PARINFO, ("ParIsEppSwWriteSupported: Negotiated Supported Leaving\n"));
  86. Extension->ProtocolModesSupported |= EPP_SW;
  87. return TRUE;
  88. }
  89. ParDump2(PARINFO, ("ParIsEppSwWriteSupported: Not Negotiated Not Supported Leaving\n"));
  90. return FALSE;
  91. }
  92. BOOLEAN
  93. ParIsEppSwReadSupported(
  94. IN PDEVICE_EXTENSION Extension
  95. )
  96. /*++
  97. Routine Description:
  98. This routine determines whether or not EPP mode is suported
  99. in the read direction (need to be able to float the data register
  100. drivers in order to do byte wide reads) by trying negotiate when asked.
  101. Arguments:
  102. Extension - The device extension.
  103. Return Value:
  104. BOOLEAN.
  105. --*/
  106. {
  107. NTSTATUS Status;
  108. if (!(Extension->HardwareCapabilities & PPT_ECP_PRESENT) &&
  109. !(Extension->HardwareCapabilities & PPT_BYTE_PRESENT)) {
  110. ParDump2(PARINFO, ("ParIsEppSwReadSupported: Hardware Not Supported Leaving\n"));
  111. // Only use EPP Software in the reverse direction if an ECR is
  112. // present or we know that we can put the data register into Byte mode.
  113. return FALSE;
  114. }
  115. if (Extension->BadProtocolModes & EPP_SW)
  116. return FALSE;
  117. if (Extension->ProtocolModesSupported & EPP_SW)
  118. return TRUE;
  119. // Must use SWEPP Enter and Terminate for this test.
  120. // Internel state machines will fail otherwise. --dvrh
  121. Status = ParEnterEppSwMode (Extension, FALSE);
  122. ParTerminateEppSwMode (Extension);
  123. if (NT_SUCCESS(Status)) {
  124. ParDump2(PARINFO, ("ParIsEppSwReadSupported: Negotiated Supported Leaving\n"));
  125. Extension->ProtocolModesSupported |= EPP_SW;
  126. return TRUE;
  127. }
  128. ParDump2(PARINFO, ("ParIsEppSwReadSupported: Not Negotiated Not Supported Leaving\n"));
  129. return FALSE;
  130. }
  131. NTSTATUS
  132. ParEnterEppSwMode(
  133. IN PDEVICE_EXTENSION Extension,
  134. IN BOOLEAN DeviceIdRequest
  135. )
  136. /*++
  137. Routine Description:
  138. This routine performs 1284 negotiation with the peripheral to the
  139. EPP mode protocol.
  140. Arguments:
  141. Controller - Supplies the port address.
  142. DeviceIdRequest - Supplies whether or not this is a request for a device
  143. id.
  144. Return Value:
  145. STATUS_SUCCESS - Successful negotiation.
  146. otherwise - Unsuccessful negotiation.
  147. --*/
  148. {
  149. NTSTATUS Status = STATUS_SUCCESS;
  150. // dvdr
  151. ParDump2(PARINFO, ("ParEnterEppSwMode: Entering\n"));
  152. // Parport Set Chip mode will put the Chip into Byte Mode if Capable
  153. // We need it for Epp Sw Mode
  154. Status = Extension->TrySetChipMode( Extension->PortContext, ECR_BYTE_PIO_MODE );
  155. if ( NT_SUCCESS(Status) ) {
  156. if ( Extension->ModeSafety == SAFE_MODE ) {
  157. if (DeviceIdRequest) {
  158. // dvdr
  159. ParDump2(PARINFO, ("ParEnterEppSwMode: Calling IeeeEnter1284Mode with DEVICE_ID_REQUEST\n"));
  160. Status = IeeeEnter1284Mode (Extension, EPP_EXTENSIBILITY | DEVICE_ID_REQ);
  161. } else {
  162. // dvdr
  163. ParDump2(PARINFO, ("ParEnterEppSwMode: Calling IeeeEnter1284Mode\n"));
  164. Status = IeeeEnter1284Mode (Extension, EPP_EXTENSIBILITY);
  165. }
  166. } else {
  167. ParDump2(PARINFO, ("ParEnterEppSwMode: In UNSAFE_MODE.\n"));
  168. Extension->Connected = TRUE;
  169. }
  170. }
  171. if ( NT_SUCCESS(Status) ) {
  172. // dvdr
  173. ParDump2(PARINFO, ("ParEnterEppSwMode: IeeeEnter1284Mode returned success\n"));
  174. Extension->CurrentPhase = PHASE_FORWARD_IDLE;
  175. Extension->IsIeeeTerminateOk = TRUE;
  176. } else {
  177. // dvdr
  178. ParDump2(PARINFO, ("ParEnterEppSwMode: IeeeEnter1284Mode returned unsuccessful\n"));
  179. Extension->CurrentPhase = PHASE_UNKNOWN;
  180. Extension->IsIeeeTerminateOk = FALSE;
  181. }
  182. ParDump2(PARINFO, ("ParEnterEppSwMode: Leaving with Status : %x \n", Status));
  183. return Status;
  184. }
  185. VOID
  186. ParTerminateEppSwMode(
  187. IN PDEVICE_EXTENSION Extension
  188. )
  189. /*++
  190. Routine Description:
  191. This routine terminates the interface back to compatibility mode.
  192. Arguments:
  193. Extension - The Device Extension which has the parallel port's controller address.
  194. Return Value:
  195. None.
  196. --*/
  197. {
  198. // dvdr
  199. ParDump2(PARINFO, ("ParTerminateEppMode: Entering\n"));
  200. if ( Extension->ModeSafety == SAFE_MODE ) {
  201. IeeeTerminate1284Mode (Extension);
  202. } else {
  203. ParDump2(PARINFO, ("ParTerminateEppMode: In UNSAFE_MODE.\n"));
  204. Extension->Connected = FALSE;
  205. }
  206. Extension->ClearChipMode( Extension->PortContext, ECR_BYTE_PIO_MODE );
  207. ParDump2(PARINFO, ("ParTerminateEppMode: Leaving\n"));
  208. return;
  209. }
  210. NTSTATUS
  211. ParEppSwWrite(
  212. IN PDEVICE_EXTENSION Extension,
  213. IN PVOID Buffer,
  214. IN ULONG BufferSize,
  215. OUT PULONG BytesTransferred
  216. )
  217. /*++
  218. Routine Description:
  219. Writes data to the peripheral using the EPP protocol under software
  220. control.
  221. Arguments:
  222. Extension - Supplies the device extension.
  223. Buffer - Supplies the buffer to write from.
  224. BufferSize - Supplies the number of bytes in the buffer.
  225. BytesTransferred - Returns the number of bytes transferred.
  226. Return Value:
  227. None.
  228. --*/
  229. {
  230. PUCHAR Controller;
  231. PUCHAR pBuffer = (PUCHAR)Buffer;
  232. NTSTATUS Status = STATUS_SUCCESS;
  233. ULONG i, j;
  234. UCHAR HDReady, HDAck, HDFinished;
  235. // dvdr
  236. ParDump2(PARINFO, ("ParEppSwWrite: Entering\n"));
  237. Controller = Extension->Controller;
  238. Extension->CurrentPhase = PHASE_FORWARD_XFER;
  239. // BIT5 of DCR needs to be low to be in BYTE forward mode
  240. HDReady = SET_DCR( INACTIVE, INACTIVE, ACTIVE, ACTIVE, INACTIVE, INACTIVE );
  241. HDAck = SET_DCR( INACTIVE, INACTIVE, ACTIVE, ACTIVE, ACTIVE, INACTIVE );
  242. HDFinished = SET_DCR( INACTIVE, INACTIVE, ACTIVE, ACTIVE, ACTIVE, ACTIVE );
  243. for (i = 0; i < BufferSize; i++) {
  244. // dvdr
  245. ParDump2(PARINFO, ("ParEppSwWrite: Writing Byte to port\n"));
  246. WRITE_PORT_BUFFER_UCHAR( Controller,
  247. pBuffer++,
  248. (ULONG)0x01 );
  249. //
  250. // Event 62
  251. //
  252. StoreControl (Controller, HDReady);
  253. // =============== Periph State 58 ===============
  254. // Should wait up to 10 micro Seconds but waiting up
  255. // to 15 micro just in case
  256. for ( j = 16; j > 0; j-- ) {
  257. if( !(GetStatus(Controller) & DSR_NOT_BUSY) )
  258. break;
  259. KeStallExecutionProcessor(1);
  260. }
  261. // see if we timed out on state 58
  262. if ( !j ) {
  263. // Time out.
  264. // Bad things happened - timed out on this state,
  265. // Mark Status as bad and let our mgr kill current mode.
  266. Status = STATUS_IO_DEVICE_ERROR;
  267. ParDump2(PARERRORS, ("ParEppSwModeWrite:Failed State 58: Controller %x\n", Controller));
  268. Extension->CurrentPhase = PHASE_UNKNOWN;
  269. break;
  270. }
  271. //
  272. // Event 63
  273. //
  274. StoreControl (Controller, HDAck);
  275. // =============== Periph State 60 ===============
  276. // Should wait up to 125 nano Seconds but waiting up
  277. // to 5 micro seconds just in case
  278. for ( j = 6; j > 0; j-- ) {
  279. if( GetStatus(Controller) & DSR_NOT_BUSY )
  280. break;
  281. KeStallExecutionProcessor(1);
  282. }
  283. if( !j ) {
  284. // Time out.
  285. // Bad things happened - timed out on this state,
  286. // Mark Status as bad and let our mgr kill current mode.
  287. Status = STATUS_IO_DEVICE_ERROR;
  288. ParDump2(PARERRORS, ("ParEppSwModeWrite:Failed State 60: Controller %x\n", Controller));
  289. Extension->CurrentPhase = PHASE_UNKNOWN;
  290. break;
  291. }
  292. //
  293. // Event 61
  294. //
  295. StoreControl (Controller, HDFinished);
  296. // Stall a little bit between data bytes
  297. KeStallExecutionProcessor(1);
  298. }
  299. *BytesTransferred = i;
  300. // dvdr
  301. ParDump2(PARINFO, ("ParEppSwWrite: Leaving with %i Bytes Transferred\n", i));
  302. if ( Status == STATUS_SUCCESS )
  303. Extension->CurrentPhase = PHASE_FORWARD_IDLE;
  304. return Status;
  305. }
  306. NTSTATUS
  307. ParEppSwRead(
  308. IN PDEVICE_EXTENSION Extension,
  309. IN PVOID Buffer,
  310. IN ULONG BufferSize,
  311. OUT PULONG BytesTransferred
  312. )
  313. /*++
  314. Routine Description:
  315. This routine performs a 1284 EPP mode read under software control
  316. into the given buffer for no more than 'BufferSize' bytes.
  317. Arguments:
  318. Extension - Supplies the device extension.
  319. Buffer - Supplies the buffer to read into.
  320. BufferSize - Supplies the number of bytes in the buffer.
  321. BytesTransferred - Returns the number of bytes transferred.
  322. --*/
  323. {
  324. PUCHAR Controller;
  325. PUCHAR pBuffer = (PUCHAR)Buffer;
  326. NTSTATUS Status = STATUS_SUCCESS;
  327. ULONG i, j;
  328. UCHAR dcr;
  329. UCHAR HDReady, HDAck;
  330. // dvdr
  331. ParDump2(PARINFO, ("ParEppSwRead: Entering\n"));
  332. Controller = Extension->Controller;
  333. Extension->CurrentPhase = PHASE_REVERSE_XFER;
  334. // Save off Control
  335. dcr = GetControl (Controller);
  336. // BIT5 of DCR needs to be high to be in BYTE reverse mode
  337. HDReady = SET_DCR( ACTIVE, INACTIVE, ACTIVE, ACTIVE, INACTIVE, ACTIVE );
  338. HDAck = SET_DCR( ACTIVE, INACTIVE, ACTIVE, ACTIVE, ACTIVE, ACTIVE );
  339. // First time to get into reverse mode quickly
  340. StoreControl (Controller, HDReady);
  341. for (i = 0; i < BufferSize; i++) {
  342. //
  343. // Event 67
  344. //
  345. StoreControl (Controller, HDReady);
  346. // =============== Periph State 58 ===============
  347. // Should wait up to 10 micro Seconds but waiting up
  348. // to 15 micro just in case
  349. for ( j = 16; j > 0; j-- ) {
  350. if( !(GetStatus(Controller) & DSR_NOT_BUSY) )
  351. break;
  352. KeStallExecutionProcessor(1);
  353. }
  354. // see if we timed out on state 58
  355. if ( !j ) {
  356. // Time out.
  357. // Bad things happened - timed out on this state,
  358. // Mark Status as bad and let our mgr kill current mode.
  359. Status = STATUS_IO_DEVICE_ERROR;
  360. ParDump2(PARERRORS, ("ParEppSwRead:Failed State 58: Controller %x\n", Controller));
  361. Extension->CurrentPhase = PHASE_UNKNOWN;
  362. break;
  363. }
  364. // Read the Byte
  365. READ_PORT_BUFFER_UCHAR( Controller,
  366. pBuffer++,
  367. (ULONG)0x01 );
  368. //
  369. // Event 63
  370. //
  371. StoreControl (Controller, HDAck);
  372. // =============== Periph State 60 ===============
  373. // Should wait up to 125 nano Seconds but waiting up
  374. // to 5 micro seconds just in case
  375. for ( j = 6; j > 0; j-- ) {
  376. if( GetStatus(Controller) & DSR_NOT_BUSY )
  377. break;
  378. KeStallExecutionProcessor(1);
  379. }
  380. if( !j ) {
  381. // Time out.
  382. // Bad things happened - timed out on this state,
  383. // Mark Status as bad and let our mgr kill current mode.
  384. Status = STATUS_IO_DEVICE_ERROR;
  385. ParDump2(PARERRORS, ("ParEppSwRead:Failed State 60: Controller %x\n", Controller));
  386. Extension->CurrentPhase = PHASE_UNKNOWN;
  387. break;
  388. }
  389. // Stall a little bit between data bytes
  390. KeStallExecutionProcessor(1);
  391. }
  392. dcr &= ~DCR_DIRECTION;
  393. StoreControl (Controller, dcr);
  394. *BytesTransferred = i;
  395. // dvdr
  396. ParDump2(PARINFO, ("ParEppSwRead: Leaving with %x Bytes Transferred\n", i));
  397. if ( Status == STATUS_SUCCESS )
  398. Extension->CurrentPhase = PHASE_FORWARD_IDLE;
  399. return Status;
  400. }