VirtualBox

source: vbox/trunk/src/VBox/Devices/Parallel/DevParallel.cpp@ 10343

Last change on this file since 10343 was 9212, checked in by vboxsync, 17 years ago

Major changes for sizeof(RTGCPTR) == uint64_t.
Introduced RCPTRTYPE for pointers valid in raw mode only (RTGCPTR32).

Disabled by default. Enable by adding VBOX_WITH_64_BITS_GUESTS to your LocalConfig.kmk.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 28.6 KB
Line 
1/* $Id: DevParallel.cpp 9212 2008-05-29 09:38:38Z vboxsync $ */
2/** @file
3 * VirtualBox Parallel Device Emulation.
4 *
5 * Contributed by: Alexander Eichner
6 */
7
8/*
9 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
20 * Clara, CA 95054 USA or visit http://www.sun.com if you need
21 * additional information or have any questions.
22 */
23
24/* based on DevSerial.cpp */
25
26/*******************************************************************************
27* Header Files *
28*******************************************************************************/
29#define LOG_GROUP LOG_GROUP_DEV_PARALLEL
30#include <VBox/pdmdev.h>
31#include <iprt/assert.h>
32#include <iprt/uuid.h>
33#include <iprt/string.h>
34#include <iprt/semaphore.h>
35#include <iprt/critsect.h>
36
37#include "Builtins.h"
38
39#define PARALLEL_SAVED_STATE_VERSION 1
40
41/* defines for accessing the register bits */
42#define LPT_STATUS_BUSY 0x80
43#define LPT_STATUS_ACK 0x40
44#define LPT_STATUS_PAPER_OUT 0x20
45#define LPT_STATUS_SELECT_IN 0x10
46#define LPT_STATUS_ERROR 0x08
47#define LPT_STATUS_IRQ 0x04
48#define LPT_STATUS_BIT1 0x02 /* reserved (only for completeness) */
49#define LPT_STATUS_EPP_TIMEOUT 0x01
50
51#define LPT_CONTROL_BIT7 0x80 /* reserved (only for completeness) */
52#define LPT_CONTROL_BIT6 0x40 /* reserved (only for completeness) */
53#define LPT_CONTROL_ENABLE_BIDIRECT 0x20
54#define LPT_CONTROL_ENABLE_IRQ_VIA_ACK 0x10
55#define LPT_CONTROL_SELECT_PRINTER 0x08
56#define LPT_CONTROL_RESET 0x04
57#define LPT_CONTROL_AUTO_LINEFEED 0x02
58#define LPT_CONTROL_STROBE 0x01
59
60/** mode defines for the extended control register */
61#define LPT_ECP_ECR_CHIPMODE_MASK 0xe0
62#define LPT_ECP_ECR_CHIPMODE_GET_BITS(reg) ((reg) >> 5)
63#define LPT_ECP_ECR_CHIPMODE_SET_BITS(val) ((val) << 5)
64#define LPT_ECP_ECR_CHIPMODE_CONFIGURATION 0x07
65#define LPT_ECP_ECR_CHIPMODE_FIFO_TEST 0x06
66#define LPT_ECP_ECR_CHIPMODE_RESERVED 0x05
67#define LPT_ECP_ECR_CHIPMODE_EPP 0x04
68#define LPT_ECP_ECR_CHIPMODE_ECP_FIFO 0x03
69#define LPT_ECP_ECR_CHIPMODE_PP_FIFO 0x02
70#define LPT_ECP_ECR_CHIPMODE_BYTE 0x01
71#define LPT_ECP_ECR_CHIPMODE_COMPAT 0x00
72
73/** FIFO status bits in extended control register */
74#define LPT_ECP_ECR_FIFO_MASK 0x03
75#define LPT_ECP_ECR_FIFO_SOME_DATA 0x00
76#define LPT_ECP_ECR_FIFO_FULL 0x02
77#define LPT_ECP_ECR_FIFO_EMPTY 0x01
78
79#define LPT_ECP_CONFIGA_FIFO_WITDH_MASK 0x70
80#define LPT_ECP_CONFIGA_FIFO_WIDTH_GET_BITS(reg) ((reg) >> 4)
81#define LPT_ECP_CONFIGA_FIFO_WIDTH_SET_BITS(val) ((val) << 4)
82#define LPT_ECP_CONFIGA_FIFO_WIDTH_16 0x00
83#define LPT_ECP_CONFIGA_FIFO_WIDTH_32 0x20
84#define LPT_ECP_CONFIGA_FIFO_WIDTH_8 0x10
85
86#define LPT_ECP_FIFO_DEPTH 2
87
88
89typedef struct ParallelState
90{
91 /** Access critical section. */
92 PDMCRITSECT CritSect;
93
94 /** Pointer to the device instance. */
95 R3R0PTRTYPE(PPDMDEVINS) pDevInsHC;
96 /** Pointer to the device instance. */
97 RCPTRTYPE(PPDMDEVINS) pDevInsGC;
98#if HC_ARCH_BITS == 64 && GC_ARCH_BITS != 64
99 RTGCPTR Alignment0;
100#endif
101 /** The base interface. */
102 R3PTRTYPE(PDMIBASE) IBase;
103 /** The host device port interface. */
104 R3PTRTYPE(PDMIHOSTPARALLELPORT) IHostParallelPort;
105 /** Pointer to the attached base driver. */
106 R3PTRTYPE(PPDMIBASE) pDrvBase;
107 /** Pointer to the attached host device. */
108 R3PTRTYPE(PPDMIHOSTPARALLELCONNECTOR) pDrvHostParallelConnector;
109
110 uint8_t reg_data;
111 uint8_t reg_status;
112 uint8_t reg_control;
113 uint8_t reg_epp_addr;
114 uint8_t reg_epp_data;
115 uint8_t reg_ecp_ecr;
116 uint8_t reg_ecp_base_plus_400h; /* has different meanings */
117 uint8_t reg_ecp_config_b;
118
119 /** The ECP FIFO implementation*/
120 uint8_t ecp_fifo[LPT_ECP_FIFO_DEPTH];
121 int act_fifo_pos_write;
122 int act_fifo_pos_read;
123
124 int irq;
125 uint8_t epp_timeout;
126
127 bool fGCEnabled;
128 bool fR0Enabled;
129 bool afAlignment[6];
130
131 RTSEMEVENT ReceiveSem;
132 uint32_t base;
133
134} DEVPARALLELSTATE, *PDEVPARALLELSTATE;
135typedef DEVPARALLELSTATE ParallelState;
136
137#ifndef VBOX_DEVICE_STRUCT_TESTCASE
138
139#define PDMIHOSTPARALLELPORT_2_PARALLELSTATE(pInstance) ( (ParallelState *)((uintptr_t)(pInterface) - RT_OFFSETOF(ParallelState, IHostParallelPort)) )
140#define PDMIHOSTDEVICEPORT_2_PARALLELSTATE(pInstance) ( (ParallelState *)((uintptr_t)(pInterface) - RT_OFFSETOF(ParallelState, IHostDevicePort)) )
141#define PDMIBASE_2_PARALLELSTATE(pInstance) ( (ParallelState *)((uintptr_t)(pInterface) - RT_OFFSETOF(ParallelState, IBase)) )
142
143__BEGIN_DECLS
144PDMBOTHCBDECL(int) parallelIOPortRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
145PDMBOTHCBDECL(int) parallelIOPortWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
146#if 0
147PDMBOTHCBDECL(int) parallelIOPortReadECP(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
148PDMBOTHCBDECL(int) parallelIOPortWriteECP(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
149#endif
150__END_DECLS
151
152#ifdef IN_RING3
153static void parallel_set_irq(ParallelState *s)
154{
155 if (s->reg_control & LPT_CONTROL_ENABLE_IRQ_VIA_ACK)
156 {
157 Log(("parallel_update_irq %d 1\n", s->irq));
158 PDMDevHlpISASetIrqNoWait(CTXSUFF(s->pDevIns), s->irq, 1);
159 }
160}
161
162static void parallel_clear_irq(ParallelState *s)
163{
164 Log(("parallel_update_irq %d 0\n", s->irq));
165 PDMDevHlpISASetIrqNoWait(CTXSUFF(s->pDevIns), s->irq, 0);
166}
167#endif
168
169static int parallel_ioport_write(void *opaque, uint32_t addr, uint32_t val)
170{
171 ParallelState *s = (ParallelState *)opaque;
172 unsigned char ch;
173
174 addr &= 7;
175 LogFlow(("parallel: write addr=0x%02x val=0x%02x\n", addr, val));
176 ch = val;
177
178 switch(addr) {
179 default:
180 case 0:
181#ifndef IN_RING3
182 NOREF(ch);
183 return VINF_IOM_HC_IOPORT_WRITE;
184#else
185 s->reg_data = ch;
186 if (RT_LIKELY(s->pDrvHostParallelConnector))
187 {
188 Log(("parallel_io_port_write: write 0x%X\n", ch));
189 size_t cbWrite = 1;
190 int rc = s->pDrvHostParallelConnector->pfnWrite(s->pDrvHostParallelConnector, &ch, &cbWrite);
191 AssertRC(rc);
192 }
193#endif
194 break;
195 case 1:
196 break;
197 case 2:
198 /* Set the reserved bits to one */
199 ch |= (LPT_CONTROL_BIT6 | LPT_CONTROL_BIT7);
200 if (ch != s->reg_control) {
201#ifndef IN_RING3
202 return VINF_IOM_HC_IOPORT_WRITE;
203#else
204 int rc = s->pDrvHostParallelConnector->pfnWriteControl(s->pDrvHostParallelConnector, ch);
205 AssertRC(rc);
206 s->reg_control = val;
207#endif
208 }
209 break;
210 case 3:
211 s->reg_epp_addr = val;
212 break;
213 case 4:
214 s->reg_epp_data = val;
215 break;
216 case 5:
217 break;
218 case 6:
219 break;
220 case 7:
221 break;
222 }
223 return VINF_SUCCESS;
224}
225
226static uint32_t parallel_ioport_read(void *opaque, uint32_t addr, int *pRC)
227{
228 ParallelState *s = (ParallelState *)opaque;
229 uint32_t ret = ~0U;
230
231 *pRC = VINF_SUCCESS;
232
233 addr &= 7;
234 switch(addr) {
235 default:
236 case 0:
237 if (!(s->reg_control & LPT_CONTROL_ENABLE_BIDIRECT))
238 ret = s->reg_data;
239 else
240 {
241#ifndef IN_RING3
242 *pRC = VINF_IOM_HC_IOPORT_READ;
243#else
244 if (RT_LIKELY(s->pDrvHostParallelConnector))
245 {
246 size_t cbRead;
247 int rc = s->pDrvHostParallelConnector->pfnRead(s->pDrvHostParallelConnector, &s->reg_data, &cbRead);
248 Log(("parallel_io_port_read: read 0x%X\n", s->reg_data));
249 AssertRC(rc);
250 }
251 ret = s->reg_data;
252#endif
253 }
254 break;
255 case 1:
256#ifndef IN_RING3
257 *pRC = VINF_IOM_HC_IOPORT_READ;
258#else
259 if (RT_LIKELY(s->pDrvHostParallelConnector))
260 {
261 int rc = s->pDrvHostParallelConnector->pfnReadStatus(s->pDrvHostParallelConnector, &s->reg_status);
262 AssertRC(rc);
263 }
264 ret = s->reg_status;
265 parallel_clear_irq(s);
266#endif
267 break;
268 case 2:
269 ret = s->reg_control;
270 break;
271 case 3:
272 ret = s->reg_epp_addr;
273 break;
274 case 4:
275 ret = s->reg_epp_data;
276 break;
277 case 5:
278 break;
279 case 6:
280 break;
281 case 7:
282 break;
283 }
284 LogFlow(("parallel: read addr=0x%02x val=0x%02x\n", addr, ret));
285 return ret;
286}
287
288#if 0
289static int parallel_ioport_write_ecp(void *opaque, uint32_t addr, uint32_t val)
290{
291 ParallelState *s = (ParallelState *)opaque;
292 unsigned char ch;
293
294 addr &= 7;
295 LogFlow(("parallel: write ecp addr=0x%02x val=0x%02x\n", addr, val));
296 ch = val;
297 switch(addr) {
298 default:
299 case 0:
300 if (LPT_ECP_ECR_CHIPMODE_GET_BITS(s->reg_ecp_ecr) == LPT_ECP_ECR_CHIPMODE_FIFO_TEST) {
301 s->ecp_fifo[s->act_fifo_pos_write] = ch;
302 s->act_fifo_pos_write++;
303 if (s->act_fifo_pos_write < LPT_ECP_FIFO_DEPTH) {
304 /* FIFO has some data (clear both FIFO bits) */
305 s->reg_ecp_ecr &= ~(LPT_ECP_ECR_FIFO_EMPTY | LPT_ECP_ECR_FIFO_FULL);
306 } else {
307 /* FIFO is full */
308 /* Clear FIFO empty bit */
309 s->reg_ecp_ecr &= ~LPT_ECP_ECR_FIFO_EMPTY;
310 /* Set FIFO full bit */
311 s->reg_ecp_ecr |= LPT_ECP_ECR_FIFO_FULL;
312 s->act_fifo_pos_write = 0;
313 }
314 } else {
315 s->reg_ecp_base_plus_400h = ch;
316 }
317 break;
318 case 1:
319 s->reg_ecp_config_b = ch;
320 break;
321 case 2:
322 /* If we change the mode clear FIFO */
323 if ((ch & LPT_ECP_ECR_CHIPMODE_MASK) != (s->reg_ecp_ecr & LPT_ECP_ECR_CHIPMODE_MASK)) {
324 /* reset the fifo */
325 s->act_fifo_pos_write = 0;
326 s->act_fifo_pos_read = 0;
327 /* Set FIFO empty bit */
328 s->reg_ecp_ecr |= LPT_ECP_ECR_FIFO_EMPTY;
329 /* Clear FIFO full bit */
330 s->reg_ecp_ecr &= ~LPT_ECP_ECR_FIFO_FULL;
331 }
332 /* Set new mode */
333 s->reg_ecp_ecr |= LPT_ECP_ECR_CHIPMODE_SET_BITS(LPT_ECP_ECR_CHIPMODE_GET_BITS(ch));
334 break;
335 case 3:
336 break;
337 case 4:
338 break;
339 case 5:
340 break;
341 case 6:
342 break;
343 case 7:
344 break;
345 }
346 return VINF_SUCCESS;
347}
348
349static uint32_t parallel_ioport_read_ecp(void *opaque, uint32_t addr, int *pRC)
350{
351 ParallelState *s = (ParallelState *)opaque;
352 uint32_t ret = ~0U;
353
354 *pRC = VINF_SUCCESS;
355
356 addr &= 7;
357 switch(addr) {
358 default:
359 case 0:
360 if (LPT_ECP_ECR_CHIPMODE_GET_BITS(s->reg_ecp_ecr) == LPT_ECP_ECR_CHIPMODE_FIFO_TEST) {
361 ret = s->ecp_fifo[s->act_fifo_pos_read];
362 s->act_fifo_pos_read++;
363 if (s->act_fifo_pos_read == LPT_ECP_FIFO_DEPTH)
364 s->act_fifo_pos_read = 0; /* end of FIFO, start at beginning */
365 if (s->act_fifo_pos_read == s->act_fifo_pos_write) {
366 /* FIFO is empty */
367 /* Set FIFO empty bit */
368 s->reg_ecp_ecr |= LPT_ECP_ECR_FIFO_EMPTY;
369 /* Clear FIFO full bit */
370 s->reg_ecp_ecr &= ~LPT_ECP_ECR_FIFO_FULL;
371 } else {
372 /* FIFO has some data (clear all FIFO bits) */
373 s->reg_ecp_ecr &= ~(LPT_ECP_ECR_FIFO_EMPTY | LPT_ECP_ECR_FIFO_FULL);
374 }
375 } else {
376 ret = s->reg_ecp_base_plus_400h;
377 }
378 break;
379 case 1:
380 ret = s->reg_ecp_config_b;
381 break;
382 case 2:
383 ret = s->reg_ecp_ecr;
384 break;
385 case 3:
386 break;
387 case 4:
388 break;
389 case 5:
390 break;
391 case 6:
392 break;
393 case 7:
394 break;
395 }
396 LogFlow(("parallel: read ecp addr=0x%02x val=0x%02x\n", addr, ret));
397 return ret;
398}
399#endif
400
401#ifdef IN_RING3
402static DECLCALLBACK(int) parallelNotifyInterrupt(PPDMIHOSTPARALLELPORT pInterface)
403{
404 ParallelState *pData = PDMIHOSTPARALLELPORT_2_PARALLELSTATE(pInterface);
405
406 PDMCritSectEnter(&pData->CritSect, VINF_SUCCESS);
407 parallel_set_irq(pData);
408 PDMCritSectLeave(&pData->CritSect);
409
410 return VINF_SUCCESS;
411}
412#endif /* IN_RING3 */
413
414/**
415 * Port I/O Handler for OUT operations.
416 *
417 * @returns VBox status code.
418 *
419 * @param pDevIns The device instance.
420 * @param pvUser User argument.
421 * @param Port Port number used for the IN operation.
422 * @param u32 The value to output.
423 * @param cb The value size in bytes.
424 */
425PDMBOTHCBDECL(int) parallelIOPortWrite(PPDMDEVINS pDevIns, void *pvUser,
426 RTIOPORT Port, uint32_t u32, unsigned cb)
427{
428 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
429 int rc = VINF_SUCCESS;
430
431 if (cb == 1)
432 {
433 rc = PDMCritSectEnter(&pData->CritSect, VINF_IOM_HC_IOPORT_WRITE);
434 if (rc == VINF_SUCCESS)
435 {
436 Log2(("%s: port %#06x val %#04x\n", __FUNCTION__, Port, u32));
437 rc = parallel_ioport_write (pData, Port, u32);
438 PDMCritSectLeave(&pData->CritSect);
439 }
440 }
441 else
442 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
443
444 return rc;
445}
446
447/**
448 * Port I/O Handler for IN operations.
449 *
450 * @returns VBox status code.
451 *
452 * @param pDevIns The device instance.
453 * @param pvUser User argument.
454 * @param Port Port number used for the IN operation.
455 * @param u32 The value to output.
456 * @param cb The value size in bytes.
457 */
458PDMBOTHCBDECL(int) parallelIOPortRead(PPDMDEVINS pDevIns, void *pvUser,
459 RTIOPORT Port, uint32_t *pu32, unsigned cb)
460{
461 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
462 int rc = VINF_SUCCESS;
463
464 if (cb == 1)
465 {
466 rc = PDMCritSectEnter(&pData->CritSect, VINF_IOM_HC_IOPORT_READ);
467 if (rc == VINF_SUCCESS)
468 {
469 *pu32 = parallel_ioport_read (pData, Port, &rc);
470 Log2(("%s: port %#06x val %#04x\n", __FUNCTION__, Port, *pu32));
471 PDMCritSectLeave(&pData->CritSect);
472 }
473 }
474 else
475 rc = VERR_IOM_IOPORT_UNUSED;
476
477 return rc;
478}
479
480#if 0
481/**
482 * Port I/O Handler for OUT operations on ECP registers.
483 *
484 * @returns VBox status code.
485 *
486 * @param pDevIns The device instance.
487 * @param pvUser User argument.
488 * @param Port Port number used for the IN operation.
489 * @param u32 The value to output.
490 * @param cb The value size in bytes.
491 */
492PDMBOTHCBDECL(int) parallelIOPortWriteECP(PPDMDEVINS pDevIns, void *pvUser,
493 RTIOPORT Port, uint32_t u32, unsigned cb)
494{
495 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
496 int rc = VINF_SUCCESS;
497
498 if (cb == 1)
499 {
500 rc = PDMCritSectEnter(&pData->CritSect, VINF_IOM_HC_IOPORT_WRITE);
501 if (rc == VINF_SUCCESS)
502 {
503 Log2(("%s: ecp port %#06x val %#04x\n", __FUNCTION__, Port, u32));
504 rc = parallel_ioport_write_ecp (pData, Port, u32);
505 PDMCritSectLeave(&pData->CritSect);
506 }
507 }
508 else
509 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
510
511 return rc;
512}
513
514/**
515 * Port I/O Handler for IN operations on ECP registers.
516 *
517 * @returns VBox status code.
518 *
519 * @param pDevIns The device instance.
520 * @param pvUser User argument.
521 * @param Port Port number used for the IN operation.
522 * @param u32 The value to output.
523 * @param cb The value size in bytes.
524 */
525PDMBOTHCBDECL(int) parallelIOPortReadECP(PPDMDEVINS pDevIns, void *pvUser,
526 RTIOPORT Port, uint32_t *pu32, unsigned cb)
527{
528 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
529 int rc = VINF_SUCCESS;
530
531 if (cb == 1)
532 {
533 rc = PDMCritSectEnter(&pData->CritSect, VINF_IOM_HC_IOPORT_READ);
534 if (rc == VINF_SUCCESS)
535 {
536 *pu32 = parallel_ioport_read_ecp (pData, Port, &rc);
537 Log2(("%s: ecp port %#06x val %#04x\n", __FUNCTION__, Port, *pu32));
538 PDMCritSectLeave(&pData->CritSect);
539 }
540 }
541 else
542 rc = VERR_IOM_IOPORT_UNUSED;
543
544 return rc;
545}
546#endif
547
548#ifdef IN_RING3
549/**
550 * Saves a state of the serial port device.
551 *
552 * @returns VBox status code.
553 * @param pDevIns The device instance.
554 * @param pSSMHandle The handle to save the state to.
555 */
556static DECLCALLBACK(int) parallelSaveExec(PPDMDEVINS pDevIns,
557 PSSMHANDLE pSSMHandle)
558{
559 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
560
561 SSMR3PutU8(pSSMHandle, pData->reg_data);
562 SSMR3PutU8(pSSMHandle, pData->reg_status);
563 SSMR3PutU8(pSSMHandle, pData->reg_control);
564 SSMR3PutS32(pSSMHandle, pData->irq);
565 SSMR3PutU32(pSSMHandle, pData->base);
566
567 return SSMR3PutU32(pSSMHandle, ~0); /* sanity/terminator */
568}
569
570/**
571 * Loads a saved serial port device state.
572 *
573 * @returns VBox status code.
574 * @param pDevIns The device instance.
575 * @param pSSMHandle The handle to the saved state.
576 * @param u32Version The data unit version number.
577 */
578static DECLCALLBACK(int) parallelLoadExec(PPDMDEVINS pDevIns,
579 PSSMHANDLE pSSMHandle,
580 uint32_t u32Version)
581{
582 int rc;
583 uint32_t u32;
584 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
585
586 if (u32Version != PARALLEL_SAVED_STATE_VERSION)
587 {
588 AssertMsgFailed(("u32Version=%d\n", u32Version));
589 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
590 }
591
592 SSMR3GetU8(pSSMHandle, &pData->reg_data);
593 SSMR3GetU8(pSSMHandle, &pData->reg_status);
594 SSMR3GetU8(pSSMHandle, &pData->reg_control);
595 SSMR3GetS32(pSSMHandle, &pData->irq);
596 SSMR3GetU32(pSSMHandle, &pData->base);
597
598 rc = SSMR3GetU32(pSSMHandle, &u32);
599 if (VBOX_FAILURE(rc))
600 return rc;
601
602 if (u32 != ~0U)
603 {
604 AssertMsgFailed(("u32=%#x expected ~0\n", u32));
605 return VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
606 }
607
608 pData->pDevInsHC = pDevIns;
609 pData->pDevInsGC = PDMDEVINS_2_GCPTR(pDevIns);
610 return VINF_SUCCESS;
611}
612
613
614/**
615 * @copydoc FNPDMDEVRELOCATE
616 */
617static DECLCALLBACK(void) parallelRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
618{
619 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
620 pData->pDevInsGC += offDelta;
621}
622
623/** @copyfrom PIBASE::pfnqueryInterface */
624static DECLCALLBACK(void *) parallelQueryInterface(PPDMIBASE pInterface, PDMINTERFACE enmInterface)
625{
626 ParallelState *pData = PDMIBASE_2_PARALLELSTATE(pInterface);
627 switch (enmInterface)
628 {
629 case PDMINTERFACE_BASE:
630 return &pData->IBase;
631 case PDMINTERFACE_HOST_PARALLEL_PORT:
632 return &pData->IHostParallelPort;
633 default:
634 return NULL;
635 }
636}
637
638/**
639 * Destruct a device instance.
640 *
641 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
642 * resources can be freed correctly.
643 *
644 * @returns VBox status.
645 * @param pDevIns The device instance data.
646 */
647static DECLCALLBACK(int) parallelDestruct(PPDMDEVINS pDevIns)
648{
649 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
650
651 PDMR3CritSectDelete(&pData->CritSect);
652 RTSemEventDestroy(pData->ReceiveSem);
653
654 return VINF_SUCCESS;
655}
656
657
658/**
659 * Construct a device instance for a VM.
660 *
661 * @returns VBox status.
662 * @param pDevIns The device instance data.
663 * If the registration structure is needed, pDevIns->pDevReg points to it.
664 * @param iInstance Instance number. Use this to figure out which registers and such to use.
665 * The device number is also found in pDevIns->iInstance, but since it's
666 * likely to be freqently used PDM passes it as parameter.
667 * @param pCfgHandle Configuration node handle for the device. Use this to obtain the configuration
668 * of the device instance. It's also found in pDevIns->pCfgHandle, but like
669 * iInstance it's expected to be used a bit in this function.
670 */
671static DECLCALLBACK(int) parallelConstruct(PPDMDEVINS pDevIns,
672 int iInstance,
673 PCFGMNODE pCfgHandle)
674{
675 int rc;
676 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState*);
677 uint16_t io_base;
678 uint8_t irq_lvl;
679
680 Assert(iInstance < 4);
681
682 pData->pDevInsHC = pDevIns;
683 pData->pDevInsGC = PDMDEVINS_2_GCPTR(pDevIns);
684
685 /*
686 * Validate configuration.
687 */
688 if (!CFGMR3AreValuesValid(pCfgHandle, "IRQ\0IOBase\0GCEnabled\0R0Enabled\0"))
689 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
690 N_("Configuration error: Unknown config key"));
691
692 rc = CFGMR3QueryBool(pCfgHandle, "GCEnabled", &pData->fGCEnabled);
693 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
694 pData->fGCEnabled = true;
695 else if (VBOX_FAILURE(rc))
696 return PDMDEV_SET_ERROR(pDevIns, rc,
697 N_("Configuration error: Failed to get the \"GCEnabled\" value"));
698
699 rc = CFGMR3QueryBool(pCfgHandle, "R0Enabled", &pData->fR0Enabled);
700 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
701 pData->fR0Enabled = true;
702 else if (VBOX_FAILURE(rc))
703 return PDMDEV_SET_ERROR(pDevIns, rc,
704 N_("Configuration error: Failed to get the \"R0Enabled\" value"));
705
706 /* IBase */
707 pData->IBase.pfnQueryInterface = parallelQueryInterface;
708
709 /* IHostParallelPort */
710 pData->IHostParallelPort.pfnNotifyInterrupt = parallelNotifyInterrupt;
711
712 rc = RTSemEventCreate(&pData->ReceiveSem);
713 AssertRC(rc);
714
715 /*
716 * Initialize critical section.
717 * This must of course be done before attaching drivers or anything else which can call us back..
718 */
719 char szName[24];
720 RTStrPrintf(szName, sizeof(szName), "Parallel#%d", iInstance);
721 rc = PDMDevHlpCritSectInit(pDevIns, &pData->CritSect, szName);
722 if (VBOX_FAILURE(rc))
723 return rc;
724
725 rc = CFGMR3QueryU8(pCfgHandle, "IRQ", &irq_lvl);
726 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
727 irq_lvl = 7;
728 else if (VBOX_FAILURE(rc))
729 return PDMDEV_SET_ERROR(pDevIns, rc,
730 N_("Configuration error: Failed to get the \"IRQ\" value"));
731
732 rc = CFGMR3QueryU16(pCfgHandle, "IOBase", &io_base);
733 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
734 io_base = 0x378;
735 else if (VBOX_FAILURE(rc))
736 return PDMDEV_SET_ERROR(pDevIns, rc,
737 N_("Configuration error: Failed to get the \"IOBase\" value"));
738
739 Log(("parallelConstruct instance %d iobase=%04x irq=%d\n", iInstance, io_base, irq_lvl));
740
741 pData->irq = irq_lvl;
742 pData->base = io_base;
743
744 /* Init parallel state */
745 pData->reg_data = 0;
746 pData->reg_ecp_ecr = LPT_ECP_ECR_CHIPMODE_COMPAT | LPT_ECP_ECR_FIFO_EMPTY;
747 pData->act_fifo_pos_read = 0;
748 pData->act_fifo_pos_write = 0;
749
750 rc = PDMDevHlpIOPortRegister(pDevIns, io_base, 8, 0,
751 parallelIOPortWrite, parallelIOPortRead,
752 NULL, NULL, "PARALLEL");
753 if (VBOX_FAILURE(rc))
754 return rc;
755
756#if 0
757 /* register ecp registers */
758 rc = PDMDevHlpIOPortRegister(pDevIns, io_base+0x400, 8, 0,
759 parallelIOPortWriteECP, parallelIOPortReadECP,
760 NULL, NULL, "PARALLEL ECP");
761 if (VBOX_FAILURE(rc))
762 return rc;
763#endif
764
765 if (pData->fGCEnabled)
766 {
767 rc = PDMDevHlpIOPortRegisterGC(pDevIns, io_base, 8, 0, "parallelIOPortWrite",
768 "parallelIOPortRead", NULL, NULL, "Parallel");
769 if (VBOX_FAILURE(rc))
770 return rc;
771
772#if 0
773 rc = PDMDevHlpIOPortRegisterGC(pDevIns, io_base+0x400, 8, 0, "parallelIOPortWriteECP",
774 "parallelIOPortReadECP", NULL, NULL, "Parallel Ecp");
775 if (VBOX_FAILURE(rc))
776 return rc;
777#endif
778 }
779
780 if (pData->fR0Enabled)
781 {
782 rc = PDMDevHlpIOPortRegisterR0(pDevIns, io_base, 8, 0, "parallelIOPortWrite",
783 "parallelIOPortRead", NULL, NULL, "Parallel");
784 if (VBOX_FAILURE(rc))
785 return rc;
786
787#if 0
788 rc = PDMDevHlpIOPortRegisterR0(pDevIns, io_base+0x400, 8, 0, "parallelIOPortWriteECP",
789 "parallelIOPortReadECP", NULL, NULL, "Parallel Ecp");
790 if (VBOX_FAILURE(rc))
791 return rc;
792#endif
793 }
794
795 /* Attach the parallel port driver and get the interfaces. For now no run-time
796 * changes are supported. */
797 rc = PDMDevHlpDriverAttach(pDevIns, 0, &pData->IBase, &pData->pDrvBase, "Parallel Host");
798 if (VBOX_SUCCESS(rc))
799 {
800 pData->pDrvHostParallelConnector = (PDMIHOSTPARALLELCONNECTOR *)pData->pDrvBase->pfnQueryInterface(pData->pDrvBase,
801 PDMINTERFACE_HOST_PARALLEL_CONNECTOR);
802 if (!pData->pDrvHostParallelConnector)
803 {
804 AssertMsgFailed(("Configuration error: instance %d has no host parallel interface!\n", iInstance));
805 return VERR_PDM_MISSING_INTERFACE;
806 }
807 /** @todo provide read notification interface!!!! */
808 }
809 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
810 {
811 pData->pDrvBase = NULL;
812 pData->pDrvHostParallelConnector = NULL;
813 LogRel(("Parallel%d: no unit\n", iInstance));
814 }
815 else
816 {
817 AssertMsgFailed(("Parallel%d: Failed to attach to host driver. rc=%Vrc\n", iInstance, rc));
818 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
819 N_("Parallel device %d cannot attach to host driver"), iInstance);
820 }
821
822 /* Set compatibility mode */
823 pData->pDrvHostParallelConnector->pfnSetMode(pData->pDrvHostParallelConnector, PDM_PARALLEL_PORT_MODE_COMPAT);
824 /* Get status of control register */
825 pData->pDrvHostParallelConnector->pfnReadControl(pData->pDrvHostParallelConnector, &pData->reg_control);
826
827 rc = PDMDevHlpSSMRegister(
828 pDevIns, /* pDevIns */
829 pDevIns->pDevReg->szDeviceName, /* pszName */
830 iInstance, /* u32Instance */
831 PARALLEL_SAVED_STATE_VERSION, /* u32Version */
832 sizeof (*pData), /* cbGuess */
833 NULL, /* pfnSavePrep */
834 parallelSaveExec, /* pfnSaveExec */
835 NULL, /* pfnSaveDone */
836 NULL, /* pfnLoadPrep */
837 parallelLoadExec, /* pfnLoadExec */
838 NULL /* pfnLoadDone */
839 );
840 if (VBOX_FAILURE(rc))
841 return rc;
842
843 return VINF_SUCCESS;
844}
845
846/**
847 * The device registration structure.
848 */
849const PDMDEVREG g_DeviceParallelPort =
850{
851 /* u32Version */
852 PDM_DEVREG_VERSION,
853 /* szDeviceName */
854 "parallel",
855 /* szGCMod */
856 "VBoxDDGC.gc",
857 /* szR0Mod */
858 "VBoxDDR0.r0",
859 /* pszDescription */
860 "Parallel Communication Port",
861 /* fFlags */
862 PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DEVREG_FLAGS_GUEST_BITS_DEFAULT | PDM_DEVREG_FLAGS_GC | PDM_DEVREG_FLAGS_R0,
863 /* fClass */
864 PDM_DEVREG_CLASS_PARALLEL,
865 /* cMaxInstances */
866 1,
867 /* cbInstance */
868 sizeof(ParallelState),
869 /* pfnConstruct */
870 parallelConstruct,
871 /* pfnDestruct */
872 parallelDestruct,
873 /* pfnRelocate */
874 parallelRelocate,
875 /* pfnIOCtl */
876 NULL,
877 /* pfnPowerOn */
878 NULL,
879 /* pfnReset */
880 NULL,
881 /* pfnSuspend */
882 NULL,
883 /* pfnResume */
884 NULL,
885 /* pfnAttach */
886 NULL,
887 /* pfnDetach */
888 NULL,
889 /* pfnQueryInterface. */
890 NULL
891};
892#endif /* IN_RING3 */
893
894
895#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
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