VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DevSB16.cpp@ 61362

Last change on this file since 61362 was 61334, checked in by vboxsync, 9 years ago

Audio: Fixed shutdown crashes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 73.9 KB
Line 
1/* $Id: DevSB16.cpp 61334 2016-05-31 14:04:04Z vboxsync $ */
2/** @file
3 * DevSB16 - VBox SB16 Audio Controller.
4 *
5 * @todo hiccups on NT4 and Win98.
6 */
7
8/*
9 * Copyright (C) 2015-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 * This code is based on: sb16.c from QEMU AUDIO subsystem (r3917).
21 * QEMU Soundblaster 16 emulation
22 *
23 * Copyright (c) 2003-2005 Vassili Karpov (malc)
24 *
25 * Permission is hereby granted, free of charge, to any person obtaining a copy
26 * of this software and associated documentation files (the "Software"), to deal
27 * in the Software without restriction, including without limitation the rights
28 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
29 * copies of the Software, and to permit persons to whom the Software is
30 * furnished to do so, subject to the following conditions:
31 *
32 * The above copyright notice and this permission notice shall be included in
33 * all copies or substantial portions of the Software.
34 *
35 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
36 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
37 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
38 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
39 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
40 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
41 * THE SOFTWARE.
42 */
43#define LOG_GROUP LOG_GROUP_DEV_SB16
44#include <VBox/log.h>
45#include <iprt/assert.h>
46#ifdef IN_RING3
47# include <iprt/mem.h>
48# include <iprt/string.h>
49# include <iprt/uuid.h>
50#endif
51
52#include <VBox/vmm/pdmdev.h>
53#include <VBox/vmm/pdmaudioifs.h>
54
55#include "VBoxDD.h"
56
57#include "AudioMixBuffer.h"
58#include "AudioMixer.h"
59#include "DrvAudio.h"
60
61/** Current saved state version. */
62#define SB16_SAVE_STATE_VERSION 2
63/** The version used in VirtualBox version 3.0 and earlier. This didn't include the config dump. */
64#define SB16_SAVE_STATE_VERSION_VBOX_30 1
65
66#define IO_READ_PROTO(name) \
67 DECLCALLBACK(int) name (PPDMDEVINS pDevIns, void *opaque, \
68 RTIOPORT nport, uint32_t *pu32, unsigned cb)
69
70#define IO_WRITE_PROTO(name) \
71 DECLCALLBACK(int) name (PPDMDEVINS pDevIns, void *opaque, \
72 RTIOPORT nport, uint32_t val, unsigned cb)
73
74static const char e3[] = "COPYRIGHT (C) CREATIVE TECHNOLOGY LTD, 1992.";
75
76typedef struct SB16OUTPUTSTREAM
77{
78 /** PCM output stream. */
79 R3PTRTYPE(PPDMAUDIOSTREAM) pStream;
80} SB16OUTPUTSTREAM, *PSB16OUTPUTSTREAM;
81
82/**
83 * Struct for maintaining a host backend driver.
84 */
85typedef struct SB16STATE *PSB16STATE;
86typedef struct SB16DRIVER
87{
88 /** Node for storing this driver in our device driver list of SB16STATE. */
89 RTLISTNODER3 Node;
90 /** Pointer to SB16 controller (state). */
91 R3PTRTYPE(PSB16STATE) pSB16State;
92 /** Driver flags. */
93 PDMAUDIODRVFLAGS Flags;
94 uint32_t PaddingFlags;
95 /** LUN # to which this driver has been assigned. */
96 uint8_t uLUN;
97 /** Whether this driver is in an attached state or not. */
98 bool fAttached;
99 uint8_t Padding[4];
100 /** Pointer to attached driver base interface. */
101 R3PTRTYPE(PPDMIBASE) pDrvBase;
102 /** Audio connector interface to the underlying host backend. */
103 R3PTRTYPE(PPDMIAUDIOCONNECTOR) pConnector;
104 /** Stream for output. */
105 SB16OUTPUTSTREAM Out;
106} SB16DRIVER, *PSB16DRIVER;
107
108typedef struct SB16STATE
109{
110#ifdef VBOX
111 /** Pointer to the device instance. */
112 PPDMDEVINSR3 pDevInsR3;
113 /** Pointer to the connector of the attached audio driver. */
114 PPDMIAUDIOCONNECTOR pDrv;
115 int irqCfg;
116 int dmaCfg;
117 int hdmaCfg;
118 int portCfg;
119 int verCfg;
120#endif
121 int irq;
122 int dma;
123 int hdma;
124 int port;
125 int ver;
126
127 int in_index;
128 int out_data_len;
129 int fmt_stereo;
130 int fmt_signed;
131 int fmt_bits;
132 PDMAUDIOFMT fmt;
133 int dma_auto;
134 int block_size;
135 int fifo;
136 int freq;
137 int time_const;
138 int speaker;
139 int needed_bytes;
140 int cmd;
141 int use_hdma;
142 int highspeed;
143 int can_write; /** @todo Value never gets 0? */
144
145 int v2x6;
146
147 uint8_t csp_param;
148 uint8_t csp_value;
149 uint8_t csp_mode;
150 uint8_t csp_regs[256];
151 uint8_t csp_index;
152 uint8_t csp_reg83[4];
153 int csp_reg83r;
154 int csp_reg83w;
155
156 uint8_t in2_data[10];
157 uint8_t out_data[50];
158 uint8_t test_reg;
159 uint8_t last_read_byte;
160 int nzero;
161
162 int left_till_irq; /** Note: Can be < 0. */
163
164 int dma_running;
165 int bytes_per_second;
166 int align;
167
168 RTLISTANCHOR lstDrv;
169 /** Number of active (running) SDn streams. */
170 uint8_t cStreamsActive;
171#ifndef VBOX_WITH_AUDIO_CALLBACKS
172 /** The timer for pumping data thru the attached LUN drivers. */
173 PTMTIMERR3 pTimerIO;
174 /** Flag indicating whether the timer is active or not. */
175 bool fTimerActive;
176 uint8_t u8Padding1[7];
177 /** The timer interval for pumping data thru the LUN drivers in timer ticks. */
178 uint64_t cTimerTicksIO;
179 /** Timestamp of the last timer callback (sb16TimerIO).
180 * Used to calculate the time actually elapsed between two timer callbacks. */
181 uint64_t uTimerTSIO;
182#endif
183 PTMTIMER pTimerIRQ;
184 /** The base interface for LUN\#0. */
185 PDMIBASE IBase;
186
187 /* mixer state */
188 int mixer_nreg;
189 uint8_t mixer_regs[256];
190} SB16STATE, *PSB16STATE;
191
192static int sb16OpenOut(PSB16STATE pThis, PPDMAUDIOSTREAMCFG pCfg);
193static void sb16CloseOut(PSB16STATE pThis);
194#ifndef VBOX_WITH_AUDIO_CALLBACKS
195static void sb16TimerMaybeStart(PSB16STATE pThis);
196static void sb16TimerMaybeStop(PSB16STATE pThis);
197#endif
198
199/**
200 * Attach command, internal version.
201 *
202 * This is called to let the device attach to a driver for a specified LUN
203 * during runtime. This is not called during VM construction, the device
204 * constructor has to attach to all the available drivers.
205 *
206 * @returns VBox status code.
207 * @param pDevIns The device instance.
208 * @param pDrv Driver to (re-)use for (re-)attaching to.
209 * If NULL is specified, a new driver will be created and appended
210 * to the driver list.
211 * @param uLUN The logical unit which is being detached.
212 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
213 */
214static DECLCALLBACK(int) sb16AttachInternal(PPDMDEVINS pDevIns, PSB16DRIVER pDrv, unsigned uLUN, uint32_t fFlags)
215{
216 PSB16STATE pThis = PDMINS_2_DATA(pDevIns, PSB16STATE);
217
218 /*
219 * Attach driver.
220 */
221 char *pszDesc = NULL;
222 if (RTStrAPrintf(&pszDesc, "Audio driver port (SB16) for LUN #%u", uLUN) <= 0)
223 AssertReleaseMsgReturn(pszDesc,
224 ("Not enough memory for SB16 driver port description of LUN #%u\n", uLUN),
225 VERR_NO_MEMORY);
226
227 PPDMIBASE pDrvBase;
228 int rc = PDMDevHlpDriverAttach(pDevIns, uLUN,
229 &pThis->IBase, &pDrvBase, pszDesc);
230 if (RT_SUCCESS(rc))
231 {
232 if (pDrv == NULL)
233 pDrv = (PSB16DRIVER)RTMemAllocZ(sizeof(SB16DRIVER));
234 if (pDrv)
235 {
236 pDrv->pDrvBase = pDrvBase;
237 pDrv->pConnector = PDMIBASE_QUERY_INTERFACE(pDrvBase, PDMIAUDIOCONNECTOR);
238 AssertMsg(pDrv->pConnector != NULL, ("Configuration error: LUN #%u has no host audio interface, rc=%Rrc\n", uLUN, rc));
239 pDrv->pSB16State = pThis;
240 pDrv->uLUN = uLUN;
241
242 /*
243 * For now we always set the driver at LUN 0 as our primary
244 * host backend. This might change in the future.
245 */
246 if (pDrv->uLUN == 0)
247 pDrv->Flags |= PDMAUDIODRVFLAG_PRIMARY;
248
249 LogFunc(("LUN#%RU8: pCon=%p, drvFlags=0x%x\n", uLUN, pDrv->pConnector, pDrv->Flags));
250
251 /* Attach to driver list if not attached yet. */
252 if (!pDrv->fAttached)
253 {
254 RTListAppend(&pThis->lstDrv, &pDrv->Node);
255 pDrv->fAttached = true;
256 }
257 }
258 else
259 rc = VERR_NO_MEMORY;
260 }
261 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
262 {
263 LogFunc(("No attached driver for LUN #%u\n", uLUN));
264 }
265 else if (RT_FAILURE(rc))
266 AssertMsgFailed(("Failed to attach SB16 LUN #%u (\"%s\"), rc=%Rrc\n",
267 uLUN, pszDesc, rc));
268
269 if (RT_FAILURE(rc))
270 {
271 /* Only free this string on failure;
272 * must remain valid for the live of the driver instance. */
273 RTStrFree(pszDesc);
274 }
275
276 LogFunc(("iLUN=%u, fFlags=0x%x, rc=%Rrc\n", uLUN, fFlags, rc));
277 return rc;
278}
279
280/**
281 * Attach command.
282 *
283 * This is called to let the device attach to a driver for a specified LUN
284 * during runtime. This is not called during VM construction, the device
285 * constructor has to attach to all the available drivers.
286 *
287 * @returns VBox status code.
288 * @param pDevIns The device instance.
289 * @param uLUN The logical unit which is being detached.
290 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
291 */
292static DECLCALLBACK(int) sb16Attach(PPDMDEVINS pDevIns, unsigned uLUN, uint32_t fFlags)
293{
294 return sb16AttachInternal(pDevIns, NULL /* pDrv */, uLUN, fFlags);
295}
296
297static DECLCALLBACK(void) sb16Detach(PPDMDEVINS pDevIns, unsigned uLUN, uint32_t fFlags)
298{
299 LogFunc(("iLUN=%u, fFlags=0x%x\n", uLUN, fFlags));
300}
301
302/**
303 * Re-attach.
304 *
305 * @returns VBox status code.
306 * @param pThis Device instance.
307 * @param pDrv Driver instance used for attaching to.
308 * If NULL is specified, a new driver will be created and appended
309 * to the driver list.
310 * @param uLUN The logical unit which is being re-detached.
311 * @param pszDriver Driver name.
312 */
313static int sb16Reattach(PSB16STATE pThis, PSB16DRIVER pDrv, uint8_t uLUN, const char *pszDriver)
314{
315 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
316 AssertPtrReturn(pszDriver, VERR_INVALID_POINTER);
317
318 PVM pVM = PDMDevHlpGetVM(pThis->pDevInsR3);
319 PCFGMNODE pRoot = CFGMR3GetRoot(pVM);
320 PCFGMNODE pDev0 = CFGMR3GetChild(pRoot, "Devices/sb16/0/");
321
322 /* Remove LUN branch. */
323 CFGMR3RemoveNode(CFGMR3GetChildF(pDev0, "LUN#%u/", uLUN));
324
325 if (pDrv)
326 {
327 /* Re-use the driver instance so detach it before. */
328 int rc = PDMDevHlpDriverDetach(pThis->pDevInsR3, PDMIBASE_2_PDMDRV(pDrv->pDrvBase), 0 /* fFlags */);
329 if (RT_FAILURE(rc))
330 return rc;
331 }
332
333#define RC_CHECK() if (RT_FAILURE(rc)) { AssertReleaseRC(rc); break; }
334
335 int rc = VINF_SUCCESS;
336 do
337 {
338 PCFGMNODE pLunL0;
339 rc = CFGMR3InsertNodeF(pDev0, &pLunL0, "LUN#%u/", uLUN); RC_CHECK();
340 rc = CFGMR3InsertString(pLunL0, "Driver", "AUDIO"); RC_CHECK();
341 rc = CFGMR3InsertNode(pLunL0, "Config/", NULL); RC_CHECK();
342
343 PCFGMNODE pLunL1, pLunL2;
344 rc = CFGMR3InsertNode (pLunL0, "AttachedDriver/", &pLunL1); RC_CHECK();
345 rc = CFGMR3InsertNode (pLunL1, "Config/", &pLunL2); RC_CHECK();
346 rc = CFGMR3InsertString(pLunL1, "Driver", pszDriver); RC_CHECK();
347
348 rc = CFGMR3InsertString(pLunL2, "AudioDriver", pszDriver); RC_CHECK();
349
350 } while (0);
351
352 if (RT_SUCCESS(rc))
353 rc = sb16AttachInternal(pThis->pDevInsR3, pDrv, uLUN, 0 /* fFlags */);
354
355 LogFunc(("pThis=%p, uLUN=%u, pszDriver=%s, rc=%Rrc\n", pThis, uLUN, pszDriver, rc));
356
357#undef RC_CHECK
358
359 return rc;
360}
361
362static void sb16AudioCallback(void *pvContext, uint32_t cbFree);
363
364static int magic_of_irq(int irq)
365{
366 switch (irq)
367 {
368 case 5:
369 return 2;
370 case 7:
371 return 4;
372 case 9:
373 return 1;
374 case 10:
375 return 8;
376 default:
377 break;
378 }
379
380 LogFlowFunc(("bad irq %d\n", irq));
381 return 2;
382}
383
384static int irq_of_magic(int magic)
385{
386 switch (magic)
387 {
388 case 1:
389 return 9;
390 case 2:
391 return 5;
392 case 4:
393 return 7;
394 case 8:
395 return 10;
396 default:
397 break;
398 }
399
400 LogFlowFunc(("bad irq magic %d\n", magic));
401 return -1;
402}
403
404#ifdef DEBUG
405static inline void log_dsp(PSB16STATE pThis)
406{
407 LogFlowFunc(("%s:%s:%d:%s:dmasize=%d:freq=%d:const=%d:speaker=%d\n",
408 pThis->fmt_stereo ? "Stereo" : "Mono",
409 pThis->fmt_signed ? "Signed" : "Unsigned",
410 pThis->fmt_bits,
411 pThis->dma_auto ? "Auto" : "Single",
412 pThis->block_size,
413 pThis->freq,
414 pThis->time_const,
415 pThis->speaker));
416}
417#endif
418
419static void sb16SpeakerControl(PSB16STATE pThis, int on)
420{
421 pThis->speaker = on;
422 /* AUD_enable (pThis->voice, on); */
423}
424
425static void sb16Control(PSB16STATE pThis, int hold)
426{
427 int dma = pThis->use_hdma ? pThis->hdma : pThis->dma;
428 pThis->dma_running = hold;
429
430 LogFlowFunc(("hold %d high %d dma %d\n", hold, pThis->use_hdma, dma));
431
432 PDMDevHlpDMASetDREQ(pThis->pDevInsR3, dma, hold);
433
434 if (hold)
435 {
436 pThis->cStreamsActive++;
437#ifndef VBOX_WITH_AUDIO_CALLBACKS
438 sb16TimerMaybeStart(pThis);
439#endif
440 PDMDevHlpDMASchedule(pThis->pDevInsR3);
441 }
442#ifndef VBOX_WITH_AUDIO_CALLBACKS
443 else
444 {
445 if (pThis->cStreamsActive)
446 pThis->cStreamsActive--;
447 sb16TimerMaybeStop(pThis);
448 }
449#endif
450
451 PSB16DRIVER pDrv;
452 RTListForEach(&pThis->lstDrv, pDrv, SB16DRIVER, Node)
453 {
454 if (!pDrv->Out.pStream)
455 continue;
456
457 int rc2 = pDrv->pConnector->pfnStreamControl(pDrv->pConnector, pDrv->Out.pStream,
458 hold == 1 ? PDMAUDIOSTREAMCMD_ENABLE : PDMAUDIOSTREAMCMD_DISABLE);
459 LogFlowFunc(("%s: rc=%Rrc\n", pDrv->Out.pStream->szName, rc2));
460 }
461}
462
463static DECLCALLBACK(void) sb16TimerIRQ(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvThis)
464{
465 PSB16STATE pThis = (PSB16STATE)pvThis;
466 pThis->can_write = 1;
467 PDMDevHlpISASetIrq(pThis->pDevInsR3, pThis->irq, 1);
468}
469
470#define DMA8_AUTO 1
471#define DMA8_HIGH 2
472
473static void continue_dma8(PSB16STATE pThis)
474{
475 if (pThis->freq > 0)
476 {
477 PDMAUDIOSTREAMCFG streamCfg;
478 RT_ZERO(streamCfg);
479 streamCfg.enmDir = PDMAUDIODIR_OUT;
480 streamCfg.DestSource.Dest = PDMAUDIOPLAYBACKDEST_FRONT;
481 streamCfg.uHz = pThis->freq;
482 streamCfg.cChannels = 1 << pThis->fmt_stereo;
483 streamCfg.enmFormat = pThis->fmt;
484 streamCfg.enmEndianness = PDMAUDIOHOSTENDIANNESS;
485
486 int rc = sb16OpenOut(pThis, &streamCfg);
487 AssertRC(rc);
488 }
489
490 sb16Control(pThis, 1);
491}
492
493static void dma_cmd8(PSB16STATE pThis, int mask, int dma_len)
494{
495 pThis->fmt = PDMAUDIOFMT_U8;
496 pThis->use_hdma = 0;
497 pThis->fmt_bits = 8;
498 pThis->fmt_signed = 0;
499 pThis->fmt_stereo = (pThis->mixer_regs[0x0e] & 2) != 0;
500
501 if (-1 == pThis->time_const)
502 {
503 if (pThis->freq <= 0)
504 pThis->freq = 11025;
505 }
506 else
507 {
508 int tmp = (256 - pThis->time_const);
509 pThis->freq = (1000000 + (tmp / 2)) / tmp;
510 }
511
512 if (dma_len != -1)
513 {
514 pThis->block_size = dma_len << pThis->fmt_stereo;
515 }
516 else
517 {
518 /* This is apparently the only way to make both Act1/PL
519 and SecondReality/FC work
520
521 r=andy Wow, actually someone who remembers Future Crew :-)
522
523 Act1 sets block size via command 0x48 and it's an odd number
524 SR does the same with even number
525 Both use stereo, and Creatives own documentation states that
526 0x48 sets block size in bytes less one.. go figure */
527 pThis->block_size &= ~pThis->fmt_stereo;
528 }
529
530 pThis->freq >>= pThis->fmt_stereo;
531 pThis->left_till_irq = pThis->block_size;
532 pThis->bytes_per_second = (pThis->freq << pThis->fmt_stereo);
533 /* pThis->highspeed = (mask & DMA8_HIGH) != 0; */
534 pThis->dma_auto = (mask & DMA8_AUTO) != 0;
535 pThis->align = (1 << pThis->fmt_stereo) - 1;
536
537 if (pThis->block_size & pThis->align)
538 LogFlowFunc(("warning: misaligned block size %d, alignment %d\n",
539 pThis->block_size, pThis->align + 1));
540
541 LogFlowFunc(("freq %d, stereo %d, sign %d, bits %d, dma %d, auto %d, fifo %d, high %d\n",
542 pThis->freq, pThis->fmt_stereo, pThis->fmt_signed, pThis->fmt_bits,
543 pThis->block_size, pThis->dma_auto, pThis->fifo, pThis->highspeed));
544
545 continue_dma8(pThis);
546 sb16SpeakerControl(pThis, 1);
547}
548
549static void dma_cmd(PSB16STATE pThis, uint8_t cmd, uint8_t d0, int dma_len)
550{
551 pThis->use_hdma = cmd < 0xc0;
552 pThis->fifo = (cmd >> 1) & 1;
553 pThis->dma_auto = (cmd >> 2) & 1;
554 pThis->fmt_signed = (d0 >> 4) & 1;
555 pThis->fmt_stereo = (d0 >> 5) & 1;
556
557 switch (cmd >> 4)
558 {
559 case 11:
560 pThis->fmt_bits = 16;
561 break;
562
563 case 12:
564 pThis->fmt_bits = 8;
565 break;
566 }
567
568 if (-1 != pThis->time_const)
569 {
570#if 1
571 int tmp = 256 - pThis->time_const;
572 pThis->freq = (1000000 + (tmp / 2)) / tmp;
573#else
574 /* pThis->freq = 1000000 / ((255 - pThis->time_const) << pThis->fmt_stereo); */
575 pThis->freq = 1000000 / ((255 - pThis->time_const));
576#endif
577 pThis->time_const = -1;
578 }
579
580 pThis->block_size = dma_len + 1;
581 pThis->block_size <<= ((pThis->fmt_bits == 16) ? 1 : 0);
582 if (!pThis->dma_auto)
583 {
584 /*
585 * It is clear that for DOOM and auto-init this value
586 * shouldn't take stereo into account, while Miles Sound Systems
587 * setsound.exe with single transfer mode wouldn't work without it
588 * wonders of SB16 yet again.
589 */
590 pThis->block_size <<= pThis->fmt_stereo;
591 }
592
593 LogFlowFunc(("freq %d, stereo %d, sign %d, bits %d, dma %d, auto %d, fifo %d, high %d\n",
594 pThis->freq, pThis->fmt_stereo, pThis->fmt_signed, pThis->fmt_bits,
595 pThis->block_size, pThis->dma_auto, pThis->fifo, pThis->highspeed));
596
597 if (16 == pThis->fmt_bits)
598 pThis->fmt = pThis->fmt_signed ? PDMAUDIOFMT_S16 : PDMAUDIOFMT_U16;
599 else
600 pThis->fmt = pThis->fmt_signed ? PDMAUDIOFMT_S8 : PDMAUDIOFMT_U8;
601
602 pThis->left_till_irq = pThis->block_size;
603
604 pThis->bytes_per_second = (pThis->freq << pThis->fmt_stereo) << ((pThis->fmt_bits == 16) ? 1 : 0);
605 pThis->highspeed = 0;
606 pThis->align = (1 << (pThis->fmt_stereo + (pThis->fmt_bits == 16))) - 1;
607 if (pThis->block_size & pThis->align)
608 {
609 LogFlowFunc(("warning: misaligned block size %d, alignment %d\n",
610 pThis->block_size, pThis->align + 1));
611 }
612
613 if (pThis->freq)
614 {
615 PDMAUDIOSTREAMCFG streamCfg;
616 RT_ZERO(streamCfg);
617 streamCfg.enmDir = PDMAUDIODIR_OUT;
618 streamCfg.DestSource.Dest = PDMAUDIOPLAYBACKDEST_FRONT;
619 streamCfg.uHz = pThis->freq;
620 streamCfg.cChannels = 1 << pThis->fmt_stereo;
621 streamCfg.enmFormat = pThis->fmt;
622 streamCfg.enmEndianness = PDMAUDIOHOSTENDIANNESS;
623
624 int rc = sb16OpenOut(pThis, &streamCfg);
625 AssertRC(rc);
626 }
627
628 sb16Control(pThis, 1);
629 sb16SpeakerControl(pThis, 1);
630}
631
632static inline void dsp_out_data (PSB16STATE pThis, uint8_t val)
633{
634 LogFlowFunc(("outdata %#x\n", val));
635 if ((size_t) pThis->out_data_len < sizeof (pThis->out_data)) {
636 pThis->out_data[pThis->out_data_len++] = val;
637 }
638}
639
640static inline uint8_t dsp_get_data (PSB16STATE pThis)
641{
642 if (pThis->in_index) {
643 return pThis->in2_data[--pThis->in_index];
644 }
645 else {
646 LogFlowFunc(("buffer underflow\n"));
647 return 0;
648 }
649}
650
651static void sb16HandleCommand(PSB16STATE pThis, uint8_t cmd)
652{
653 LogFlowFunc(("command %#x\n", cmd));
654
655 if (cmd > 0xaf && cmd < 0xd0)
656 {
657 if (cmd & 8) /** @todo Handle recording. */
658 LogFlowFunc(("ADC not yet supported (command %#x)\n", cmd));
659
660 switch (cmd >> 4)
661 {
662 case 11:
663 case 12:
664 break;
665 default:
666 LogFlowFunc(("%#x wrong bits\n", cmd));
667 }
668
669 pThis->needed_bytes = 3;
670 }
671 else
672 {
673 pThis->needed_bytes = 0;
674
675 switch (cmd)
676 {
677 case 0x03:
678 dsp_out_data(pThis, 0x10); /* pThis->csp_param); */
679 goto warn;
680
681 case 0x04:
682 pThis->needed_bytes = 1;
683 goto warn;
684
685 case 0x05:
686 pThis->needed_bytes = 2;
687 goto warn;
688
689 case 0x08:
690 /* __asm__ ("int3"); */
691 goto warn;
692
693 case 0x0e:
694 pThis->needed_bytes = 2;
695 goto warn;
696
697 case 0x09:
698 dsp_out_data(pThis, 0xf8);
699 goto warn;
700
701 case 0x0f:
702 pThis->needed_bytes = 1;
703 goto warn;
704
705 case 0x10:
706 pThis->needed_bytes = 1;
707 goto warn;
708
709 case 0x14:
710 pThis->needed_bytes = 2;
711 pThis->block_size = 0;
712 break;
713
714 case 0x1c: /* Auto-Initialize DMA DAC, 8-bit */
715 dma_cmd8(pThis, DMA8_AUTO, -1);
716 break;
717
718 case 0x20: /* Direct ADC, Juice/PL */
719 dsp_out_data(pThis, 0xff);
720 goto warn;
721
722 case 0x35:
723 LogFlowFunc(("0x35 - MIDI command not implemented\n"));
724 break;
725
726 case 0x40:
727 pThis->freq = -1;
728 pThis->time_const = -1;
729 pThis->needed_bytes = 1;
730 break;
731
732 case 0x41:
733 pThis->freq = -1;
734 pThis->time_const = -1;
735 pThis->needed_bytes = 2;
736 break;
737
738 case 0x42:
739 pThis->freq = -1;
740 pThis->time_const = -1;
741 pThis->needed_bytes = 2;
742 goto warn;
743
744 case 0x45:
745 dsp_out_data(pThis, 0xaa);
746 goto warn;
747
748 case 0x47: /* Continue Auto-Initialize DMA 16bit */
749 break;
750
751 case 0x48:
752 pThis->needed_bytes = 2;
753 break;
754
755 case 0x74:
756 pThis->needed_bytes = 2; /* DMA DAC, 4-bit ADPCM */
757 LogFlowFunc(("0x75 - DMA DAC, 4-bit ADPCM not implemented\n"));
758 break;
759
760 case 0x75: /* DMA DAC, 4-bit ADPCM Reference */
761 pThis->needed_bytes = 2;
762 LogFlowFunc(("0x74 - DMA DAC, 4-bit ADPCM Reference not implemented\n"));
763 break;
764
765 case 0x76: /* DMA DAC, 2.6-bit ADPCM */
766 pThis->needed_bytes = 2;
767 LogFlowFunc(("0x74 - DMA DAC, 2.6-bit ADPCM not implemented\n"));
768 break;
769
770 case 0x77: /* DMA DAC, 2.6-bit ADPCM Reference */
771 pThis->needed_bytes = 2;
772 LogFlowFunc(("0x74 - DMA DAC, 2.6-bit ADPCM Reference not implemented\n"));
773 break;
774
775 case 0x7d:
776 LogFlowFunc(("0x7d - Autio-Initialize DMA DAC, 4-bit ADPCM Reference\n"));
777 LogFlowFunc(("not implemented\n"));
778 break;
779
780 case 0x7f:
781 LogFlowFunc(("0x7d - Autio-Initialize DMA DAC, 2.6-bit ADPCM Reference\n"));
782 LogFlowFunc(("not implemented\n"));
783 break;
784
785 case 0x80:
786 pThis->needed_bytes = 2;
787 break;
788
789 case 0x90:
790 case 0x91:
791 dma_cmd8(pThis, (((cmd & 1) == 0) ? 1 : 0) | DMA8_HIGH, -1);
792 break;
793
794 case 0xd0: /* halt DMA operation. 8bit */
795 sb16Control(pThis, 0);
796 break;
797
798 case 0xd1: /* speaker on */
799 sb16SpeakerControl(pThis, 1);
800 break;
801
802 case 0xd3: /* speaker off */
803 sb16SpeakerControl(pThis, 0);
804 break;
805
806 case 0xd4: /* continue DMA operation. 8bit */
807 /* KQ6 (or maybe Sierras audblst.drv in general) resets
808 the frequency between halt/continue */
809 continue_dma8(pThis);
810 break;
811
812 case 0xd5: /* halt DMA operation. 16bit */
813 sb16Control(pThis, 0);
814 break;
815
816 case 0xd6: /* continue DMA operation. 16bit */
817 sb16Control(pThis, 1);
818 break;
819
820 case 0xd9: /* exit auto-init DMA after this block. 16bit */
821 pThis->dma_auto = 0;
822 break;
823
824 case 0xda: /* exit auto-init DMA after this block. 8bit */
825 pThis->dma_auto = 0;
826 break;
827
828 case 0xe0: /* DSP identification */
829 pThis->needed_bytes = 1;
830 break;
831
832 case 0xe1:
833 dsp_out_data(pThis, pThis->ver & 0xff);
834 dsp_out_data(pThis, pThis->ver >> 8);
835 break;
836
837 case 0xe2:
838 pThis->needed_bytes = 1;
839 goto warn;
840
841 case 0xe3:
842 {
843 for (int i = sizeof (e3) - 1; i >= 0; --i)
844 dsp_out_data(pThis, e3[i]);
845
846 break;
847 }
848
849 case 0xe4: /* write test reg */
850 pThis->needed_bytes = 1;
851 break;
852
853 case 0xe7:
854 LogFlowFunc(("Attempt to probe for ESS (0xe7)?\n"));
855 break;
856
857 case 0xe8: /* read test reg */
858 dsp_out_data(pThis, pThis->test_reg);
859 break;
860
861 case 0xf2:
862 case 0xf3:
863 dsp_out_data(pThis, 0xaa);
864 pThis->mixer_regs[0x82] |= (cmd == 0xf2) ? 1 : 2;
865 PDMDevHlpISASetIrq(pThis->pDevInsR3, pThis->irq, 1);
866 break;
867
868 case 0xf8:
869 /* Undocumented, used by old Creative diagnostic programs. */
870 dsp_out_data (pThis, 0);
871 goto warn;
872
873 case 0xf9:
874 pThis->needed_bytes = 1;
875 goto warn;
876
877 case 0xfa:
878 dsp_out_data (pThis, 0);
879 goto warn;
880
881 case 0xfc: /* FIXME */
882 dsp_out_data (pThis, 0);
883 goto warn;
884
885 default:
886 LogFlowFunc(("Unrecognized command %#x\n", cmd));
887 break;
888 }
889 }
890
891 if (!pThis->needed_bytes)
892 LogFlow(("\n"));
893
894exit:
895
896 if (!pThis->needed_bytes)
897 pThis->cmd = -1;
898 else
899 pThis->cmd = cmd;
900
901 return;
902
903warn:
904 LogFlowFunc(("warning: command %#x,%d is not truly understood yet\n",
905 cmd, pThis->needed_bytes));
906 goto exit;
907}
908
909static uint16_t dsp_get_lohi (PSB16STATE pThis)
910{
911 uint8_t hi = dsp_get_data (pThis);
912 uint8_t lo = dsp_get_data (pThis);
913 return (hi << 8) | lo;
914}
915
916static uint16_t dsp_get_hilo (PSB16STATE pThis)
917{
918 uint8_t lo = dsp_get_data (pThis);
919 uint8_t hi = dsp_get_data (pThis);
920 return (hi << 8) | lo;
921}
922
923static void complete(PSB16STATE pThis)
924{
925 int d0, d1, d2;
926 LogFlowFunc(("complete command %#x, in_index %d, needed_bytes %d\n",
927 pThis->cmd, pThis->in_index, pThis->needed_bytes));
928
929 if (pThis->cmd > 0xaf && pThis->cmd < 0xd0)
930 {
931 d2 = dsp_get_data (pThis);
932 d1 = dsp_get_data (pThis);
933 d0 = dsp_get_data (pThis);
934
935 if (pThis->cmd & 8)
936 LogFlowFunc(("ADC params cmd = %#x d0 = %d, d1 = %d, d2 = %d\n", pThis->cmd, d0, d1, d2));
937 else
938 {
939 LogFlowFunc(("cmd = %#x d0 = %d, d1 = %d, d2 = %d\n", pThis->cmd, d0, d1, d2));
940 dma_cmd(pThis, pThis->cmd, d0, d1 + (d2 << 8));
941 }
942 }
943 else
944 {
945 switch (pThis->cmd)
946 {
947 case 0x04:
948 pThis->csp_mode = dsp_get_data (pThis);
949 pThis->csp_reg83r = 0;
950 pThis->csp_reg83w = 0;
951 LogFlowFunc(("CSP command 0x04: mode=%#x\n", pThis->csp_mode));
952 break;
953
954 case 0x05:
955 pThis->csp_param = dsp_get_data (pThis);
956 pThis->csp_value = dsp_get_data (pThis);
957 LogFlowFunc(("CSP command 0x05: param=%#x value=%#x\n",
958 pThis->csp_param,
959 pThis->csp_value));
960 break;
961
962 case 0x0e:
963 {
964 d0 = dsp_get_data(pThis);
965 d1 = dsp_get_data(pThis);
966 LogFlowFunc(("write CSP register %d <- %#x\n", d1, d0));
967 if (d1 == 0x83)
968 {
969 LogFlowFunc(("0x83[%d] <- %#x\n", pThis->csp_reg83r, d0));
970 pThis->csp_reg83[pThis->csp_reg83r % 4] = d0;
971 pThis->csp_reg83r += 1;
972 }
973 else
974 pThis->csp_regs[d1] = d0;
975 break;
976 }
977
978 case 0x0f:
979 d0 = dsp_get_data(pThis);
980 LogFlowFunc(("read CSP register %#x -> %#x, mode=%#x\n", d0, pThis->csp_regs[d0], pThis->csp_mode));
981 if (d0 == 0x83)
982 {
983 LogFlowFunc(("0x83[%d] -> %#x\n",
984 pThis->csp_reg83w,
985 pThis->csp_reg83[pThis->csp_reg83w % 4]));
986 dsp_out_data (pThis, pThis->csp_reg83[pThis->csp_reg83w % 4]);
987 pThis->csp_reg83w += 1;
988 }
989 else
990 dsp_out_data(pThis, pThis->csp_regs[d0]);
991 break;
992
993 case 0x10:
994 d0 = dsp_get_data(pThis);
995 LogFlowFunc(("cmd 0x10 d0=%#x\n", d0));
996 break;
997
998 case 0x14:
999 dma_cmd8(pThis, 0, dsp_get_lohi (pThis) + 1);
1000 break;
1001
1002 case 0x40:
1003 pThis->time_const = dsp_get_data(pThis);
1004 LogFlowFunc(("set time const %d\n", pThis->time_const));
1005 break;
1006
1007 case 0x42: /* FT2 sets output freq with this, go figure */
1008#if 0
1009 LogFlowFunc(("cmd 0x42 might not do what it think it should\n"));
1010#endif
1011 case 0x41:
1012 pThis->freq = dsp_get_hilo(pThis);
1013 LogFlowFunc(("set freq %d\n", pThis->freq));
1014 break;
1015
1016 case 0x48:
1017 pThis->block_size = dsp_get_lohi(pThis) + 1;
1018 LogFlowFunc(("set dma block len %d\n", pThis->block_size));
1019 break;
1020
1021 case 0x74:
1022 case 0x75:
1023 case 0x76:
1024 case 0x77:
1025 /* ADPCM stuff, ignore */
1026 break;
1027
1028 case 0x80:
1029 {
1030 int freq, samples, bytes;
1031 uint64_t ticks;
1032
1033 freq = pThis->freq > 0 ? pThis->freq : 11025;
1034 samples = dsp_get_lohi (pThis) + 1;
1035 bytes = samples << pThis->fmt_stereo << ((pThis->fmt_bits == 16) ? 1 : 0);
1036 ticks = (bytes * TMTimerGetFreq(pThis->pTimerIRQ)) / freq;
1037 if (ticks < TMTimerGetFreq(pThis->pTimerIRQ) / 1024)
1038 PDMDevHlpISASetIrq(pThis->pDevInsR3, pThis->irq, 1);
1039 else
1040 TMTimerSet(pThis->pTimerIRQ, TMTimerGet(pThis->pTimerIRQ) + ticks);
1041 LogFlowFunc(("mix silence %d %d % %RU64\n", samples, bytes, ticks));
1042 break;
1043 }
1044
1045 case 0xe0:
1046 d0 = dsp_get_data(pThis);
1047 pThis->out_data_len = 0;
1048 LogFlowFunc(("E0 data = %#x\n", d0));
1049 dsp_out_data(pThis, ~d0);
1050 break;
1051
1052 case 0xe2:
1053 d0 = dsp_get_data(pThis);
1054 LogFlow(("SB16:E2 = %#x\n", d0));
1055 break;
1056
1057 case 0xe4:
1058 pThis->test_reg = dsp_get_data(pThis);
1059 break;
1060
1061 case 0xf9:
1062 d0 = dsp_get_data(pThis);
1063 LogFlowFunc(("command 0xf9 with %#x\n", d0));
1064 switch (d0) {
1065 case 0x0e:
1066 dsp_out_data(pThis, 0xff);
1067 break;
1068
1069 case 0x0f:
1070 dsp_out_data(pThis, 0x07);
1071 break;
1072
1073 case 0x37:
1074 dsp_out_data(pThis, 0x38);
1075 break;
1076
1077 default:
1078 dsp_out_data(pThis, 0x00);
1079 break;
1080 }
1081 break;
1082
1083 default:
1084 LogFlowFunc(("complete: unrecognized command %#x\n", pThis->cmd));
1085 return;
1086 }
1087 }
1088
1089 LogFlow(("\n"));
1090 pThis->cmd = -1;
1091 return;
1092}
1093
1094static uint8_t sb16MixRegToVol(PSB16STATE pThis, int reg)
1095{
1096 /* The SB16 mixer has a 0 to -62dB range in 32 levels (2dB each step).
1097 * We use a 0 to -96dB range in 256 levels (0.375dB each step).
1098 * Only the top 5 bits of a mixer register are used.
1099 */
1100 uint8_t steps = 31 - (pThis->mixer_regs[reg] >> 3);
1101 uint8_t vol = 255 - steps * 16 / 3; /* (2dB*8) / (0.375dB*8) */
1102 return vol;
1103}
1104
1105static void sb16SetMasterVolume(PSB16STATE pThis)
1106{
1107 /* There's no mute switch, only volume controls. */
1108 uint8_t lvol = sb16MixRegToVol(pThis, 0x30);
1109 uint8_t rvol = sb16MixRegToVol(pThis, 0x31);
1110 PDMAUDIOVOLUME vol = { false, lvol, rvol };
1111
1112// AudioMixerSetMasterVolume(pThis->pMixer, &vol);
1113}
1114
1115static void sb16SetPcmOutVolume(PSB16STATE pThis)
1116{
1117 /* There's no mute switch, only volume controls. */
1118 uint8_t lvol = sb16MixRegToVol(pThis, 0x32);
1119 uint8_t rvol = sb16MixRegToVol(pThis, 0x33);
1120 PDMAUDIOVOLUME vol = { false, lvol, rvol };
1121
1122 // AudioMixerSinkSetVolume(pThis->pSinkOutput, &vol);
1123}
1124
1125static void sb16ResetLegacy(PSB16STATE pThis)
1126{
1127 sb16CloseOut(pThis);
1128
1129 pThis->freq = 11025;
1130 pThis->fmt_signed = 0;
1131 pThis->fmt_bits = 8;
1132 pThis->fmt_stereo = 0;
1133
1134 PDMAUDIOSTREAMCFG streamCfg;
1135 RT_ZERO(streamCfg);
1136 streamCfg.enmDir = PDMAUDIODIR_OUT;
1137 streamCfg.DestSource.Dest = PDMAUDIOPLAYBACKDEST_FRONT;
1138 streamCfg.uHz = pThis->freq;
1139 streamCfg.cChannels = 1; /* Mono */
1140 streamCfg.enmFormat = PDMAUDIOFMT_U8;
1141 streamCfg.enmEndianness = PDMAUDIOHOSTENDIANNESS;
1142
1143 int rc2 = sb16OpenOut(pThis, &streamCfg);
1144 AssertRC(rc2);
1145}
1146
1147static void sb16Reset(PSB16STATE pThis)
1148{
1149 PDMDevHlpISASetIrq(pThis->pDevInsR3, pThis->irq, 0);
1150 if (pThis->dma_auto)
1151 {
1152 PDMDevHlpISASetIrq(pThis->pDevInsR3, pThis->irq, 1);
1153 PDMDevHlpISASetIrq(pThis->pDevInsR3, pThis->irq, 0);
1154 }
1155
1156 pThis->mixer_regs[0x82] = 0;
1157 pThis->dma_auto = 0;
1158 pThis->in_index = 0;
1159 pThis->out_data_len = 0;
1160 pThis->left_till_irq = 0;
1161 pThis->needed_bytes = 0;
1162 pThis->block_size = -1;
1163 pThis->nzero = 0;
1164 pThis->highspeed = 0;
1165 pThis->v2x6 = 0;
1166 pThis->cmd = -1;
1167
1168 dsp_out_data(pThis, 0xaa);
1169 sb16SpeakerControl(pThis, 0);
1170 sb16Control(pThis, 0);
1171 sb16ResetLegacy(pThis);
1172}
1173
1174static IO_WRITE_PROTO(dsp_write)
1175{
1176 PSB16STATE pThis = (PSB16STATE)opaque;
1177 int iport = nport - pThis->port;
1178
1179 LogFlowFunc(("write %#x <- %#x\n", nport, val));
1180 switch (iport)
1181 {
1182 case 0x06:
1183 switch (val)
1184 {
1185 case 0x00:
1186 {
1187 if (pThis->v2x6 == 1)
1188 {
1189 if (0 && pThis->highspeed)
1190 {
1191 pThis->highspeed = 0;
1192 PDMDevHlpISASetIrq(pThis->pDevInsR3, pThis->irq, 0);
1193 sb16Control(pThis, 0);
1194 }
1195 else
1196 sb16Reset(pThis);
1197 }
1198 pThis->v2x6 = 0;
1199 break;
1200 }
1201
1202 case 0x01:
1203 case 0x03: /* FreeBSD kludge */
1204 pThis->v2x6 = 1;
1205 break;
1206
1207 case 0xc6:
1208 pThis->v2x6 = 0; /* Prince of Persia, csp.sys, diagnose.exe */
1209 break;
1210
1211 case 0xb8: /* Panic */
1212 sb16Reset(pThis);
1213 break;
1214
1215 case 0x39:
1216 dsp_out_data(pThis, 0x38);
1217 sb16Reset(pThis);
1218 pThis->v2x6 = 0x39;
1219 break;
1220
1221 default:
1222 pThis->v2x6 = val;
1223 break;
1224 }
1225 break;
1226
1227 case 0x0c: /* Write data or command | write status */
1228#if 0
1229 if (pThis->highspeed)
1230 break;
1231#endif
1232 if (0 == pThis->needed_bytes)
1233 {
1234 sb16HandleCommand(pThis, val);
1235#if 0
1236 if (0 == pThis->needed_bytes) {
1237 log_dsp (pThis);
1238 }
1239#endif
1240 }
1241 else
1242 {
1243 if (pThis->in_index == sizeof (pThis->in2_data))
1244 {
1245 LogFlowFunc(("in data overrun\n"));
1246 }
1247 else
1248 {
1249 pThis->in2_data[pThis->in_index++] = val;
1250 if (pThis->in_index == pThis->needed_bytes)
1251 {
1252 pThis->needed_bytes = 0;
1253 complete (pThis);
1254#if 0
1255 log_dsp (pThis);
1256#endif
1257 }
1258 }
1259 }
1260 break;
1261
1262 default:
1263 LogFlowFunc(("nport=%#x, val=%#x)\n", nport, val));
1264 break;
1265 }
1266
1267 return VINF_SUCCESS;
1268}
1269
1270static IO_READ_PROTO(dsp_read)
1271{
1272 PSB16STATE pThis = (PSB16STATE)opaque;
1273 int iport, retval, ack = 0;
1274
1275 iport = nport - pThis->port;
1276
1277 /** @todo reject non-byte access?
1278 * The spec does not mention a non-byte access so we should check how real hardware behaves. */
1279
1280 switch (iport)
1281 {
1282 case 0x06: /* reset */
1283 retval = 0xff;
1284 break;
1285
1286 case 0x0a: /* read data */
1287 if (pThis->out_data_len)
1288 {
1289 retval = pThis->out_data[--pThis->out_data_len];
1290 pThis->last_read_byte = retval;
1291 }
1292 else
1293 {
1294 if (pThis->cmd != -1)
1295 LogFlowFunc(("empty output buffer for command %#x\n", pThis->cmd));
1296 retval = pThis->last_read_byte;
1297 /* goto error; */
1298 }
1299 break;
1300
1301 case 0x0c: /* 0 can write */
1302 retval = pThis->can_write ? 0 : 0x80;
1303 break;
1304
1305 case 0x0d: /* timer interrupt clear */
1306 /* LogFlowFunc(("timer interrupt clear\n")); */
1307 retval = 0;
1308 break;
1309
1310 case 0x0e: /* data available status | irq 8 ack */
1311 retval = (!pThis->out_data_len || pThis->highspeed) ? 0 : 0x80;
1312 if (pThis->mixer_regs[0x82] & 1)
1313 {
1314 ack = 1;
1315 pThis->mixer_regs[0x82] &= ~1;
1316 PDMDevHlpISASetIrq(pThis->pDevInsR3, pThis->irq, 0);
1317 }
1318 break;
1319
1320 case 0x0f: /* irq 16 ack */
1321 retval = 0xff;
1322 if (pThis->mixer_regs[0x82] & 2)
1323 {
1324 ack = 1;
1325 pThis->mixer_regs[0x82] &= ~2;
1326 PDMDevHlpISASetIrq(pThis->pDevInsR3, pThis->irq, 0);
1327 }
1328 break;
1329
1330 default:
1331 goto error;
1332 }
1333
1334 if (!ack)
1335 LogFlowFunc(("read %#x -> %#x\n", nport, retval));
1336
1337 *pu32 = retval;
1338 return VINF_SUCCESS;
1339
1340 error:
1341 LogFlowFunc(("warning: dsp_read %#x error\n", nport));
1342 return VERR_IOM_IOPORT_UNUSED;
1343}
1344
1345static void sb16MixerReset(PSB16STATE pThis)
1346{
1347 memset(pThis->mixer_regs, 0xff, 0x7f);
1348 memset(pThis->mixer_regs + 0x83, 0xff, sizeof (pThis->mixer_regs) - 0x83);
1349
1350 pThis->mixer_regs[0x02] = 4; /* master volume 3bits */
1351 pThis->mixer_regs[0x06] = 4; /* MIDI volume 3bits */
1352 pThis->mixer_regs[0x08] = 0; /* CD volume 3bits */
1353 pThis->mixer_regs[0x0a] = 0; /* voice volume 2bits */
1354
1355 /* d5=input filt, d3=lowpass filt, d1,d2=input source */
1356 pThis->mixer_regs[0x0c] = 0;
1357
1358 /* d5=output filt, d1=stereo switch */
1359 pThis->mixer_regs[0x0e] = 0;
1360
1361 /* voice volume L d5,d7, R d1,d3 */
1362 pThis->mixer_regs[0x04] = (12 << 4) | 12;
1363 /* master ... */
1364 pThis->mixer_regs[0x22] = (12 << 4) | 12;
1365 /* MIDI ... */
1366 pThis->mixer_regs[0x26] = (12 << 4) | 12;
1367
1368 /* master/voice/MIDI L/R volume */
1369 for (int i = 0x30; i < 0x36; i++)
1370 pThis->mixer_regs[i] = 24 << 3; /* -14 dB */
1371
1372 /* treble/bass */
1373 for (int i = 0x44; i < 0x48; i++)
1374 pThis->mixer_regs[i] = 0x80;
1375
1376 /* Update the master (mixer) and PCM out volumes. */
1377 sb16SetMasterVolume(pThis);
1378 sb16SetPcmOutVolume(pThis);
1379}
1380
1381static IO_WRITE_PROTO(mixer_write_indexb)
1382{
1383 PSB16STATE pThis = (PSB16STATE)opaque;
1384 (void) nport;
1385 pThis->mixer_nreg = val;
1386
1387 return VINF_SUCCESS;
1388}
1389
1390uint32_t popcount(uint32_t u) /** @todo r=andy WTF? */
1391{
1392 u = ((u&0x55555555) + ((u>>1)&0x55555555));
1393 u = ((u&0x33333333) + ((u>>2)&0x33333333));
1394 u = ((u&0x0f0f0f0f) + ((u>>4)&0x0f0f0f0f));
1395 u = ((u&0x00ff00ff) + ((u>>8)&0x00ff00ff));
1396 u = ( u&0x0000ffff) + (u>>16);
1397 return u;
1398}
1399
1400uint32_t lsbindex(uint32_t u)
1401{
1402 return popcount((u & -(int32_t)u) - 1);
1403}
1404
1405/* Convert SB16 to SB Pro mixer volume (left). */
1406static inline void sb16ConvVolumeL(PSB16STATE pThis, unsigned reg, uint8_t val)
1407{
1408 /* High nibble in SBP mixer. */
1409 pThis->mixer_regs[reg] = (pThis->mixer_regs[reg] & 0x0f) | (val & 0xf0);
1410}
1411
1412/* Convert SB16 to SB Pro mixer volume (right). */
1413static inline void sb16ConvVolumeR(PSB16STATE pThis, unsigned reg, uint8_t val)
1414{
1415 /* Low nibble in SBP mixer. */
1416 pThis->mixer_regs[reg] = (pThis->mixer_regs[reg] & 0xf0) | (val >> 4);
1417}
1418
1419/* Convert SB Pro to SB16 mixer volume (left + right). */
1420static inline void sb16ConvVolumeOldToNew(PSB16STATE pThis, unsigned reg, uint8_t val)
1421{
1422 /* Left channel. */
1423 pThis->mixer_regs[reg + 0] = (val & 0xf0) | RT_BIT(3);
1424 /* Right channel (the register immediately following). */
1425 pThis->mixer_regs[reg + 1] = (val << 4) | RT_BIT(3);
1426}
1427
1428static IO_WRITE_PROTO(mixer_write_datab)
1429{
1430 PSB16STATE pThis = (PSB16STATE)opaque;
1431 bool fUpdateMaster = false;
1432 bool fUpdateStream = false;
1433
1434 (void) nport;
1435 LogFlowFunc(("mixer_write [%#x] <- %#x\n", pThis->mixer_nreg, val));
1436
1437 switch (pThis->mixer_nreg)
1438 {
1439 case 0x00:
1440 sb16MixerReset(pThis);
1441 /* And update the actual volume, too. */
1442 fUpdateMaster = true;
1443 fUpdateStream = true;
1444 break;
1445
1446 case 0x04: /* Translate from old style voice volume (L/R). */
1447 sb16ConvVolumeOldToNew(pThis, 0x32, val);
1448 fUpdateStream = true;
1449 break;
1450
1451 case 0x22: /* Translate from old style master volume (L/R). */
1452 sb16ConvVolumeOldToNew(pThis, 0x30, val);
1453 fUpdateMaster = true;
1454 break;
1455
1456 case 0x26: /* Translate from old style MIDI volume (L/R). */
1457 sb16ConvVolumeOldToNew(pThis, 0x34, val);
1458 break;
1459
1460 case 0x28: /* Translate from old style CD volume (L/R). */
1461 sb16ConvVolumeOldToNew(pThis, 0x36, val);
1462 break;
1463
1464 case 0x2E: /* Translate from old style line volume (L/R). */
1465 sb16ConvVolumeOldToNew(pThis, 0x38, val);
1466 break;
1467
1468 case 0x30: /* Translate to old style master volume (L). */
1469 sb16ConvVolumeL(pThis, 0x22, val);
1470 fUpdateMaster = true;
1471 break;
1472
1473 case 0x31: /* Translate to old style master volume (R). */
1474 sb16ConvVolumeR(pThis, 0x22, val);
1475 fUpdateMaster = true;
1476 break;
1477
1478 case 0x32: /* Translate to old style voice volume (L). */
1479 sb16ConvVolumeL(pThis, 0x04, val);
1480 fUpdateStream = true;
1481 break;
1482
1483 case 0x33: /* Translate to old style voice volume (R). */
1484 sb16ConvVolumeR(pThis, 0x04, val);
1485 fUpdateStream = true;
1486 break;
1487
1488 case 0x34: /* Translate to old style MIDI volume (L). */
1489 sb16ConvVolumeL(pThis, 0x26, val);
1490 break;
1491
1492 case 0x35: /* Translate to old style MIDI volume (R). */
1493 sb16ConvVolumeR(pThis, 0x26, val);
1494 break;
1495
1496 case 0x36: /* Translate to old style CD volume (L). */
1497 sb16ConvVolumeL(pThis, 0x28, val);
1498 break;
1499
1500 case 0x37: /* Translate to old style CD volume (R). */
1501 sb16ConvVolumeR(pThis, 0x28, val);
1502 break;
1503
1504 case 0x38: /* Translate to old style line volume (L). */
1505 sb16ConvVolumeL(pThis, 0x2E, val);
1506 break;
1507
1508 case 0x39: /* Translate to old style line volume (R). */
1509 sb16ConvVolumeR(pThis, 0x2E, val);
1510 break;
1511
1512 case 0x80:
1513 {
1514 int irq = irq_of_magic(val);
1515 LogFlowFunc(("setting irq to %d (val=%#x)\n", irq, val));
1516 if (irq > 0)
1517 pThis->irq = irq;
1518 break;
1519 }
1520
1521 case 0x81:
1522 {
1523 int dma, hdma;
1524
1525 dma = lsbindex (val & 0xf);
1526 hdma = lsbindex (val & 0xf0);
1527 if (dma != pThis->dma || hdma != pThis->hdma)
1528 LogFlow(("SB16: attempt to change DMA 8bit %d(%d), 16bit %d(%d) (val=%#x)\n",
1529 dma, pThis->dma, hdma, pThis->hdma, val));
1530#if 0
1531 pThis->dma = dma;
1532 pThis->hdma = hdma;
1533#endif
1534 break;
1535 }
1536
1537 case 0x82:
1538 LogFlowFunc(("attempt to write into IRQ status register (val=%#x)\n", val));
1539 return VINF_SUCCESS;
1540
1541 default:
1542 if (pThis->mixer_nreg >= 0x80)
1543 LogFlowFunc(("attempt to write mixer[%#x] <- %#x\n", pThis->mixer_nreg, val));
1544 break;
1545 }
1546
1547 pThis->mixer_regs[pThis->mixer_nreg] = val;
1548
1549 /* Update the master (mixer) volume. */
1550 if (fUpdateMaster)
1551 sb16SetMasterVolume(pThis);
1552
1553 /* Update the stream (PCM) volume. */
1554 if (fUpdateStream)
1555 sb16SetPcmOutVolume(pThis);
1556
1557 return VINF_SUCCESS;
1558}
1559
1560static IO_WRITE_PROTO(mixer_write)
1561{
1562 PSB16STATE pThis = (PSB16STATE)opaque;
1563 int iport = nport - pThis->port;
1564 switch (cb)
1565 {
1566 case 1:
1567 switch (iport)
1568 {
1569 case 4:
1570 mixer_write_indexb (pDevIns, opaque, nport, val, 1);
1571 break;
1572 case 5:
1573 mixer_write_datab (pDevIns, opaque, nport, val, 1);
1574 break;
1575 }
1576 break;
1577 case 2:
1578 mixer_write_indexb (pDevIns, opaque, nport, val & 0xff, 1);
1579 mixer_write_datab (pDevIns, opaque, nport, (val >> 8) & 0xff, 1);
1580 break;
1581 default:
1582 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", nport, cb, val));
1583 break;
1584 }
1585 return VINF_SUCCESS;
1586}
1587
1588static IO_READ_PROTO(mixer_read)
1589{
1590 PSB16STATE pThis = (PSB16STATE)opaque;
1591
1592 (void) nport;
1593#ifndef DEBUG_SB16_MOST
1594 if (pThis->mixer_nreg != 0x82) {
1595 LogFlowFunc(("mixer_read[%#x] -> %#x\n",
1596 pThis->mixer_nreg, pThis->mixer_regs[pThis->mixer_nreg]));
1597 }
1598#else
1599 LogFlowFunc(("mixer_read[%#x] -> %#x\n",
1600 pThis->mixer_nreg, pThis->mixer_regs[pThis->mixer_nreg]));
1601#endif
1602 *pu32 = pThis->mixer_regs[pThis->mixer_nreg];
1603 return VINF_SUCCESS;
1604}
1605
1606static int sb16WriteAudio(PSB16STATE pThis, int nchan, uint32_t dma_pos,
1607 uint32_t dma_len, int len)
1608{
1609 uint8_t tmpbuf[_4K]; /** @todo Have a buffer on the heap. */
1610 uint32_t cbToWrite = len;
1611 uint32_t cbWrittenTotal = 0;
1612
1613 while (cbToWrite)
1614 {
1615 uint32_t cbToRead;
1616 uint32_t cbRead;
1617
1618 cbToRead = RT_MIN(dma_len - dma_pos, cbToWrite);
1619 if (cbToRead > sizeof(tmpbuf))
1620 cbToRead = sizeof(tmpbuf);
1621
1622 int rc = PDMDevHlpDMAReadMemory(pThis->pDevInsR3, nchan, tmpbuf, dma_pos, cbToRead, &cbRead);
1623 AssertMsgRC(rc, ("DMAReadMemory -> %Rrc\n", rc));
1624
1625 /*
1626 * Write data to the backends.
1627 */
1628 uint32_t cbWritten = 0;
1629
1630 PSB16DRIVER pDrv;
1631 RTListForEach(&pThis->lstDrv, pDrv, SB16DRIVER, Node)
1632 {
1633 int rc2 = pDrv->pConnector->pfnStreamWrite(pDrv->pConnector, pDrv->Out.pStream, tmpbuf, cbToRead, &cbWritten);
1634 if (RT_FAILURE(rc2))
1635 LogFlowFunc(("Failed writing to stream '%s': %Rrc\n", &pDrv->Out.pStream->szName, rc2));
1636 }
1637
1638 LogFlowFunc(("\tcbToRead=%RU32, cbToWrite=%RU32, cbWritten=%RU32, cbLeft=%RU32, rc=%Rrc\n",
1639 cbToRead, cbToWrite, cbWritten, cbToWrite - cbWrittenTotal, rc));
1640
1641 Assert(cbToWrite >= cbToRead);
1642 cbToWrite -= cbToRead;
1643 dma_pos = (dma_pos + cbToRead) % dma_len;
1644 cbWrittenTotal += cbToRead;
1645
1646 if (!cbRead)
1647 break;
1648 }
1649
1650 return cbWrittenTotal;
1651}
1652
1653static DECLCALLBACK(uint32_t) sb16DMARead(PPDMDEVINS pDevIns, void *opaque, unsigned nchan, uint32_t dma_pos, uint32_t dma_len)
1654{
1655 PSB16STATE pThis = (PSB16STATE)opaque;
1656 int till, copy, written, free;
1657
1658 if (pThis->block_size <= 0)
1659 {
1660 LogFlowFunc(("invalid block size=%d nchan=%d dma_pos=%d dma_len=%d\n",
1661 pThis->block_size, nchan, dma_pos, dma_len));
1662 return dma_pos;
1663 }
1664
1665 if (pThis->left_till_irq < 0)
1666 pThis->left_till_irq = pThis->block_size;
1667
1668 PSB16DRIVER pDrv;
1669 RTListForEach(&pThis->lstDrv, pDrv, SB16DRIVER, Node)
1670 pDrv->pConnector->pfnStreamIterate(pDrv->pConnector, pDrv->Out.pStream);
1671
1672 free = dma_len;
1673
1674 if (free <= 0)
1675 return dma_pos;
1676
1677 copy = free;
1678 till = pThis->left_till_irq;
1679
1680#ifdef DEBUG_SB16_MOST
1681 LogFlowFunc(("pos:%06d %d till:%d len:%d\n", dma_pos, free, till, dma_len));
1682#endif
1683
1684 if (copy >= till)
1685 {
1686 if (0 == pThis->dma_auto)
1687 {
1688 copy = till;
1689 }
1690 else
1691 {
1692 if (copy >= till + pThis->block_size)
1693 copy = till; /* Make sure we won't skip IRQs. */
1694 }
1695 }
1696
1697 written = sb16WriteAudio(pThis, nchan, dma_pos, dma_len, copy);
1698 dma_pos = (dma_pos + written) % dma_len;
1699 pThis->left_till_irq -= written;
1700
1701 if (pThis->left_till_irq <= 0)
1702 {
1703 pThis->mixer_regs[0x82] |= (nchan & 4) ? 2 : 1;
1704 PDMDevHlpISASetIrq(pThis->pDevInsR3, pThis->irq, 1);
1705 if (0 == pThis->dma_auto)
1706 {
1707 sb16Control(pThis, 0);
1708 sb16SpeakerControl(pThis, 0);
1709 }
1710 }
1711
1712#ifdef DEBUG_SB16_MOST
1713 LogFlowFunc(("pos %5d free %5d size %5d till % 5d copy %5d written %5d size %5d\n",
1714 dma_pos, free, dma_len, pThis->left_till_irq, copy, written,
1715 pThis->block_size));
1716#endif
1717
1718 while (pThis->left_till_irq <= 0)
1719 pThis->left_till_irq += pThis->block_size;
1720
1721 return dma_pos;
1722}
1723
1724#ifndef VBOX_WITH_AUDIO_CALLBACKS
1725
1726static void sb16TimerMaybeStart(PSB16STATE pThis)
1727{
1728 LogFlowFunc(("cStreamsActive=%RU8\n", pThis->cStreamsActive));
1729
1730 if (pThis->cStreamsActive == 0) /* Only start the timer if there are no active streams. */
1731 return;
1732
1733 if (!pThis->pTimerIO)
1734 return;
1735
1736 /* Set timer flag. */
1737 ASMAtomicXchgBool(&pThis->fTimerActive, true);
1738
1739 /* Update current time timestamp. */
1740 pThis->uTimerTSIO = TMTimerGet(pThis->pTimerIO);
1741
1742 /* Fire off timer. */
1743 TMTimerSet(pThis->pTimerIO, TMTimerGet(pThis->pTimerIO) + pThis->cTimerTicksIO);
1744}
1745
1746static void sb16TimerMaybeStop(PSB16STATE pThis)
1747{
1748 LogFlowFunc(("cStreamsActive=%RU8\n", pThis->cStreamsActive));
1749
1750 if (pThis->cStreamsActive) /* Some streams still active? Bail out. */
1751 return;
1752
1753 if (!pThis->pTimerIO)
1754 return;
1755
1756 /* Set timer flag. */
1757 ASMAtomicXchgBool(&pThis->fTimerActive, false);
1758}
1759
1760static DECLCALLBACK(void) sb16TimerIO(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
1761{
1762 PSB16STATE pThis = (PSB16STATE)pvUser;
1763 Assert(pThis == PDMINS_2_DATA(pDevIns, PSB16STATE));
1764 AssertPtr(pThis);
1765
1766 uint64_t cTicksNow = TMTimerGet(pTimer);
1767 uint64_t cTicksElapsed = cTicksNow - pThis->uTimerTSIO;
1768 uint64_t cTicksPerSec = TMTimerGetFreq(pTimer);
1769
1770 pThis->uTimerTSIO = cTicksNow;
1771
1772 bool fIsPlaying = false;
1773
1774 LogFlowFuncEnter();
1775
1776 PSB16DRIVER pDrv;
1777 RTListForEach(&pThis->lstDrv, pDrv, SB16DRIVER, Node)
1778 {
1779 PPDMAUDIOSTREAM pStream = pDrv->Out.pStream;
1780 if (!pStream)
1781 continue;
1782
1783 PDMAUDIOSTRMSTS strmSts = pDrv->pConnector->pfnStreamGetStatus(pDrv->pConnector, pStream);
1784 fIsPlaying |= ( (strmSts & PDMAUDIOSTRMSTS_FLAG_ENABLED)
1785 || (strmSts & PDMAUDIOSTRMSTS_FLAG_PENDING_DISABLE));
1786
1787 LogFlowFunc(("%s: strmSts=0x%x -> fIsPlaying=%RTbool\n", pStream->szName, strmSts, fIsPlaying));
1788
1789 int rc2 = pDrv->pConnector->pfnStreamIterate(pDrv->pConnector, pStream);
1790 if (RT_SUCCESS(rc2))
1791 {
1792 if (pStream->enmDir == PDMAUDIODIR_IN)
1793 {
1794 /** @todo Implement this! */
1795 }
1796 else
1797 {
1798 rc2 = pDrv->pConnector->pfnStreamPlay(pDrv->pConnector, pStream, NULL /* cPlayed */);
1799 if (RT_FAILURE(rc2))
1800 LogFlowFunc(("%s: Failed playing stream, rc=%Rrc\n", pStream->szName, rc2));
1801 }
1802 }
1803 }
1804
1805 if ( ASMAtomicReadBool(&pThis->fTimerActive)
1806 || fIsPlaying)
1807 {
1808 /* Schedule the next transfer. */
1809 PDMDevHlpDMASchedule(pThis->pDevInsR3);
1810
1811 /* Kick the timer again. */
1812 uint64_t cTicks = pThis->cTimerTicksIO;
1813 /** @todo adjust cTicks down by now much cbOutMin represents. */
1814 TMTimerSet(pThis->pTimerIO, cTicksNow + cTicks);
1815 }
1816}
1817
1818#endif /* !VBOX_WITH_AUDIO_CALLBACKS */
1819
1820static void sb16Save(PSSMHANDLE pSSM, PSB16STATE pThis)
1821{
1822 SSMR3PutS32(pSSM, pThis->irq);
1823 SSMR3PutS32(pSSM, pThis->dma);
1824 SSMR3PutS32(pSSM, pThis->hdma);
1825 SSMR3PutS32(pSSM, pThis->port);
1826 SSMR3PutS32(pSSM, pThis->ver);
1827 SSMR3PutS32(pSSM, pThis->in_index);
1828 SSMR3PutS32(pSSM, pThis->out_data_len);
1829 SSMR3PutS32(pSSM, pThis->fmt_stereo);
1830 SSMR3PutS32(pSSM, pThis->fmt_signed);
1831 SSMR3PutS32(pSSM, pThis->fmt_bits);
1832
1833 SSMR3PutU32(pSSM, pThis->fmt);
1834
1835 SSMR3PutS32(pSSM, pThis->dma_auto);
1836 SSMR3PutS32(pSSM, pThis->block_size);
1837 SSMR3PutS32(pSSM, pThis->fifo);
1838 SSMR3PutS32(pSSM, pThis->freq);
1839 SSMR3PutS32(pSSM, pThis->time_const);
1840 SSMR3PutS32(pSSM, pThis->speaker);
1841 SSMR3PutS32(pSSM, pThis->needed_bytes);
1842 SSMR3PutS32(pSSM, pThis->cmd);
1843 SSMR3PutS32(pSSM, pThis->use_hdma);
1844 SSMR3PutS32(pSSM, pThis->highspeed);
1845 SSMR3PutS32(pSSM, pThis->can_write);
1846 SSMR3PutS32(pSSM, pThis->v2x6);
1847
1848 SSMR3PutU8 (pSSM, pThis->csp_param);
1849 SSMR3PutU8 (pSSM, pThis->csp_value);
1850 SSMR3PutU8 (pSSM, pThis->csp_mode);
1851 SSMR3PutU8 (pSSM, pThis->csp_param); /* Bug compatible! */
1852 SSMR3PutMem(pSSM, pThis->csp_regs, 256);
1853 SSMR3PutU8 (pSSM, pThis->csp_index);
1854 SSMR3PutMem(pSSM, pThis->csp_reg83, 4);
1855 SSMR3PutS32(pSSM, pThis->csp_reg83r);
1856 SSMR3PutS32(pSSM, pThis->csp_reg83w);
1857
1858 SSMR3PutMem(pSSM, pThis->in2_data, sizeof (pThis->in2_data));
1859 SSMR3PutMem(pSSM, pThis->out_data, sizeof (pThis->out_data));
1860 SSMR3PutU8 (pSSM, pThis->test_reg);
1861 SSMR3PutU8 (pSSM, pThis->last_read_byte);
1862
1863 SSMR3PutS32(pSSM, pThis->nzero);
1864 SSMR3PutS32(pSSM, pThis->left_till_irq);
1865 SSMR3PutS32(pSSM, pThis->dma_running);
1866 SSMR3PutS32(pSSM, pThis->bytes_per_second);
1867 SSMR3PutS32(pSSM, pThis->align);
1868
1869 SSMR3PutS32(pSSM, pThis->mixer_nreg);
1870 SSMR3PutMem(pSSM, pThis->mixer_regs, 256);
1871
1872}
1873
1874static int sb16Load(PSSMHANDLE pSSM, PSB16STATE pThis, int version_id)
1875{
1876 SSMR3GetS32(pSSM, &pThis->irq);
1877 SSMR3GetS32(pSSM, &pThis->dma);
1878 SSMR3GetS32(pSSM, &pThis->hdma);
1879 SSMR3GetS32(pSSM, &pThis->port);
1880 SSMR3GetS32(pSSM, &pThis->ver);
1881 SSMR3GetS32(pSSM, &pThis->in_index);
1882 SSMR3GetS32(pSSM, &pThis->out_data_len);
1883 SSMR3GetS32(pSSM, &pThis->fmt_stereo);
1884 SSMR3GetS32(pSSM, &pThis->fmt_signed);
1885 SSMR3GetS32(pSSM, &pThis->fmt_bits);
1886
1887 SSMR3GetU32(pSSM, (uint32_t *)&pThis->fmt);
1888
1889 SSMR3GetS32(pSSM, &pThis->dma_auto);
1890 SSMR3GetS32(pSSM, &pThis->block_size);
1891 SSMR3GetS32(pSSM, &pThis->fifo);
1892 SSMR3GetS32(pSSM, &pThis->freq);
1893 SSMR3GetS32(pSSM, &pThis->time_const);
1894 SSMR3GetS32(pSSM, &pThis->speaker);
1895 SSMR3GetS32(pSSM, &pThis->needed_bytes);
1896 SSMR3GetS32(pSSM, &pThis->cmd);
1897 SSMR3GetS32(pSSM, &pThis->use_hdma);
1898 SSMR3GetS32(pSSM, &pThis->highspeed);
1899 SSMR3GetS32(pSSM, &pThis->can_write);
1900 SSMR3GetS32(pSSM, &pThis->v2x6);
1901
1902 SSMR3GetU8 (pSSM, &pThis->csp_param);
1903 SSMR3GetU8 (pSSM, &pThis->csp_value);
1904 SSMR3GetU8 (pSSM, &pThis->csp_mode);
1905 SSMR3GetU8 (pSSM, &pThis->csp_param); /* Bug compatible! */
1906 SSMR3GetMem(pSSM, pThis->csp_regs, 256);
1907 SSMR3GetU8 (pSSM, &pThis->csp_index);
1908 SSMR3GetMem(pSSM, pThis->csp_reg83, 4);
1909 SSMR3GetS32(pSSM, &pThis->csp_reg83r);
1910 SSMR3GetS32(pSSM, &pThis->csp_reg83w);
1911
1912 SSMR3GetMem(pSSM, pThis->in2_data, sizeof (pThis->in2_data));
1913 SSMR3GetMem(pSSM, pThis->out_data, sizeof (pThis->out_data));
1914 SSMR3GetU8 (pSSM, &pThis->test_reg);
1915 SSMR3GetU8 (pSSM, &pThis->last_read_byte);
1916
1917 SSMR3GetS32(pSSM, &pThis->nzero);
1918 SSMR3GetS32(pSSM, &pThis->left_till_irq);
1919 SSMR3GetS32(pSSM, &pThis->dma_running);
1920 SSMR3GetS32(pSSM, &pThis->bytes_per_second);
1921 SSMR3GetS32(pSSM, &pThis->align);
1922
1923 SSMR3GetS32(pSSM, &pThis->mixer_nreg);
1924 SSMR3GetMem(pSSM, pThis->mixer_regs, 256);
1925
1926#if 0
1927 PSB16DRIVER pDrv;
1928 RTListForEach(&pThis->lstDrv, pDrv, SB16DRIVER, Node)
1929 {
1930 if (pDrv->Out.pStream)
1931 {
1932 pDrv->pConnector->pfnCloseOut(pThis->pDrv, pDrv->Out.pStream);
1933 pDrv->Out.pStream = NULL;
1934 }
1935 }
1936#endif
1937
1938 if (pThis->dma_running)
1939 {
1940 if (pThis->freq)
1941 {
1942 PDMAUDIOSTREAMCFG streamCfg;
1943 RT_ZERO(streamCfg);
1944 streamCfg.enmDir = PDMAUDIODIR_OUT;
1945 streamCfg.DestSource.Dest = PDMAUDIOPLAYBACKDEST_FRONT;
1946 streamCfg.uHz = pThis->freq;
1947 streamCfg.cChannels = 1 << pThis->fmt_stereo;
1948 streamCfg.enmFormat = pThis->fmt;
1949 streamCfg.enmEndianness = PDMAUDIOHOSTENDIANNESS;
1950
1951 int rc = sb16OpenOut(pThis, &streamCfg);
1952 AssertRC(rc);
1953 }
1954
1955 sb16Control(pThis, 1);
1956 sb16SpeakerControl(pThis, pThis->speaker);
1957 }
1958
1959 /* Update the master (mixer) and PCM out volumes. */
1960 sb16SetMasterVolume(pThis);
1961 sb16SetPcmOutVolume(pThis);
1962
1963 return VINF_SUCCESS;
1964}
1965
1966static DECLCALLBACK(int) sb16LiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
1967{
1968 PSB16STATE pThis = PDMINS_2_DATA(pDevIns, PSB16STATE);
1969
1970 SSMR3PutS32(pSSM, pThis->irqCfg);
1971 SSMR3PutS32(pSSM, pThis->dmaCfg);
1972 SSMR3PutS32(pSSM, pThis->hdmaCfg);
1973 SSMR3PutS32(pSSM, pThis->portCfg);
1974 SSMR3PutS32(pSSM, pThis->verCfg);
1975 return VINF_SSM_DONT_CALL_AGAIN;
1976}
1977
1978static DECLCALLBACK(int) sb16SaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
1979{
1980 PSB16STATE pThis = PDMINS_2_DATA(pDevIns, PSB16STATE);
1981
1982 sb16LiveExec(pDevIns, pSSM, 0);
1983 sb16Save(pSSM, pThis);
1984 return VINF_SUCCESS;
1985}
1986
1987static DECLCALLBACK(int) sb16LoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
1988{
1989 PSB16STATE pThis = PDMINS_2_DATA(pDevIns, PSB16STATE);
1990
1991 AssertMsgReturn( uVersion == SB16_SAVE_STATE_VERSION
1992 || uVersion == SB16_SAVE_STATE_VERSION_VBOX_30,
1993 ("%u\n", uVersion),
1994 VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION);
1995 if (uVersion > SB16_SAVE_STATE_VERSION_VBOX_30)
1996 {
1997 int32_t irq;
1998 SSMR3GetS32 (pSSM, &irq);
1999 int32_t dma;
2000 SSMR3GetS32 (pSSM, &dma);
2001 int32_t hdma;
2002 SSMR3GetS32 (pSSM, &hdma);
2003 int32_t port;
2004 SSMR3GetS32 (pSSM, &port);
2005 int32_t ver;
2006 int rc = SSMR3GetS32 (pSSM, &ver);
2007 AssertRCReturn (rc, rc);
2008
2009 if ( irq != pThis->irqCfg
2010 || dma != pThis->dmaCfg
2011 || hdma != pThis->hdmaCfg
2012 || port != pThis->portCfg
2013 || ver != pThis->verCfg)
2014 {
2015 return SSMR3SetCfgError(pSSM, RT_SRC_POS,
2016 N_("config changed: irq=%x/%x dma=%x/%x hdma=%x/%x port=%x/%x ver=%x/%x (saved/config)"),
2017 irq, pThis->irqCfg,
2018 dma, pThis->dmaCfg,
2019 hdma, pThis->hdmaCfg,
2020 port, pThis->portCfg,
2021 ver, pThis->verCfg);
2022 }
2023 }
2024
2025 if (uPass != SSM_PASS_FINAL)
2026 return VINF_SUCCESS;
2027
2028 sb16Load(pSSM, pThis, uVersion);
2029 return VINF_SUCCESS;
2030}
2031
2032static int sb16OpenOut(PSB16STATE pThis, PPDMAUDIOSTREAMCFG pCfg)
2033{
2034 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2035 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
2036
2037 AssertReturn(pCfg->enmDir == PDMAUDIODIR_OUT, VERR_INVALID_PARAMETER);
2038
2039 int rc = VINF_SUCCESS;
2040
2041 /* Set a default audio format for the host. */
2042 PDMAUDIOSTREAMCFG CfgHost;
2043 CfgHost.enmDir = PDMAUDIODIR_OUT;
2044 CfgHost.DestSource.Dest = PDMAUDIOPLAYBACKDEST_FRONT;
2045 CfgHost.uHz = 44100;
2046 CfgHost.cChannels = 2;
2047 CfgHost.enmFormat = PDMAUDIOFMT_S16;
2048 CfgHost.enmEndianness = PDMAUDIOHOSTENDIANNESS;
2049
2050 RTStrPrintf(CfgHost.szName, sizeof(CfgHost.szName), "sb16.po");
2051
2052 uint8_t uLUN = 0;
2053
2054 PSB16DRIVER pDrv;
2055 RTListForEach(&pThis->lstDrv, pDrv, SB16DRIVER, Node)
2056 {
2057 if (!RTStrPrintf(pCfg->szName, sizeof(pCfg->szName), "[LUN#%RU8] sb16.po (%RU32Hz, %RU8 %s)",
2058 pDrv->uLUN, pCfg->uHz, pCfg->cChannels, pCfg->cChannels > 1 ? "Channels" : "Channel"))
2059 {
2060 rc = VERR_BUFFER_OVERFLOW;
2061 break;
2062 }
2063
2064 if (pDrv->Out.pStream)
2065 {
2066 pDrv->pConnector->pfnStreamRelease(pDrv->pConnector, pDrv->Out.pStream);
2067
2068 int rc3 = pDrv->pConnector->pfnStreamDestroy(pDrv->pConnector, pDrv->Out.pStream);
2069 AssertRC(rc3);
2070
2071 pDrv->Out.pStream = NULL;
2072 }
2073
2074 int rc2 = pDrv->pConnector->pfnStreamCreate(pDrv->pConnector, &CfgHost, pCfg, &pDrv->Out.pStream);
2075 if (RT_SUCCESS(rc2))
2076 pDrv->pConnector->pfnStreamAddRef(pDrv->pConnector, pDrv->Out.pStream);
2077
2078 LogFlowFunc(("LUN#%RU8: Created output \"%s\", rc=%Rrc\n", pDrv->uLUN, pCfg->szName, rc2));
2079
2080 uLUN++;
2081 }
2082
2083 return rc;
2084}
2085
2086static void sb16CloseOut(PSB16STATE pThis)
2087{
2088 AssertPtrReturnVoid(pThis);
2089
2090 LogFlowFuncEnter();
2091
2092 PSB16DRIVER pDrv;
2093 RTListForEach(&pThis->lstDrv, pDrv, SB16DRIVER, Node)
2094 {
2095 if (pDrv->Out.pStream)
2096 {
2097 pDrv->pConnector->pfnStreamRelease(pDrv->pConnector, pDrv->Out.pStream);
2098
2099 pDrv->pConnector->pfnStreamDestroy(pDrv->pConnector, pDrv->Out.pStream);
2100 pDrv->Out.pStream = NULL;
2101 }
2102 }
2103}
2104
2105/**
2106 * @interface_method_impl{PDMDEVREG,pfnReset}
2107 */
2108static DECLCALLBACK(void) sb16DevReset(PPDMDEVINS pDevIns)
2109{
2110 PSB16STATE pThis = PDMINS_2_DATA(pDevIns, PSB16STATE);
2111
2112 /* Bring back the device to initial state, and especially make
2113 * sure there's no interrupt or DMA activity.
2114 */
2115 PDMDevHlpISASetIrq(pThis->pDevInsR3, pThis->irq, 0);
2116
2117 pThis->mixer_regs[0x82] = 0;
2118 pThis->csp_regs[5] = 1;
2119 pThis->csp_regs[9] = 0xf8;
2120
2121 pThis->dma_auto = 0;
2122 pThis->in_index = 0;
2123 pThis->out_data_len = 0;
2124 pThis->left_till_irq = 0;
2125 pThis->needed_bytes = 0;
2126 pThis->block_size = -1;
2127 pThis->nzero = 0;
2128 pThis->highspeed = 0;
2129 pThis->v2x6 = 0;
2130 pThis->cmd = -1;
2131
2132 sb16MixerReset(pThis);
2133 sb16SpeakerControl(pThis, 0);
2134 sb16Control(pThis, 0);
2135 sb16ResetLegacy(pThis);
2136}
2137
2138/**
2139 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
2140 */
2141static DECLCALLBACK(void *) sb16QueryInterface(struct PDMIBASE *pInterface, const char *pszIID)
2142{
2143 PSB16STATE pThis = RT_FROM_MEMBER(pInterface, SB16STATE, IBase);
2144 Assert(&pThis->IBase == pInterface);
2145
2146 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->IBase);
2147 return NULL;
2148}
2149
2150/**
2151 * Powers off the device.
2152 *
2153 * @param pDevIns Device instance to power off.
2154 */
2155static DECLCALLBACK(void) sb16PowerOff(PPDMDEVINS pDevIns)
2156{
2157 PSB16STATE pThis = PDMINS_2_DATA(pDevIns, PSB16STATE);
2158
2159 LogRel2(("SB16: Powering off ...\n"));
2160
2161 sb16CloseOut(pThis);
2162}
2163
2164/**
2165 * @interface_method_impl{PDMDEVREG,pfnDestruct}
2166 */
2167static DECLCALLBACK(int) sb16Destruct(PPDMDEVINS pDevIns)
2168{
2169 PSB16STATE pThis = PDMINS_2_DATA(pDevIns, PSB16STATE);
2170
2171 LogFlowFuncEnter();
2172
2173 PSB16DRIVER pDrv;
2174 while (!RTListIsEmpty(&pThis->lstDrv))
2175 {
2176 pDrv = RTListGetFirst(&pThis->lstDrv, SB16DRIVER, Node);
2177
2178 RTListNodeRemove(&pDrv->Node);
2179 RTMemFree(pDrv);
2180 }
2181
2182 return VINF_SUCCESS;
2183}
2184
2185static DECLCALLBACK(int) sb16Construct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
2186{
2187 PSB16STATE pThis = PDMINS_2_DATA(pDevIns, PSB16STATE);
2188
2189 /*
2190 * Validations.
2191 */
2192 Assert(iInstance == 0);
2193 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
2194 if (!CFGMR3AreValuesValid(pCfg,
2195 "IRQ\0"
2196 "DMA\0"
2197 "DMA16\0"
2198 "Port\0"
2199 "Version\0"
2200 "TimerHz\0"))
2201 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
2202 N_("Invalid configuration for SB16 device"));
2203
2204 /*
2205 * Read config data.
2206 */
2207 int rc = CFGMR3QuerySIntDef(pCfg, "IRQ", &pThis->irq, 5);
2208 if (RT_FAILURE(rc))
2209 return PDMDEV_SET_ERROR(pDevIns, rc,
2210 N_("SB16 configuration error: Failed to get the \"IRQ\" value"));
2211 pThis->irqCfg = pThis->irq;
2212
2213 rc = CFGMR3QuerySIntDef(pCfg, "DMA", &pThis->dma, 1);
2214 if (RT_FAILURE(rc))
2215 return PDMDEV_SET_ERROR(pDevIns, rc,
2216 N_("SB16 configuration error: Failed to get the \"DMA\" value"));
2217 pThis->dmaCfg = pThis->dma;
2218
2219 rc = CFGMR3QuerySIntDef(pCfg, "DMA16", &pThis->hdma, 5);
2220 if (RT_FAILURE(rc))
2221 return PDMDEV_SET_ERROR(pDevIns, rc,
2222 N_("SB16 configuration error: Failed to get the \"DMA16\" value"));
2223 pThis->hdmaCfg = pThis->hdma;
2224
2225 RTIOPORT Port;
2226 rc = CFGMR3QueryPortDef(pCfg, "Port", &Port, 0x220);
2227 if (RT_FAILURE(rc))
2228 return PDMDEV_SET_ERROR(pDevIns, rc,
2229 N_("SB16 configuration error: Failed to get the \"Port\" value"));
2230 pThis->port = Port;
2231 pThis->portCfg = Port;
2232
2233 uint16_t u16Version;
2234 rc = CFGMR3QueryU16Def(pCfg, "Version", &u16Version, 0x0405);
2235 if (RT_FAILURE(rc))
2236 return PDMDEV_SET_ERROR(pDevIns, rc,
2237 N_("SB16 configuration error: Failed to get the \"Version\" value"));
2238
2239#ifndef VBOX_WITH_AUDIO_CALLBACKS
2240 uint16_t uTimerHz;
2241 rc = CFGMR3QueryU16Def(pCfg, "TimerHz", &uTimerHz, 25 /* Hz */);
2242 if (RT_FAILURE(rc))
2243 return PDMDEV_SET_ERROR(pDevIns, rc,
2244 N_("SB16 configuration error: failed to read Hertz (Hz) rate as unsigned integer"));
2245#endif
2246
2247 pThis->ver = u16Version;
2248 pThis->verCfg = u16Version;
2249
2250 /*
2251 * Init instance data.
2252 */
2253 pThis->pDevInsR3 = pDevIns;
2254 pThis->IBase.pfnQueryInterface = sb16QueryInterface;
2255 pThis->cmd = -1;
2256
2257 pThis->mixer_regs[0x80] = magic_of_irq (pThis->irq);
2258 pThis->mixer_regs[0x81] = (1 << pThis->dma) | (1 << pThis->hdma);
2259 pThis->mixer_regs[0x82] = 2 << 5;
2260
2261 pThis->csp_regs[5] = 1;
2262 pThis->csp_regs[9] = 0xf8;
2263
2264 RTListInit(&pThis->lstDrv);
2265
2266 sb16MixerReset(pThis);
2267
2268 /*
2269 * Create timer(s), register & attach stuff.
2270 */
2271 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, sb16TimerIRQ, pThis,
2272 TMTIMER_FLAGS_DEFAULT_CRIT_SECT, "SB16 IRQ timer", &pThis->pTimerIRQ);
2273 if (RT_FAILURE(rc))
2274 AssertMsgFailedReturn(("Error creating IRQ timer, rc=%Rrc\n", rc), rc);
2275
2276 rc = PDMDevHlpIOPortRegister(pDevIns, pThis->port + 0x04, 2, pThis,
2277 mixer_write, mixer_read, NULL, NULL, "SB16");
2278 if (RT_FAILURE(rc))
2279 return rc;
2280 rc = PDMDevHlpIOPortRegister(pDevIns, pThis->port + 0x06, 10, pThis,
2281 dsp_write, dsp_read, NULL, NULL, "SB16");
2282 if (RT_FAILURE(rc))
2283 return rc;
2284
2285 rc = PDMDevHlpDMARegister(pDevIns, pThis->hdma, sb16DMARead, pThis);
2286 if (RT_FAILURE(rc))
2287 return rc;
2288 rc = PDMDevHlpDMARegister(pDevIns, pThis->dma, sb16DMARead, pThis);
2289 if (RT_FAILURE(rc))
2290 return rc;
2291
2292 pThis->can_write = 1;
2293
2294 rc = PDMDevHlpSSMRegister3(pDevIns, SB16_SAVE_STATE_VERSION, sizeof(SB16STATE), sb16LiveExec, sb16SaveExec, sb16LoadExec);
2295 if (RT_FAILURE(rc))
2296 return rc;
2297
2298 /*
2299 * Attach driver.
2300 */
2301 uint8_t uLUN;
2302 for (uLUN = 0; uLUN < UINT8_MAX; ++uLUN)
2303 {
2304 LogFunc(("Trying to attach driver for LUN #%RU8 ...\n", uLUN));
2305 rc = sb16AttachInternal(pDevIns, NULL /* pDrv */, uLUN, 0 /* fFlags */);
2306 if (RT_FAILURE(rc))
2307 {
2308 if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
2309 rc = VINF_SUCCESS;
2310 else if (rc == VERR_AUDIO_BACKEND_INIT_FAILED)
2311 {
2312 sb16Reattach(pThis, NULL /* pDrv */, uLUN, "NullAudio");
2313 PDMDevHlpVMSetRuntimeError(pDevIns, 0 /*fFlags*/, "HostAudioNotResponding",
2314 N_("No audio devices could be opened. Selecting the NULL audio backend "
2315 "with the consequence that no sound is audible"));
2316 /* attaching to the NULL audio backend will never fail */
2317 rc = VINF_SUCCESS;
2318 }
2319 break;
2320 }
2321 }
2322
2323 LogFunc(("cLUNs=%RU8, rc=%Rrc\n", uLUN, rc));
2324
2325 sb16ResetLegacy(pThis);
2326
2327 PSB16DRIVER pDrv;
2328 RTListForEach(&pThis->lstDrv, pDrv, SB16DRIVER, Node)
2329 {
2330 /*
2331 * Only primary drivers are critical for the VM to run. Everything else
2332 * might not worth showing an own error message box in the GUI.
2333 */
2334 if (!(pDrv->Flags & PDMAUDIODRVFLAG_PRIMARY))
2335 continue;
2336
2337 PPDMIAUDIOCONNECTOR pCon = pDrv->pConnector;
2338 AssertPtr(pCon);
2339
2340 /** @todo No input streams available for SB16 yet. */
2341 bool fValidOut = pCon->pfnStreamGetStatus(pCon, pDrv->Out.pStream) & PDMAUDIOSTRMSTS_FLAG_INITIALIZED;
2342 if (!fValidOut)
2343 {
2344 LogRel(("SB16: Falling back to NULL backend (no sound audible)\n"));
2345
2346 sb16ResetLegacy(pThis);
2347 sb16Reattach(pThis, pDrv, pDrv->uLUN, "NullAudio");
2348
2349 PDMDevHlpVMSetRuntimeError(pDevIns, 0 /*fFlags*/, "HostAudioNotResponding",
2350 N_("No audio devices could be opened. Selecting the NULL audio backend "
2351 "with the consequence that no sound is audible"));
2352 }
2353 }
2354
2355#ifndef VBOX_WITH_AUDIO_CALLBACKS
2356 if (RT_SUCCESS(rc))
2357 {
2358 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, sb16TimerIO, pThis,
2359 TMTIMER_FLAGS_DEFAULT_CRIT_SECT, "SB16 IO timer", &pThis->pTimerIO);
2360 if (RT_SUCCESS(rc))
2361 {
2362 pThis->cTimerTicksIO = TMTimerGetFreq(pThis->pTimerIO) / uTimerHz;
2363 pThis->uTimerTSIO = TMTimerGet(pThis->pTimerIO);
2364 LogFunc(("Timer ticks=%RU64 (%RU16 Hz)\n", pThis->cTimerTicksIO, uTimerHz));
2365
2366 sb16TimerMaybeStart(pThis);
2367 }
2368 else
2369 AssertMsgFailedReturn(("Error creating I/O timer, rc=%Rrc\n", rc), rc);
2370 }
2371#else
2372 if (RT_SUCCESS(rc))
2373 {
2374 /** @todo Merge this callback registration with the validation block above once
2375 * this becomes the standard. */
2376 PSB16DRIVER pDrv;
2377 RTListForEach(&pThis->lstDrv, pDrv, SB16DRIVER, Node)
2378 {
2379 /* Only register primary driver.
2380 * The device emulation does the output multiplexing then. */
2381 if (pDrv->Flags != PDMAUDIODRVFLAG_PRIMARY)
2382 continue;
2383
2384 PDMAUDIOCALLBACK AudioCallbacks[2];
2385
2386 SB16CALLBACKCTX Ctx = { pThis, pDrv };
2387
2388 AudioCallbacks[0].enmType = PDMAUDIOCALLBACKTYPE_INPUT;
2389 AudioCallbacks[0].pfnCallback = sb16CallbackInput;
2390 AudioCallbacks[0].pvCtx = &Ctx;
2391 AudioCallbacks[0].cbCtx = sizeof(SB16CALLBACKCTX);
2392
2393 AudioCallbacks[1].enmType = PDMAUDIOCALLBACKTYPE_OUTPUT;
2394 AudioCallbacks[1].pfnCallback = sb16CallbackOutput;
2395 AudioCallbacks[1].pvCtx = &Ctx;
2396 AudioCallbacks[1].cbCtx = sizeof(SB16CALLBACKCTX);
2397
2398 rc = pDrv->pConnector->pfnRegisterCallbacks(pDrv->pConnector, AudioCallbacks, RT_ELEMENTS(AudioCallbacks));
2399 if (RT_FAILURE(rc))
2400 break;
2401 }
2402 }
2403#endif
2404
2405 return VINF_SUCCESS;
2406}
2407
2408const PDMDEVREG g_DeviceSB16 =
2409{
2410 /* u32Version */
2411 PDM_DEVREG_VERSION,
2412 /* szName */
2413 "sb16",
2414 /* szRCMod */
2415 "",
2416 /* szR0Mod */
2417 "",
2418 /* pszDescription */
2419 "Sound Blaster 16 Controller",
2420 /* fFlags */
2421 PDM_DEVREG_FLAGS_DEFAULT_BITS,
2422 /* fClass */
2423 PDM_DEVREG_CLASS_AUDIO,
2424 /* cMaxInstances */
2425 1,
2426 /* cbInstance */
2427 sizeof(SB16STATE),
2428 /* pfnConstruct */
2429 sb16Construct,
2430 /* pfnDestruct */
2431 sb16Destruct,
2432 /* pfnRelocate */
2433 NULL,
2434 /* pfnMemSetup */
2435 NULL,
2436 /* pfnPowerOn */
2437 NULL,
2438 /* pfnReset */
2439 sb16DevReset,
2440 /* pfnSuspend */
2441 NULL,
2442 /* pfnResume */
2443 NULL,
2444 /* pfnAttach */
2445 sb16Attach,
2446 /* pfnDetach */
2447 sb16Detach,
2448 /* pfnQueryInterface */
2449 NULL,
2450 /* pfnInitComplete */
2451 NULL,
2452 /* pfnPowerOff */
2453 sb16PowerOff,
2454 /* pfnSoftReset */
2455 NULL,
2456 /* u32VersionEnd */
2457 PDM_DEVREG_VERSION
2458};
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