VirtualBox

source: vbox/trunk/src/VBox/Devices/Parallel/DrvHostParallel.cpp@ 62993

Last change on this file since 62993 was 62993, checked in by vboxsync, 8 years ago

Devices: warnings

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 36.9 KB
Line 
1/* $Id: DrvHostParallel.cpp 62993 2016-08-04 15:11:32Z vboxsync $ */
2/** @file
3 * VirtualBox Host Parallel Port Driver.
4 *
5 * Initial Linux-only code contributed by: Alexander Eichner
6 */
7
8/*
9 * Copyright (C) 2006-2016 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 */
19
20
21/*********************************************************************************************************************************
22* Header Files *
23*********************************************************************************************************************************/
24#define LOG_GROUP LOG_GROUP_DRV_HOST_PARALLEL
25#include <VBox/vmm/pdmdrv.h>
26#include <VBox/vmm/pdmthread.h>
27#include <iprt/asm.h>
28#include <iprt/assert.h>
29#include <iprt/file.h>
30#include <iprt/pipe.h>
31#include <iprt/semaphore.h>
32#include <iprt/stream.h>
33#include <iprt/uuid.h>
34#include <iprt/cdefs.h>
35#include <iprt/ctype.h>
36
37#ifdef RT_OS_LINUX
38# include <sys/ioctl.h>
39# include <sys/types.h>
40# include <sys/stat.h>
41# include <sys/poll.h>
42# include <fcntl.h>
43# include <unistd.h>
44# include <linux/ppdev.h>
45# include <linux/parport.h>
46# include <errno.h>
47#endif
48
49/** @def VBOX_WITH_WIN_PARPORT_SUP *
50 * Indicates whether to use the generic direct hardware access or host specific
51 * code to access the parallel port.
52 */
53#if defined(RT_OS_LINUX)
54# undef VBOX_WITH_WIN_PARPORT_SUP
55#elif defined(RT_OS_WINDOWS)
56#else
57# error "Not ported"
58#endif
59
60#if defined(VBOX_WITH_WIN_PARPORT_SUP) && defined(IN_RING0)
61# include <iprt/asm-amd64-x86.h>
62#endif
63
64#if defined(VBOX_WITH_WIN_PARPORT_SUP) && defined(IN_RING3)
65# include <iprt/win/windows.h>
66# include <iprt/win/setupapi.h>
67# include <cfgmgr32.h>
68# include <iprt/mem.h>
69# include <iprt/ctype.h>
70# include <iprt/path.h>
71# include <iprt/string.h>
72#endif
73
74#include "VBoxDD.h"
75
76
77/*********************************************************************************************************************************
78* Structures and Typedefs *
79*********************************************************************************************************************************/
80/**
81 * Host parallel port driver instance data.
82 * @implements PDMIHOSTPARALLELCONNECTOR
83 */
84typedef struct DRVHOSTPARALLEL
85{
86 /** Pointer to the driver instance structure. */
87 PPDMDRVINS pDrvIns;
88 /** Pointer to the driver instance. */
89 PPDMDRVINSR3 pDrvInsR3;
90 PPDMDRVINSR0 pDrvInsR0;
91 /** Pointer to the char port interface of the driver/device above us. */
92 PPDMIHOSTPARALLELPORT pDrvHostParallelPort;
93 /** Our host device interface. */
94 PDMIHOSTPARALLELCONNECTOR IHostParallelConnector;
95 /** Our host device interface. */
96 PDMIHOSTPARALLELCONNECTOR IHostParallelConnectorR3;
97 /** Device Path */
98 char *pszDevicePath;
99 /** Device Handle */
100 RTFILE hFileDevice;
101#ifndef VBOX_WITH_WIN_PARPORT_SUP
102 /** Thread waiting for interrupts. */
103 PPDMTHREAD pMonitorThread;
104 /** Wakeup pipe read end. */
105 RTPIPE hWakeupPipeR;
106 /** Wakeup pipe write end. */
107 RTPIPE hWakeupPipeW;
108 /** Current mode the parallel port is in. */
109 PDMPARALLELPORTMODE enmModeCur;
110#endif
111
112#ifdef VBOX_WITH_WIN_PARPORT_SUP
113 /** Data register. */
114 RTIOPORT PortDirectData;
115 /** Status register. */
116 RTIOPORT PortDirectStatus;
117 /** Control register. */
118 RTIOPORT PortDirectControl;
119 /** Control read result buffer. */
120 uint8_t bReadInControl;
121 /** Status read result buffer. */
122 uint8_t bReadInStatus;
123 /** Data buffer for reads and writes. */
124 uint8_t abDataBuf[32];
125#endif /* VBOX_WITH_WIN_PARPORT_SUP */
126} DRVHOSTPARALLEL, *PDRVHOSTPARALLEL;
127
128
129/**
130 * Ring-0 operations.
131 */
132typedef enum DRVHOSTPARALLELR0OP
133{
134 /** Invalid zero value. */
135 DRVHOSTPARALLELR0OP_INVALID = 0,
136 /** Perform R0 initialization. */
137 DRVHOSTPARALLELR0OP_INITR0STUFF,
138 /** Read data into the data buffer (abDataBuf). */
139 DRVHOSTPARALLELR0OP_READ,
140 /** Read status register. */
141 DRVHOSTPARALLELR0OP_READSTATUS,
142 /** Read control register. */
143 DRVHOSTPARALLELR0OP_READCONTROL,
144 /** Write data from the data buffer (abDataBuf). */
145 DRVHOSTPARALLELR0OP_WRITE,
146 /** Write control register. */
147 DRVHOSTPARALLELR0OP_WRITECONTROL,
148 /** Set port direction. */
149 DRVHOSTPARALLELR0OP_SETPORTDIRECTION
150} DRVHOSTPARALLELR0OP;
151
152/** Converts a pointer to DRVHOSTPARALLEL::IHostDeviceConnector to a PDRHOSTPARALLEL. */
153#define PDMIHOSTPARALLELCONNECTOR_2_DRVHOSTPARALLEL(pInterface) ( (PDRVHOSTPARALLEL)((uintptr_t)pInterface - RT_OFFSETOF(DRVHOSTPARALLEL, CTX_SUFF(IHostParallelConnector))) )
154
155
156/*********************************************************************************************************************************
157* Defined Constants And Macros *
158*********************************************************************************************************************************/
159#define CTRL_REG_OFFSET 2
160#define STATUS_REG_OFFSET 1
161#define LPT_CONTROL_ENABLE_BIDIRECT 0x20
162
163
164
165#ifdef VBOX_WITH_WIN_PARPORT_SUP
166# ifdef IN_RING0
167
168/**
169 * R0 mode function to write byte value to data port.
170 *
171 * @returns VBox status code.
172 * @param pThis Pointer to the instance data.
173 * @param u64Arg The number of bytes to write (from abDataBuf).
174 */
175static int drvR0HostParallelReqWrite(PDRVHOSTPARALLEL pThis, uint64_t u64Arg)
176{
177 LogFlowFunc(("write %#RX64 bytes to data (%#x)\n", u64Arg, pThis->PortDirectData));
178
179 AssertReturn(u64Arg > 0 && u64Arg <= sizeof(pThis->abDataBuf), VERR_OUT_OF_RANGE);
180 uint8_t const *pbSrc = pThis->abDataBuf;
181 while (u64Arg-- > 0)
182 {
183 ASMOutU8(pThis->PortDirectData, *pbSrc);
184 pbSrc++;
185 }
186
187 return VINF_SUCCESS;
188}
189
190/**
191 * R0 mode function to write byte value to parallel port control register.
192 *
193 * @returns VBox status code.
194 * @param pThis Pointer to the instance data.
195 * @param u64Arg Data to be written to control register.
196 */
197static int drvR0HostParallelReqWriteControl(PDRVHOSTPARALLEL pThis, uint64_t u64Arg)
198{
199 LogFlowFunc(("write to ctrl port=%#x val=%#x\n", pThis->PortDirectControl, u64Arg));
200 ASMOutU8(pThis->PortDirectControl, (uint8_t)(u64Arg));
201 return VINF_SUCCESS;
202}
203
204/**
205 * R0 mode function to ready byte value from the parallel port data register.
206 *
207 * @returns VBox status code.
208 * @param pThis Pointer to the instance data.
209 * @param u64Arg The number of bytes to read into abDataBuf.
210 */
211static int drvR0HostParallelReqRead(PDRVHOSTPARALLEL pThis, uint64_t u64Arg)
212{
213 LogFlowFunc(("read %#RX64 bytes to data (%#x)\n", u64Arg, pThis->PortDirectData));
214
215 AssertReturn(u64Arg > 0 && u64Arg <= sizeof(pThis->abDataBuf), VERR_OUT_OF_RANGE);
216 uint8_t *pbDst = pThis->abDataBuf;
217 while (u64Arg-- > 0)
218 *pbDst++ = ASMInU8(pThis->PortDirectData);
219
220 return VINF_SUCCESS;
221}
222
223/**
224 * R0 mode function to ready byte value from the parallel port control register.
225 *
226 * @returns VBox status code.
227 * @param pThis Pointer to the instance data.
228 */
229static int drvR0HostParallelReqReadControl(PDRVHOSTPARALLEL pThis)
230{
231 uint8_t u8Data = ASMInU8(pThis->PortDirectControl);
232 LogFlowFunc(("read from ctrl port=%#x val=%#x\n", pThis->PortDirectControl, u8Data));
233 pThis->bReadInControl = u8Data;
234 return VINF_SUCCESS;
235}
236
237/**
238 * R0 mode function to ready byte value from the parallel port status register.
239 *
240 * @returns VBox status code.
241 * @param pThis Pointer to the instance data.
242 */
243static int drvR0HostParallelReqReadStatus(PDRVHOSTPARALLEL pThis)
244{
245 uint8_t u8Data = ASMInU8(pThis->PortDirectStatus);
246 LogFlowFunc(("read from status port=%#x val=%#x\n", pThis->PortDirectStatus, u8Data));
247 pThis->bReadInStatus = u8Data;
248 return VINF_SUCCESS;
249}
250
251/**
252 * R0 mode function to set the direction of parallel port -
253 * operate in bidirectional mode or single direction.
254 *
255 * @returns VBox status code.
256 * @param pThis Pointer to the instance data.
257 * @param u64Arg Mode.
258 */
259static int drvR0HostParallelReqSetPortDir(PDRVHOSTPARALLEL pThis, uint64_t u64Arg)
260{
261 uint8_t bCtl = ASMInU8(pThis->PortDirectControl);
262 if (u64Arg)
263 bCtl |= LPT_CONTROL_ENABLE_BIDIRECT; /* enable input direction */
264 else
265 bCtl &= ~LPT_CONTROL_ENABLE_BIDIRECT; /* disable input direction */
266 ASMOutU8(pThis->PortDirectControl, bCtl);
267
268 return VINF_SUCCESS;
269}
270
271/**
272 * @interface_method_impl{FNPDMDRVREQHANDLERR0}
273 */
274PDMBOTHCBDECL(int) drvR0HostParallelReqHandler(PPDMDRVINS pDrvIns, uint32_t uOperation, uint64_t u64Arg)
275{
276 PDRVHOSTPARALLEL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPARALLEL);
277 int rc;
278 LogFlowFuncEnter();
279
280 if (pThis->PortDirectData != 0)
281 {
282 switch ((DRVHOSTPARALLELR0OP)uOperation)
283 {
284 case DRVHOSTPARALLELR0OP_READ:
285 rc = drvR0HostParallelReqRead(pThis, u64Arg);
286 break;
287 case DRVHOSTPARALLELR0OP_READSTATUS:
288 rc = drvR0HostParallelReqReadStatus(pThis);
289 break;
290 case DRVHOSTPARALLELR0OP_READCONTROL:
291 rc = drvR0HostParallelReqReadControl(pThis);
292 break;
293 case DRVHOSTPARALLELR0OP_WRITE:
294 rc = drvR0HostParallelReqWrite(pThis, u64Arg);
295 break;
296 case DRVHOSTPARALLELR0OP_WRITECONTROL:
297 rc = drvR0HostParallelReqWriteControl(pThis, u64Arg);
298 break;
299 case DRVHOSTPARALLELR0OP_SETPORTDIRECTION:
300 rc = drvR0HostParallelReqSetPortDir(pThis, u64Arg);
301 break;
302 default:
303 rc = VERR_INVALID_FUNCTION;
304 break;
305 }
306 }
307 else
308 rc = VERR_WRONG_ORDER;
309
310 LogFlowFuncLeaveRC(rc);
311 return rc;
312}
313
314# endif /* IN_RING0 */
315#endif /* VBOX_WITH_WIN_PARPORT_SUP */
316
317#ifdef IN_RING3
318# ifdef VBOX_WITH_WIN_PARPORT_SUP
319
320/**
321 * Find IO port range for the parallel port and return the lower address.
322 *
323 * @returns Parallel base I/O port.
324 * @param DevInst Device instance (dword/handle) for the parallel port.
325 */
326static RTIOPORT drvHostParallelGetWinHostIoPortsSub(const DEVINST DevInst)
327{
328 RTIOPORT PortBase = 0;
329
330 /* Get handle of the first logical configuration. */
331 LOG_CONF hFirstLogConf;
332 CONFIGRET rcCm = CM_Get_First_Log_Conf(&hFirstLogConf, DevInst, ALLOC_LOG_CONF);
333 if (rcCm != CR_SUCCESS)
334 rcCm = CM_Get_First_Log_Conf(&hFirstLogConf, DevInst, BOOT_LOG_CONF);
335 if (rcCm == CR_SUCCESS)
336 {
337 /*
338 * This loop is based on the "fact" that only one I/O resource is assigned
339 * to the LPT port. Should there ever be multiple resources, we'll pick
340 * the last one for some silly reason.
341 */
342
343 /* Get the first resource descriptor handle. */
344 LOG_CONF hCurLogConf = 0;
345 rcCm = CM_Get_Next_Res_Des(&hCurLogConf, hFirstLogConf, ResType_IO, 0, 0);
346 if (rcCm == CR_SUCCESS)
347 {
348 for (;;)
349 {
350 ULONG cbData;
351 rcCm = CM_Get_Res_Des_Data_Size(&cbData, hCurLogConf, 0);
352 if (rcCm != CR_SUCCESS)
353 cbData = 0;
354 cbData = RT_MAX(cbData, sizeof(IO_DES));
355 IO_DES *pIoDesc = (IO_DES *)RTMemAllocZ(cbData);
356 if (pIoDesc)
357 {
358 rcCm = CM_Get_Res_Des_Data(hCurLogConf, pIoDesc, cbData, 0L);
359 if (rcCm == CR_SUCCESS)
360 {
361 LogRel(("drvHostParallelGetWinHostIoPortsSub: Count=%#u Type=%#x Base=%#RX64 End=%#RX64 Flags=%#x\n",
362 pIoDesc->IOD_Count, pIoDesc->IOD_Type, (uint64_t)pIoDesc->IOD_Alloc_Base,
363 (uint64_t)pIoDesc->IOD_Alloc_End, pIoDesc->IOD_DesFlags));
364 PortBase = (RTIOPORT)pIoDesc->IOD_Alloc_Base;
365 }
366 else
367 LogRel(("drvHostParallelGetWinHostIoPortsSub: CM_Get_Res_Des_Data(,,%u,0) failed: %u\n", cbData, rcCm));
368 RTMemFree(pIoDesc);
369 }
370 else
371 LogRel(("drvHostParallelGetWinHostIoPortsSub: failed to allocate %#x bytes\n", cbData));
372
373 /* Next */
374 RES_DES hFreeResDesc = hCurLogConf;
375 rcCm = CM_Get_Next_Res_Des(&hCurLogConf, hCurLogConf, ResType_IO, 0, 0);
376 CM_Free_Res_Des_Handle(hFreeResDesc);
377 if (rcCm != CR_SUCCESS)
378 {
379 if (rcCm != CR_NO_MORE_RES_DES)
380 LogRel(("drvHostParallelGetWinHostIoPortsSub: CM_Get_Next_Res_Des failed: %u\n", rcCm));
381 break;
382 }
383 }
384 }
385 else
386 LogRel(("drvHostParallelGetWinHostIoPortsSub: Initial CM_Get_Next_Res_Des failed: %u\n", rcCm));
387 CM_Free_Log_Conf_Handle(hFirstLogConf);
388 }
389 LogFlowFunc(("return PortBase=%#x", PortBase));
390 return PortBase;
391}
392
393/**
394 * Get Parallel port address and update the shared data structure.
395 *
396 * @returns VBox status code.
397 * @param pThis The host parallel port instance data.
398 */
399static int drvHostParallelGetWinHostIoPorts(PDRVHOSTPARALLEL pThis)
400{
401 /*
402 * Assume the host device path is on the form "\\.\PIPE\LPT1", then get the "LPT1" part.
403 */
404 const char * const pszCfgPortName = RTPathFilename(pThis->pszDevicePath);
405 AssertReturn(pszCfgPortName, VERR_INTERNAL_ERROR_3);
406 size_t const cchCfgPortName = strlen(pszCfgPortName);
407 if ( cchCfgPortName != 4
408 || RTStrNICmp(pszCfgPortName, "LPT", 3) != 0
409 || !RT_C_IS_DIGIT(pszCfgPortName[3]) )
410 {
411 LogRel(("drvHostParallelGetWinHostIoPorts: The configured device name '%s' is not on the expected 'LPTx' form!\n",
412 pszCfgPortName));
413 return VERR_INVALID_NAME;
414 }
415
416 /*
417 * Get a list of devices then enumerate it looking for the LPT port we're using.
418 */
419 HDEVINFO hDevInfo = SetupDiGetClassDevs(NULL, 0, 0, DIGCF_PRESENT | DIGCF_ALLCLASSES);
420 if (hDevInfo == INVALID_HANDLE_VALUE)
421 {
422 DWORD dwErr = GetLastError();
423 LogRel(("drvHostParallelGetWinHostIoPorts: SetupDiGetClassDevs failed: %u\n", dwErr));
424 return RTErrConvertFromWin32(dwErr);
425 }
426
427 int rc = VINF_SUCCESS;
428 char *pszBuf = NULL;
429 DWORD cbBuf = 0;
430 for (int32_t idxDevInfo = 0;; idxDevInfo++)
431 {
432 /*
433 * Query the next device info.
434 */
435 SP_DEVINFO_DATA DeviceInfoData;
436 DeviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA);
437 if (!SetupDiEnumDeviceInfo(hDevInfo, idxDevInfo, &DeviceInfoData))
438 {
439 DWORD dwErr = GetLastError();
440 if (dwErr != ERROR_NO_MORE_ITEMS && dwErr != NO_ERROR)
441 {
442 LogRel(("drvHostParallelGetWinHostIoPorts: SetupDiEnumDeviceInfo failed: %u\n", dwErr));
443 rc = RTErrConvertFromWin32(dwErr);
444 }
445 break;
446 }
447
448 /* Get the friendly name of the device. */
449 DWORD dwDataType;
450 DWORD cbBufActual;
451 BOOL fOk;
452 while (!(fOk = SetupDiGetDeviceRegistryProperty(hDevInfo, &DeviceInfoData, SPDRP_FRIENDLYNAME,
453 &dwDataType, (PBYTE)pszBuf, cbBuf, &cbBufActual)))
454 {
455 DWORD dwErr = GetLastError();
456 if (dwErr == ERROR_INSUFFICIENT_BUFFER)
457 {
458 LogFlow(("ERROR_INSUFF_BUFF = %d. dwBufSz = %d\n", dwErr, cbBuf));
459 void *pvNew = RTMemRealloc(pszBuf, RT_MAX(RT_ALIGN_Z(cbBufActual + 16, 64), 256));
460 if (pvNew)
461 pszBuf = (char *)pvNew;
462 else
463 {
464 LogFlow(("GetDevProp Error = %d & cbBufActual = %d\n", dwErr, cbBufActual));
465 break;
466 }
467 }
468 else
469 {
470 /* No need to bother about this error (in most cases its errno=13,
471 * INVALID_DATA . Just break from here and proceed to next device
472 * enumerated item
473 */
474 LogFlow(("GetDevProp Error = %d & cbBufActual = %d\n", dwErr, cbBufActual));
475 break;
476 }
477 }
478 if ( fOk
479 && pszBuf)
480 {
481 pszBuf[cbBuf - 1] = '\0';
482
483 /*
484 * Does this look like the port name we're looking for.
485 *
486 * We're expecting either "Parallel Port (LPT1)" or just "LPT1", though we'll make do
487 * with anything that includes the name we're looking for as a separate word.
488 */
489 char *pszMatch;
490 do
491 pszMatch = RTStrIStr(pszBuf, pszCfgPortName);
492 while ( pszMatch != NULL
493 && !( ( pszMatch == pszBuf
494 || pszMatch[-1] == '('
495 || RT_C_IS_BLANK(pszMatch[-1]))
496 && ( pszMatch[cchCfgPortName] == '\0'
497 || pszMatch[cchCfgPortName] == ')'
498 || RT_C_IS_BLANK(pszMatch[cchCfgPortName])) ) );
499 if (pszMatch != NULL)
500 {
501 RTIOPORT Port = drvHostParallelGetWinHostIoPortsSub(DeviceInfoData.DevInst);
502 if (Port != 0)
503 {
504 pThis->PortDirectData = Port;
505 pThis->PortDirectControl = Port + CTRL_REG_OFFSET;
506 pThis->PortDirectStatus = Port + STATUS_REG_OFFSET;
507 break;
508 }
509 LogRel(("drvHostParallelGetWinHostIoPorts: Addr not found for '%s'\n", pszBuf));
510 }
511 }
512 }
513
514 /* Cleanup. */
515 RTMemFree(pszBuf);
516 SetupDiDestroyDeviceInfoList(hDevInfo);
517 return rc;
518
519}
520# endif /* VBOX_WITH_WIN_PARPORT_SUP */
521
522/**
523 * Changes the current mode of the host parallel port.
524 *
525 * @returns VBox status code.
526 * @param pThis The host parallel port instance data.
527 * @param enmMode The mode to change the port to.
528 */
529static int drvHostParallelSetMode(PDRVHOSTPARALLEL pThis, PDMPARALLELPORTMODE enmMode)
530{
531 LogFlowFunc(("mode=%d\n", enmMode));
532# ifndef VBOX_WITH_WIN_PARPORT_SUP
533 int rc = VINF_SUCCESS;
534 int iMode = 0;
535 int rcLnx;
536 if (pThis->enmModeCur != enmMode)
537 {
538 switch (enmMode)
539 {
540 case PDM_PARALLEL_PORT_MODE_SPP:
541 iMode = IEEE1284_MODE_COMPAT;
542 break;
543 case PDM_PARALLEL_PORT_MODE_EPP_DATA:
544 iMode = IEEE1284_MODE_EPP | IEEE1284_DATA;
545 break;
546 case PDM_PARALLEL_PORT_MODE_EPP_ADDR:
547 iMode = IEEE1284_MODE_EPP | IEEE1284_ADDR;
548 break;
549 case PDM_PARALLEL_PORT_MODE_ECP:
550 case PDM_PARALLEL_PORT_MODE_INVALID:
551 default:
552 return VERR_NOT_SUPPORTED;
553 }
554
555 rcLnx = ioctl(RTFileToNative(pThis->hFileDevice), PPSETMODE, &iMode);
556 if (RT_UNLIKELY(rcLnx < 0))
557 rc = RTErrConvertFromErrno(errno);
558 else
559 pThis->enmModeCur = enmMode;
560 }
561
562 return rc;
563# else /* VBOX_WITH_WIN_PARPORT_SUP */
564 RT_NOREF(pThis, enmMode);
565 return VINF_SUCCESS;
566# endif /* VBOX_WITH_WIN_PARPORT_SUP */
567}
568
569/* -=-=-=-=- IBase -=-=-=-=- */
570
571/**
572 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
573 */
574static DECLCALLBACK(void *) drvHostParallelQueryInterface(PPDMIBASE pInterface, const char *pszIID)
575{
576 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
577 PDRVHOSTPARALLEL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPARALLEL);
578
579 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
580 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIHOSTPARALLELCONNECTOR, &pThis->CTX_SUFF(IHostParallelConnector));
581 return NULL;
582}
583
584
585/* -=-=-=-=- IHostDeviceConnector -=-=-=-=- */
586
587/**
588 * @interface_method_impl{PDMIHOSTPARALLELCONNECTOR,pfnWrite}
589 */
590static DECLCALLBACK(int)
591drvHostParallelWrite(PPDMIHOSTPARALLELCONNECTOR pInterface, const void *pvBuf, size_t cbWrite, PDMPARALLELPORTMODE enmMode)
592{
593 PDRVHOSTPARALLEL pThis = RT_FROM_MEMBER(pInterface, DRVHOSTPARALLEL, CTX_SUFF(IHostParallelConnector));
594 int rc = VINF_SUCCESS;
595
596 LogFlowFunc(("pvBuf=%#p cbWrite=%d\n", pvBuf, cbWrite));
597
598 rc = drvHostParallelSetMode(pThis, enmMode);
599 if (RT_FAILURE(rc))
600 return rc;
601# ifndef VBOX_WITH_WIN_PARPORT_SUP
602 int rcLnx = 0;
603 if (enmMode == PDM_PARALLEL_PORT_MODE_SPP)
604 {
605 /* Set the data lines directly. */
606 rcLnx = ioctl(RTFileToNative(pThis->hFileDevice), PPWDATA, pvBuf);
607 }
608 else
609 {
610 /* Use write interface. */
611 rcLnx = write(RTFileToNative(pThis->hFileDevice), pvBuf, cbWrite);
612 }
613 if (RT_UNLIKELY(rcLnx < 0))
614 rc = RTErrConvertFromErrno(errno);
615
616# else /* VBOX_WITH_WIN_PARPORT_SUP */
617 if (pThis->PortDirectData != 0)
618 {
619 while (cbWrite > 0)
620 {
621 size_t cbToWrite = RT_MIN(cbWrite, sizeof(pThis->abDataBuf));
622 LogFlowFunc(("Calling R0 to write %#zu bytes of data\n", cbToWrite));
623 memcpy(pThis->abDataBuf, pvBuf, cbToWrite);
624 rc = PDMDrvHlpCallR0(pThis->CTX_SUFF(pDrvIns), DRVHOSTPARALLELR0OP_WRITE, cbToWrite);
625 AssertRC(rc);
626 pvBuf = (uint8_t const *)pvBuf + cbToWrite;
627 cbWrite -= cbToWrite;
628 }
629 }
630# endif /* VBOX_WITH_WIN_PARPORT_SUP */
631 return rc;
632}
633
634/**
635 * @interface_method_impl{PDMIHOSTPARALLELCONNECTOR,pfnRead}
636 */
637static DECLCALLBACK(int)
638drvHostParallelRead(PPDMIHOSTPARALLELCONNECTOR pInterface, void *pvBuf, size_t cbRead, PDMPARALLELPORTMODE enmMode)
639{
640 PDRVHOSTPARALLEL pThis = RT_FROM_MEMBER(pInterface, DRVHOSTPARALLEL, CTX_SUFF(IHostParallelConnector));
641 int rc = VINF_SUCCESS;
642
643# ifndef VBOX_WITH_WIN_PARPORT_SUP
644 int rcLnx = 0;
645 LogFlowFunc(("pvBuf=%#p cbRead=%d\n", pvBuf, cbRead));
646
647 rc = drvHostParallelSetMode(pThis, enmMode);
648 if (RT_FAILURE(rc))
649 return rc;
650
651 if (enmMode == PDM_PARALLEL_PORT_MODE_SPP)
652 {
653 /* Set the data lines directly. */
654 rcLnx = ioctl(RTFileToNative(pThis->hFileDevice), PPWDATA, pvBuf);
655 }
656 else
657 {
658 /* Use write interface. */
659 rcLnx = read(RTFileToNative(pThis->hFileDevice), pvBuf, cbRead);
660 }
661 if (RT_UNLIKELY(rcLnx < 0))
662 rc = RTErrConvertFromErrno(errno);
663
664# else /* VBOX_WITH_WIN_PARPORT_SUP */
665 RT_NOREF(enmMode);
666 if (pThis->PortDirectData != 0)
667 {
668 while (cbRead > 0)
669 {
670 size_t cbToRead = RT_MIN(cbRead, sizeof(pThis->abDataBuf));
671 LogFlowFunc(("Calling R0 to read %#zu bytes of data\n", cbToRead));
672 memset(pThis->abDataBuf, 0, cbToRead);
673 rc = PDMDrvHlpCallR0(pThis->CTX_SUFF(pDrvIns), DRVHOSTPARALLELR0OP_READ, cbToRead);
674 AssertRC(rc);
675 memcpy(pvBuf, pThis->abDataBuf, cbToRead);
676 pvBuf = (uint8_t *)pvBuf + cbToRead;
677 cbRead -= cbToRead;
678 }
679 }
680# endif /* VBOX_WITH_WIN_PARPORT_SUP */
681 return rc;
682}
683
684/**
685 * @interface_method_impl{PDMIHOSTPARALLELCONNECTOR,pfnSetPortDirection}
686 */
687static DECLCALLBACK(int) drvHostParallelSetPortDirection(PPDMIHOSTPARALLELCONNECTOR pInterface, bool fForward)
688{
689 PDRVHOSTPARALLEL pThis = RT_FROM_MEMBER(pInterface, DRVHOSTPARALLEL, CTX_SUFF(IHostParallelConnector));
690 int rc = VINF_SUCCESS;
691 int iMode = 0;
692 if (!fForward)
693 iMode = 1;
694# ifndef VBOX_WITH_WIN_PARPORT_SUP
695 int rcLnx = ioctl(RTFileToNative(pThis->hFileDevice), PPDATADIR, &iMode);
696 if (RT_UNLIKELY(rcLnx < 0))
697 rc = RTErrConvertFromErrno(errno);
698
699# else /* VBOX_WITH_WIN_PARPORT_SUP */
700 if (pThis->PortDirectData != 0)
701 {
702 LogFlowFunc(("calling R0 to write CTRL, data=%#x\n", iMode));
703 rc = PDMDrvHlpCallR0(pThis->CTX_SUFF(pDrvIns), DRVHOSTPARALLELR0OP_SETPORTDIRECTION, iMode);
704 AssertRC(rc);
705 }
706# endif /* VBOX_WITH_WIN_PARPORT_SUP */
707 return rc;
708}
709
710/**
711 * @interface_method_impl{PDMIHOSTPARALLELCONNECTOR,pfnWriteControl}
712 */
713static DECLCALLBACK(int) drvHostParallelWriteControl(PPDMIHOSTPARALLELCONNECTOR pInterface, uint8_t fReg)
714{
715 PDRVHOSTPARALLEL pThis = RT_FROM_MEMBER(pInterface, DRVHOSTPARALLEL, CTX_SUFF(IHostParallelConnector));
716 int rc = VINF_SUCCESS;
717
718 LogFlowFunc(("fReg=%#x\n", fReg));
719# ifndef VBOX_WITH_WIN_PARPORT_SUP
720 int rcLnx = ioctl(RTFileToNative(pThis->hFileDevice), PPWCONTROL, &fReg);
721 if (RT_UNLIKELY(rcLnx < 0))
722 rc = RTErrConvertFromErrno(errno);
723# else /* VBOX_WITH_WIN_PARPORT_SUP */
724 if (pThis->PortDirectData != 0)
725 {
726 LogFlowFunc(("calling R0 to write CTRL, data=%#x\n", fReg));
727 rc = PDMDrvHlpCallR0(pThis->CTX_SUFF(pDrvIns), DRVHOSTPARALLELR0OP_WRITECONTROL, fReg);
728 AssertRC(rc);
729 }
730# endif /* VBOX_WITH_WIN_PARPORT_SUP */
731 return rc;
732}
733
734
735/**
736 * @interface_method_impl{PDMIHOSTPARALLELCONNECTOR,pfnReadControl}
737 */
738static DECLCALLBACK(int) drvHostParallelReadControl(PPDMIHOSTPARALLELCONNECTOR pInterface, uint8_t *pfReg)
739{
740 PDRVHOSTPARALLEL pThis = RT_FROM_MEMBER(pInterface, DRVHOSTPARALLEL, CTX_SUFF(IHostParallelConnector));
741 int rc = VINF_SUCCESS;
742
743# ifndef VBOX_WITH_WIN_PARPORT_SUP
744 uint8_t fReg = 0;
745 int rcLnx = ioctl(RTFileToNative(pThis->hFileDevice), PPRCONTROL, &fReg);
746 if (RT_UNLIKELY(rcLnx < 0))
747 rc = RTErrConvertFromErrno(errno);
748 else
749 {
750 LogFlowFunc(("fReg=%#x\n", fReg));
751 *pfReg = fReg;
752 }
753# else /* VBOX_WITH_WIN_PARPORT_SUP */
754 *pfReg = 0; /* Initialize the buffer*/
755 if (pThis->PortDirectData != 0)
756 {
757 LogFlowFunc(("calling R0 to read control from parallel port\n"));
758 rc = PDMDrvHlpCallR0(pThis-> CTX_SUFF(pDrvIns), DRVHOSTPARALLELR0OP_READCONTROL, 0);
759 AssertRC(rc);
760 *pfReg = pThis->bReadInControl;
761 }
762# endif /* VBOX_WITH_WIN_PARPORT_SUP */
763 return rc;
764}
765
766/**
767 * @interface_method_impl{PDMIHOSTPARALLELCONNECTOR,pfnReadStatus}
768 */
769static DECLCALLBACK(int) drvHostParallelReadStatus(PPDMIHOSTPARALLELCONNECTOR pInterface, uint8_t *pfReg)
770{
771 PDRVHOSTPARALLEL pThis = RT_FROM_MEMBER(pInterface, DRVHOSTPARALLEL, CTX_SUFF(IHostParallelConnector));
772 int rc = VINF_SUCCESS;
773# ifndef VBOX_WITH_WIN_PARPORT_SUP
774 uint8_t fReg = 0;
775 int rcLnx = ioctl(RTFileToNative(pThis->hFileDevice), PPRSTATUS, &fReg);
776 if (RT_UNLIKELY(rcLnx < 0))
777 rc = RTErrConvertFromErrno(errno);
778 else
779 {
780 LogFlowFunc(("fReg=%#x\n", fReg));
781 *pfReg = fReg;
782 }
783# else /* VBOX_WITH_WIN_PARPORT_SUP */
784 *pfReg = 0; /* Intialize the buffer. */
785 if (pThis->PortDirectData != 0)
786 {
787 LogFlowFunc(("calling R0 to read status from parallel port\n"));
788 rc = PDMDrvHlpCallR0(pThis->CTX_SUFF(pDrvIns), DRVHOSTPARALLELR0OP_READSTATUS, 0);
789 AssertRC(rc);
790 *pfReg = pThis->bReadInStatus;
791 }
792# endif /* VBOX_WITH_WIN_PARPORT_SUP */
793 return rc;
794}
795
796# ifndef VBOX_WITH_WIN_PARPORT_SUP
797
798static DECLCALLBACK(int) drvHostParallelMonitorThread(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
799{
800 PDRVHOSTPARALLEL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPARALLEL);
801 struct pollfd aFDs[2];
802
803 /*
804 * We can wait for interrupts using poll on linux hosts.
805 */
806 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
807 {
808 int rc;
809
810 aFDs[0].fd = RTFileToNative(pThis->hFileDevice);
811 aFDs[0].events = POLLIN;
812 aFDs[0].revents = 0;
813 aFDs[1].fd = RTPipeToNative(pThis->hWakeupPipeR);
814 aFDs[1].events = POLLIN | POLLERR | POLLHUP;
815 aFDs[1].revents = 0;
816 rc = poll(aFDs, RT_ELEMENTS(aFDs), -1);
817 if (rc < 0)
818 {
819 AssertMsgFailed(("poll failed with rc=%d\n", RTErrConvertFromErrno(errno)));
820 return RTErrConvertFromErrno(errno);
821 }
822
823 if (pThread->enmState != PDMTHREADSTATE_RUNNING)
824 break;
825 if (rc > 0 && aFDs[1].revents)
826 {
827 if (aFDs[1].revents & (POLLHUP | POLLERR | POLLNVAL))
828 break;
829 /* notification to terminate -- drain the pipe */
830 char ch;
831 size_t cbRead;
832 RTPipeRead(pThis->hWakeupPipeR, &ch, 1, &cbRead);
833 continue;
834 }
835
836 /* Interrupt occurred. */
837 rc = pThis->pDrvHostParallelPort->pfnNotifyInterrupt(pThis->pDrvHostParallelPort);
838 AssertRC(rc);
839 }
840
841 return VINF_SUCCESS;
842}
843
844/**
845 * Unblock the monitor thread so it can respond to a state change.
846 *
847 * @returns a VBox status code.
848 * @param pDrvIns The driver instance.
849 * @param pThread The send thread.
850 */
851static DECLCALLBACK(int) drvHostParallelWakeupMonitorThread(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
852{
853 PDRVHOSTPARALLEL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPARALLEL);
854 size_t cbIgnored;
855 return RTPipeWrite(pThis->hWakeupPipeW, "", 1, &cbIgnored);
856}
857
858# endif /* VBOX_WITH_WIN_PARPORT_SUP */
859
860/**
861 * Destruct a host parallel driver instance.
862 *
863 * Most VM resources are freed by the VM. This callback is provided so that
864 * any non-VM resources can be freed correctly.
865 *
866 * @param pDrvIns The driver instance data.
867 */
868static DECLCALLBACK(void) drvHostParallelDestruct(PPDMDRVINS pDrvIns)
869{
870 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
871 LogFlowFunc(("iInstance=%d\n", pDrvIns->iInstance));
872
873#ifndef VBOX_WITH_WIN_PARPORT_SUP
874 PDRVHOSTPARALLEL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPARALLEL);
875 if (pThis->hFileDevice != NIL_RTFILE)
876 ioctl(RTFileToNative(pThis->hFileDevice), PPRELEASE);
877
878 if (pThis->hWakeupPipeW != NIL_RTPIPE)
879 {
880 int rc = RTPipeClose(pThis->hWakeupPipeW); AssertRC(rc);
881 pThis->hWakeupPipeW = NIL_RTPIPE;
882 }
883
884 if (pThis->hWakeupPipeR != NIL_RTPIPE)
885 {
886 int rc = RTPipeClose(pThis->hWakeupPipeR); AssertRC(rc);
887 pThis->hWakeupPipeR = NIL_RTPIPE;
888 }
889
890 if (pThis->hFileDevice != NIL_RTFILE)
891 {
892 int rc = RTFileClose(pThis->hFileDevice); AssertRC(rc);
893 pThis->hFileDevice = NIL_RTFILE;
894 }
895
896 if (pThis->pszDevicePath)
897 {
898 MMR3HeapFree(pThis->pszDevicePath);
899 pThis->pszDevicePath = NULL;
900 }
901#endif /* !VBOX_WITH_WIN_PARPORT_SUP */
902}
903
904/**
905 * Construct a host parallel driver instance.
906 *
907 * @copydoc FNPDMDRVCONSTRUCT
908 */
909static DECLCALLBACK(int) drvHostParallelConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
910{
911 RT_NOREF(fFlags);
912 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
913 LogFlowFunc(("iInstance=%d\n", pDrvIns->iInstance));
914 PDRVHOSTPARALLEL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTPARALLEL);
915
916
917 /*
918 * Init basic data members and interfaces.
919 *
920 * Must be done before returning any failure because we've got a destructor.
921 */
922 pThis->hFileDevice = NIL_RTFILE;
923#ifndef VBOX_WITH_WIN_PARPORT_SUP
924 pThis->hWakeupPipeR = NIL_RTPIPE;
925 pThis->hWakeupPipeW = NIL_RTPIPE;
926#endif
927
928 pThis->pDrvInsR3 = pDrvIns;
929#ifdef VBOX_WITH_DRVINTNET_IN_R0
930 pThis->pDrvInsR0 = PDMDRVINS_2_R0PTR(pDrvIns);
931#endif
932
933 /* IBase. */
934 pDrvIns->IBase.pfnQueryInterface = drvHostParallelQueryInterface;
935 /* IHostParallelConnector. */
936 pThis->IHostParallelConnectorR3.pfnWrite = drvHostParallelWrite;
937 pThis->IHostParallelConnectorR3.pfnRead = drvHostParallelRead;
938 pThis->IHostParallelConnectorR3.pfnSetPortDirection = drvHostParallelSetPortDirection;
939 pThis->IHostParallelConnectorR3.pfnWriteControl = drvHostParallelWriteControl;
940 pThis->IHostParallelConnectorR3.pfnReadControl = drvHostParallelReadControl;
941 pThis->IHostParallelConnectorR3.pfnReadStatus = drvHostParallelReadStatus;
942
943 /*
944 * Validate the config.
945 */
946 if (!CFGMR3AreValuesValid(pCfg, "DevicePath\0"))
947 return PDMDRV_SET_ERROR(pDrvIns, VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES,
948 N_("Unknown host parallel configuration option, only supports DevicePath"));
949
950 /*
951 * Query configuration.
952 */
953 /* Device */
954 int rc = CFGMR3QueryStringAlloc(pCfg, "DevicePath", &pThis->pszDevicePath);
955 if (RT_FAILURE(rc))
956 {
957 AssertMsgFailed(("Configuration error: query for \"DevicePath\" string returned %Rra.\n", rc));
958 return rc;
959 }
960
961 /*
962 * Open the device
963 */
964 /** @todo exclusive access on windows? */
965 rc = RTFileOpen(&pThis->hFileDevice, pThis->pszDevicePath, RTFILE_O_READWRITE | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
966 if (RT_FAILURE(rc))
967 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS, N_("Parallel#%d could not open '%s'"),
968 pDrvIns->iInstance, pThis->pszDevicePath);
969
970#ifndef VBOX_WITH_WIN_PARPORT_SUP
971 /*
972 * Try to get exclusive access to parallel port
973 */
974 rc = ioctl(RTFileToNative(pThis->hFileDevice), PPEXCL);
975 if (rc < 0)
976 return PDMDrvHlpVMSetError(pDrvIns, RTErrConvertFromErrno(errno), RT_SRC_POS,
977 N_("Parallel#%d could not get exclusive access for parallel port '%s'"
978 "Be sure that no other process or driver accesses this port"),
979 pDrvIns->iInstance, pThis->pszDevicePath);
980
981 /*
982 * Claim the parallel port
983 */
984 rc = ioctl(RTFileToNative(pThis->hFileDevice), PPCLAIM);
985 if (rc < 0)
986 return PDMDrvHlpVMSetError(pDrvIns, RTErrConvertFromErrno(errno), RT_SRC_POS,
987 N_("Parallel#%d could not claim parallel port '%s'"
988 "Be sure that no other process or driver accesses this port"),
989 pDrvIns->iInstance, pThis->pszDevicePath);
990
991 /*
992 * Get the IHostParallelPort interface of the above driver/device.
993 */
994 pThis->pDrvHostParallelPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIHOSTPARALLELPORT);
995 if (!pThis->pDrvHostParallelPort)
996 return PDMDrvHlpVMSetError(pDrvIns, VERR_PDM_MISSING_INTERFACE_ABOVE, RT_SRC_POS, N_("Parallel#%d has no parallel port interface above"),
997 pDrvIns->iInstance);
998
999 /*
1000 * Create wakeup pipe.
1001 */
1002 rc = RTPipeCreate(&pThis->hWakeupPipeR, &pThis->hWakeupPipeW, 0 /*fFlags*/);
1003 AssertRCReturn(rc, rc);
1004
1005 /*
1006 * Start in SPP mode.
1007 */
1008 pThis->enmModeCur = PDM_PARALLEL_PORT_MODE_INVALID;
1009 rc = drvHostParallelSetMode(pThis, PDM_PARALLEL_PORT_MODE_SPP);
1010 if (RT_FAILURE(rc))
1011 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS, N_("HostParallel#%d cannot change mode of parallel mode to SPP"), pDrvIns->iInstance);
1012
1013 /*
1014 * Start waiting for interrupts.
1015 */
1016 rc = PDMDrvHlpThreadCreate(pDrvIns, &pThis->pMonitorThread, pThis, drvHostParallelMonitorThread, drvHostParallelWakeupMonitorThread, 0,
1017 RTTHREADTYPE_IO, "ParMon");
1018 if (RT_FAILURE(rc))
1019 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS, N_("HostParallel#%d cannot create monitor thread"), pDrvIns->iInstance);
1020
1021#else /* VBOX_WITH_WIN_PARPORT_SUP */
1022
1023 pThis->PortDirectData = 0;
1024 pThis->PortDirectControl = 0;
1025 pThis->PortDirectStatus = 0;
1026 rc = drvHostParallelGetWinHostIoPorts(pThis);
1027 if (RT_FAILURE(rc))
1028 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1029 N_("HostParallel#%d: Could not get direct access to the host parallel port!! (rc=%Rrc)"),
1030 pDrvIns->iInstance, rc);
1031
1032#endif /* VBOX_WITH_WIN_PARPORT_SUP */
1033 return VINF_SUCCESS;
1034}
1035
1036
1037/**
1038 * Char driver registration record.
1039 */
1040const PDMDRVREG g_DrvHostParallel =
1041{
1042 /* u32Version */
1043 PDM_DRVREG_VERSION,
1044 /* szName */
1045 "HostParallel",
1046 /* szRCMod */
1047 "",
1048 /* szR0Mod */
1049 "VBoxDDR0.r0",
1050 /* pszDescription */
1051 "Parallel host driver.",
1052 /* fFlags */
1053# if defined(VBOX_WITH_WIN_PARPORT_SUP)
1054 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DRVREG_FLAGS_R0,
1055# else
1056 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1057# endif
1058 /* fClass. */
1059 PDM_DRVREG_CLASS_CHAR,
1060 /* cMaxInstances */
1061 ~0U,
1062 /* cbInstance */
1063 sizeof(DRVHOSTPARALLEL),
1064 /* pfnConstruct */
1065 drvHostParallelConstruct,
1066 /* pfnDestruct */
1067 drvHostParallelDestruct,
1068 /* pfnRelocate */
1069 NULL,
1070 /* pfnIOCtl */
1071 NULL,
1072 /* pfnPowerOn */
1073 NULL,
1074 /* pfnReset */
1075 NULL,
1076 /* pfnSuspend */
1077 NULL,
1078 /* pfnResume */
1079 NULL,
1080 /* pfnAttach */
1081 NULL,
1082 /* pfnDetach */
1083 NULL,
1084 /* pfnPowerOff */
1085 NULL,
1086 /* pfnSoftReset */
1087 NULL,
1088 /* u32EndVersion */
1089 PDM_DRVREG_VERSION
1090};
1091#endif /*IN_RING3*/
1092
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette