VirtualBox

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

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

Devices: warnings (clang)

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