VirtualBox

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

Last change on this file since 8567 was 8155, checked in by vboxsync, 17 years ago

The Big Sun Rebranding Header Change

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 28.6 KB
Line 
1/* $Id: DevParallel.cpp 8155 2008-04-18 15:16:47Z 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 GCPTRTYPE(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 NOREF(ch);
203 return VINF_IOM_HC_IOPORT_WRITE;
204#else
205 int rc = s->pDrvHostParallelConnector->pfnWriteControl(s->pDrvHostParallelConnector, ch);
206 AssertRC(rc);
207 s->reg_control = val;
208#endif
209 }
210 break;
211 case 3:
212 s->reg_epp_addr = val;
213 break;
214 case 4:
215 s->reg_epp_data = val;
216 break;
217 case 5:
218 break;
219 case 6:
220 break;
221 case 7:
222 break;
223 }
224 return VINF_SUCCESS;
225}
226
227static uint32_t parallel_ioport_read(void *opaque, uint32_t addr, int *pRC)
228{
229 ParallelState *s = (ParallelState *)opaque;
230 uint32_t ret = ~0U;
231
232 *pRC = VINF_SUCCESS;
233
234 addr &= 7;
235 switch(addr) {
236 default:
237 case 0:
238 if (!(s->reg_control & LPT_CONTROL_ENABLE_BIDIRECT))
239 ret = s->reg_data;
240 else
241 {
242#ifndef IN_RING3
243 *pRC = VINF_IOM_HC_IOPORT_READ;
244#else
245 if (RT_LIKELY(s->pDrvHostParallelConnector))
246 {
247 size_t cbRead;
248 int rc = s->pDrvHostParallelConnector->pfnRead(s->pDrvHostParallelConnector, &s->reg_data, &cbRead);
249 Log(("parallel_io_port_read: read 0x%X\n", s->reg_data));
250 AssertRC(rc);
251 }
252 ret = s->reg_data;
253#endif
254 }
255 break;
256 case 1:
257#ifndef IN_RING3
258 *pRC = VINF_IOM_HC_IOPORT_READ;
259#else
260 if (RT_LIKELY(s->pDrvHostParallelConnector))
261 {
262 int rc = s->pDrvHostParallelConnector->pfnReadStatus(s->pDrvHostParallelConnector, &s->reg_status);
263 AssertRC(rc);
264 }
265 ret = s->reg_status;
266 parallel_clear_irq(s);
267#endif
268 break;
269 case 2:
270 ret = s->reg_control;
271 break;
272 case 3:
273 ret = s->reg_epp_addr;
274 break;
275 case 4:
276 ret = s->reg_epp_data;
277 break;
278 case 5:
279 break;
280 case 6:
281 break;
282 case 7:
283 break;
284 }
285 LogFlow(("parallel: read addr=0x%02x val=0x%02x\n", addr, ret));
286 return ret;
287}
288
289#if 0
290static int parallel_ioport_write_ecp(void *opaque, uint32_t addr, uint32_t val)
291{
292 ParallelState *s = (ParallelState *)opaque;
293 unsigned char ch;
294
295 addr &= 7;
296 LogFlow(("parallel: write ecp addr=0x%02x val=0x%02x\n", addr, val));
297 ch = val;
298 switch(addr) {
299 default:
300 case 0:
301 if (LPT_ECP_ECR_CHIPMODE_GET_BITS(s->reg_ecp_ecr) == LPT_ECP_ECR_CHIPMODE_FIFO_TEST) {
302 s->ecp_fifo[s->act_fifo_pos_write] = ch;
303 s->act_fifo_pos_write++;
304 if (s->act_fifo_pos_write < LPT_ECP_FIFO_DEPTH) {
305 /* FIFO has some data (clear both FIFO bits) */
306 s->reg_ecp_ecr &= ~(LPT_ECP_ECR_FIFO_EMPTY | LPT_ECP_ECR_FIFO_FULL);
307 } else {
308 /* FIFO is full */
309 /* Clear FIFO empty bit */
310 s->reg_ecp_ecr &= ~LPT_ECP_ECR_FIFO_EMPTY;
311 /* Set FIFO full bit */
312 s->reg_ecp_ecr |= LPT_ECP_ECR_FIFO_FULL;
313 s->act_fifo_pos_write = 0;
314 }
315 } else {
316 s->reg_ecp_base_plus_400h = ch;
317 }
318 break;
319 case 1:
320 s->reg_ecp_config_b = ch;
321 break;
322 case 2:
323 /* If we change the mode clear FIFO */
324 if ((ch & LPT_ECP_ECR_CHIPMODE_MASK) != (s->reg_ecp_ecr & LPT_ECP_ECR_CHIPMODE_MASK)) {
325 /* reset the fifo */
326 s->act_fifo_pos_write = 0;
327 s->act_fifo_pos_read = 0;
328 /* Set FIFO empty bit */
329 s->reg_ecp_ecr |= LPT_ECP_ECR_FIFO_EMPTY;
330 /* Clear FIFO full bit */
331 s->reg_ecp_ecr &= ~LPT_ECP_ECR_FIFO_FULL;
332 }
333 /* Set new mode */
334 s->reg_ecp_ecr |= LPT_ECP_ECR_CHIPMODE_SET_BITS(LPT_ECP_ECR_CHIPMODE_GET_BITS(ch));
335 break;
336 case 3:
337 break;
338 case 4:
339 break;
340 case 5:
341 break;
342 case 6:
343 break;
344 case 7:
345 break;
346 }
347 return VINF_SUCCESS;
348}
349
350static uint32_t parallel_ioport_read_ecp(void *opaque, uint32_t addr, int *pRC)
351{
352 ParallelState *s = (ParallelState *)opaque;
353 uint32_t ret = ~0U;
354
355 *pRC = VINF_SUCCESS;
356
357 addr &= 7;
358 switch(addr) {
359 default:
360 case 0:
361 if (LPT_ECP_ECR_CHIPMODE_GET_BITS(s->reg_ecp_ecr) == LPT_ECP_ECR_CHIPMODE_FIFO_TEST) {
362 ret = s->ecp_fifo[s->act_fifo_pos_read];
363 s->act_fifo_pos_read++;
364 if (s->act_fifo_pos_read == LPT_ECP_FIFO_DEPTH)
365 s->act_fifo_pos_read = 0; /* end of FIFO, start at beginning */
366 if (s->act_fifo_pos_read == s->act_fifo_pos_write) {
367 /* FIFO is empty */
368 /* Set FIFO empty bit */
369 s->reg_ecp_ecr |= LPT_ECP_ECR_FIFO_EMPTY;
370 /* Clear FIFO full bit */
371 s->reg_ecp_ecr &= ~LPT_ECP_ECR_FIFO_FULL;
372 } else {
373 /* FIFO has some data (clear all FIFO bits) */
374 s->reg_ecp_ecr &= ~(LPT_ECP_ECR_FIFO_EMPTY | LPT_ECP_ECR_FIFO_FULL);
375 }
376 } else {
377 ret = s->reg_ecp_base_plus_400h;
378 }
379 break;
380 case 1:
381 ret = s->reg_ecp_config_b;
382 break;
383 case 2:
384 ret = s->reg_ecp_ecr;
385 break;
386 case 3:
387 break;
388 case 4:
389 break;
390 case 5:
391 break;
392 case 6:
393 break;
394 case 7:
395 break;
396 }
397 LogFlow(("parallel: read ecp addr=0x%02x val=0x%02x\n", addr, ret));
398 return ret;
399}
400#endif
401
402#ifdef IN_RING3
403static DECLCALLBACK(int) parallelNotifyInterrupt(PPDMIHOSTPARALLELPORT pInterface)
404{
405 ParallelState *pData = PDMIHOSTPARALLELPORT_2_PARALLELSTATE(pInterface);
406
407 PDMCritSectEnter(&pData->CritSect, VINF_SUCCESS);
408 parallel_set_irq(pData);
409 PDMCritSectLeave(&pData->CritSect);
410
411 return VINF_SUCCESS;
412}
413#endif /* IN_RING3 */
414
415/**
416 * Port I/O Handler for OUT operations.
417 *
418 * @returns VBox status code.
419 *
420 * @param pDevIns The device instance.
421 * @param pvUser User argument.
422 * @param Port Port number used for the IN operation.
423 * @param u32 The value to output.
424 * @param cb The value size in bytes.
425 */
426PDMBOTHCBDECL(int) parallelIOPortWrite(PPDMDEVINS pDevIns, void *pvUser,
427 RTIOPORT Port, uint32_t u32, unsigned cb)
428{
429 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
430 int rc = VINF_SUCCESS;
431
432 if (cb == 1)
433 {
434 rc = PDMCritSectEnter(&pData->CritSect, VINF_IOM_HC_IOPORT_WRITE);
435 if (rc == VINF_SUCCESS)
436 {
437 Log2(("%s: port %#06x val %#04x\n", __FUNCTION__, Port, u32));
438 rc = parallel_ioport_write (pData, Port, u32);
439 PDMCritSectLeave(&pData->CritSect);
440 }
441 }
442 else
443 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
444
445 return rc;
446}
447
448/**
449 * Port I/O Handler for IN operations.
450 *
451 * @returns VBox status code.
452 *
453 * @param pDevIns The device instance.
454 * @param pvUser User argument.
455 * @param Port Port number used for the IN operation.
456 * @param u32 The value to output.
457 * @param cb The value size in bytes.
458 */
459PDMBOTHCBDECL(int) parallelIOPortRead(PPDMDEVINS pDevIns, void *pvUser,
460 RTIOPORT Port, uint32_t *pu32, unsigned cb)
461{
462 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
463 int rc = VINF_SUCCESS;
464
465 if (cb == 1)
466 {
467 rc = PDMCritSectEnter(&pData->CritSect, VINF_IOM_HC_IOPORT_READ);
468 if (rc == VINF_SUCCESS)
469 {
470 *pu32 = parallel_ioport_read (pData, Port, &rc);
471 Log2(("%s: port %#06x val %#04x\n", __FUNCTION__, Port, *pu32));
472 PDMCritSectLeave(&pData->CritSect);
473 }
474 }
475 else
476 rc = VERR_IOM_IOPORT_UNUSED;
477
478 return rc;
479}
480
481#if 0
482/**
483 * Port I/O Handler for OUT operations on ECP registers.
484 *
485 * @returns VBox status code.
486 *
487 * @param pDevIns The device instance.
488 * @param pvUser User argument.
489 * @param Port Port number used for the IN operation.
490 * @param u32 The value to output.
491 * @param cb The value size in bytes.
492 */
493PDMBOTHCBDECL(int) parallelIOPortWriteECP(PPDMDEVINS pDevIns, void *pvUser,
494 RTIOPORT Port, uint32_t u32, unsigned cb)
495{
496 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
497 int rc = VINF_SUCCESS;
498
499 if (cb == 1)
500 {
501 rc = PDMCritSectEnter(&pData->CritSect, VINF_IOM_HC_IOPORT_WRITE);
502 if (rc == VINF_SUCCESS)
503 {
504 Log2(("%s: ecp port %#06x val %#04x\n", __FUNCTION__, Port, u32));
505 rc = parallel_ioport_write_ecp (pData, Port, u32);
506 PDMCritSectLeave(&pData->CritSect);
507 }
508 }
509 else
510 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
511
512 return rc;
513}
514
515/**
516 * Port I/O Handler for IN operations on ECP registers.
517 *
518 * @returns VBox status code.
519 *
520 * @param pDevIns The device instance.
521 * @param pvUser User argument.
522 * @param Port Port number used for the IN operation.
523 * @param u32 The value to output.
524 * @param cb The value size in bytes.
525 */
526PDMBOTHCBDECL(int) parallelIOPortReadECP(PPDMDEVINS pDevIns, void *pvUser,
527 RTIOPORT Port, uint32_t *pu32, unsigned cb)
528{
529 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
530 int rc = VINF_SUCCESS;
531
532 if (cb == 1)
533 {
534 rc = PDMCritSectEnter(&pData->CritSect, VINF_IOM_HC_IOPORT_READ);
535 if (rc == VINF_SUCCESS)
536 {
537 *pu32 = parallel_ioport_read_ecp (pData, Port, &rc);
538 Log2(("%s: ecp port %#06x val %#04x\n", __FUNCTION__, Port, *pu32));
539 PDMCritSectLeave(&pData->CritSect);
540 }
541 }
542 else
543 rc = VERR_IOM_IOPORT_UNUSED;
544
545 return rc;
546}
547#endif
548
549#ifdef IN_RING3
550/**
551 * Saves a state of the serial port device.
552 *
553 * @returns VBox status code.
554 * @param pDevIns The device instance.
555 * @param pSSMHandle The handle to save the state to.
556 */
557static DECLCALLBACK(int) parallelSaveExec(PPDMDEVINS pDevIns,
558 PSSMHANDLE pSSMHandle)
559{
560 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
561
562 SSMR3PutU8(pSSMHandle, pData->reg_data);
563 SSMR3PutU8(pSSMHandle, pData->reg_status);
564 SSMR3PutU8(pSSMHandle, pData->reg_control);
565 SSMR3PutS32(pSSMHandle, pData->irq);
566 SSMR3PutU32(pSSMHandle, pData->base);
567
568 return SSMR3PutU32(pSSMHandle, ~0); /* sanity/terminator */
569}
570
571/**
572 * Loads a saved serial port device state.
573 *
574 * @returns VBox status code.
575 * @param pDevIns The device instance.
576 * @param pSSMHandle The handle to the saved state.
577 * @param u32Version The data unit version number.
578 */
579static DECLCALLBACK(int) parallelLoadExec(PPDMDEVINS pDevIns,
580 PSSMHANDLE pSSMHandle,
581 uint32_t u32Version)
582{
583 int rc;
584 uint32_t u32;
585 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
586
587 if (u32Version != PARALLEL_SAVED_STATE_VERSION)
588 {
589 AssertMsgFailed(("u32Version=%d\n", u32Version));
590 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
591 }
592
593 SSMR3GetU8(pSSMHandle, &pData->reg_data);
594 SSMR3GetU8(pSSMHandle, &pData->reg_status);
595 SSMR3GetU8(pSSMHandle, &pData->reg_control);
596 SSMR3GetS32(pSSMHandle, &pData->irq);
597 SSMR3GetU32(pSSMHandle, &pData->base);
598
599 rc = SSMR3GetU32(pSSMHandle, &u32);
600 if (VBOX_FAILURE(rc))
601 return rc;
602
603 if (u32 != ~0U)
604 {
605 AssertMsgFailed(("u32=%#x expected ~0\n", u32));
606 return VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
607 }
608
609 pData->pDevInsHC = pDevIns;
610 pData->pDevInsGC = PDMDEVINS_2_GCPTR(pDevIns);
611 return VINF_SUCCESS;
612}
613
614
615/**
616 * @copydoc FNPDMDEVRELOCATE
617 */
618static DECLCALLBACK(void) parallelRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
619{
620 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
621 pData->pDevInsGC += offDelta;
622}
623
624/** @copyfrom PIBASE::pfnqueryInterface */
625static DECLCALLBACK(void *) parallelQueryInterface(PPDMIBASE pInterface, PDMINTERFACE enmInterface)
626{
627 ParallelState *pData = PDMIBASE_2_PARALLELSTATE(pInterface);
628 switch (enmInterface)
629 {
630 case PDMINTERFACE_BASE:
631 return &pData->IBase;
632 case PDMINTERFACE_HOST_PARALLEL_PORT:
633 return &pData->IHostParallelPort;
634 default:
635 return NULL;
636 }
637}
638
639/**
640 * Destruct a device instance.
641 *
642 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
643 * resources can be freed correctly.
644 *
645 * @returns VBox status.
646 * @param pDevIns The device instance data.
647 */
648static DECLCALLBACK(int) parallelDestruct(PPDMDEVINS pDevIns)
649{
650 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState *);
651
652 PDMR3CritSectDelete(&pData->CritSect);
653/** @todo destroy the event sem? */
654
655 return VINF_SUCCESS;
656}
657
658
659/**
660 * Construct a device instance for a VM.
661 *
662 * @returns VBox status.
663 * @param pDevIns The device instance data.
664 * If the registration structure is needed, pDevIns->pDevReg points to it.
665 * @param iInstance Instance number. Use this to figure out which registers and such to use.
666 * The device number is also found in pDevIns->iInstance, but since it's
667 * likely to be freqently used PDM passes it as parameter.
668 * @param pCfgHandle Configuration node handle for the device. Use this to obtain the configuration
669 * of the device instance. It's also found in pDevIns->pCfgHandle, but like
670 * iInstance it's expected to be used a bit in this function.
671 */
672static DECLCALLBACK(int) parallelConstruct(PPDMDEVINS pDevIns,
673 int iInstance,
674 PCFGMNODE pCfgHandle)
675{
676 int rc;
677 ParallelState *pData = PDMINS2DATA(pDevIns, ParallelState*);
678 uint16_t io_base;
679 uint8_t irq_lvl;
680
681 Assert(iInstance < 4);
682
683 pData->pDevInsHC = pDevIns;
684 pData->pDevInsGC = PDMDEVINS_2_GCPTR(pDevIns);
685
686 /*
687 * Validate configuration.
688 */
689 if (!CFGMR3AreValuesValid(pCfgHandle, "IRQ\0IOBase\0GCEnabled\0R0Enabled\0"))
690 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
691 N_("Configuration error: Unknown config key"));
692
693 rc = CFGMR3QueryBool(pCfgHandle, "GCEnabled", &pData->fGCEnabled);
694 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
695 pData->fGCEnabled = true;
696 else if (VBOX_FAILURE(rc))
697 return PDMDEV_SET_ERROR(pDevIns, rc,
698 N_("Configuration error: Failed to get the \"GCEnabled\" value"));
699
700 rc = CFGMR3QueryBool(pCfgHandle, "R0Enabled", &pData->fR0Enabled);
701 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
702 pData->fR0Enabled = true;
703 else if (VBOX_FAILURE(rc))
704 return PDMDEV_SET_ERROR(pDevIns, rc,
705 N_("Configuration error: Failed to get the \"R0Enabled\" value"));
706
707 /* IBase */
708 pData->IBase.pfnQueryInterface = parallelQueryInterface;
709
710 /* IHostParallelPort */
711 pData->IHostParallelPort.pfnNotifyInterrupt = parallelNotifyInterrupt;
712
713 rc = RTSemEventCreate(&pData->ReceiveSem);
714 AssertRC(rc);
715
716 /*
717 * Initialize critical section.
718 * This must of course be done before attaching drivers or anything else which can call us back..
719 */
720 char szName[24];
721 RTStrPrintf(szName, sizeof(szName), "Parallel#%d", iInstance);
722 rc = PDMDevHlpCritSectInit(pDevIns, &pData->CritSect, szName);
723 if (VBOX_FAILURE(rc))
724 return rc;
725
726 rc = CFGMR3QueryU8(pCfgHandle, "IRQ", &irq_lvl);
727 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
728 irq_lvl = 7;
729 else if (VBOX_FAILURE(rc))
730 return PDMDEV_SET_ERROR(pDevIns, rc,
731 N_("Configuration error: Failed to get the \"IRQ\" value"));
732
733 rc = CFGMR3QueryU16(pCfgHandle, "IOBase", &io_base);
734 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
735 io_base = 0x378;
736 else if (VBOX_FAILURE(rc))
737 return PDMDEV_SET_ERROR(pDevIns, rc,
738 N_("Configuration error: Failed to get the \"IOBase\" value"));
739
740 Log(("parallelConstruct instance %d iobase=%04x irq=%d\n", iInstance, io_base, irq_lvl));
741
742 pData->irq = irq_lvl;
743 pData->base = io_base;
744
745 /* Init parallel state */
746 pData->reg_data = 0;
747 pData->reg_ecp_ecr = LPT_ECP_ECR_CHIPMODE_COMPAT | LPT_ECP_ECR_FIFO_EMPTY;
748 pData->act_fifo_pos_read = 0;
749 pData->act_fifo_pos_write = 0;
750
751 rc = PDMDevHlpIOPortRegister(pDevIns, io_base, 8, 0,
752 parallelIOPortWrite, parallelIOPortRead,
753 NULL, NULL, "PARALLEL");
754 if (VBOX_FAILURE(rc))
755 return rc;
756
757#if 0
758 /* register ecp registers */
759 rc = PDMDevHlpIOPortRegister(pDevIns, io_base+0x400, 8, 0,
760 parallelIOPortWriteECP, parallelIOPortReadECP,
761 NULL, NULL, "PARALLEL ECP");
762 if (VBOX_FAILURE(rc))
763 return rc;
764#endif
765
766 if (pData->fGCEnabled)
767 {
768 rc = PDMDevHlpIOPortRegisterGC(pDevIns, io_base, 8, 0, "parallelIOPortWrite",
769 "parallelIOPortRead", NULL, NULL, "Parallel");
770 if (VBOX_FAILURE(rc))
771 return rc;
772
773#if 0
774 rc = PDMDevHlpIOPortRegisterGC(pDevIns, io_base+0x400, 8, 0, "parallelIOPortWriteECP",
775 "parallelIOPortReadECP", NULL, NULL, "Parallel Ecp");
776 if (VBOX_FAILURE(rc))
777 return rc;
778#endif
779 }
780
781 if (pData->fR0Enabled)
782 {
783 rc = PDMDevHlpIOPortRegisterR0(pDevIns, io_base, 8, 0, "parallelIOPortWrite",
784 "parallelIOPortRead", NULL, NULL, "Parallel");
785 if (VBOX_FAILURE(rc))
786 return rc;
787
788#if 0
789 rc = PDMDevHlpIOPortRegisterR0(pDevIns, io_base+0x400, 8, 0, "parallelIOPortWriteECP",
790 "parallelIOPortReadECP", NULL, NULL, "Parallel Ecp");
791 if (VBOX_FAILURE(rc))
792 return rc;
793#endif
794 }
795
796 /* Attach the parallel port driver and get the interfaces. For now no run-time
797 * changes are supported. */
798 rc = PDMDevHlpDriverAttach(pDevIns, 0, &pData->IBase, &pData->pDrvBase, "Parallel Host");
799 if (VBOX_SUCCESS(rc))
800 {
801 pData->pDrvHostParallelConnector = (PDMIHOSTPARALLELCONNECTOR *)pData->pDrvBase->pfnQueryInterface(pData->pDrvBase,
802 PDMINTERFACE_HOST_PARALLEL_CONNECTOR);
803 if (!pData->pDrvHostParallelConnector)
804 {
805 AssertMsgFailed(("Configuration error: instance %d has no host parallel interface!\n", iInstance));
806 return VERR_PDM_MISSING_INTERFACE;
807 }
808 /** @todo provide read notification interface!!!! */
809 }
810 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
811 {
812 pData->pDrvBase = NULL;
813 pData->pDrvHostParallelConnector = NULL;
814 LogRel(("Parallel%d: no unit\n", iInstance));
815 }
816 else
817 {
818 AssertMsgFailed(("Parallel%d: Failed to attach to host driver. rc=%Vrc\n", iInstance, rc));
819 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
820 N_("Parallel device %d cannot attach to host driver"), iInstance);
821 }
822
823 /* Set compatibility mode */
824 pData->pDrvHostParallelConnector->pfnSetMode(pData->pDrvHostParallelConnector, PDM_PARALLEL_PORT_MODE_COMPAT);
825 /* Get status of control register */
826 pData->pDrvHostParallelConnector->pfnReadControl(pData->pDrvHostParallelConnector, &pData->reg_control);
827
828 rc = PDMDevHlpSSMRegister(
829 pDevIns, /* pDevIns */
830 pDevIns->pDevReg->szDeviceName, /* pszName */
831 iInstance, /* u32Instance */
832 PARALLEL_SAVED_STATE_VERSION, /* u32Version */
833 sizeof (*pData), /* cbGuess */
834 NULL, /* pfnSavePrep */
835 parallelSaveExec, /* pfnSaveExec */
836 NULL, /* pfnSaveDone */
837 NULL, /* pfnLoadPrep */
838 parallelLoadExec, /* pfnLoadExec */
839 NULL /* pfnLoadDone */
840 );
841 if (VBOX_FAILURE(rc))
842 return rc;
843
844 return VINF_SUCCESS;
845}
846
847/**
848 * The device registration structure.
849 */
850const PDMDEVREG g_DeviceParallelPort =
851{
852 /* u32Version */
853 PDM_DEVREG_VERSION,
854 /* szDeviceName */
855 "parallel",
856 /* szGCMod */
857 "VBoxDDGC.gc",
858 /* szR0Mod */
859 "VBoxDDR0.r0",
860 /* pszDescription */
861 "Parallel Communication Port",
862 /* fFlags */
863 PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DEVREG_FLAGS_GUEST_BITS_DEFAULT | PDM_DEVREG_FLAGS_GC | PDM_DEVREG_FLAGS_R0,
864 /* fClass */
865 PDM_DEVREG_CLASS_PARALLEL,
866 /* cMaxInstances */
867 1,
868 /* cbInstance */
869 sizeof(ParallelState),
870 /* pfnConstruct */
871 parallelConstruct,
872 /* pfnDestruct */
873 parallelDestruct,
874 /* pfnRelocate */
875 parallelRelocate,
876 /* pfnIOCtl */
877 NULL,
878 /* pfnPowerOn */
879 NULL,
880 /* pfnReset */
881 NULL,
882 /* pfnSuspend */
883 NULL,
884 /* pfnResume */
885 NULL,
886 /* pfnAttach */
887 NULL,
888 /* pfnDetach */
889 NULL,
890 /* pfnQueryInterface. */
891 NULL
892};
893#endif /* IN_RING3 */
894
895
896#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