VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/DevIchAc97.cpp@ 71733

Last change on this file since 71733 was 71247, checked in by vboxsync, 7 years ago

DevIchAc97: Critsect alignment fix (32-bit).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 137.9 KB
Line 
1/* $Id: DevIchAc97.cpp 71247 2018-03-06 17:16:24Z vboxsync $ */
2/** @file
3 * DevIchAc97 - VBox ICH AC97 Audio Controller.
4 */
5
6/*
7 * Copyright (C) 2006-2017 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
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_DEV_AC97
23#include <VBox/log.h>
24#include <VBox/vmm/pdmdev.h>
25#include <VBox/vmm/pdmaudioifs.h>
26
27#include <iprt/assert.h>
28#ifdef IN_RING3
29# ifdef DEBUG
30# include <iprt/file.h>
31# endif
32# include <iprt/mem.h>
33# include <iprt/semaphore.h>
34# include <iprt/string.h>
35# include <iprt/uuid.h>
36#endif
37
38#include "VBoxDD.h"
39
40#include "AudioMixBuffer.h"
41#include "AudioMixer.h"
42#include "DrvAudio.h"
43
44
45/*********************************************************************************************************************************
46* Defined Constants And Macros *
47*********************************************************************************************************************************/
48
49/** Current saved state version. */
50#define AC97_SSM_VERSION 1
51
52/** Default timer frequency (in Hz). */
53#define AC97_TIMER_HZ 200
54
55/** Maximum FIFO size (in bytes). */
56#define AC97_FIFO_MAX 256
57
58#define AC97_SR_FIFOE RT_BIT(4) /* rwc, FIFO error. */
59#define AC97_SR_BCIS RT_BIT(3) /* rwc, Buffer completion interrupt status. */
60#define AC97_SR_LVBCI RT_BIT(2) /* rwc, Last valid buffer completion interrupt. */
61#define AC97_SR_CELV RT_BIT(1) /* ro, Current equals last valid. */
62#define AC97_SR_DCH RT_BIT(0) /* ro, Controller halted. */
63#define AC97_SR_VALID_MASK (RT_BIT(5) - 1)
64#define AC97_SR_WCLEAR_MASK (AC97_SR_FIFOE | AC97_SR_BCIS | AC97_SR_LVBCI)
65#define AC97_SR_RO_MASK (AC97_SR_DCH | AC97_SR_CELV)
66#define AC97_SR_INT_MASK (AC97_SR_FIFOE | AC97_SR_BCIS | AC97_SR_LVBCI)
67
68#define AC97_CR_IOCE RT_BIT(4) /* rw, Interrupt On Completion Enable. */
69#define AC97_CR_FEIE RT_BIT(3) /* rw FIFO Error Interrupt Enable. */
70#define AC97_CR_LVBIE RT_BIT(2) /* rw Last Valid Buffer Interrupt Enable. */
71#define AC97_CR_RR RT_BIT(1) /* rw Reset Registers. */
72#define AC97_CR_RPBM RT_BIT(0) /* rw Run/Pause Bus Master. */
73#define AC97_CR_VALID_MASK (RT_BIT(5) - 1)
74#define AC97_CR_DONT_CLEAR_MASK (AC97_CR_IOCE | AC97_CR_FEIE | AC97_CR_LVBIE)
75
76#define AC97_GC_WR 4 /* rw Warm reset. */
77#define AC97_GC_CR 2 /* rw Cold reset. */
78#define AC97_GC_VALID_MASK (RT_BIT(6) - 1)
79
80#define AC97_GS_MD3 RT_BIT(17) /* rw */
81#define AC97_GS_AD3 RT_BIT(16) /* rw */
82#define AC97_GS_RCS RT_BIT(15) /* rwc */
83#define AC97_GS_B3S12 RT_BIT(14) /* ro */
84#define AC97_GS_B2S12 RT_BIT(13) /* ro */
85#define AC97_GS_B1S12 RT_BIT(12) /* ro */
86#define AC97_GS_S1R1 RT_BIT(11) /* rwc */
87#define AC97_GS_S0R1 RT_BIT(10) /* rwc */
88#define AC97_GS_S1CR RT_BIT(9) /* ro */
89#define AC97_GS_S0CR RT_BIT(8) /* ro */
90#define AC97_GS_MINT RT_BIT(7) /* ro */
91#define AC97_GS_POINT RT_BIT(6) /* ro */
92#define AC97_GS_PIINT RT_BIT(5) /* ro */
93#define AC97_GS_RSRVD (RT_BIT(4)|RT_BIT(3))
94#define AC97_GS_MOINT RT_BIT(2) /* ro */
95#define AC97_GS_MIINT RT_BIT(1) /* ro */
96#define AC97_GS_GSCI RT_BIT(0) /* rwc */
97#define AC97_GS_RO_MASK (AC97_GS_B3S12 | \
98 AC97_GS_B2S12 | \
99 AC97_GS_B1S12 | \
100 AC97_GS_S1CR | \
101 AC97_GS_S0CR | \
102 AC97_GS_MINT | \
103 AC97_GS_POINT | \
104 AC97_GS_PIINT | \
105 AC97_GS_RSRVD | \
106 AC97_GS_MOINT | \
107 AC97_GS_MIINT)
108#define AC97_GS_VALID_MASK (RT_BIT(18) - 1)
109#define AC97_GS_WCLEAR_MASK (AC97_GS_RCS|AC97_GS_S1R1|AC97_GS_S0R1|AC97_GS_GSCI)
110
111/** @name Buffer Descriptor (BD).
112 * @{ */
113#define AC97_BD_IOC RT_BIT(31) /**< Interrupt on Completion. */
114#define AC97_BD_BUP RT_BIT(30) /**< Buffer Underrun Policy. */
115
116#define AC97_BD_LEN_MASK 0xFFFF /**< Mask for the BDL buffer length. */
117
118#define AC97_MAX_BDLE 32 /**< Maximum number of BDLEs. */
119/** @} */
120
121/** @name Extended Audio Status and Control Register (EACS).
122 * @{ */
123#define AC97_EACS_VRA 1 /**< Variable Rate Audio (4.2.1.1). */
124#define AC97_EACS_VRM 8 /**< Variable Rate Mic Audio (4.2.1.1). */
125/** @} */
126
127/** @name Baseline Audio Register Set (BARS).
128 * @{ */
129#define AC97_BARS_VOL_MASK 0x1f /**< Volume mask for the Baseline Audio Register Set (5.7.2). */
130#define AC97_BARS_VOL_STEPS 31 /**< Volume steps for the Baseline Audio Register Set (5.7.2). */
131#define AC97_BARS_VOL_MUTE_SHIFT 15 /**< Mute bit shift for the Baseline Audio Register Set (5.7.2). */
132/** @} */
133
134/* AC'97 uses 1.5dB steps, we use 0.375dB steps: 1 AC'97 step equals 4 PDM steps. */
135#define AC97_DB_FACTOR 4
136
137#define AC97_REC_MASK 7
138enum
139{
140 AC97_REC_MIC = 0,
141 AC97_REC_CD,
142 AC97_REC_VIDEO,
143 AC97_REC_AUX,
144 AC97_REC_LINE_IN,
145 AC97_REC_STEREO_MIX,
146 AC97_REC_MONO_MIX,
147 AC97_REC_PHONE
148};
149
150enum
151{
152 AC97_Reset = 0x00,
153 AC97_Master_Volume_Mute = 0x02,
154 AC97_Headphone_Volume_Mute = 0x04, /** Also known as AUX, see table 16, section 5.7. */
155 AC97_Master_Volume_Mono_Mute = 0x06,
156 AC97_Master_Tone_RL = 0x08,
157 AC97_PC_BEEP_Volume_Mute = 0x0A,
158 AC97_Phone_Volume_Mute = 0x0C,
159 AC97_Mic_Volume_Mute = 0x0E,
160 AC97_Line_In_Volume_Mute = 0x10,
161 AC97_CD_Volume_Mute = 0x12,
162 AC97_Video_Volume_Mute = 0x14,
163 AC97_Aux_Volume_Mute = 0x16,
164 AC97_PCM_Out_Volume_Mute = 0x18,
165 AC97_Record_Select = 0x1A,
166 AC97_Record_Gain_Mute = 0x1C,
167 AC97_Record_Gain_Mic_Mute = 0x1E,
168 AC97_General_Purpose = 0x20,
169 AC97_3D_Control = 0x22,
170 AC97_AC_97_RESERVED = 0x24,
171 AC97_Powerdown_Ctrl_Stat = 0x26,
172 AC97_Extended_Audio_ID = 0x28,
173 AC97_Extended_Audio_Ctrl_Stat = 0x2A,
174 AC97_PCM_Front_DAC_Rate = 0x2C,
175 AC97_PCM_Surround_DAC_Rate = 0x2E,
176 AC97_PCM_LFE_DAC_Rate = 0x30,
177 AC97_PCM_LR_ADC_Rate = 0x32,
178 AC97_MIC_ADC_Rate = 0x34,
179 AC97_6Ch_Vol_C_LFE_Mute = 0x36,
180 AC97_6Ch_Vol_L_R_Surround_Mute = 0x38,
181 AC97_Vendor_Reserved = 0x58,
182 AC97_AD_Misc = 0x76,
183 AC97_Vendor_ID1 = 0x7c,
184 AC97_Vendor_ID2 = 0x7e
185};
186
187/* Codec models. */
188typedef enum
189{
190 AC97_CODEC_STAC9700 = 0, /* SigmaTel STAC9700 */
191 AC97_CODEC_AD1980, /* Analog Devices AD1980 */
192 AC97_CODEC_AD1981B /* Analog Devices AD1981B */
193} AC97CODEC;
194
195/* Analog Devices miscellaneous regiter bits used in AD1980. */
196#define AC97_AD_MISC_LOSEL RT_BIT(5) /* Surround (rear) goes to line out outputs. */
197#define AC97_AD_MISC_HPSEL RT_BIT(10) /* PCM (front) goes to headphone outputs. */
198
199#define ICHAC97STATE_2_DEVINS(a_pAC97) ((a_pAC97)->pDevInsR3)
200
201enum
202{
203 BUP_SET = RT_BIT(0),
204 BUP_LAST = RT_BIT(1)
205};
206
207/** Emits registers for a specific (Native Audio Bus Master BAR) NABMBAR. */
208#define AC97_NABMBAR_REGS(prefix, off) \
209 enum { \
210 prefix ## _BDBAR = off, /* Buffer Descriptor Base Address */ \
211 prefix ## _CIV = off + 4, /* Current Index Value */ \
212 prefix ## _LVI = off + 5, /* Last Valid Index */ \
213 prefix ## _SR = off + 6, /* Status Register */ \
214 prefix ## _PICB = off + 8, /* Position in Current Buffer */ \
215 prefix ## _PIV = off + 10, /* Prefetched Index Value */ \
216 prefix ## _CR = off + 11 /* Control Register */ \
217 }
218
219#ifndef VBOX_DEVICE_STRUCT_TESTCASE
220typedef enum
221{
222 AC97SOUNDSOURCE_PI_INDEX = 0, /** PCM in */
223 AC97SOUNDSOURCE_PO_INDEX, /** PCM out */
224 AC97SOUNDSOURCE_MC_INDEX, /** Mic in */
225 AC97SOUNDSOURCE_LAST_INDEX
226} AC97SOUNDSOURCE;
227
228AC97_NABMBAR_REGS(PI, AC97SOUNDSOURCE_PI_INDEX * 16);
229AC97_NABMBAR_REGS(PO, AC97SOUNDSOURCE_PO_INDEX * 16);
230AC97_NABMBAR_REGS(MC, AC97SOUNDSOURCE_MC_INDEX * 16);
231#endif
232
233enum
234{
235 /** NABMBAR: Global Control Register. */
236 AC97_GLOB_CNT = 0x2c,
237 /** NABMBAR Global Status. */
238 AC97_GLOB_STA = 0x30,
239 /** Codec Access Semaphore Register. */
240 AC97_CAS = 0x34
241};
242
243#define AC97_PORT2IDX(a_idx) ( ((a_idx) >> 4) & 3 )
244
245
246/*********************************************************************************************************************************
247* Structures and Typedefs *
248*********************************************************************************************************************************/
249
250/**
251 * Buffer Descriptor List Entry (BDLE).
252 */
253typedef struct AC97BDLE
254{
255 uint32_t addr;
256 uint32_t ctl_len;
257} AC97BDLE, *PAC97BDLE;
258
259/**
260 * Bus master register set for an audio stream.
261 */
262typedef struct AC97BMREGS
263{
264 uint32_t bdbar; /** rw 0, Buffer Descriptor List: BAR (Base Address Register). */
265 uint8_t civ; /** ro 0, Current index value. */
266 uint8_t lvi; /** rw 0, Last valid index. */
267 uint16_t sr; /** rw 1, Status register. */
268 uint16_t picb; /** ro 0, Position in current buffer (in samples). */
269 uint8_t piv; /** ro 0, Prefetched index value. */
270 uint8_t cr; /** rw 0, Control register. */
271 int bd_valid; /** Whether current BDLE is initialized or not. */
272 AC97BDLE bd; /** Current Buffer Descriptor List Entry (BDLE). */
273} AC97BMREGS, *PAC97BMREGS;
274
275#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
276/**
277 * Structure keeping the AC'97 stream's state for asynchronous I/O.
278 */
279typedef struct AC97STREAMSTATEAIO
280{
281 /** Thread handle for the actual I/O thread. */
282 RTTHREAD Thread;
283 /** Event for letting the thread know there is some data to process. */
284 RTSEMEVENT Event;
285 /** Critical section for synchronizing access. */
286 RTCRITSECT CritSect;
287 /** Started indicator. */
288 volatile bool fStarted;
289 /** Shutdown indicator. */
290 volatile bool fShutdown;
291 /** Whether the thread should do any data processing or not. */
292 volatile bool fEnabled;
293 uint32_t Padding1;
294} AC97STREAMSTATEAIO, *PAC97STREAMSTATEAIO;
295#endif
296
297/**
298 * Structure for keeping the internal state of an AC'97 stream.
299 */
300typedef struct AC97STREAMSTATE
301{
302 /** Circular buffer (FIFO) for holding DMA'ed data. */
303 R3PTRTYPE(PRTCIRCBUF) pCircBuf;
304 /** Criticial section for this stream. */
305 RTCRITSECT CritSect;
306 /** The stream's current configuration. */
307 PDMAUDIOSTREAMCFG Cfg;
308 uint32_t Padding;
309#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
310 /** Asynchronous I/O state members. */
311 AC97STREAMSTATEAIO AIO;
312#endif
313} AC97STREAMSTATE, *PAC97STREAMSTATE;
314
315/**
316 * Structure for an AC'97 stream.
317 */
318typedef struct AC97STREAM
319{
320 /** Stream number (SDn). */
321 uint8_t u8SD;
322 /** Bus master registers of this stream. */
323 AC97BMREGS Regs;
324 /** Internal state of this stream. */
325 AC97STREAMSTATE State;
326} AC97STREAM, *PAC97STREAM;
327
328typedef struct AC97STATE *PAC97STATE;
329#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
330/**
331 * Structure for the async I/O thread context.
332 */
333typedef struct AC97STREAMTHREADCTX
334{
335 PAC97STATE pThis;
336 PAC97STREAM pStream;
337} AC97STREAMTHREADCTX, *PAC97STREAMTHREADCTX;
338#endif
339
340/**
341 * Structure defining a (host backend) driver stream.
342 * Each driver has its own instances of audio mixer streams, which then
343 * can go into the same (or even different) audio mixer sinks.
344 */
345typedef struct AC97DRIVERSTREAM
346{
347 union
348 {
349 /** Desired playback destination (for an output stream). */
350 PDMAUDIOPLAYBACKDEST Dest;
351 /** Desired recording source (for an input stream). */
352 PDMAUDIORECSOURCE Source;
353 } DestSource;
354 uint8_t Padding1[4];
355 /** Associated mixer stream handle. */
356 R3PTRTYPE(PAUDMIXSTREAM) pMixStrm;
357} AC97DRIVERSTREAM, *PAC97DRIVERSTREAM;
358
359/**
360 * Struct for maintaining a host backend driver.
361 */
362typedef struct AC97DRIVER
363{
364 /** Node for storing this driver in our device driver list of AC97STATE. */
365 RTLISTNODER3 Node;
366 /** Pointer to AC97 controller (state). */
367 R3PTRTYPE(PAC97STATE) pAC97State;
368 /** Driver flags. */
369 PDMAUDIODRVFLAGS fFlags;
370 uint32_t PaddingFlags;
371 /** LUN # to which this driver has been assigned. */
372 uint8_t uLUN;
373 /** Whether this driver is in an attached state or not. */
374 bool fAttached;
375 uint8_t Padding[4];
376 /** Pointer to attached driver base interface. */
377 R3PTRTYPE(PPDMIBASE) pDrvBase;
378 /** Audio connector interface to the underlying host backend. */
379 R3PTRTYPE(PPDMIAUDIOCONNECTOR) pConnector;
380 /** Driver stream for line input. */
381 AC97DRIVERSTREAM LineIn;
382 /** Driver stream for mic input. */
383 AC97DRIVERSTREAM MicIn;
384 /** Driver stream for output. */
385 AC97DRIVERSTREAM Out;
386} AC97DRIVER, *PAC97DRIVER;
387
388/**
389 * Structure for maintaining an AC'97 device state.
390 */
391typedef struct AC97STATE
392{
393 /** The PCI device state. */
394 PDMPCIDEV PciDev;
395 /** Critical section protecting the AC'97 state. */
396 PDMCRITSECT CritSect;
397 /** R3 Pointer to the device instance. */
398 PPDMDEVINSR3 pDevInsR3;
399 /** Global Control (Bus Master Control Register). */
400 uint32_t glob_cnt;
401 /** Global Status (Bus Master Control Register). */
402 uint32_t glob_sta;
403 /** Codec Access Semaphore Register (Bus Master Control Register). */
404 uint32_t cas;
405 uint32_t last_samp;
406 uint8_t mixer_data[256];
407 /** AC'97 stream for line-in. */
408 AC97STREAM StreamLineIn;
409 /** AC'97 stream for microphone-in. */
410 AC97STREAM StreamMicIn;
411 /** AC'97 stream for output. */
412 AC97STREAM StreamOut;
413 /** Number of active (running) SDn streams. */
414 uint8_t cStreamsActive;
415#ifndef VBOX_WITH_AUDIO_AC97_CALLBACKS
416 /** The timer for pumping data thru the attached LUN drivers. */
417 PTMTIMERR3 pTimer;
418# if HC_ARCH_BITS == 32
419 uint32_t Padding0;
420# endif
421 /** Flag indicating whether the timer is active or not. */
422 bool fTimerActive;
423 uint8_t u8Padding1[7];
424 /** The timer interval for pumping data thru the LUN drivers in timer ticks. */
425 uint64_t cTimerTicks;
426 /** Timestamp of the last timer callback (ac97Timer).
427 * Used to calculate the time actually elapsed between two timer callbacks. */
428 uint64_t uTimerTS;
429#endif
430#ifdef VBOX_WITH_STATISTICS
431 STAMPROFILE StatTimer;
432 STAMPROFILE StatIn;
433 STAMPROFILE StatOut;
434 STAMCOUNTER StatBytesRead;
435 STAMCOUNTER StatBytesWritten;
436#endif
437 /** List of associated LUN drivers (AC97DRIVER). */
438 RTLISTANCHOR lstDrv;
439 /** The device's software mixer. */
440 R3PTRTYPE(PAUDIOMIXER) pMixer;
441 /** Audio sink for PCM output. */
442 R3PTRTYPE(PAUDMIXSINK) pSinkOut;
443 /** Audio sink for line input. */
444 R3PTRTYPE(PAUDMIXSINK) pSinkLineIn;
445 /** Audio sink for microphone input. */
446 R3PTRTYPE(PAUDMIXSINK) pSinkMicIn;
447 uint8_t silence[128];
448 int bup_flag;
449 /** The base interface for LUN\#0. */
450 PDMIBASE IBase;
451 /** Base port of the I/O space region. */
452 RTIOPORT IOPortBase[2];
453 /** Codec model. */
454 uint32_t uCodecModel;
455} AC97STATE, *PAC97STATE;
456
457/**
458 * Acquires the AC'97 lock.
459 */
460#define DEVAC97_LOCK(a_pThis) \
461 do { \
462 int rcLock = PDMCritSectEnter(&(a_pThis)->CritSect, VERR_IGNORED); \
463 AssertRC(rcLock); \
464 } while (0)
465
466/**
467 * Acquires the AC'97 lock or returns.
468 */
469# define DEVAC97_LOCK_RETURN(a_pThis, a_rcBusy) \
470 do { \
471 int rcLock = PDMCritSectEnter(&(a_pThis)->CritSect, a_rcBusy); \
472 if (rcLock != VINF_SUCCESS) \
473 { \
474 AssertRC(rcLock); \
475 return rcLock; \
476 } \
477 } while (0)
478
479/**
480 * Acquires the AC'97 lock or returns.
481 */
482# define DEVAC97_LOCK_RETURN_VOID(a_pThis) \
483 do { \
484 int rcLock = PDMCritSectEnter(&(a_pThis)->CritSect, VERR_IGNORED); \
485 if (rcLock != VINF_SUCCESS) \
486 { \
487 AssertRC(rcLock); \
488 return; \
489 } \
490 } while (0)
491
492/**
493 * Releases the AC'97 lock.
494 */
495#define DEVAC97_UNLOCK(a_pThis) \
496 do { PDMCritSectLeave(&(a_pThis)->CritSect); } while (0)
497
498/**
499 * Acquires the TM lock and AC'97 lock, returns on failure.
500 */
501#define DEVAC97_LOCK_BOTH_RETURN_VOID(a_pThis) \
502 do { \
503 int rcLock = TMTimerLock((a_pThis)->pTimer, VERR_IGNORED); \
504 if (rcLock != VINF_SUCCESS) \
505 { \
506 AssertRC(rcLock); \
507 return; \
508 } \
509 rcLock = PDMCritSectEnter(&(a_pThis)->CritSect, VERR_IGNORED); \
510 if (rcLock != VINF_SUCCESS) \
511 { \
512 AssertRC(rcLock); \
513 TMTimerUnlock((a_pThis)->pTimer); \
514 return; \
515 } \
516 } while (0)
517
518/**
519 * Acquires the TM lock and AC'97 lock, returns on failure.
520 */
521#define DEVAC97_LOCK_BOTH_RETURN(a_pThis, a_rcBusy) \
522 do { \
523 int rcLock = TMTimerLock((a_pThis)->pTimer, (a_rcBusy)); \
524 if (rcLock != VINF_SUCCESS) \
525 return rcLock; \
526 rcLock = PDMCritSectEnter(&(a_pThis)->CritSect, (a_rcBusy)); \
527 if (rcLock != VINF_SUCCESS) \
528 { \
529 AssertRC(rcLock); \
530 TMTimerUnlock((a_pThis)->pTimer); \
531 return rcLock; \
532 } \
533 } while (0)
534
535/**
536 * Releases the AC'97 lock and TM lock.
537 */
538#define DEVAC97_UNLOCK_BOTH(a_pThis) \
539 do { \
540 PDMCritSectLeave(&(a_pThis)->CritSect); \
541 TMTimerUnlock((a_pThis)->pTimer); \
542 } while (0)
543
544#ifdef VBOX_WITH_STATISTICS
545AssertCompileMemberAlignment(AC97STATE, StatTimer, 8);
546AssertCompileMemberAlignment(AC97STATE, StatBytesRead, 8);
547AssertCompileMemberAlignment(AC97STATE, StatBytesWritten, 8);
548#endif
549
550#ifndef VBOX_DEVICE_STRUCT_TESTCASE
551
552DECLINLINE(PAC97STREAM) ichac97GetStreamFromIdx(PAC97STATE pThis, uint32_t uIdx);
553static int ichac97StreamCreate(PAC97STATE pThis, PAC97STREAM pStream, uint8_t u8Strm);
554static void ichac97StreamDestroy(PAC97STATE pThis, PAC97STREAM pStream);
555static int ichac97StreamOpen(PAC97STATE pThis, PAC97STREAM pStream);
556static int ichac97StreamReOpen(PAC97STATE pThis, PAC97STREAM pStream);
557static int ichac97StreamClose(PAC97STATE pThis, PAC97STREAM pStream);
558static void ichac97StreamReset(PAC97STATE pThis, PAC97STREAM pStream);
559static void ichac97StreamLock(PAC97STREAM pStream);
560static void ichac97StreamUnlock(PAC97STREAM pStream);
561static uint32_t ichac97StreamGetUsed(PAC97STREAM pStream);
562static uint32_t ichac97StreamGetFree(PAC97STREAM pStream);
563static int ichac97StreamTransfer(PAC97STATE pThis, PAC97STREAM pStream, uint32_t cbToProcessMax);
564static void ichac97StreamUpdate(PAC97STATE pThis, PAC97STREAM pStream, bool fInTimer);
565
566static DECLCALLBACK(void) ichac97Reset(PPDMDEVINS pDevIns);
567#ifndef VBOX_WITH_AUDIO_AC97_CALLBACKS
568static int ichac97TimerStart(PAC97STATE pThis);
569static int ichac97TimerMaybeStart(PAC97STATE pThis);
570static int ichac97TimerStop(PAC97STATE pThis);
571static int ichac97TimerMaybeStop(PAC97STATE pThis);
572static void ichac97TimerMain(PAC97STATE pThis);
573static DECLCALLBACK(void) ichac97Timer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser);
574#endif
575static void ichac97DoTransfers(PAC97STATE pThis);
576
577static int ichac97MixerAddDrvStream(PAC97STATE pThis, PAUDMIXSINK pMixSink, PPDMAUDIOSTREAMCFG pCfg, PAC97DRIVER pDrv);
578static int ichac97MixerAddDrvStreams(PAC97STATE pThis, PAUDMIXSINK pMixSink, PPDMAUDIOSTREAMCFG pCfg);
579static void ichac97MixerRemoveDrvStream(PAC97STATE pThis, PAUDMIXSINK pMixSink, PDMAUDIODIR enmDir, PDMAUDIODESTSOURCE dstSrc, PAC97DRIVER pDrv);
580static void ichac97MixerRemoveDrvStreams(PAC97STATE pThis, PAUDMIXSINK pMixSink, PDMAUDIODIR enmDir, PDMAUDIODESTSOURCE dstSrc);
581
582#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
583static DECLCALLBACK(int) ichac97StreamAsyncIOThread(RTTHREAD hThreadSelf, void *pvUser);
584static int ichac97StreamAsyncIOCreate(PAC97STATE pThis, PAC97STREAM pStream);
585static int ichac97StreamAsyncIODestroy(PAC97STATE pThis, PAC97STREAM pStream);
586static int ichac97StreamAsyncIONotify(PAC97STATE pThis, PAC97STREAM pStream);
587static void ichac97StreamAsyncIOLock(PAC97STREAM pStream);
588static void ichac97StreamAsyncIOUnlock(PAC97STREAM pStream);
589static void ichac97StreamAsyncIOEnable(PAC97STREAM pStream, bool fEnable);
590#endif
591
592static void ichac97WarmReset(PAC97STATE pThis)
593{
594 NOREF(pThis);
595}
596
597static void ichac97ColdReset(PAC97STATE pThis)
598{
599 NOREF(pThis);
600}
601
602/**
603 * Retrieves the audio mixer sink of a corresponding AC'97 stream index.
604 *
605 * @returns Pointer to audio mixer sink if found, or NULL if not found / invalid.
606 * @param pThis AC'97 state.
607 * @param uIndex Stream index to get audio mixer sink for.
608 */
609DECLINLINE(PAUDMIXSINK) ichac97IndexToSink(PAC97STATE pThis, uint8_t uIndex)
610{
611 AssertPtrReturn(pThis, NULL);
612
613 switch (uIndex)
614 {
615 case AC97SOUNDSOURCE_PI_INDEX: return pThis->pSinkLineIn; break;
616 case AC97SOUNDSOURCE_PO_INDEX: return pThis->pSinkOut; break;
617 case AC97SOUNDSOURCE_MC_INDEX: return pThis->pSinkMicIn; break;
618 default: break;
619 }
620
621 AssertMsgFailed(("Wrong index %RU8\n", uIndex));
622 return NULL;
623}
624
625/**
626 * Fetches the current BDLE (Buffer Descriptor List Entry) of an AC'97 audio stream.
627 *
628 * @returns IPRT status code.
629 * @param pThis AC'97 state.
630 * @param pStream AC'97 stream to fetch BDLE for.
631 *
632 * @remark Uses CIV as BDLE index.
633 */
634static void ichac97StreamFetchBDLE(PAC97STATE pThis, PAC97STREAM pStream)
635{
636 PPDMDEVINS pDevIns = ICHAC97STATE_2_DEVINS(pThis);
637 PAC97BMREGS pRegs = &pStream->Regs;
638
639 uint32_t u32[2];
640
641 PDMDevHlpPhysRead(pDevIns, pRegs->bdbar + pRegs->civ * 8, &u32[0], sizeof(u32));
642 pRegs->bd_valid = 1;
643#if !defined(RT_ARCH_X86) && !defined(RT_ARCH_AMD64)
644# error Please adapt the code (audio buffers are little endian)!
645#else
646 pRegs->bd.addr = RT_H2LE_U32(u32[0] & ~3);
647 pRegs->bd.ctl_len = RT_H2LE_U32(u32[1]);
648#endif
649 pRegs->picb = pRegs->bd.ctl_len & AC97_BD_LEN_MASK;
650 LogFlowFunc(("bd %2d addr=%#x ctl=%#06x len=%#x(%d bytes)\n",
651 pRegs->civ, pRegs->bd.addr, pRegs->bd.ctl_len >> 16,
652 pRegs->bd.ctl_len & AC97_BD_LEN_MASK,
653 (pRegs->bd.ctl_len & AC97_BD_LEN_MASK) << 1)); /** @todo r=andy Assumes 16bit samples. */
654}
655
656/**
657 * Updates the status register (SR) of an AC'97 audio stream.
658 *
659 * @param pThis AC'97 state.
660 * @param pStream AC'97 stream to update SR for.
661 * @param new_sr New value for status register (SR).
662 */
663static void ichac97StreamUpdateSR(PAC97STATE pThis, PAC97STREAM pStream, uint32_t new_sr)
664{
665 PPDMDEVINS pDevIns = ICHAC97STATE_2_DEVINS(pThis);
666 PAC97BMREGS pRegs = &pStream->Regs;
667
668 bool fSignal = false;
669 int iIRQL = 0;
670
671 uint32_t new_mask = new_sr & AC97_SR_INT_MASK;
672 uint32_t old_mask = pRegs->sr & AC97_SR_INT_MASK;
673
674 static uint32_t const masks[] = { AC97_GS_PIINT, AC97_GS_POINT, AC97_GS_MINT };
675
676 if (new_mask ^ old_mask)
677 {
678 /** @todo Is IRQ deasserted when only one of status bits is cleared? */
679 if (!new_mask)
680 {
681 fSignal = true;
682 iIRQL = 0;
683 }
684 else if ((new_mask & AC97_SR_LVBCI) && (pRegs->cr & AC97_CR_LVBIE))
685 {
686 fSignal = true;
687 iIRQL = 1;
688 }
689 else if ((new_mask & AC97_SR_BCIS) && (pRegs->cr & AC97_CR_IOCE))
690 {
691 fSignal = true;
692 iIRQL = 1;
693 }
694 }
695
696 pRegs->sr = new_sr;
697
698 LogFlowFunc(("IOC%d, LVB%d, sr=%#x, fSignal=%RTbool, IRQL=%d\n",
699 pRegs->sr & AC97_SR_BCIS, pRegs->sr & AC97_SR_LVBCI, pRegs->sr, fSignal, iIRQL));
700
701 if (fSignal)
702 {
703 if (iIRQL)
704 pThis->glob_sta |= masks[pStream->u8SD];
705 else
706 pThis->glob_sta &= ~masks[pStream->u8SD];
707
708 LogFlowFunc(("Setting IRQ level=%d\n", iIRQL));
709 PDMDevHlpPCISetIrq(pDevIns, 0, iIRQL);
710 }
711}
712
713/**
714 * Returns whether an AC'97 stream is enabled or not.
715 *
716 * @returns IPRT status code.
717 * @param pThis AC'97 device state.
718 * @param pStream Stream to return status for.
719 */
720static bool ichac97StreamIsEnabled(PAC97STATE pThis, PAC97STREAM pStream)
721{
722 AssertPtrReturn(pThis, false);
723 AssertPtrReturn(pStream, false);
724
725 PAUDMIXSINK pSink = ichac97IndexToSink(pThis, pStream->u8SD);
726 bool fIsEnabled = RT_BOOL(AudioMixerSinkGetStatus(pSink) & AUDMIXSINK_STS_RUNNING);
727
728 LogFunc(("[SD%RU8] fIsEnabled=%RTbool\n", pStream->u8SD, fIsEnabled));
729 return fIsEnabled;
730}
731
732/**
733 * Enables or disables an AC'97 audio stream.
734 *
735 * @returns IPRT status code.
736 * @param pThis AC'97 state.
737 * @param pStream AC'97 stream to enable or disable.
738 * @param fEnable Whether to enable or disable the stream.
739 *
740 */
741static int ichac97StreamEnable(PAC97STATE pThis, PAC97STREAM pStream, bool fEnable)
742{
743 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
744 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
745
746 ichac97StreamLock(pStream);
747
748 int rc = VINF_SUCCESS;
749
750#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
751 if (fEnable)
752 rc = ichac97StreamAsyncIOCreate(pThis, pStream);
753 if (RT_SUCCESS(rc))
754 {
755 ichac97StreamAsyncIOLock(pStream);
756 ichac97StreamAsyncIOEnable(pStream, fEnable);
757 }
758#endif
759
760 if (fEnable)
761 {
762 if (pStream->State.pCircBuf)
763 RTCircBufReset(pStream->State.pCircBuf);
764
765 rc = ichac97StreamOpen(pThis, pStream);
766 }
767 else
768 rc = ichac97StreamClose(pThis, pStream);
769
770 if (RT_SUCCESS(rc))
771 {
772 /* First, enable or disable the stream and the stream's sink, if any. */
773 rc = AudioMixerSinkCtl(ichac97IndexToSink(pThis, pStream->u8SD),
774 fEnable ? AUDMIXSINKCMD_ENABLE : AUDMIXSINKCMD_DISABLE);
775 }
776
777#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
778 ichac97StreamAsyncIOUnlock(pStream);
779#endif
780
781 /* Make sure to leave the lock before (eventually) starting the timer. */
782 ichac97StreamUnlock(pStream);
783
784#ifndef VBOX_WITH_AUDIO_AC97_CALLBACKS
785 /* Second, see if we need to start or stop the timer. */
786 if (!fEnable)
787 ichac97TimerMaybeStop(pThis);
788 else
789 ichac97TimerMaybeStart(pThis);
790#endif
791
792 LogFunc(("[SD%RU8] cStreamsActive=%RU8, fEnable=%RTbool, rc=%Rrc\n", pStream->u8SD, pThis->cStreamsActive, fEnable, rc));
793 return rc;
794}
795
796/**
797 * Resets an AC'97 stream.
798 *
799 * @param pThis AC'97 state.
800 * @param pStream AC'97 stream to reset.
801 *
802 */
803static void ichac97StreamReset(PAC97STATE pThis, PAC97STREAM pStream)
804{
805 AssertPtrReturnVoid(pThis);
806 AssertPtrReturnVoid(pStream);
807
808 ichac97StreamLock(pStream);
809
810 LogFunc(("[SD%RU8]\n", pStream->u8SD));
811
812 if (pStream->State.pCircBuf)
813 RTCircBufReset(pStream->State.pCircBuf);
814
815 PAC97BMREGS pRegs = &pStream->Regs;
816
817 pRegs->bdbar = 0;
818 pRegs->civ = 0;
819 pRegs->lvi = 0;
820
821 pRegs->picb = 0;
822 pRegs->piv = 0;
823 pRegs->cr = pRegs->cr & AC97_CR_DONT_CLEAR_MASK;
824 pRegs->bd_valid = 0;
825
826 RT_ZERO(pThis->silence);
827
828 ichac97StreamUnlock(pStream);
829}
830
831/**
832 * Creates an AC'97 audio stream.
833 *
834 * @returns IPRT status code.
835 * @param pThis AC'97 state.
836 * @param pStream AC'97 stream to create.
837 * @param u8Strm Stream ID to assign AC'97 stream to.
838 */
839static int ichac97StreamCreate(PAC97STATE pThis, PAC97STREAM pStream, uint8_t u8Strm)
840{
841 RT_NOREF(pThis);
842 AssertPtrReturn(pStream, VERR_INVALID_PARAMETER);
843 /** @todo Validate u8Strm. */
844
845 LogFunc(("[SD%RU8] pStream=%p\n", u8Strm, pStream));
846
847 pStream->u8SD = u8Strm;
848
849 int rc = RTCritSectInit(&pStream->State.CritSect);
850 if (RT_SUCCESS(rc))
851 rc = RTCircBufCreate(&pStream->State.pCircBuf, _4K); /** @todo Make this configurable. */
852
853 return rc;
854}
855
856/**
857 * Destroys an AC'97 audio stream.
858 *
859 * @returns IPRT status code.
860 * @param pThis AC'97 state.
861 * @param pStream AC'97 stream to destroy.
862 */
863static void ichac97StreamDestroy(PAC97STATE pThis, PAC97STREAM pStream)
864{
865 LogFlowFunc(("[SD%RU8]\n", pStream->u8SD));
866
867 int rc2 = RTCritSectDelete(&pStream->State.CritSect);
868 AssertRC(rc2);
869
870 if (pStream->State.pCircBuf)
871 {
872 RTCircBufDestroy(pStream->State.pCircBuf);
873 pStream->State.pCircBuf = NULL;
874 }
875
876#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
877 rc2 = ichac97StreamAsyncIODestroy(pThis, pStream);
878 AssertRC(rc2);
879#else
880 RT_NOREF(pThis);
881#endif
882
883 LogFlowFuncLeave();
884}
885
886/**
887 * Destroys all AC'97 audio streams of the device.
888 *
889 * @param pThis AC'97 state.
890 */
891static void ichac97StreamsDestroy(PAC97STATE pThis)
892{
893 LogFlowFuncEnter();
894
895 /*
896 * Destroy all AC'97 streams.
897 */
898
899 ichac97StreamDestroy(pThis, &pThis->StreamLineIn);
900 ichac97StreamDestroy(pThis, &pThis->StreamMicIn);
901 ichac97StreamDestroy(pThis, &pThis->StreamOut);
902
903 /*
904 * Destroy all sinks.
905 */
906
907 PDMAUDIODESTSOURCE dstSrc;
908 if (pThis->pSinkLineIn)
909 {
910 dstSrc.Source = PDMAUDIORECSOURCE_LINE;
911 ichac97MixerRemoveDrvStreams(pThis, pThis->pSinkLineIn, PDMAUDIODIR_IN, dstSrc);
912
913 AudioMixerSinkDestroy(pThis->pSinkLineIn);
914 pThis->pSinkLineIn = NULL;
915 }
916
917 if (pThis->pSinkMicIn)
918 {
919 dstSrc.Source = PDMAUDIORECSOURCE_MIC;
920 ichac97MixerRemoveDrvStreams(pThis, pThis->pSinkMicIn, PDMAUDIODIR_IN, dstSrc);
921
922 AudioMixerSinkDestroy(pThis->pSinkMicIn);
923 pThis->pSinkMicIn = NULL;
924 }
925
926 if (pThis->pSinkOut)
927 {
928 dstSrc.Dest = PDMAUDIOPLAYBACKDEST_FRONT;
929 ichac97MixerRemoveDrvStreams(pThis, pThis->pSinkOut, PDMAUDIODIR_OUT, dstSrc);
930
931 AudioMixerSinkDestroy(pThis->pSinkOut);
932 pThis->pSinkOut = NULL;
933 }
934}
935
936/**
937 * Writes audio data from a mixer sink into an AC'97 stream's DMA buffer.
938 *
939 * @returns IPRT status code.
940 * @param pThis AC'97 state.
941 * @param pDstStream AC'97 stream to write to.
942 * @param pSrcMixSink Mixer sink to get audio data to write from.
943 * @param cbToWrite Number of bytes to write.
944 * @param pcbWritten Number of bytes written. Optional.
945 */
946static int ichac97StreamWrite(PAC97STATE pThis, PAC97STREAM pDstStream, PAUDMIXSINK pSrcMixSink, uint32_t cbToWrite,
947 uint32_t *pcbWritten)
948{
949 RT_NOREF(pThis);
950 AssertPtrReturn(pDstStream, VERR_INVALID_POINTER);
951 AssertPtrReturn(pSrcMixSink, VERR_INVALID_POINTER);
952 AssertReturn(cbToWrite, VERR_INVALID_PARAMETER);
953 /* pcbWritten is optional. */
954
955 PRTCIRCBUF pCircBuf = pDstStream->State.pCircBuf;
956 AssertPtr(pCircBuf);
957
958 void *pvDst;
959 size_t cbDst;
960
961 uint32_t cbRead = 0;
962
963 RTCircBufAcquireWriteBlock(pCircBuf, cbToWrite, &pvDst, &cbDst);
964
965 if (cbDst)
966 {
967 int rc2 = AudioMixerSinkRead(pSrcMixSink, AUDMIXOP_COPY, pvDst, (uint32_t)cbDst, &cbRead);
968 AssertRC(rc2);
969
970#ifdef VBOX_AUDIO_DEBUG_DUMP_PCM_DATA
971 RTFILE fh;
972 RTFileOpen(&fh, VBOX_AUDIO_DEBUG_DUMP_PCM_DATA_PATH "ichac97StreamWrite.pcm",
973 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
974 RTFileWrite(fh, pvDst, cbRead, NULL);
975 RTFileClose(fh);
976#endif
977 }
978
979 RTCircBufReleaseWriteBlock(pCircBuf, cbRead);
980
981 if (pcbWritten)
982 *pcbWritten = cbRead;
983
984 return VINF_SUCCESS;
985}
986
987/**
988 * Reads audio data from an AC'97 stream's DMA buffer and writes into a specified mixer sink.
989 *
990 * @returns IPRT status code.
991 * @param pThis AC'97 state.
992 * @param pSrcStream AC'97 stream to read audio data from.
993 * @param pDstMixSink Mixer sink to write audio data to.
994 * @param cbToRead Number of bytes to read.
995 * @param pcbRead Number of bytes read. Optional.
996 */
997static int ichac97StreamRead(PAC97STATE pThis, PAC97STREAM pSrcStream, PAUDMIXSINK pDstMixSink, uint32_t cbToRead,
998 uint32_t *pcbRead)
999{
1000 RT_NOREF(pThis);
1001 AssertPtrReturn(pSrcStream, VERR_INVALID_POINTER);
1002 AssertPtrReturn(pDstMixSink, VERR_INVALID_POINTER);
1003 AssertReturn(cbToRead, VERR_INVALID_PARAMETER);
1004 /* pcbRead is optional. */
1005
1006 int rc = VINF_SUCCESS;
1007
1008 uint32_t cbReadTotal = 0;
1009
1010 PRTCIRCBUF pCircBuf = pSrcStream->State.pCircBuf;
1011 AssertPtr(pCircBuf);
1012
1013 void *pvSrc;
1014 size_t cbSrc;
1015
1016 while (cbToRead)
1017 {
1018 uint32_t cbWritten = 0;
1019
1020 RTCircBufAcquireReadBlock(pCircBuf, cbToRead, &pvSrc, &cbSrc);
1021
1022 if (cbSrc)
1023 {
1024#ifdef VBOX_AUDIO_DEBUG_DUMP_PCM_DATA
1025 RTFILE fh;
1026 RTFileOpen(&fh, VBOX_AUDIO_DEBUG_DUMP_PCM_DATA_PATH "ac97StreamRead.pcm",
1027 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
1028 RTFileWrite(fh, pvSrc, cbSrc, NULL);
1029 RTFileClose(fh);
1030#endif
1031 rc = AudioMixerSinkWrite(pDstMixSink, AUDMIXOP_COPY, pvSrc, (uint32_t)cbSrc, &cbWritten);
1032 if (RT_SUCCESS(rc))
1033 {
1034 Assert(cbWritten <= cbSrc);
1035
1036 cbReadTotal += cbWritten;
1037
1038 Assert(cbToRead >= cbWritten);
1039 cbToRead -= cbWritten;
1040 }
1041 }
1042
1043 RTCircBufReleaseReadBlock(pCircBuf, cbWritten);
1044
1045 if ( !cbWritten
1046 || !RTCircBufUsed(pCircBuf))
1047 break;
1048
1049 if (RT_FAILURE(rc))
1050 break;
1051 }
1052
1053 if (pcbRead)
1054 *pcbRead = cbReadTotal;
1055
1056 return rc;
1057}
1058
1059#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
1060/**
1061 * Asynchronous I/O thread for an AC'97 stream.
1062 * This will do the heavy lifting work for us as soon as it's getting notified by another thread.
1063 *
1064 * @returns IPRT status code.
1065 * @param hThreadSelf Thread handle.
1066 * @param pvUser User argument. Must be of type PAC97STREAMTHREADCTX.
1067 */
1068static DECLCALLBACK(int) ichac97StreamAsyncIOThread(RTTHREAD hThreadSelf, void *pvUser)
1069{
1070 PAC97STREAMTHREADCTX pCtx = (PAC97STREAMTHREADCTX)pvUser;
1071 AssertPtr(pCtx);
1072
1073 PAC97STATE pThis = pCtx->pThis;
1074 AssertPtr(pThis);
1075
1076 PAC97STREAM pStream = pCtx->pStream;
1077 AssertPtr(pStream);
1078
1079 PAC97STREAMSTATEAIO pAIO = &pCtx->pStream->State.AIO;
1080
1081 PRTCIRCBUF pCircBuf = pStream->State.pCircBuf;
1082 AssertPtr(pCircBuf);
1083
1084 PAUDMIXSINK pMixSink = ichac97IndexToSink(pThis, pStream->u8SD);
1085 AssertPtr(pMixSink);
1086
1087 ASMAtomicXchgBool(&pAIO->fStarted, true);
1088
1089 RTThreadUserSignal(hThreadSelf);
1090
1091 LogFunc(("[SD%RU8] Started\n", pStream->u8SD));
1092
1093 for (;;)
1094 {
1095 Log2Func(("[SD%RU8] Waiting ...\n", pStream->u8SD));
1096
1097 int rc2 = RTSemEventWait(pAIO->Event, RT_INDEFINITE_WAIT);
1098 if (RT_FAILURE(rc2))
1099 break;
1100
1101 if (ASMAtomicReadBool(&pAIO->fShutdown))
1102 break;
1103
1104 rc2 = RTCritSectEnter(&pAIO->CritSect);
1105 if (RT_SUCCESS(rc2))
1106 {
1107 if (!pAIO->fEnabled)
1108 {
1109 RTCritSectLeave(&pAIO->CritSect);
1110 continue;
1111 }
1112
1113 ichac97StreamUpdate(pThis, pStream, false /* fInTimer */);
1114
1115 int rc3 = RTCritSectLeave(&pAIO->CritSect);
1116 AssertRC(rc3);
1117 }
1118
1119 AssertRC(rc2);
1120 }
1121
1122 LogFunc(("[SD%RU8] Ended\n", pStream->u8SD));
1123
1124 ASMAtomicXchgBool(&pAIO->fStarted, false);
1125
1126 return VINF_SUCCESS;
1127}
1128
1129/**
1130 * Creates the async I/O thread for a specific AC'97 audio stream.
1131 *
1132 * @returns IPRT status code.
1133 * @param pThis AC'97 state.
1134 * @param pStream AC'97 audio stream to create the async I/O thread for.
1135 */
1136static int ichac97StreamAsyncIOCreate(PAC97STATE pThis, PAC97STREAM pStream)
1137{
1138 PAC97STREAMSTATEAIO pAIO = &pStream->State.AIO;
1139
1140 int rc;
1141
1142 if (!ASMAtomicReadBool(&pAIO->fStarted))
1143 {
1144 pAIO->fShutdown = false;
1145
1146 rc = RTSemEventCreate(&pAIO->Event);
1147 if (RT_SUCCESS(rc))
1148 {
1149 rc = RTCritSectInit(&pAIO->CritSect);
1150 if (RT_SUCCESS(rc))
1151 {
1152 AC97STREAMTHREADCTX Ctx = { pThis, pStream };
1153
1154 char szThreadName[64];
1155 RTStrPrintf2(szThreadName, sizeof(szThreadName), "ac97AIO%RU8", pStream->u8SD);
1156
1157 rc = RTThreadCreate(&pAIO->Thread, ichac97StreamAsyncIOThread, &Ctx,
1158 0, RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, szThreadName);
1159 if (RT_SUCCESS(rc))
1160 rc = RTThreadUserWait(pAIO->Thread, 10 * 1000 /* 10s timeout */);
1161 }
1162 }
1163 }
1164 else
1165 rc = VINF_SUCCESS;
1166
1167 LogFunc(("[SD%RU8] Returning %Rrc\n", pStream->u8SD, rc));
1168 return rc;
1169}
1170
1171/**
1172 * Destroys the async I/O thread of a specific AC'97 audio stream.
1173 *
1174 * @returns IPRT status code.
1175 * @param pThis AC'97 state.
1176 * @param pStream AC'97 audio stream to destroy the async I/O thread for.
1177 */
1178static int ichac97StreamAsyncIODestroy(PAC97STATE pThis, PAC97STREAM pStream)
1179{
1180 PAC97STREAMSTATEAIO pAIO = &pStream->State.AIO;
1181
1182 if (!ASMAtomicReadBool(&pAIO->fStarted))
1183 return VINF_SUCCESS;
1184
1185 ASMAtomicWriteBool(&pAIO->fShutdown, true);
1186
1187 int rc = ichac97StreamAsyncIONotify(pThis, pStream);
1188 AssertRC(rc);
1189
1190 int rcThread;
1191 rc = RTThreadWait(pAIO->Thread, 30 * 1000 /* 30s timeout */, &rcThread);
1192 LogFunc(("Async I/O thread ended with %Rrc (%Rrc)\n", rc, rcThread));
1193
1194 if (RT_SUCCESS(rc))
1195 {
1196 rc = RTCritSectDelete(&pAIO->CritSect);
1197 AssertRC(rc);
1198
1199 rc = RTSemEventDestroy(pAIO->Event);
1200 AssertRC(rc);
1201
1202 pAIO->fStarted = false;
1203 pAIO->fShutdown = false;
1204 pAIO->fEnabled = false;
1205 }
1206
1207 LogFunc(("[SD%RU8] Returning %Rrc\n", pStream->u8SD, rc));
1208 return rc;
1209}
1210
1211/**
1212 * Lets the stream's async I/O thread know that there is some data to process.
1213 *
1214 * @returns IPRT status code.
1215 * @param pThis AC'97 state.
1216 * @param pStream AC'97 stream to notify async I/O thread for.
1217 */
1218static int ichac97StreamAsyncIONotify(PAC97STATE pThis, PAC97STREAM pStream)
1219{
1220 RT_NOREF(pThis);
1221
1222 LogFunc(("[SD%RU8]\n", pStream->u8SD));
1223 return RTSemEventSignal(pStream->State.AIO.Event);
1224}
1225
1226/**
1227 * Locks the async I/O thread of a specific AC'97 audio stream.
1228 *
1229 * @param pStream AC'97 stream to lock async I/O thread for.
1230 */
1231static void ichac97StreamAsyncIOLock(PAC97STREAM pStream)
1232{
1233 PAC97STREAMSTATEAIO pAIO = &pStream->State.AIO;
1234
1235 if (!ASMAtomicReadBool(&pAIO->fStarted))
1236 return;
1237
1238 int rc2 = RTCritSectEnter(&pAIO->CritSect);
1239 AssertRC(rc2);
1240}
1241
1242/**
1243 * Unlocks the async I/O thread of a specific AC'97 audio stream.
1244 *
1245 * @param pStream AC'97 stream to unlock async I/O thread for.
1246 */
1247static void ichac97StreamAsyncIOUnlock(PAC97STREAM pStream)
1248{
1249 PAC97STREAMSTATEAIO pAIO = &pStream->State.AIO;
1250
1251 if (!ASMAtomicReadBool(&pAIO->fStarted))
1252 return;
1253
1254 int rc2 = RTCritSectLeave(&pAIO->CritSect);
1255 AssertRC(rc2);
1256}
1257
1258/**
1259 * Enables (resumes) or disables (pauses) the async I/O thread.
1260 *
1261 * @param pStream AC'97 stream to enable/disable async I/O thread for.
1262 * @param fEnable Whether to enable or disable the I/O thread.
1263 *
1264 * @remarks Does not do locking.
1265 */
1266static void ichac97StreamAsyncIOEnable(PAC97STREAM pStream, bool fEnable)
1267{
1268 PAC97STREAMSTATEAIO pAIO = &pStream->State.AIO;
1269 ASMAtomicXchgBool(&pAIO->fEnabled, fEnable);
1270}
1271#endif /* VBOX_WITH_AUDIO_AC97_ASYNC_IO */
1272
1273/**
1274 * Updates an AC'97 stream by doing its required data transfers.
1275 * The host sink(s) set the overall pace.
1276 *
1277 * This routine is called by both, the synchronous and the asynchronous, implementations.
1278 *
1279 * @param pThis AC'97 state.
1280 * @param pStream AC'97 stream to update.
1281 * @param fInTimer Whether to this function was called from the timer
1282 * context or an asynchronous I/O stream thread (if supported).
1283 */
1284static void ichac97StreamUpdate(PAC97STATE pThis, PAC97STREAM pStream, bool fInTimer)
1285{
1286 PAUDMIXSINK pSink = ichac97IndexToSink(pThis, pStream->u8SD);
1287 AssertPtr(pSink);
1288
1289 if (!AudioMixerSinkIsActive(pSink)) /* No sink available? Bail out. */
1290 return;
1291
1292 int rc2;
1293
1294 if (pStream->u8SD == AC97SOUNDSOURCE_PO_INDEX) /* Output (SDO). */
1295 {
1296 /* Is the AC'97 stream ready to be written (guest output data) to? If so, by how much? */
1297 const uint32_t cbFree = ichac97StreamGetFree(pStream);
1298
1299 if ( fInTimer
1300 && cbFree)
1301 {
1302 Log3Func(("[SD%RU8] cbFree=%RU32\n", pStream->u8SD, cbFree));
1303
1304 /* Do the DMA transfer. */
1305 rc2 = ichac97StreamTransfer(pThis, pStream, cbFree);
1306 AssertRC(rc2);
1307 }
1308
1309 /* How much (guest output) data is available at the moment for the AC'97 stream? */
1310 uint32_t cbUsed = ichac97StreamGetUsed(pStream);
1311
1312#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
1313 if ( fInTimer
1314 && cbUsed)
1315 {
1316 rc2 = ichac97StreamAsyncIONotify(pThis, pStream);
1317 AssertRC(rc2);
1318 }
1319 else
1320 {
1321#endif
1322 const uint32_t cbSinkWritable = AudioMixerSinkGetWritable(pSink);
1323
1324 /* Do not write more than the sink can hold at the moment.
1325 * The host sets the overall pace. */
1326 if (cbUsed > cbSinkWritable)
1327 cbUsed = cbSinkWritable;
1328
1329 if (cbUsed)
1330 {
1331 /* Read (guest output) data and write it to the stream's sink. */
1332 uint32_t cbRead;
1333 rc2 = ichac97StreamRead(pThis, pStream, pSink, cbUsed, &cbRead);
1334 AssertRC(rc2);
1335
1336 AssertMsg(cbUsed == cbRead, ("[SD%RU8] %RU32 bytes announced to be ready to read but %RU32 bytes read\n",
1337 pStream->u8SD, cbUsed, cbRead));
1338 }
1339
1340 /* When running synchronously, update the associated sink here.
1341 * Otherwise this will be done in the device timer. */
1342 rc2 = AudioMixerSinkUpdate(pSink);
1343 AssertRC(rc2);
1344
1345#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
1346 }
1347#endif
1348 }
1349 else /* Input (SDI). */
1350 {
1351#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
1352 if (fInTimer)
1353 {
1354 rc2 = ichac97StreamAsyncIONotify(pThis, pStream);
1355 AssertRC(rc2);
1356 }
1357 else
1358 {
1359#endif
1360 rc2 = AudioMixerSinkUpdate(pSink);
1361 AssertRC(rc2);
1362
1363 /* Is the sink ready to be read (host input data) from? If so, by how much? */
1364 const uint32_t cbReadable = AudioMixerSinkGetReadable(pSink);
1365
1366 /* How much (guest input) data is free at the moment? */
1367 uint32_t cbFree = ichac97StreamGetFree(pStream);
1368
1369 Log3Func(("[SD%RU8] cbReadable=%RU32, cbFree=%RU32\n", pStream->u8SD, cbReadable, cbFree));
1370
1371 /* Do not read more than the sink can provide at the moment.
1372 * The host sets the overall pace. */
1373 if (cbFree > cbReadable)
1374 cbFree = cbReadable;
1375
1376 if (cbFree)
1377 {
1378 /* Write (guest input) data to the stream which was read from stream's sink before. */
1379 rc2 = ichac97StreamWrite(pThis, pStream, pSink, cbFree, NULL /* pcbWritten */);
1380 AssertRC(rc2);
1381 }
1382#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
1383 }
1384#endif
1385
1386#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
1387 if (fInTimer)
1388 {
1389#endif
1390 const uint32_t cbToTransfer = ichac97StreamGetUsed(pStream);
1391 if (cbToTransfer)
1392 {
1393 /* When running synchronously, do the DMA data transfers here.
1394 * Otherwise this will be done in the stream's async I/O thread. */
1395 rc2 = ichac97StreamTransfer(pThis, pStream, cbToTransfer);
1396 AssertRC(rc2);
1397 }
1398#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
1399 }
1400#endif
1401 }
1402}
1403
1404/**
1405 * Sets a AC'97 mixer control to a specific value.
1406 *
1407 * @returns IPRT status code.
1408 * @param pThis AC'97 state.
1409 * @param uMixerIdx Mixer control to set value for.
1410 * @param uVal Value to set.
1411 */
1412static void ichac97MixerSet(PAC97STATE pThis, uint8_t uMixerIdx, uint16_t uVal)
1413{
1414 if (size_t(uMixerIdx + 2) > sizeof(pThis->mixer_data))
1415 {
1416 AssertMsgFailed(("Index %RU8 out of bounds(%zu)\n", uMixerIdx, sizeof(pThis->mixer_data)));
1417 return;
1418 }
1419
1420 pThis->mixer_data[uMixerIdx + 0] = RT_LO_U8(uVal);
1421 pThis->mixer_data[uMixerIdx + 1] = RT_HI_U8(uVal);
1422}
1423
1424/**
1425 * Gets a value from a specific AC'97 mixer control.
1426 *
1427 * @returns Retrieved mixer control value.
1428 * @param pThis AC'97 state.
1429 * @param uMixerIdx Mixer control to get value for.
1430 */
1431static uint16_t ichac97MixerGet(PAC97STATE pThis, uint32_t uMixerIdx)
1432{
1433 uint16_t uVal;
1434
1435 if (size_t(uMixerIdx + 2) > sizeof(pThis->mixer_data))
1436 {
1437 AssertMsgFailed(("Index %RU8 out of bounds (%zu)\n", uMixerIdx, sizeof(pThis->mixer_data)));
1438 uVal = UINT16_MAX;
1439 }
1440 else
1441 uVal = RT_MAKE_U16(pThis->mixer_data[uMixerIdx + 0], pThis->mixer_data[uMixerIdx + 1]);
1442
1443 return uVal;
1444}
1445
1446/**
1447 * Retrieves a specific driver stream of a AC'97 driver.
1448 *
1449 * @returns Pointer to driver stream if found, or NULL if not found.
1450 * @param pThis AC'97 state.
1451 * @param pDrv Driver to retrieve driver stream for.
1452 * @param enmDir Stream direction to retrieve.
1453 * @param dstSrc Stream destination / source to retrieve.
1454 */
1455static PAC97DRIVERSTREAM ichac97MixerGetDrvStream(PAC97STATE pThis, PAC97DRIVER pDrv,
1456 PDMAUDIODIR enmDir, PDMAUDIODESTSOURCE dstSrc)
1457{
1458 RT_NOREF(pThis);
1459
1460 PAC97DRIVERSTREAM pDrvStream = NULL;
1461
1462 if (enmDir == PDMAUDIODIR_IN)
1463 {
1464 LogFunc(("enmRecSource=%d\n", dstSrc.Source));
1465
1466 switch (dstSrc.Source)
1467 {
1468 case PDMAUDIORECSOURCE_LINE:
1469 pDrvStream = &pDrv->LineIn;
1470 break;
1471 case PDMAUDIORECSOURCE_MIC:
1472 pDrvStream = &pDrv->MicIn;
1473 break;
1474 default:
1475 AssertFailed();
1476 break;
1477 }
1478 }
1479 else if (enmDir == PDMAUDIODIR_OUT)
1480 {
1481 LogFunc(("enmPlaybackDest=%d\n", dstSrc.Dest));
1482
1483 switch (dstSrc.Dest)
1484 {
1485 case PDMAUDIOPLAYBACKDEST_FRONT:
1486 pDrvStream = &pDrv->Out;
1487 break;
1488 default:
1489 AssertFailed();
1490 break;
1491 }
1492 }
1493 else
1494 AssertFailed();
1495
1496 return pDrvStream;
1497}
1498
1499/**
1500 * Adds a driver stream to a specific mixer sink.
1501 *
1502 * @returns IPRT status code.
1503 * @param pThis AC'97 state.
1504 * @param pMixSink Mixer sink to add driver stream to.
1505 * @param pCfg Stream configuration to use.
1506 * @param pDrv Driver stream to add.
1507 */
1508static int ichac97MixerAddDrvStream(PAC97STATE pThis, PAUDMIXSINK pMixSink, PPDMAUDIOSTREAMCFG pCfg, PAC97DRIVER pDrv)
1509{
1510 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1511 AssertPtrReturn(pMixSink, VERR_INVALID_POINTER);
1512 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1513
1514 PPDMAUDIOSTREAMCFG pStreamCfg = DrvAudioHlpStreamCfgDup(pCfg);
1515 if (!pStreamCfg)
1516 return VERR_NO_MEMORY;
1517
1518 if (!RTStrPrintf(pStreamCfg->szName, sizeof(pStreamCfg->szName), "%s", pCfg->szName))
1519 {
1520 RTMemFree(pStreamCfg);
1521 return VERR_BUFFER_OVERFLOW;
1522 }
1523
1524 LogFunc(("[LUN#%RU8] %s\n", pDrv->uLUN, pStreamCfg->szName));
1525
1526 int rc;
1527
1528 PAC97DRIVERSTREAM pDrvStream = ichac97MixerGetDrvStream(pThis, pDrv, pStreamCfg->enmDir, pStreamCfg->DestSource);
1529 if (pDrvStream)
1530 {
1531 AssertMsg(pDrvStream->pMixStrm == NULL, ("[LUN#%RU8] Driver stream already present when it must not\n", pDrv->uLUN));
1532
1533 PAUDMIXSTREAM pMixStrm;
1534 rc = AudioMixerSinkCreateStream(pMixSink, pDrv->pConnector, pStreamCfg, 0 /* fFlags */, &pMixStrm);
1535 if (RT_SUCCESS(rc))
1536 {
1537 rc = AudioMixerSinkAddStream(pMixSink, pMixStrm);
1538 LogFlowFunc(("LUN#%RU8: Created stream \"%s\", rc=%Rrc\n", pDrv->uLUN, pCfg->szName, rc));
1539 }
1540
1541 if (RT_SUCCESS(rc))
1542 pDrvStream->pMixStrm = pMixStrm;
1543 }
1544 else
1545 rc = VERR_INVALID_PARAMETER;
1546
1547 if (pStreamCfg)
1548 {
1549 RTMemFree(pStreamCfg);
1550 pStreamCfg = NULL;
1551 }
1552
1553 LogFlowFuncLeaveRC(rc);
1554 return rc;
1555}
1556
1557/**
1558 * Adds all current driver streams to a specific mixer sink.
1559 *
1560 * @returns IPRT status code.
1561 * @param pThis AC'97 state.
1562 * @param pMixSink Mixer sink to add stream to.
1563 * @param pCfg Stream configuration to use.
1564 */
1565static int ichac97MixerAddDrvStreams(PAC97STATE pThis, PAUDMIXSINK pMixSink, PPDMAUDIOSTREAMCFG pCfg)
1566{
1567 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1568 AssertPtrReturn(pMixSink, VERR_INVALID_POINTER);
1569 AssertPtrReturn(pCfg, VERR_INVALID_POINTER);
1570
1571 if (!DrvAudioHlpStreamCfgIsValid(pCfg))
1572 return VERR_INVALID_PARAMETER;
1573
1574 int rc = AudioMixerSinkSetFormat(pMixSink, &pCfg->Props);
1575 if (RT_FAILURE(rc))
1576 return rc;
1577
1578 PAC97DRIVER pDrv;
1579 RTListForEach(&pThis->lstDrv, pDrv, AC97DRIVER, Node)
1580 {
1581 int rc2 = ichac97MixerAddDrvStream(pThis, pMixSink, pCfg, pDrv);
1582 if (RT_FAILURE(rc2))
1583 LogFunc(("Attaching stream failed with %Rrc\n", rc2));
1584
1585 /* Do not pass failure to rc here, as there might be drivers which aren't
1586 * configured / ready yet. */
1587 }
1588
1589 LogFlowFuncLeaveRC(rc);
1590 return rc;
1591}
1592
1593/**
1594 * Removes a driver stream from a specific mixer sink.
1595 *
1596 * @param pThis AC'97 state.
1597 * @param pMixSink Mixer sink to remove audio streams from.
1598 * @param enmDir Stream direction to remove.
1599 * @param dstSrc Stream destination / source to remove.
1600 * @param pDrv Driver stream to remove.
1601 */
1602static void ichac97MixerRemoveDrvStream(PAC97STATE pThis, PAUDMIXSINK pMixSink,
1603 PDMAUDIODIR enmDir, PDMAUDIODESTSOURCE dstSrc, PAC97DRIVER pDrv)
1604{
1605 AssertPtrReturnVoid(pThis);
1606 AssertPtrReturnVoid(pMixSink);
1607
1608 PAC97DRIVERSTREAM pDrvStream = ichac97MixerGetDrvStream(pThis, pDrv, enmDir, dstSrc);
1609 if (pDrvStream)
1610 {
1611 if (pDrvStream->pMixStrm)
1612 {
1613 AudioMixerSinkRemoveStream(pMixSink, pDrvStream->pMixStrm);
1614
1615 AudioMixerStreamDestroy(pDrvStream->pMixStrm);
1616 pDrvStream->pMixStrm = NULL;
1617 }
1618 }
1619}
1620
1621/**
1622 * Removes all driver streams from a specific mixer sink.
1623 *
1624 * @param pThis AC'97 state.
1625 * @param pMixSink Mixer sink to remove audio streams from.
1626 * @param enmDir Stream direction to remove.
1627 * @param dstSrc Stream destination / source to remove.
1628 */
1629static void ichac97MixerRemoveDrvStreams(PAC97STATE pThis, PAUDMIXSINK pMixSink,
1630 PDMAUDIODIR enmDir, PDMAUDIODESTSOURCE dstSrc)
1631{
1632 AssertPtrReturnVoid(pThis);
1633 AssertPtrReturnVoid(pMixSink);
1634
1635 PAC97DRIVER pDrv;
1636 RTListForEach(&pThis->lstDrv, pDrv, AC97DRIVER, Node)
1637 ichac97MixerRemoveDrvStream(pThis, pMixSink, enmDir, dstSrc, pDrv);
1638}
1639
1640/**
1641 * Opens an AC'97 stream with its current mixer settings.
1642 *
1643 * This will open an AC'97 stream with 2 (stereo) channels, 16-bit samples and
1644 * the last set sample rate in the AC'97 mixer for this stream.
1645 *
1646 * @returns IPRT status code.
1647 * @param pThis AC'97 state.
1648 * @param pStream AC'97 Stream to open.
1649 */
1650static int ichac97StreamOpen(PAC97STATE pThis, PAC97STREAM pStream)
1651{
1652 int rc = VINF_SUCCESS;
1653
1654 LogFunc(("[SD%RU8]\n", pStream->u8SD));
1655
1656 RT_ZERO(pStream->State.Cfg);
1657
1658 PPDMAUDIOSTREAMCFG pCfg = &pStream->State.Cfg;
1659 PAUDMIXSINK pMixSink = NULL;
1660
1661 switch (pStream->u8SD)
1662 {
1663 case AC97SOUNDSOURCE_PI_INDEX:
1664 {
1665 pCfg->Props.uHz = ichac97MixerGet(pThis, AC97_PCM_LR_ADC_Rate);
1666 pCfg->enmDir = PDMAUDIODIR_IN;
1667 pCfg->DestSource.Source = PDMAUDIORECSOURCE_LINE;
1668 pCfg->enmLayout = PDMAUDIOSTREAMLAYOUT_NON_INTERLEAVED;
1669
1670 RTStrPrintf2(pCfg->szName, sizeof(pCfg->szName), "Line-In");
1671
1672 pMixSink = pThis->pSinkLineIn;
1673 break;
1674 }
1675
1676 case AC97SOUNDSOURCE_MC_INDEX:
1677 {
1678 pCfg->Props.uHz = ichac97MixerGet(pThis, AC97_MIC_ADC_Rate);
1679 pCfg->enmDir = PDMAUDIODIR_IN;
1680 pCfg->DestSource.Source = PDMAUDIORECSOURCE_MIC;
1681 pCfg->enmLayout = PDMAUDIOSTREAMLAYOUT_NON_INTERLEAVED;
1682
1683 RTStrPrintf2(pCfg->szName, sizeof(pCfg->szName), "Mic-In");
1684
1685 pMixSink = pThis->pSinkMicIn;
1686 break;
1687 }
1688
1689 case AC97SOUNDSOURCE_PO_INDEX:
1690 {
1691 pCfg->Props.uHz = ichac97MixerGet(pThis, AC97_PCM_Front_DAC_Rate);
1692 pCfg->enmDir = PDMAUDIODIR_OUT;
1693 pCfg->DestSource.Dest = PDMAUDIOPLAYBACKDEST_FRONT;
1694 pCfg->enmLayout = PDMAUDIOSTREAMLAYOUT_NON_INTERLEAVED;
1695
1696 RTStrPrintf2(pCfg->szName, sizeof(pCfg->szName), "Output");
1697
1698 pMixSink = pThis->pSinkOut;
1699 break;
1700 }
1701
1702 default:
1703 rc = VERR_NOT_SUPPORTED;
1704 break;
1705 }
1706
1707 if (RT_SUCCESS(rc))
1708 {
1709 ichac97MixerRemoveDrvStreams(pThis, pMixSink, pCfg->enmDir, pCfg->DestSource);
1710
1711 if (pCfg->Props.uHz)
1712 {
1713 Assert(pCfg->enmDir != PDMAUDIODIR_UNKNOWN);
1714
1715 pCfg->Props.cChannels = 2;
1716 pCfg->Props.cBits = 16;
1717 pCfg->Props.fSigned = true;
1718 pCfg->Props.cShift = PDMAUDIOPCMPROPS_MAKE_SHIFT_PARMS(pCfg->Props.cBits, pCfg->Props.cChannels);
1719
1720 rc = ichac97MixerAddDrvStreams(pThis, pMixSink, pCfg);
1721 }
1722 }
1723
1724 LogFlowFunc(("[SD%RU8] rc=%Rrc\n", pStream->u8SD, rc));
1725 return rc;
1726}
1727
1728/**
1729 * Closes an AC'97 stream.
1730 *
1731 * @returns IPRT status code.
1732 * @param pThis AC'97 state.
1733 * @param pStream AC'97 stream to close.
1734 */
1735static int ichac97StreamClose(PAC97STATE pThis, PAC97STREAM pStream)
1736{
1737 RT_NOREF(pThis);
1738 RT_NOREF(pStream);
1739
1740 LogFlowFunc(("[SD%RU8]\n", pStream->u8SD));
1741
1742 return VINF_SUCCESS;
1743}
1744
1745/**
1746 * Re-opens (that is, closes and opens again) an AC'97 stream on the backend
1747 * side with the current AC'97 mixer settings for this stream.
1748 *
1749 * @returns IPRT status code.
1750 * @param pThis AC'97 device state.
1751 * @param pStream AC'97 stream to re-open.
1752 */
1753static int ichac97StreamReOpen(PAC97STATE pThis, PAC97STREAM pStream)
1754{
1755 LogFlowFunc(("[SD%RU8]\n", pStream->u8SD));
1756
1757 int rc = ichac97StreamClose(pThis, pStream);
1758 if (RT_SUCCESS(rc))
1759 rc = ichac97StreamOpen(pThis, pStream);
1760
1761 return rc;
1762}
1763
1764/**
1765 * Locks an AC'97 stream for serialized access.
1766 *
1767 * @returns IPRT status code.
1768 * @param pStream AC'97 stream to lock.
1769 */
1770static void ichac97StreamLock(PAC97STREAM pStream)
1771{
1772 AssertPtrReturnVoid(pStream);
1773 int rc2 = RTCritSectEnter(&pStream->State.CritSect);
1774 AssertRC(rc2);
1775}
1776
1777/**
1778 * Unlocks a formerly locked AC'97 stream.
1779 *
1780 * @returns IPRT status code.
1781 * @param pStream AC'97 stream to unlock.
1782 */
1783static void ichac97StreamUnlock(PAC97STREAM pStream)
1784{
1785 AssertPtrReturnVoid(pStream);
1786 int rc2 = RTCritSectLeave(&pStream->State.CritSect);
1787 AssertRC(rc2);
1788}
1789
1790/**
1791 * Retrieves the available size of (buffered) audio data (in bytes) of a given AC'97 stream.
1792 *
1793 * @returns Available data (in bytes).
1794 * @param pStream AC'97 stream to retrieve size for.
1795 */
1796static uint32_t ichac97StreamGetUsed(PAC97STREAM pStream)
1797{
1798 AssertPtrReturn(pStream, 0);
1799
1800 if (!pStream->State.pCircBuf)
1801 return 0;
1802
1803 return (uint32_t)RTCircBufUsed(pStream->State.pCircBuf);
1804}
1805
1806/**
1807 * Retrieves the free size of audio data (in bytes) of a given AC'97 stream.
1808 *
1809 * @returns Free data (in bytes).
1810 * @param pStream AC'97 stream to retrieve size for.
1811 */
1812static uint32_t ichac97StreamGetFree(PAC97STREAM pStream)
1813{
1814 AssertPtrReturn(pStream, 0);
1815
1816 if (!pStream->State.pCircBuf)
1817 return 0;
1818
1819 return (uint32_t)RTCircBufFree(pStream->State.pCircBuf);
1820}
1821
1822/**
1823 * Sets the volume of a specific AC'97 mixer control.
1824 *
1825 * This currently only supports attenuation -- gain support is currently not implemented.
1826 *
1827 * @returns IPRT status code.
1828 * @param pThis AC'97 state.
1829 * @param index AC'97 mixer index to set volume for.
1830 * @param enmMixerCtl Corresponding audio mixer sink.
1831 * @param uVal Volume value to set.
1832 */
1833static int ichac97MixerSetVolume(PAC97STATE pThis, int index, PDMAUDIOMIXERCTL enmMixerCtl, uint32_t uVal)
1834{
1835 /*
1836 * From AC'97 SoundMax Codec AD1981A/AD1981B:
1837 * "Because AC '97 defines 6-bit volume registers, to maintain compatibility whenever the
1838 * D5 or D13 bits are set to 1, their respective lower five volume bits are automatically
1839 * set to 1 by the Codec logic. On readback, all lower 5 bits will read ones whenever
1840 * these bits are set to 1."
1841 *
1842 * Linux ALSA depends on this behavior.
1843 */
1844 /// @todo Does this apply to anything other than the master volume control?
1845 if (uVal & RT_BIT(5)) /* D5 bit set? */
1846 uVal |= RT_BIT(4) | RT_BIT(3) | RT_BIT(2) | RT_BIT(1) | RT_BIT(0);
1847 if (uVal & RT_BIT(13)) /* D13 bit set? */
1848 uVal |= RT_BIT(12) | RT_BIT(11) | RT_BIT(10) | RT_BIT(9) | RT_BIT(8);
1849
1850 const bool fCtlMuted = (uVal >> AC97_BARS_VOL_MUTE_SHIFT) & 1;
1851 uint8_t uCtlAttLeft = (uVal >> 8) & AC97_BARS_VOL_MASK;
1852 uint8_t uCtlAttRight = uVal & AC97_BARS_VOL_MASK;
1853
1854 /* For the master and headphone volume, 0 corresponds to 0dB attenuation. For the other
1855 * volume controls, 0 means 12dB gain and 8 means unity gain.
1856 */
1857 if (index != AC97_Master_Volume_Mute && index != AC97_Headphone_Volume_Mute)
1858 {
1859#ifndef VBOX_WITH_AC97_GAIN_SUPPORT
1860 /* NB: Currently there is no gain support, only attenuation. */
1861 uCtlAttLeft = uCtlAttLeft < 8 ? 0 : uCtlAttLeft - 8;
1862 uCtlAttRight = uCtlAttRight < 8 ? 0 : uCtlAttRight - 8;
1863#endif
1864 }
1865 Assert(uCtlAttLeft <= 255 / AC97_DB_FACTOR);
1866 Assert(uCtlAttRight <= 255 / AC97_DB_FACTOR);
1867
1868 LogFunc(("index=0x%x, uVal=%RU32, enmMixerCtl=%RU32\n", index, uVal, enmMixerCtl));
1869 LogFunc(("uCtlAttLeft=%RU8, uCtlAttRight=%RU8 ", uCtlAttLeft, uCtlAttRight));
1870
1871 /*
1872 * For AC'97 volume controls, each additional step means -1.5dB attenuation with
1873 * zero being maximum. In contrast, we're internally using 255 (PDMAUDIO_VOLUME_MAX)
1874 * steps, each -0.375dB, where 0 corresponds to -96dB and 255 corresponds to 0dB.
1875 */
1876 uint8_t lVol = PDMAUDIO_VOLUME_MAX - uCtlAttLeft * AC97_DB_FACTOR;
1877 uint8_t rVol = PDMAUDIO_VOLUME_MAX - uCtlAttRight * AC97_DB_FACTOR;
1878
1879 Log(("-> fMuted=%RTbool, lVol=%RU8, rVol=%RU8\n", fCtlMuted, lVol, rVol));
1880
1881 int rc = VINF_SUCCESS;
1882
1883 if (pThis->pMixer) /* Device can be in reset state, so no mixer available. */
1884 {
1885 PDMAUDIOVOLUME Vol = { fCtlMuted, lVol, rVol };
1886 PAUDMIXSINK pSink = NULL;
1887
1888 switch (enmMixerCtl)
1889 {
1890 case PDMAUDIOMIXERCTL_VOLUME_MASTER:
1891 rc = AudioMixerSetMasterVolume(pThis->pMixer, &Vol);
1892 break;
1893
1894 case PDMAUDIOMIXERCTL_FRONT:
1895 pSink = pThis->pSinkOut;
1896 break;
1897
1898 case PDMAUDIOMIXERCTL_MIC_IN:
1899 pSink = pThis->pSinkMicIn;
1900 break;
1901
1902 case PDMAUDIOMIXERCTL_LINE_IN:
1903 pSink = pThis->pSinkLineIn;
1904 break;
1905
1906 default:
1907 AssertFailed();
1908 rc = VERR_NOT_SUPPORTED;
1909 break;
1910 }
1911
1912 if (pSink)
1913 rc = AudioMixerSinkSetVolume(pSink, &Vol);
1914 }
1915
1916 ichac97MixerSet(pThis, index, uVal);
1917
1918 if (RT_FAILURE(rc))
1919 LogFlowFunc(("Failed with %Rrc\n", rc));
1920
1921 return rc;
1922}
1923
1924/**
1925 * Converts an AC'97 recording source index to a PDM audio recording source.
1926 *
1927 * @returns PDM audio recording source.
1928 * @param uIdx AC'97 index to convert.
1929 */
1930static PDMAUDIORECSOURCE ichac97IdxToRecSource(uint8_t uIdx)
1931{
1932 switch (uIdx)
1933 {
1934 case AC97_REC_MIC: return PDMAUDIORECSOURCE_MIC;
1935 case AC97_REC_CD: return PDMAUDIORECSOURCE_CD;
1936 case AC97_REC_VIDEO: return PDMAUDIORECSOURCE_VIDEO;
1937 case AC97_REC_AUX: return PDMAUDIORECSOURCE_AUX;
1938 case AC97_REC_LINE_IN: return PDMAUDIORECSOURCE_LINE;
1939 case AC97_REC_PHONE: return PDMAUDIORECSOURCE_PHONE;
1940 default:
1941 break;
1942 }
1943
1944 LogFlowFunc(("Unknown record source %d, using MIC\n", uIdx));
1945 return PDMAUDIORECSOURCE_MIC;
1946}
1947
1948/**
1949 * Converts a PDM audio recording source to an AC'97 recording source index.
1950 *
1951 * @returns AC'97 recording source index.
1952 * @param enmRecSrc PDM audio recording source to convert.
1953 */
1954static uint8_t ichac97RecSourceToIdx(PDMAUDIORECSOURCE enmRecSrc)
1955{
1956 switch (enmRecSrc)
1957 {
1958 case PDMAUDIORECSOURCE_MIC: return AC97_REC_MIC;
1959 case PDMAUDIORECSOURCE_CD: return AC97_REC_CD;
1960 case PDMAUDIORECSOURCE_VIDEO: return AC97_REC_VIDEO;
1961 case PDMAUDIORECSOURCE_AUX: return AC97_REC_AUX;
1962 case PDMAUDIORECSOURCE_LINE: return AC97_REC_LINE_IN;
1963 case PDMAUDIORECSOURCE_PHONE: return AC97_REC_PHONE;
1964 default:
1965 break;
1966 }
1967
1968 LogFlowFunc(("Unknown audio recording source %d using MIC\n", enmRecSrc));
1969 return AC97_REC_MIC;
1970}
1971
1972/**
1973 * Retrieves an AC'97 audio stream from an AC'97 stream index.
1974 *
1975 * @returns Pointer to AC'97 audio stream if found, or NULL if not found / invalid.
1976 * @param pThis AC'97 state.
1977 * @param uIdx AC'97 stream index to retrieve AC'97 audio stream for.
1978 */
1979DECLINLINE(PAC97STREAM) ichac97GetStreamFromIdx(PAC97STATE pThis, uint32_t uIdx)
1980{
1981 switch (uIdx)
1982 {
1983 case AC97SOUNDSOURCE_PI_INDEX: return &pThis->StreamLineIn;
1984 case AC97SOUNDSOURCE_MC_INDEX: return &pThis->StreamMicIn;
1985 case AC97SOUNDSOURCE_PO_INDEX: return &pThis->StreamOut;
1986 default: break;
1987 }
1988
1989 return NULL;
1990}
1991
1992/**
1993 * Performs an AC'97 mixer record select to switch to a different recording
1994 * source.
1995 *
1996 * @param pThis AC'97 state.
1997 * @param val AC'97 recording source index to set.
1998 */
1999static void ichac97MixerRecordSelect(PAC97STATE pThis, uint32_t val)
2000{
2001 uint8_t rs = val & AC97_REC_MASK;
2002 uint8_t ls = (val >> 8) & AC97_REC_MASK;
2003 PDMAUDIORECSOURCE ars = ichac97IdxToRecSource(rs);
2004 PDMAUDIORECSOURCE als = ichac97IdxToRecSource(ls);
2005 rs = ichac97RecSourceToIdx(ars);
2006 ls = ichac97RecSourceToIdx(als);
2007 ichac97MixerSet(pThis, AC97_Record_Select, rs | (ls << 8));
2008}
2009
2010/**
2011 * Resets the AC'97 mixer.
2012 *
2013 * @returns IPRT status code.
2014 * @param pThis AC'97 state.
2015 */
2016static int ichac97MixerReset(PAC97STATE pThis)
2017{
2018 AssertPtrReturn(pThis, VERR_INVALID_PARAMETER);
2019
2020 LogFlowFuncEnter();
2021
2022 RT_ZERO(pThis->mixer_data);
2023
2024 /* Note: Make sure to reset all registers first before bailing out on error. */
2025
2026 ichac97MixerSet(pThis, AC97_Reset , 0x0000); /* 6940 */
2027 ichac97MixerSet(pThis, AC97_Master_Volume_Mono_Mute , 0x8000);
2028 ichac97MixerSet(pThis, AC97_PC_BEEP_Volume_Mute , 0x0000);
2029
2030 ichac97MixerSet(pThis, AC97_Phone_Volume_Mute , 0x8008);
2031 ichac97MixerSet(pThis, AC97_Mic_Volume_Mute , 0x8008);
2032 ichac97MixerSet(pThis, AC97_CD_Volume_Mute , 0x8808);
2033 ichac97MixerSet(pThis, AC97_Aux_Volume_Mute , 0x8808);
2034 ichac97MixerSet(pThis, AC97_Record_Gain_Mic_Mute , 0x8000);
2035 ichac97MixerSet(pThis, AC97_General_Purpose , 0x0000);
2036 ichac97MixerSet(pThis, AC97_3D_Control , 0x0000);
2037 ichac97MixerSet(pThis, AC97_Powerdown_Ctrl_Stat , 0x000f);
2038
2039 ichac97MixerSet(pThis, AC97_Extended_Audio_ID , 0x0809);
2040 ichac97MixerSet(pThis, AC97_Extended_Audio_Ctrl_Stat, 0x0009);
2041 ichac97MixerSet(pThis, AC97_PCM_Front_DAC_Rate , 0xbb80);
2042 ichac97MixerSet(pThis, AC97_PCM_Surround_DAC_Rate , 0xbb80);
2043 ichac97MixerSet(pThis, AC97_PCM_LFE_DAC_Rate , 0xbb80);
2044 ichac97MixerSet(pThis, AC97_PCM_LR_ADC_Rate , 0xbb80);
2045 ichac97MixerSet(pThis, AC97_MIC_ADC_Rate , 0xbb80);
2046
2047 if (pThis->uCodecModel == AC97_CODEC_AD1980)
2048 {
2049 /* Analog Devices 1980 (AD1980) */
2050 ichac97MixerSet(pThis, AC97_Reset , 0x0010); /* Headphones. */
2051 ichac97MixerSet(pThis, AC97_Vendor_ID1 , 0x4144);
2052 ichac97MixerSet(pThis, AC97_Vendor_ID2 , 0x5370);
2053 ichac97MixerSet(pThis, AC97_Headphone_Volume_Mute , 0x8000);
2054 }
2055 else if (pThis->uCodecModel == AC97_CODEC_AD1981B)
2056 {
2057 /* Analog Devices 1981B (AD1981B) */
2058 ichac97MixerSet(pThis, AC97_Vendor_ID1 , 0x4144);
2059 ichac97MixerSet(pThis, AC97_Vendor_ID2 , 0x5374);
2060 }
2061 else
2062 {
2063 /* Sigmatel 9700 (STAC9700) */
2064 ichac97MixerSet(pThis, AC97_Vendor_ID1 , 0x8384);
2065 ichac97MixerSet(pThis, AC97_Vendor_ID2 , 0x7600); /* 7608 */
2066 }
2067 ichac97MixerRecordSelect(pThis, 0);
2068
2069 /* The default value is 8000h, which corresponds to 0 dB attenuation with mute on. */
2070 ichac97MixerSetVolume(pThis, AC97_Master_Volume_Mute, PDMAUDIOMIXERCTL_VOLUME_MASTER, 0x8000);
2071
2072 /* The default value for stereo registers is 8808h, which corresponds to 0 dB gain with mute on.*/
2073 ichac97MixerSetVolume(pThis, AC97_PCM_Out_Volume_Mute, PDMAUDIOMIXERCTL_FRONT, 0x8808);
2074 ichac97MixerSetVolume(pThis, AC97_Line_In_Volume_Mute, PDMAUDIOMIXERCTL_LINE_IN, 0x8808);
2075 ichac97MixerSetVolume(pThis, AC97_Mic_Volume_Mute, PDMAUDIOMIXERCTL_MIC_IN, 0x8808);
2076
2077 return VINF_SUCCESS;
2078}
2079
2080/* Unused */
2081#if 0
2082static void ichac97WriteBUP(PAC97STATE pThis, uint32_t cbElapsed)
2083{
2084 LogFlowFunc(("cbElapsed=%RU32\n", cbElapsed));
2085
2086 if (!(pThis->bup_flag & BUP_SET))
2087 {
2088 if (pThis->bup_flag & BUP_LAST)
2089 {
2090 unsigned int i;
2091 uint32_t *p = (uint32_t*)pThis->silence;
2092 for (i = 0; i < sizeof(pThis->silence) / 4; i++) /** @todo r=andy Assumes 16-bit samples, stereo. */
2093 *p++ = pThis->last_samp;
2094 }
2095 else
2096 RT_ZERO(pThis->silence);
2097
2098 pThis->bup_flag |= BUP_SET;
2099 }
2100
2101 while (cbElapsed)
2102 {
2103 uint32_t cbToWrite = RT_MIN(cbElapsed, (uint32_t)sizeof(pThis->silence));
2104 uint32_t cbWrittenToStream;
2105
2106 int rc2 = AudioMixerSinkWrite(pThis->pSinkOut, AUDMIXOP_COPY,
2107 pThis->silence, cbToWrite, &cbWrittenToStream);
2108 if (RT_SUCCESS(rc2))
2109 {
2110 if (cbWrittenToStream < cbToWrite) /* Lagging behind? */
2111 LogFlowFunc(("Warning: Only written %RU32 / %RU32 bytes, expect lags\n", cbWrittenToStream, cbToWrite));
2112 }
2113
2114 /* Always report all data as being written;
2115 * backends who were not able to catch up have to deal with it themselves. */
2116 Assert(cbElapsed >= cbToWrite);
2117 cbElapsed -= cbToWrite;
2118 }
2119}
2120#endif /* Unused */
2121
2122#ifndef VBOX_WITH_AUDIO_AC97_CALLBACKS
2123/**
2124 * Starts the internal audio device timer.
2125 *
2126 * @return IPRT status code.
2127 * @param pThis AC'97 state.
2128 */
2129static int ichac97TimerStart(PAC97STATE pThis)
2130{
2131 LogFlowFuncEnter();
2132
2133 DEVAC97_LOCK_BOTH_RETURN(pThis, VINF_IOM_R3_MMIO_WRITE);
2134
2135 AssertPtr(pThis->pTimer);
2136
2137 if (!pThis->fTimerActive)
2138 {
2139 LogRel2(("AC97: Starting transfers\n"));
2140
2141 pThis->fTimerActive = true;
2142
2143 /* Start transfers. */
2144 ichac97TimerMain(pThis);
2145 }
2146
2147 DEVAC97_UNLOCK_BOTH(pThis);
2148
2149 return VINF_SUCCESS;
2150}
2151
2152/**
2153 * Starts the internal audio device timer (if not started yet).
2154 *
2155 * @return IPRT status code.
2156 * @param pThis AC'97 state.
2157 */
2158static int ichac97TimerMaybeStart(PAC97STATE pThis)
2159{
2160 LogFlowFuncEnter();
2161
2162 if (!pThis->pTimer)
2163 return VERR_WRONG_ORDER;
2164
2165 pThis->cStreamsActive++;
2166
2167 /* Only start the timer at the first active stream. */
2168 if (pThis->cStreamsActive == 1)
2169 return ichac97TimerStart(pThis);
2170
2171 return VINF_SUCCESS;
2172}
2173
2174/**
2175 * Stops the internal audio device timer.
2176 *
2177 * @return IPRT status code.
2178 * @param pThis AC'97 state.
2179 */
2180static int ichac97TimerStop(PAC97STATE pThis)
2181{
2182 LogFlowFuncEnter();
2183
2184 if (!pThis->pTimer) /* Only can happen on device construction time, so no locking needed here. */
2185 return VINF_SUCCESS;
2186
2187 DEVAC97_LOCK_BOTH_RETURN(pThis, VINF_IOM_R3_MMIO_WRITE);
2188
2189 if (pThis->fTimerActive)
2190 {
2191 LogRel2(("AC97: Stopping transfers ...\n"));
2192
2193 pThis->fTimerActive = false;
2194
2195 /* Note: Do not stop the timer via TMTimerStop() here, as there still might
2196 * be queued audio data which needs to be handled (e.g. played back) first
2197 * before actually stopping the timer for good. */
2198 }
2199
2200 DEVAC97_UNLOCK_BOTH(pThis);
2201
2202 return VINF_SUCCESS;
2203}
2204
2205/**
2206 * Decreases the active AC'97 streams count by one and
2207 * then checks if the internal audio device timer can be
2208 * stopped.
2209 *
2210 * @return IPRT status code.
2211 * @param pThis AC'97 state.
2212 */
2213static int ichac97TimerMaybeStop(PAC97STATE pThis)
2214{
2215 LogFlowFuncEnter();
2216
2217 if (!pThis->pTimer)
2218 return VERR_WRONG_ORDER;
2219
2220 if (pThis->cStreamsActive) /* Function can be called mupltiple times. */
2221 {
2222 pThis->cStreamsActive--;
2223
2224 if (pThis->cStreamsActive == 0)
2225 return ichac97TimerStop(pThis);
2226 }
2227
2228 return VINF_SUCCESS;
2229}
2230
2231/**
2232 * Main routine for the device timer.
2233 *
2234 * @param pThis AC'97 state.
2235 */
2236static void ichac97TimerMain(PAC97STATE pThis)
2237{
2238 STAM_PROFILE_START(&pThis->StatTimer, a);
2239
2240 DEVAC97_LOCK_BOTH_RETURN_VOID(pThis);
2241
2242 uint64_t cTicksNow = TMTimerGet(pThis->pTimer);
2243
2244 /* Update current time timestamp. */
2245 pThis->uTimerTS = cTicksNow;
2246
2247 /* Flag indicating whether to kick the timer again for the next DMA transfer or sink processing. */
2248 bool fKickTimer = false;
2249
2250 ichac97DoTransfers(pThis);
2251
2252 /* Do we need to kick the timer again? */
2253 if ( AudioMixerSinkIsActive(ichac97IndexToSink(pThis, pThis->StreamLineIn.u8SD))
2254 || AudioMixerSinkIsActive(ichac97IndexToSink(pThis, pThis->StreamMicIn.u8SD))
2255 || AudioMixerSinkIsActive(ichac97IndexToSink(pThis, pThis->StreamOut.u8SD)))
2256 {
2257 fKickTimer = true;
2258 }
2259
2260 if ( ASMAtomicReadBool(&pThis->fTimerActive)
2261 || fKickTimer)
2262 {
2263 /* Kick the timer again. */
2264 uint64_t cTicks = pThis->cTimerTicks;
2265 /** @todo adjust cTicks down by now much cbOutMin represents. */
2266 TMTimerSet(pThis->pTimer, cTicksNow + cTicks);
2267 }
2268 else
2269 LogRel2(("AC97: Stopped transfers\n"));
2270
2271 DEVAC97_UNLOCK_BOTH(pThis);
2272
2273 STAM_PROFILE_STOP(&pThis->StatTimer, a);
2274}
2275
2276/**
2277 * Timer callback which handles the audio data transfers on a periodic basis.
2278 *
2279 * @param pDevIns Device instance.
2280 * @param pTimer Timer which was used when calling this.
2281 * @param pvUser User argument as PAC97STATE.
2282 */
2283static DECLCALLBACK(void) ichac97Timer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
2284{
2285 RT_NOREF(pDevIns, pTimer);
2286
2287 PAC97STATE pThis = (PAC97STATE)pvUser;
2288 Assert(pThis == PDMINS_2_DATA(pDevIns, PAC97STATE));
2289 AssertPtr(pThis);
2290
2291 ichac97TimerMain(pThis);
2292}
2293#endif /* !VBOX_WITH_AUDIO_AC97_CALLBACKS */
2294
2295/**
2296 * Main routine to perform the actual audio data transfers from the AC'97 streams
2297 * to the backend(s) and vice versa.
2298 *
2299 * @param pThis AC'97 state.
2300 */
2301static void ichac97DoTransfers(PAC97STATE pThis)
2302{
2303 AssertPtrReturnVoid(pThis);
2304
2305 ichac97StreamUpdate(pThis, &pThis->StreamLineIn, true /* fInTimer */);
2306 ichac97StreamUpdate(pThis, &pThis->StreamMicIn, true /* fInTimer */);
2307 ichac97StreamUpdate(pThis, &pThis->StreamOut, true /* fInTimer */);
2308}
2309
2310/**
2311 * Transfers data of an AC'97 stream according to its usage (input / output).
2312 *
2313 * For an SDO (output) stream this means reading DMA data from the device to
2314 * the AC'97 stream's internal FIFO buffer.
2315 *
2316 * For an SDI (input) stream this is reading audio data from the AC'97 stream's
2317 * internal FIFO buffer and writing it as DMA data to the device.
2318 *
2319 * @returns IPRT status code.
2320 * @param pThis AC'97 state.
2321 * @param pStream AC'97 stream to update.
2322 * @param cbToProcessMax Maximum of data (in bytes) to process.
2323 */
2324static int ichac97StreamTransfer(PAC97STATE pThis, PAC97STREAM pStream, uint32_t cbToProcessMax)
2325{
2326 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2327 AssertPtrReturn(pStream, VERR_INVALID_POINTER);
2328 AssertReturn(cbToProcessMax, VERR_INVALID_PARAMETER);
2329
2330 ichac97StreamLock(pStream);
2331
2332 PAC97BMREGS pRegs = &pStream->Regs;
2333
2334 if (pRegs->sr & AC97_SR_DCH) /* Controller halted? */
2335 {
2336 if (pRegs->cr & AC97_CR_RPBM) /* Bus master operation starts. */
2337 {
2338 switch (pStream->u8SD)
2339 {
2340 case AC97SOUNDSOURCE_PO_INDEX:
2341 /*ichac97WriteBUP(pThis, cbToProcess);*/
2342 break;
2343
2344 default:
2345 break;
2346 }
2347 }
2348
2349 ichac97StreamUnlock(pStream);
2350 return VINF_SUCCESS;
2351 }
2352
2353 /* BCIS flag still set? Skip iteration. */
2354 if (pRegs->sr & AC97_SR_BCIS)
2355 {
2356 Log3Func(("[SD%RU8] BCIS set\n", pStream->u8SD));
2357
2358 ichac97StreamUnlock(pStream);
2359 return VINF_SUCCESS;
2360 }
2361
2362 uint32_t cbLeft = RT_MIN((uint32_t)(pRegs->picb << 1), cbToProcessMax); /** @todo r=andy Assumes 16bit samples. */
2363 uint32_t cbProcessedTotal = 0;
2364
2365 PRTCIRCBUF pCircBuf = pStream->State.pCircBuf;
2366 AssertPtr(pCircBuf);
2367
2368 int rc = VINF_SUCCESS;
2369
2370 Log3Func(("[SD%RU8] cbToProcessMax=%RU32, cbLeft=%RU32\n", pStream->u8SD, cbToProcessMax, cbLeft));
2371
2372 while (cbLeft)
2373 {
2374 if (!pRegs->bd_valid)
2375 {
2376 Log3Func(("Invalid buffer descriptor, fetching next one ...\n"));
2377 ichac97StreamFetchBDLE(pThis, pStream);
2378 }
2379
2380 if (!pRegs->picb) /* Got a new buffer descriptor, that is, the position is 0? */
2381 {
2382 Log3Func(("Fresh buffer descriptor %RU8 is empty, addr=%#x, len=%#x, skipping\n",
2383 pRegs->civ, pRegs->bd.addr, pRegs->bd.ctl_len));
2384 if (pRegs->civ == pRegs->lvi)
2385 {
2386 pRegs->sr |= AC97_SR_DCH; /** @todo r=andy Also set CELV? */
2387 pThis->bup_flag = 0;
2388
2389 rc = VINF_EOF;
2390 break;
2391 }
2392
2393 pRegs->sr &= ~AC97_SR_CELV;
2394 pRegs->civ = pRegs->piv;
2395 pRegs->piv = (pRegs->piv + 1) % AC97_MAX_BDLE;
2396
2397 ichac97StreamFetchBDLE(pThis, pStream);
2398 continue;
2399 }
2400
2401 uint32_t cbChunk = RT_MIN((uint32_t)(pRegs->picb << 1), cbLeft); /** @todo r=andy Assumes 16bit samples. */
2402 Assert(cbChunk);
2403
2404 switch (pStream->u8SD)
2405 {
2406 case AC97SOUNDSOURCE_PO_INDEX: /* Output */
2407 {
2408 void *pvDst;
2409 size_t cbDst;
2410
2411 RTCircBufAcquireWriteBlock(pCircBuf, cbChunk, &pvDst, &cbDst);
2412
2413 if (cbDst)
2414 {
2415 int rc2 = PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), pRegs->bd.addr, (uint8_t *)pvDst, cbDst);
2416 AssertRC(rc2);
2417
2418#ifdef VBOX_AUDIO_DEBUG_DUMP_PCM_DATA
2419 RTFILE fh;
2420 RTFileOpen(&fh, VBOX_AUDIO_DEBUG_DUMP_PCM_DATA_PATH "ac97DMARead.pcm",
2421 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
2422 RTFileWrite(fh, pvDst, cbDst, NULL);
2423 RTFileClose(fh);
2424#endif
2425 }
2426
2427 RTCircBufReleaseWriteBlock(pCircBuf, cbDst);
2428
2429 cbChunk = (uint32_t)cbDst; /* Update the current chunk size to what really has been written. */
2430 break;
2431 }
2432
2433 case AC97SOUNDSOURCE_PI_INDEX: /* Input */
2434 case AC97SOUNDSOURCE_MC_INDEX: /* Input */
2435 {
2436 void *pvSrc;
2437 size_t cbSrc;
2438
2439 RTCircBufAcquireReadBlock(pCircBuf, cbChunk, &pvSrc, &cbSrc);
2440
2441 if (cbSrc)
2442 {
2443 int rc2 = PDMDevHlpPhysWrite(pThis->CTX_SUFF(pDevIns), pRegs->bd.addr, (uint8_t *)pvSrc, cbSrc);
2444 AssertRC(rc2);
2445
2446#ifdef VBOX_AUDIO_DEBUG_DUMP_PCM_DATA
2447 RTFILE fh;
2448 RTFileOpen(&fh, VBOX_AUDIO_DEBUG_DUMP_PCM_DATA_PATH "ac97DMAWrite.pcm",
2449 RTFILE_O_OPEN_CREATE | RTFILE_O_APPEND | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
2450 RTFileWrite(fh, pvSrc, cbSrc, NULL);
2451 RTFileClose(fh);
2452#endif
2453 }
2454
2455 RTCircBufReleaseReadBlock(pCircBuf, cbSrc);
2456
2457 cbChunk = (uint32_t)cbSrc; /* Update the current chunk size to what really has been read. */
2458 break;
2459 }
2460
2461 default:
2462 AssertMsgFailed(("Stream #%RU8 not supported\n", pStream->u8SD));
2463 rc = VERR_NOT_SUPPORTED;
2464 break;
2465 }
2466
2467 if (RT_FAILURE(rc))
2468 break;
2469
2470 if (cbChunk)
2471 {
2472 cbProcessedTotal += cbChunk;
2473 Assert(cbProcessedTotal <= cbToProcessMax);
2474 Assert(cbLeft >= cbChunk);
2475 cbLeft -= cbChunk;
2476 Assert((cbChunk & 1) == 0); /* Else the following shift won't work */
2477
2478 pRegs->picb -= (cbChunk >> 1); /** @todo r=andy Assumes 16bit samples. */
2479 pRegs->bd.addr += cbChunk;
2480 }
2481
2482 LogFlowFunc(("[SD%RU8] cbChunk=%RU32, cbLeft=%RU32, cbTotal=%RU32, rc=%Rrc\n",
2483 pStream->u8SD, cbChunk, cbLeft, cbProcessedTotal, rc));
2484
2485 if (!pRegs->picb)
2486 {
2487 uint32_t new_sr = pRegs->sr & ~AC97_SR_CELV;
2488
2489 if (pRegs->bd.ctl_len & AC97_BD_IOC)
2490 {
2491 new_sr |= AC97_SR_BCIS;
2492 }
2493
2494 if (pRegs->civ == pRegs->lvi)
2495 {
2496 /* Did we run out of data? */
2497 LogFunc(("Underrun CIV (%RU8) == LVI (%RU8)\n", pRegs->civ, pRegs->lvi));
2498
2499 new_sr |= AC97_SR_LVBCI | AC97_SR_DCH | AC97_SR_CELV;
2500 pThis->bup_flag = (pRegs->bd.ctl_len & AC97_BD_BUP) ? BUP_LAST : 0;
2501
2502 rc = VINF_EOF;
2503 }
2504 else
2505 {
2506 pRegs->civ = pRegs->piv;
2507 pRegs->piv = (pRegs->piv + 1) % AC97_MAX_BDLE;
2508 ichac97StreamFetchBDLE(pThis, pStream);
2509 }
2510
2511 ichac97StreamUpdateSR(pThis, pStream, new_sr);
2512 }
2513
2514 if (/* All data processed? */
2515 rc == VINF_EOF
2516 /* ... or an error occurred? */
2517 || RT_FAILURE(rc))
2518 {
2519 break;
2520 }
2521 }
2522
2523 ichac97StreamUnlock(pStream);
2524
2525 LogFlowFuncLeaveRC(rc);
2526 return rc;
2527}
2528
2529/**
2530 * Port I/O Handler for IN operations.
2531 *
2532 * @returns VINF_SUCCESS or VINF_EM_*.
2533 * @returns VERR_IOM_IOPORT_UNUSED if the port is really unused and a ~0 value should be returned.
2534 *
2535 * @param pDevIns The device instance.
2536 * @param pvUser User argument.
2537 * @param uPort Port number used for the IN operation.
2538 * @param pu32Val Where to store the result. This is always a 32-bit
2539 * variable regardless of what @a cbVal might say.
2540 * @param cbVal Number of bytes read.
2541 * @remarks Caller enters the device critical section.
2542 */
2543static DECLCALLBACK(int) ichac97IOPortNABMRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT uPort,
2544 uint32_t *pu32Val, unsigned cbVal)
2545{
2546 RT_NOREF(pDevIns);
2547
2548 PAC97STATE pThis = (PAC97STATE)pvUser;
2549
2550 DEVAC97_LOCK(pThis);
2551
2552 /* Get the index of the NABMBAR port. */
2553 const uint32_t uPortIdx = uPort - pThis->IOPortBase[1];
2554
2555 PAC97STREAM pStream = ichac97GetStreamFromIdx(pThis, AC97_PORT2IDX(uPortIdx));
2556 PAC97BMREGS pRegs = NULL;
2557
2558 if (pStream) /* Can be NULL, depending on the index (port). */
2559 pRegs = &pStream->Regs;
2560
2561 int rc = VINF_SUCCESS;
2562
2563 switch (cbVal)
2564 {
2565 case 1:
2566 {
2567 switch (uPortIdx)
2568 {
2569 case AC97_CAS:
2570 /* Codec Access Semaphore Register */
2571 Log3Func(("CAS %d\n", pThis->cas));
2572 *pu32Val = pThis->cas;
2573 pThis->cas = 1;
2574 break;
2575 case PI_CIV:
2576 case PO_CIV:
2577 case MC_CIV:
2578 /* Current Index Value Register */
2579 *pu32Val = pRegs->civ;
2580 Log3Func(("CIV[%d] -> %#x\n", AC97_PORT2IDX(uPortIdx), *pu32Val));
2581 break;
2582 case PI_LVI:
2583 case PO_LVI:
2584 case MC_LVI:
2585 /* Last Valid Index Register */
2586 *pu32Val = pRegs->lvi;
2587 Log3Func(("LVI[%d] -> %#x\n", AC97_PORT2IDX(uPortIdx), *pu32Val));
2588 break;
2589 case PI_PIV:
2590 case PO_PIV:
2591 case MC_PIV:
2592 /* Prefetched Index Value Register */
2593 *pu32Val = pRegs->piv;
2594 Log3Func(("PIV[%d] -> %#x\n", AC97_PORT2IDX(uPortIdx), *pu32Val));
2595 break;
2596 case PI_CR:
2597 case PO_CR:
2598 case MC_CR:
2599 /* Control Register */
2600 *pu32Val = pRegs->cr;
2601 Log3Func(("CR[%d] -> %#x\n", AC97_PORT2IDX(uPortIdx), *pu32Val));
2602 break;
2603 case PI_SR:
2604 case PO_SR:
2605 case MC_SR:
2606 /* Status Register (lower part) */
2607 *pu32Val = RT_LO_U8(pRegs->sr);
2608 Log3Func(("SRb[%d] -> %#x\n", AC97_PORT2IDX(uPortIdx), *pu32Val));
2609 break;
2610 default:
2611 *pu32Val = UINT32_MAX;
2612 LogFunc(("U nabm readb %#x -> %#x\n", uPort, *pu32Val));
2613 break;
2614 }
2615 break;
2616 }
2617
2618 case 2:
2619 {
2620 switch (uPortIdx)
2621 {
2622 case PI_SR:
2623 case PO_SR:
2624 case MC_SR:
2625 /* Status Register */
2626 *pu32Val = pRegs->sr;
2627 Log3Func(("SR[%d] -> %#x\n", AC97_PORT2IDX(uPortIdx), *pu32Val));
2628 break;
2629 case PI_PICB:
2630 case PO_PICB:
2631 case MC_PICB:
2632 /* Position in Current Buffer */
2633 *pu32Val = pRegs->picb;
2634 Log3Func(("PICB[%d] -> %#x\n", AC97_PORT2IDX(uPortIdx), *pu32Val));
2635 break;
2636 default:
2637 *pu32Val = UINT32_MAX;
2638 LogFunc(("U nabm readw %#x -> %#x\n", uPort, *pu32Val));
2639 break;
2640 }
2641 break;
2642 }
2643
2644 case 4:
2645 {
2646 switch (uPortIdx)
2647 {
2648 case PI_BDBAR:
2649 case PO_BDBAR:
2650 case MC_BDBAR:
2651 /* Buffer Descriptor Base Address Register */
2652 *pu32Val = pRegs->bdbar;
2653 Log3Func(("BMADDR[%d] -> %#x\n", AC97_PORT2IDX(uPortIdx), *pu32Val));
2654 break;
2655 case PI_CIV:
2656 case PO_CIV:
2657 case MC_CIV:
2658 /* 32-bit access: Current Index Value Register +
2659 * Last Valid Index Register +
2660 * Status Register */
2661 *pu32Val = pRegs->civ | (pRegs->lvi << 8) | (pRegs->sr << 16); /** @todo r=andy Use RT_MAKE_U32_FROM_U8. */
2662 Log3Func(("CIV LVI SR[%d] -> %#x, %#x, %#x\n",
2663 AC97_PORT2IDX(uPortIdx), pRegs->civ, pRegs->lvi, pRegs->sr));
2664 break;
2665 case PI_PICB:
2666 case PO_PICB:
2667 case MC_PICB:
2668 /* 32-bit access: Position in Current Buffer Register +
2669 * Prefetched Index Value Register +
2670 * Control Register */
2671 *pu32Val = pRegs->picb | (pRegs->piv << 16) | (pRegs->cr << 24); /** @todo r=andy Use RT_MAKE_U32_FROM_U8. */
2672 Log3Func(("PICB PIV CR[%d] -> %#x %#x %#x %#x\n",
2673 AC97_PORT2IDX(uPortIdx), *pu32Val, pRegs->picb, pRegs->piv, pRegs->cr));
2674 break;
2675 case AC97_GLOB_CNT:
2676 /* Global Control */
2677 *pu32Val = pThis->glob_cnt;
2678 Log3Func(("glob_cnt -> %#x\n", *pu32Val));
2679 break;
2680 case AC97_GLOB_STA:
2681 /* Global Status */
2682 *pu32Val = pThis->glob_sta | AC97_GS_S0CR;
2683 Log3Func(("glob_sta -> %#x\n", *pu32Val));
2684 break;
2685 default:
2686 *pu32Val = UINT32_MAX;
2687 LogFunc(("U nabm readl %#x -> %#x\n", uPort, *pu32Val));
2688 break;
2689 }
2690 break;
2691 }
2692
2693 default:
2694 {
2695 AssertFailed();
2696 rc = VERR_IOM_IOPORT_UNUSED;
2697 }
2698 }
2699
2700 DEVAC97_UNLOCK(pThis);
2701
2702 return rc;
2703}
2704
2705/**
2706 * Port I/O Handler for OUT operations.
2707 *
2708 * @returns VINF_SUCCESS or VINF_EM_*.
2709 *
2710 * @param pDevIns The device instance.
2711 * @param pvUser User argument.
2712 * @param uPort Port number used for the OUT operation.
2713 * @param u32Val The value to output.
2714 * @param cbVal The value size in bytes.
2715 * @remarks Caller enters the device critical section.
2716 */
2717static DECLCALLBACK(int) ichac97IOPortNABMWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT uPort,
2718 uint32_t u32Val, unsigned cbVal)
2719{
2720 RT_NOREF(pDevIns);
2721
2722 PAC97STATE pThis = (PAC97STATE)pvUser;
2723
2724 DEVAC97_LOCK_BOTH_RETURN(pThis, VINF_IOM_R3_IOPORT_WRITE);
2725
2726 /* Get the index of the NABMBAR register. */
2727 const uint32_t uPortIdx = uPort - pThis->IOPortBase[1];
2728
2729 PAC97STREAM pStream = ichac97GetStreamFromIdx(pThis, AC97_PORT2IDX(uPortIdx));
2730 PAC97BMREGS pRegs = NULL;
2731
2732 if (pStream) /* Can be NULL, depending on the index (port). */
2733 pRegs = &pStream->Regs;
2734
2735 switch (cbVal)
2736 {
2737 case 1:
2738 {
2739 switch (uPortIdx)
2740 {
2741 /*
2742 * Last Valid Index.
2743 */
2744 case PI_LVI:
2745 case PO_LVI:
2746 case MC_LVI:
2747 {
2748 if ( (pRegs->cr & AC97_CR_RPBM)
2749 && (pRegs->sr & AC97_SR_DCH))
2750 {
2751 pRegs->sr &= ~(AC97_SR_DCH | AC97_SR_CELV);
2752 pRegs->civ = pRegs->piv;
2753 pRegs->piv = (pRegs->piv + 1) % AC97_MAX_BDLE;
2754
2755 ichac97StreamFetchBDLE(pThis, pStream);
2756 }
2757 pRegs->lvi = u32Val % AC97_MAX_BDLE;
2758 Log3Func(("[SD%RU8] LVI <- %#x\n", pStream->u8SD, u32Val));
2759 break;
2760 }
2761
2762 /*
2763 * Control Registers.
2764 */
2765 case PI_CR:
2766 case PO_CR:
2767 case MC_CR:
2768 {
2769 Log3Func(("[SD%RU8] CR <- %#x (cr %#x)\n", pStream->u8SD, u32Val, pRegs->cr));
2770
2771 if (u32Val & AC97_CR_RR) /* Busmaster reset. */
2772 {
2773 Log3Func(("[SD%RU8] Reset\n", pStream->u8SD));
2774
2775 /* Make sure that Run/Pause Bus Master bit (RPBM) is cleared (0). */
2776 Assert((pRegs->cr & AC97_CR_RPBM) == 0);
2777
2778 ichac97StreamEnable(pThis, pStream, false /* fEnable */);
2779 ichac97StreamReset(pThis, pStream);
2780
2781 ichac97StreamUpdateSR(pThis, pStream, AC97_SR_DCH); /** @todo Do we need to do that? */
2782 }
2783 else
2784 {
2785 pRegs->cr = u32Val & AC97_CR_VALID_MASK;
2786
2787 if (!(pRegs->cr & AC97_CR_RPBM))
2788 {
2789 Log3Func(("[SD%RU8] Disable\n", pStream->u8SD));
2790
2791 ichac97StreamEnable(pThis, pStream, false /* fEnable */);
2792
2793 pRegs->sr |= AC97_SR_DCH;
2794 }
2795 else
2796 {
2797 Log3Func(("[SD%RU8] Enable\n", pStream->u8SD));
2798
2799 pRegs->civ = pRegs->piv;
2800 pRegs->piv = (pRegs->piv + 1) % AC97_MAX_BDLE;
2801
2802 pRegs->sr &= ~AC97_SR_DCH;
2803
2804 /* Fetch the initial BDLE descriptor. */
2805 ichac97StreamFetchBDLE(pThis, pStream);
2806
2807 ichac97StreamEnable(pThis, pStream, true /* fEnable */);
2808 }
2809 }
2810 break;
2811 }
2812
2813 /*
2814 * Status Registers.
2815 */
2816 case PI_SR:
2817 case PO_SR:
2818 case MC_SR:
2819 {
2820 pRegs->sr |= u32Val & ~(AC97_SR_RO_MASK | AC97_SR_WCLEAR_MASK);
2821 ichac97StreamUpdateSR(pThis, pStream, pRegs->sr & ~(u32Val & AC97_SR_WCLEAR_MASK));
2822 Log3Func(("[SD%RU8] SR <- %#x (sr %#x)\n", pStream->u8SD, u32Val, pRegs->sr));
2823 break;
2824 }
2825
2826 default:
2827 LogRel2(("AC97: Warning: Unimplemented NABMWrite (%u byte) portIdx=%#x <- %#x\n", cbVal, uPortIdx, u32Val));
2828 break;
2829 }
2830 break;
2831 }
2832
2833 case 2:
2834 {
2835 switch (uPortIdx)
2836 {
2837 case PI_SR:
2838 case PO_SR:
2839 case MC_SR:
2840 /* Status Register */
2841 pRegs->sr |= u32Val & ~(AC97_SR_RO_MASK | AC97_SR_WCLEAR_MASK);
2842 ichac97StreamUpdateSR(pThis, pStream, pRegs->sr & ~(u32Val & AC97_SR_WCLEAR_MASK));
2843 Log3Func(("[SD%RU8] SR <- %#x (sr %#x)\n", pStream->u8SD, u32Val, pRegs->sr));
2844 break;
2845 default:
2846 LogRel2(("AC97: Warning: Unimplemented NABMWrite (%u byte) portIdx=%#x <- %#x\n", cbVal, uPortIdx, u32Val));
2847 break;
2848 }
2849 break;
2850 }
2851
2852 case 4:
2853 {
2854 switch (uPortIdx)
2855 {
2856 case PI_BDBAR:
2857 case PO_BDBAR:
2858 case MC_BDBAR:
2859 /* Buffer Descriptor list Base Address Register */
2860 pRegs->bdbar = u32Val & ~3;
2861 Log3Func(("[SD%RU8] BDBAR <- %#x (bdbar %#x)\n", AC97_PORT2IDX(uPortIdx), u32Val, pRegs->bdbar));
2862 break;
2863 case AC97_GLOB_CNT:
2864 /* Global Control */
2865 if (u32Val & AC97_GC_WR)
2866 ichac97WarmReset(pThis);
2867 if (u32Val & AC97_GC_CR)
2868 ichac97ColdReset(pThis);
2869 if (!(u32Val & (AC97_GC_WR | AC97_GC_CR)))
2870 pThis->glob_cnt = u32Val & AC97_GC_VALID_MASK;
2871 Log3Func(("glob_cnt <- %#x (glob_cnt %#x)\n", u32Val, pThis->glob_cnt));
2872 break;
2873 case AC97_GLOB_STA:
2874 /* Global Status */
2875 pThis->glob_sta &= ~(u32Val & AC97_GS_WCLEAR_MASK);
2876 pThis->glob_sta |= (u32Val & ~(AC97_GS_WCLEAR_MASK | AC97_GS_RO_MASK)) & AC97_GS_VALID_MASK;
2877 Log3Func(("glob_sta <- %#x (glob_sta %#x)\n", u32Val, pThis->glob_sta));
2878 break;
2879 default:
2880 LogRel2(("AC97: Warning: Unimplemented NABMWrite (%u byte) portIdx=%#x <- %#x\n", cbVal, uPortIdx, u32Val));
2881 break;
2882 }
2883 break;
2884 }
2885
2886 default:
2887 LogRel2(("AC97: Warning: Unimplemented NABMWrite (%u byte) portIdx=%#x <- %#x\n", cbVal, uPortIdx, u32Val));
2888 break;
2889 }
2890
2891 DEVAC97_UNLOCK_BOTH(pThis);
2892
2893 return VINF_SUCCESS;
2894}
2895
2896/**
2897 * Port I/O Handler for IN operations.
2898 *
2899 * @returns VINF_SUCCESS or VINF_EM_*.
2900 * @returns VERR_IOM_IOPORT_UNUSED if the port is really unused and a ~0 value should be returned.
2901 *
2902 * @param pDevIns The device instance.
2903 * @param pvUser User argument.
2904 * @param uPort Port number used for the IN operation.
2905 * @param pu32Val Where to store the result. This is always a 32-bit
2906 * variable regardless of what @a cbVal might say.
2907 * @param cbVal Number of bytes read.
2908 * @remarks Caller enters the device critical section.
2909 */
2910static DECLCALLBACK(int) ichac97IOPortNAMRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT uPort, uint32_t *pu32Val, unsigned cbVal)
2911{
2912 RT_NOREF(pDevIns);
2913 PAC97STATE pThis = (PAC97STATE)pvUser;
2914
2915 DEVAC97_LOCK(pThis);
2916
2917 int rc = VINF_SUCCESS;
2918
2919 uint32_t index = uPort - pThis->IOPortBase[0];
2920
2921 switch (cbVal)
2922 {
2923 case 1:
2924 {
2925 LogRel2(("AC97: Warning: Unimplemented read (%u byte) port=%#x, idx=%RU32\n", cbVal, uPort, index));
2926 pThis->cas = 0;
2927 *pu32Val = UINT32_MAX;
2928 break;
2929 }
2930
2931 case 2:
2932 {
2933 *pu32Val = UINT32_MAX;
2934 pThis->cas = 0;
2935
2936 switch (index)
2937 {
2938 default:
2939 *pu32Val = ichac97MixerGet(pThis, index);
2940 break;
2941 }
2942 break;
2943 }
2944
2945 case 4:
2946 {
2947 LogRel2(("AC97: Warning: Unimplemented read (%u byte) port=%#x, idx=%RU32\n", cbVal, uPort, index));
2948 pThis->cas = 0;
2949 *pu32Val = UINT32_MAX;
2950 break;
2951 }
2952
2953 default:
2954 {
2955 AssertFailed();
2956 rc = VERR_IOM_IOPORT_UNUSED;
2957 }
2958 }
2959
2960 DEVAC97_UNLOCK(pThis);
2961
2962 return rc;
2963}
2964
2965/**
2966 * Port I/O Handler for OUT operations.
2967 *
2968 * @returns VINF_SUCCESS or VINF_EM_*.
2969 *
2970 * @param pDevIns The device instance.
2971 * @param pvUser User argument.
2972 * @param uPort Port number used for the OUT operation.
2973 * @param u32Val The value to output.
2974 * @param cbVal The value size in bytes.
2975 * @remarks Caller enters the device critical section.
2976 */
2977static DECLCALLBACK(int) ichac97IOPortNAMWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT uPort,
2978 uint32_t u32Val, unsigned cbVal)
2979{
2980 RT_NOREF(pDevIns);
2981 PAC97STATE pThis = (PAC97STATE)pvUser;
2982
2983 DEVAC97_LOCK_BOTH_RETURN(pThis, VINF_IOM_R3_IOPORT_WRITE);
2984
2985 uint32_t uPortIdx = uPort - pThis->IOPortBase[0];
2986
2987 switch (cbVal)
2988 {
2989 case 1:
2990 {
2991 LogRel2(("AC97: Warning: Unimplemented NAMWrite (%u byte) port=%#x, idx=0x%x <- %#x\n", cbVal, uPort, uPortIdx, u32Val));
2992 pThis->cas = 0;
2993 break;
2994 }
2995
2996 case 2:
2997 {
2998 pThis->cas = 0;
2999 switch (uPortIdx)
3000 {
3001 case AC97_Reset:
3002 ichac97Reset(pThis->CTX_SUFF(pDevIns));
3003 break;
3004 case AC97_Powerdown_Ctrl_Stat:
3005 u32Val &= ~0xf;
3006 u32Val |= ichac97MixerGet(pThis, uPortIdx) & 0xf;
3007 ichac97MixerSet(pThis, uPortIdx, u32Val);
3008 break;
3009 case AC97_Master_Volume_Mute:
3010 if (pThis->uCodecModel == AC97_CODEC_AD1980)
3011 {
3012 if (ichac97MixerGet(pThis, AC97_AD_Misc) & AC97_AD_MISC_LOSEL)
3013 break; /* Register controls surround (rear), do nothing. */
3014 }
3015 ichac97MixerSetVolume(pThis, uPortIdx, PDMAUDIOMIXERCTL_VOLUME_MASTER, u32Val);
3016 break;
3017 case AC97_Headphone_Volume_Mute:
3018 if (pThis->uCodecModel == AC97_CODEC_AD1980)
3019 {
3020 if (ichac97MixerGet(pThis, AC97_AD_Misc) & AC97_AD_MISC_HPSEL)
3021 {
3022 /* Register controls PCM (front) outputs. */
3023 ichac97MixerSetVolume(pThis, uPortIdx, PDMAUDIOMIXERCTL_VOLUME_MASTER, u32Val);
3024 }
3025 }
3026 break;
3027 case AC97_PCM_Out_Volume_Mute:
3028 ichac97MixerSetVolume(pThis, uPortIdx, PDMAUDIOMIXERCTL_FRONT, u32Val);
3029 break;
3030 case AC97_Line_In_Volume_Mute:
3031 ichac97MixerSetVolume(pThis, uPortIdx, PDMAUDIOMIXERCTL_LINE_IN, u32Val);
3032 break;
3033 case AC97_Record_Select:
3034 ichac97MixerRecordSelect(pThis, u32Val);
3035 break;
3036 case AC97_Record_Gain_Mute:
3037 /* Newer Ubuntu guests rely on that when controlling gain and muting
3038 * the recording (capturing) levels. */
3039 ichac97MixerSetVolume(pThis, uPortIdx, PDMAUDIOMIXERCTL_LINE_IN, u32Val);
3040 break;
3041 case AC97_Record_Gain_Mic_Mute:
3042 /* Ditto; see note above. */
3043 ichac97MixerSetVolume(pThis, uPortIdx, PDMAUDIOMIXERCTL_MIC_IN, u32Val);
3044 break;
3045 case AC97_Vendor_ID1:
3046 case AC97_Vendor_ID2:
3047 LogFunc(("Attempt to write vendor ID to %#x\n", u32Val));
3048 break;
3049 case AC97_Extended_Audio_ID:
3050 LogFunc(("Attempt to write extended audio ID to %#x\n", u32Val));
3051 break;
3052 case AC97_Extended_Audio_Ctrl_Stat:
3053 if (!(u32Val & AC97_EACS_VRA))
3054 {
3055 ichac97MixerSet(pThis, AC97_PCM_Front_DAC_Rate, 48000);
3056 ichac97StreamReOpen(pThis, &pThis->StreamOut);
3057
3058 ichac97MixerSet(pThis, AC97_PCM_LR_ADC_Rate, 48000);
3059 ichac97StreamReOpen(pThis, &pThis->StreamLineIn);
3060 }
3061 else
3062 LogRel2(("AC97: Variable rate audio (VRA) is not supported\n"));
3063
3064 if (!(u32Val & AC97_EACS_VRM))
3065 {
3066 ichac97MixerSet(pThis, AC97_MIC_ADC_Rate, 48000);
3067 ichac97StreamReOpen(pThis, &pThis->StreamMicIn);
3068 }
3069 else
3070 LogRel2(("AC97: Variable rate microphone audio (VRM) is not supported\n"));
3071
3072 LogFunc(("Setting extended audio control to %#x\n", u32Val));
3073 ichac97MixerSet(pThis, AC97_Extended_Audio_Ctrl_Stat, u32Val);
3074 break;
3075 case AC97_PCM_Front_DAC_Rate:
3076 if (ichac97MixerGet(pThis, AC97_Extended_Audio_Ctrl_Stat) & AC97_EACS_VRA)
3077 {
3078 ichac97MixerSet(pThis, uPortIdx, u32Val);
3079 LogFunc(("Set front DAC rate to %RU32\n", u32Val));
3080 ichac97StreamReOpen(pThis, &pThis->StreamOut);
3081 }
3082 else
3083 AssertMsgFailed(("Attempt to set front DAC rate to %RU32, but VRA is not set\n", u32Val));
3084 break;
3085 case AC97_MIC_ADC_Rate:
3086 if (ichac97MixerGet(pThis, AC97_Extended_Audio_Ctrl_Stat) & AC97_EACS_VRM)
3087 {
3088 ichac97MixerSet(pThis, uPortIdx, u32Val);
3089 LogFunc(("Set MIC ADC rate to %RU32\n", u32Val));
3090 ichac97StreamReOpen(pThis, &pThis->StreamMicIn);
3091 }
3092 else
3093 AssertMsgFailed(("Attempt to set MIC ADC rate to %RU32, but VRM is not set\n", u32Val));
3094 break;
3095 case AC97_PCM_LR_ADC_Rate:
3096 if (ichac97MixerGet(pThis, AC97_Extended_Audio_Ctrl_Stat) & AC97_EACS_VRA)
3097 {
3098 ichac97MixerSet(pThis, uPortIdx, u32Val);
3099 LogFunc(("Set front LR ADC rate to %RU32\n", u32Val));
3100 ichac97StreamReOpen(pThis, &pThis->StreamLineIn);
3101 }
3102 else
3103 AssertMsgFailed(("Attempt to set LR ADC rate to %RU32, but VRA is not set\n", u32Val));
3104 break;
3105 default:
3106 LogRel2(("AC97: Warning: Unimplemented NAMWrite (%u byte) port=%#x, idx=0x%x <- %#x\n", cbVal, uPort, uPortIdx, u32Val));
3107 ichac97MixerSet(pThis, uPortIdx, u32Val);
3108 break;
3109 }
3110 break;
3111 }
3112
3113 case 4:
3114 {
3115 LogRel2(("AC97: Warning: Unimplemented NAMWrite (%u byte) port=%#x, idx=0x%x <- %#x\n", cbVal, uPort, uPortIdx, u32Val));
3116 pThis->cas = 0;
3117 break;
3118 }
3119
3120 default:
3121 AssertMsgFailed(("Unhandled NAMWrite port=%#x, cbVal=%u u32Val=%#x\n", uPort, cbVal, u32Val));
3122 break;
3123 }
3124
3125 DEVAC97_UNLOCK_BOTH(pThis);
3126
3127 return VINF_SUCCESS;
3128}
3129
3130
3131/**
3132 * @callback_method_impl{FNPCIIOREGIONMAP}
3133 */
3134static DECLCALLBACK(int) ichac97IOPortMap(PPDMDEVINS pDevIns, PPDMPCIDEV pPciDev, uint32_t iRegion,
3135 RTGCPHYS GCPhysAddress, RTGCPHYS cb, PCIADDRESSSPACE enmType)
3136{
3137 RT_NOREF(cb, enmType);
3138
3139 Assert(enmType == PCI_ADDRESS_SPACE_IO);
3140 Assert(cb >= 0x20);
3141
3142 if (iRegion > 1) /* We support 2 regions max. at the moment. */
3143 return VERR_INVALID_PARAMETER;
3144
3145 PAC97STATE pThis = RT_FROM_MEMBER(pPciDev, AC97STATE, PciDev);
3146 RTIOPORT Port = (RTIOPORT)GCPhysAddress;
3147
3148 int rc;
3149 if (iRegion == 0)
3150 rc = PDMDevHlpIOPortRegister(pDevIns, Port, 256, pThis,
3151 ichac97IOPortNAMWrite, ichac97IOPortNAMRead,
3152 NULL, NULL, "ICHAC97 NAM");
3153 else
3154 rc = PDMDevHlpIOPortRegister(pDevIns, Port, 64, pThis,
3155 ichac97IOPortNABMWrite, ichac97IOPortNABMRead,
3156 NULL, NULL, "ICHAC97 NABM");
3157 if (RT_FAILURE(rc))
3158 return rc;
3159
3160 pThis->IOPortBase[iRegion] = Port;
3161 return VINF_SUCCESS;
3162}
3163
3164#ifdef IN_RING3
3165/**
3166 * Saves (serializes) an AC'97 stream using SSM.
3167 *
3168 * @returns IPRT status code.
3169 * @param pDevIns Device instance.
3170 * @param pSSM Saved state manager (SSM) handle to use.
3171 * @param pStream AC'97 stream to save.
3172 */
3173static int ichac97SaveStream(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, PAC97STREAM pStream)
3174{
3175 RT_NOREF(pDevIns);
3176 PAC97BMREGS pRegs = &pStream->Regs;
3177
3178 SSMR3PutU32(pSSM, pRegs->bdbar);
3179 SSMR3PutU8( pSSM, pRegs->civ);
3180 SSMR3PutU8( pSSM, pRegs->lvi);
3181 SSMR3PutU16(pSSM, pRegs->sr);
3182 SSMR3PutU16(pSSM, pRegs->picb);
3183 SSMR3PutU8( pSSM, pRegs->piv);
3184 SSMR3PutU8( pSSM, pRegs->cr);
3185 SSMR3PutS32(pSSM, pRegs->bd_valid);
3186 SSMR3PutU32(pSSM, pRegs->bd.addr);
3187 SSMR3PutU32(pSSM, pRegs->bd.ctl_len);
3188
3189 return VINF_SUCCESS;
3190}
3191
3192/**
3193 * @callback_method_impl{FNSSMDEVSAVEEXEC}
3194 */
3195static DECLCALLBACK(int) ichac97SaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
3196{
3197 PAC97STATE pThis = PDMINS_2_DATA(pDevIns, PAC97STATE);
3198
3199 LogFlowFuncEnter();
3200
3201 SSMR3PutU32(pSSM, pThis->glob_cnt);
3202 SSMR3PutU32(pSSM, pThis->glob_sta);
3203 SSMR3PutU32(pSSM, pThis->cas);
3204
3205 /** @todo r=andy For the next saved state version, add unique stream identifiers and a stream count. */
3206 /* Note: The order the streams are saved here is critical, so don't touch. */
3207 int rc2 = ichac97SaveStream(pDevIns, pSSM, &pThis->StreamLineIn);
3208 AssertRC(rc2);
3209 rc2 = ichac97SaveStream(pDevIns, pSSM, &pThis->StreamOut);
3210 AssertRC(rc2);
3211 rc2 = ichac97SaveStream(pDevIns, pSSM, &pThis->StreamMicIn);
3212 AssertRC(rc2);
3213
3214 SSMR3PutMem(pSSM, pThis->mixer_data, sizeof(pThis->mixer_data));
3215
3216 uint8_t active[AC97SOUNDSOURCE_LAST_INDEX];
3217
3218 active[AC97SOUNDSOURCE_PI_INDEX] = ichac97StreamIsEnabled(pThis, &pThis->StreamLineIn) ? 1 : 0;
3219 active[AC97SOUNDSOURCE_PO_INDEX] = ichac97StreamIsEnabled(pThis, &pThis->StreamOut) ? 1 : 0;
3220 active[AC97SOUNDSOURCE_MC_INDEX] = ichac97StreamIsEnabled(pThis, &pThis->StreamMicIn) ? 1 : 0;
3221
3222 SSMR3PutMem(pSSM, active, sizeof(active));
3223
3224 LogFlowFuncLeaveRC(VINF_SUCCESS);
3225 return VINF_SUCCESS;
3226}
3227
3228/**
3229 * Loads an AC'97 stream from SSM.
3230 *
3231 * @returns IPRT status code.
3232 * @param pDevIns Device instance.
3233 * @param pSSM Saved state manager (SSM) handle to use.
3234 * @param pStream AC'97 stream to load.
3235 */
3236static int ichac97LoadStream(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, PAC97STREAM pStream)
3237{
3238 RT_NOREF(pDevIns);
3239 PAC97BMREGS pRegs = &pStream->Regs;
3240
3241 SSMR3GetU32(pSSM, &pRegs->bdbar);
3242 SSMR3GetU8( pSSM, &pRegs->civ);
3243 SSMR3GetU8( pSSM, &pRegs->lvi);
3244 SSMR3GetU16(pSSM, &pRegs->sr);
3245 SSMR3GetU16(pSSM, &pRegs->picb);
3246 SSMR3GetU8( pSSM, &pRegs->piv);
3247 SSMR3GetU8( pSSM, &pRegs->cr);
3248 SSMR3GetS32(pSSM, &pRegs->bd_valid);
3249 SSMR3GetU32(pSSM, &pRegs->bd.addr);
3250 SSMR3GetU32(pSSM, &pRegs->bd.ctl_len);
3251
3252 return VINF_SUCCESS;
3253}
3254
3255/**
3256 * @callback_method_impl{FNSSMDEVLOADEXEC}
3257 */
3258static DECLCALLBACK(int) ichac97LoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
3259{
3260 PAC97STATE pThis = PDMINS_2_DATA(pDevIns, PAC97STATE);
3261
3262 LogRel2(("ichac97LoadExec: uVersion=%RU32, uPass=0x%x\n", uVersion, uPass));
3263
3264 AssertMsgReturn (uVersion == AC97_SSM_VERSION, ("%RU32\n", uVersion), VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION);
3265 Assert(uPass == SSM_PASS_FINAL); NOREF(uPass);
3266
3267 SSMR3GetU32(pSSM, &pThis->glob_cnt);
3268 SSMR3GetU32(pSSM, &pThis->glob_sta);
3269 SSMR3GetU32(pSSM, &pThis->cas);
3270
3271 /** @todo r=andy For the next saved state version, add unique stream identifiers and a stream count. */
3272 /* Note: The order the streams are loaded here is critical, so don't touch. */
3273 int rc2 = ichac97LoadStream(pDevIns, pSSM, &pThis->StreamLineIn);
3274 AssertRC(rc2);
3275 rc2 = ichac97LoadStream(pDevIns, pSSM, &pThis->StreamOut);
3276 AssertRC(rc2);
3277 rc2 = ichac97LoadStream(pDevIns, pSSM, &pThis->StreamMicIn);
3278 AssertRC(rc2);
3279
3280 SSMR3GetMem(pSSM, pThis->mixer_data, sizeof(pThis->mixer_data));
3281
3282 /** @todo r=andy Stream IDs are hardcoded to certain streams. */
3283 uint8_t uaStrmsActive[AC97SOUNDSOURCE_LAST_INDEX];
3284 SSMR3GetMem(pSSM, uaStrmsActive, sizeof(uaStrmsActive));
3285
3286 ichac97MixerRecordSelect(pThis, ichac97MixerGet(pThis, AC97_Record_Select));
3287# define V_(a, b) ichac97MixerSetVolume(pThis, a, b, ichac97MixerGet(pThis, a))
3288 V_(AC97_Master_Volume_Mute, PDMAUDIOMIXERCTL_VOLUME_MASTER);
3289 V_(AC97_PCM_Out_Volume_Mute, PDMAUDIOMIXERCTL_FRONT);
3290 V_(AC97_Line_In_Volume_Mute, PDMAUDIOMIXERCTL_LINE_IN);
3291 V_(AC97_Mic_Volume_Mute, PDMAUDIOMIXERCTL_MIC_IN);
3292# undef V_
3293 if (pThis->uCodecModel == AC97_CODEC_AD1980)
3294 if (ichac97MixerGet(pThis, AC97_AD_Misc) & AC97_AD_MISC_HPSEL)
3295 ichac97MixerSetVolume(pThis, AC97_Headphone_Volume_Mute, PDMAUDIOMIXERCTL_VOLUME_MASTER,
3296 ichac97MixerGet(pThis, AC97_Headphone_Volume_Mute));
3297
3298 /** @todo r=andy Stream IDs are hardcoded to certain streams. */
3299 rc2 = ichac97StreamEnable(pThis, &pThis->StreamLineIn, RT_BOOL(uaStrmsActive[AC97SOUNDSOURCE_PI_INDEX]));
3300 if (RT_SUCCESS(rc2))
3301 rc2 = ichac97StreamEnable(pThis, &pThis->StreamMicIn, RT_BOOL(uaStrmsActive[AC97SOUNDSOURCE_MC_INDEX]));
3302 if (RT_SUCCESS(rc2))
3303 rc2 = ichac97StreamEnable(pThis, &pThis->StreamOut, RT_BOOL(uaStrmsActive[AC97SOUNDSOURCE_PO_INDEX]));
3304
3305 pThis->bup_flag = 0;
3306 pThis->last_samp = 0;
3307
3308 return VINF_SUCCESS;
3309}
3310
3311
3312/**
3313 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
3314 */
3315static DECLCALLBACK(void *) ichac97QueryInterface(struct PDMIBASE *pInterface, const char *pszIID)
3316{
3317 PAC97STATE pThis = RT_FROM_MEMBER(pInterface, AC97STATE, IBase);
3318 Assert(&pThis->IBase == pInterface);
3319
3320 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->IBase);
3321 return NULL;
3322}
3323
3324
3325/**
3326 * Powers off the device.
3327 *
3328 * @param pDevIns Device instance to power off.
3329 */
3330static DECLCALLBACK(void) ichac97PowerOff(PPDMDEVINS pDevIns)
3331{
3332 PAC97STATE pThis = PDMINS_2_DATA(pDevIns, PAC97STATE);
3333
3334 LogRel2(("AC97: Powering off ...\n"));
3335
3336 /* Note: Involves mixer stream / sink destruction, so also do this here
3337 * instead of in ichac97Destruct(). */
3338 ichac97StreamsDestroy(pThis);
3339
3340 /**
3341 * Note: Destroy the mixer while powering off and *not* in ichac97Destruct,
3342 * giving the mixer the chance to release any references held to
3343 * PDM audio streams it maintains.
3344 */
3345 if (pThis->pMixer)
3346 {
3347 AudioMixerDestroy(pThis->pMixer);
3348 pThis->pMixer = NULL;
3349 }
3350}
3351
3352
3353/**
3354 * @interface_method_impl{PDMDEVREG,pfnReset}
3355 *
3356 * @remarks The original sources didn't install a reset handler, but it seems to
3357 * make sense to me so we'll do it.
3358 */
3359static DECLCALLBACK(void) ichac97Reset(PPDMDEVINS pDevIns)
3360{
3361 PAC97STATE pThis = PDMINS_2_DATA(pDevIns, PAC97STATE);
3362
3363 LogRel(("AC97: Reset\n"));
3364
3365 /*
3366 * Reset the mixer too. The Windows XP driver seems to rely on
3367 * this. At least it wants to read the vendor id before it resets
3368 * the codec manually.
3369 */
3370 ichac97MixerReset(pThis);
3371
3372 /*
3373 * Reset all streams.
3374 */
3375 ichac97StreamEnable(pThis, &pThis->StreamLineIn, false /* fEnable */);
3376 ichac97StreamReset(pThis, &pThis->StreamLineIn);
3377
3378 ichac97StreamEnable(pThis, &pThis->StreamMicIn, false /* fEnable */);
3379 ichac97StreamReset(pThis, &pThis->StreamMicIn);
3380
3381 ichac97StreamEnable(pThis, &pThis->StreamOut, false /* fEnable */);
3382 ichac97StreamReset(pThis, &pThis->StreamOut);
3383
3384 /*
3385 * Reset mixer sinks.
3386 *
3387 * Do the reset here instead of in ichac97StreamReset();
3388 * the mixer sink(s) might still have data to be processed when an audio stream gets reset.
3389 */
3390 AudioMixerSinkReset(pThis->pSinkLineIn);
3391 AudioMixerSinkReset(pThis->pSinkMicIn);
3392 AudioMixerSinkReset(pThis->pSinkOut);
3393
3394# ifndef VBOX_WITH_AUDIO_AC97_CALLBACKS
3395 /*
3396 * Stop the timer, if any.
3397 */
3398 ichac97TimerStop(pThis);
3399
3400 pThis->cStreamsActive = 0;
3401# endif
3402}
3403
3404
3405/**
3406 * @interface_method_impl{PDMDEVREG,pfnDestruct}
3407 */
3408static DECLCALLBACK(int) ichac97Destruct(PPDMDEVINS pDevIns)
3409{
3410 PAC97STATE pThis = PDMINS_2_DATA(pDevIns, PAC97STATE);
3411
3412 LogFlowFuncEnter();
3413
3414 PAC97DRIVER pDrv, pDrvNext;
3415 RTListForEachSafe(&pThis->lstDrv, pDrv, pDrvNext, AC97DRIVER, Node)
3416 {
3417 RTListNodeRemove(&pDrv->Node);
3418 RTMemFree(pDrv);
3419 }
3420
3421 /* Sanity. */
3422 Assert(RTListIsEmpty(&pThis->lstDrv));
3423
3424 return VINF_SUCCESS;
3425}
3426
3427
3428/**
3429 * Attach command, internal version.
3430 *
3431 * This is called to let the device attach to a driver for a specified LUN
3432 * during runtime. This is not called during VM construction, the device
3433 * constructor has to attach to all the available drivers.
3434 *
3435 * @returns VBox status code.
3436 * @param pThis AC'97 state.
3437 * @param uLUN The logical unit which is being detached.
3438 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
3439 * @param ppDrv Attached driver instance on success. Optional.
3440 */
3441static int ichac97AttachInternal(PAC97STATE pThis, unsigned uLUN, uint32_t fFlags, PAC97DRIVER *ppDrv)
3442{
3443 RT_NOREF(fFlags);
3444
3445 /*
3446 * Attach driver.
3447 */
3448 char *pszDesc;
3449 if (RTStrAPrintf(&pszDesc, "Audio driver port (AC'97) for LUN #%u", uLUN) <= 0)
3450 AssertLogRelFailedReturn(VERR_NO_MEMORY);
3451
3452 PPDMIBASE pDrvBase;
3453 int rc = PDMDevHlpDriverAttach(pThis->pDevInsR3, uLUN,
3454 &pThis->IBase, &pDrvBase, pszDesc);
3455 if (RT_SUCCESS(rc))
3456 {
3457 PAC97DRIVER pDrv = (PAC97DRIVER)RTMemAllocZ(sizeof(AC97DRIVER));
3458 if (pDrv)
3459 {
3460 pDrv->pDrvBase = pDrvBase;
3461 pDrv->pConnector = PDMIBASE_QUERY_INTERFACE(pDrvBase, PDMIAUDIOCONNECTOR);
3462 AssertMsg(pDrv->pConnector != NULL, ("Configuration error: LUN #%u has no host audio interface, rc=%Rrc\n", uLUN, rc));
3463 pDrv->pAC97State = pThis;
3464 pDrv->uLUN = uLUN;
3465
3466 /*
3467 * For now we always set the driver at LUN 0 as our primary
3468 * host backend. This might change in the future.
3469 */
3470 if (pDrv->uLUN == 0)
3471 pDrv->fFlags |= PDMAUDIODRVFLAGS_PRIMARY;
3472
3473 LogFunc(("LUN#%RU8: pCon=%p, drvFlags=0x%x\n", uLUN, pDrv->pConnector, pDrv->fFlags));
3474
3475 /* Attach to driver list if not attached yet. */
3476 if (!pDrv->fAttached)
3477 {
3478 RTListAppend(&pThis->lstDrv, &pDrv->Node);
3479 pDrv->fAttached = true;
3480 }
3481
3482 if (ppDrv)
3483 *ppDrv = pDrv;
3484 }
3485 else
3486 rc = VERR_NO_MEMORY;
3487 }
3488 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
3489 LogFunc(("No attached driver for LUN #%u\n", uLUN));
3490
3491 if (RT_FAILURE(rc))
3492 {
3493 /* Only free this string on failure;
3494 * must remain valid for the live of the driver instance. */
3495 RTStrFree(pszDesc);
3496 }
3497
3498 LogFunc(("uLUN=%u, fFlags=0x%x, rc=%Rrc\n", uLUN, fFlags, rc));
3499 return rc;
3500}
3501
3502/**
3503 * Detach command, internal version.
3504 *
3505 * This is called to let the device detach from a driver for a specified LUN
3506 * during runtime.
3507 *
3508 * @returns VBox status code.
3509 * @param pThis AC'97 state.
3510 * @param pDrv Driver to detach device from.
3511 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
3512 */
3513static int ichac97DetachInternal(PAC97STATE pThis, PAC97DRIVER pDrv, uint32_t fFlags)
3514{
3515 RT_NOREF(fFlags);
3516
3517 AudioMixerSinkRemoveStream(pThis->pSinkMicIn, pDrv->MicIn.pMixStrm);
3518 AudioMixerStreamDestroy(pDrv->MicIn.pMixStrm);
3519 pDrv->MicIn.pMixStrm = NULL;
3520
3521 AudioMixerSinkRemoveStream(pThis->pSinkLineIn, pDrv->LineIn.pMixStrm);
3522 AudioMixerStreamDestroy(pDrv->LineIn.pMixStrm);
3523 pDrv->LineIn.pMixStrm = NULL;
3524
3525 AudioMixerSinkRemoveStream(pThis->pSinkOut, pDrv->Out.pMixStrm);
3526 AudioMixerStreamDestroy(pDrv->Out.pMixStrm);
3527 pDrv->Out.pMixStrm = NULL;
3528
3529 RTListNodeRemove(&pDrv->Node);
3530
3531 LogFunc(("uLUN=%u, fFlags=0x%x\n", pDrv->uLUN, fFlags));
3532 return VINF_SUCCESS;
3533}
3534
3535/**
3536 * @interface_method_impl{PDMDEVREG,pfnAttach}
3537 */
3538static DECLCALLBACK(int) ichac97Attach(PPDMDEVINS pDevIns, unsigned uLUN, uint32_t fFlags)
3539{
3540 PAC97STATE pThis = PDMINS_2_DATA(pDevIns, PAC97STATE);
3541
3542 LogFunc(("uLUN=%u, fFlags=0x%x\n", uLUN, fFlags));
3543
3544 DEVAC97_LOCK(pThis);
3545
3546 PAC97DRIVER pDrv;
3547 int rc2 = ichac97AttachInternal(pThis, uLUN, fFlags, &pDrv);
3548 if (RT_SUCCESS(rc2))
3549 {
3550 if (DrvAudioHlpStreamCfgIsValid(&pThis->StreamLineIn.State.Cfg))
3551 ichac97MixerAddDrvStream(pThis, pThis->pSinkLineIn, &pThis->StreamLineIn.State.Cfg, pDrv);
3552
3553 if (DrvAudioHlpStreamCfgIsValid(&pThis->StreamMicIn.State.Cfg))
3554 ichac97MixerAddDrvStream(pThis, pThis->pSinkMicIn, &pThis->StreamMicIn.State.Cfg, pDrv);
3555
3556 if (DrvAudioHlpStreamCfgIsValid(&pThis->StreamOut.State.Cfg))
3557 ichac97MixerAddDrvStream(pThis, pThis->pSinkOut, &pThis->StreamOut.State.Cfg, pDrv);
3558 }
3559
3560 DEVAC97_UNLOCK(pThis);
3561
3562 return VINF_SUCCESS;
3563}
3564
3565/**
3566 * @interface_method_impl{PDMDEVREG,pfnDetach}
3567 */
3568static DECLCALLBACK(void) ichac97Detach(PPDMDEVINS pDevIns, unsigned uLUN, uint32_t fFlags)
3569{
3570 PAC97STATE pThis = PDMINS_2_DATA(pDevIns, PAC97STATE);
3571
3572 LogFunc(("uLUN=%u, fFlags=0x%x\n", uLUN, fFlags));
3573
3574 DEVAC97_LOCK(pThis);
3575
3576 PAC97DRIVER pDrv, pDrvNext;
3577 RTListForEachSafe(&pThis->lstDrv, pDrv, pDrvNext, AC97DRIVER, Node)
3578 {
3579 if (pDrv->uLUN == uLUN)
3580 {
3581 int rc2 = ichac97DetachInternal(pThis, pDrv, fFlags);
3582 if (RT_SUCCESS(rc2))
3583 {
3584 RTMemFree(pDrv);
3585 pDrv = NULL;
3586 }
3587
3588 break;
3589 }
3590 }
3591
3592 DEVAC97_UNLOCK(pThis);
3593}
3594
3595/**
3596 * Re-attaches (replaces) a driver with a new driver.
3597 *
3598 * @returns VBox status code.
3599 * @param pThis Device instance.
3600 * @param pDrv Driver instance used for attaching to.
3601 * If NULL is specified, a new driver will be created and appended
3602 * to the driver list.
3603 * @param uLUN The logical unit which is being re-detached.
3604 * @param pszDriver New driver name to attach.
3605 */
3606static int ichac97ReattachInternal(PAC97STATE pThis, PAC97DRIVER pDrv, uint8_t uLUN, const char *pszDriver)
3607{
3608 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
3609 AssertPtrReturn(pszDriver, VERR_INVALID_POINTER);
3610
3611 int rc;
3612
3613 if (pDrv)
3614 {
3615 rc = ichac97DetachInternal(pThis, pDrv, 0 /* fFlags */);
3616 if (RT_SUCCESS(rc))
3617 rc = PDMDevHlpDriverDetach(pThis->pDevInsR3, PDMIBASE_2_PDMDRV(pDrv->pDrvBase), 0 /* fFlags */);
3618
3619 if (RT_FAILURE(rc))
3620 return rc;
3621
3622 pDrv = NULL;
3623 }
3624
3625 PVM pVM = PDMDevHlpGetVM(pThis->pDevInsR3);
3626 PCFGMNODE pRoot = CFGMR3GetRoot(pVM);
3627 PCFGMNODE pDev0 = CFGMR3GetChild(pRoot, "Devices/ichac97/0/");
3628
3629 /* Remove LUN branch. */
3630 CFGMR3RemoveNode(CFGMR3GetChildF(pDev0, "LUN#%u/", uLUN));
3631
3632#define RC_CHECK() if (RT_FAILURE(rc)) { AssertReleaseRC(rc); break; }
3633
3634 do
3635 {
3636 PCFGMNODE pLunL0;
3637 rc = CFGMR3InsertNodeF(pDev0, &pLunL0, "LUN#%u/", uLUN); RC_CHECK();
3638 rc = CFGMR3InsertString(pLunL0, "Driver", "AUDIO"); RC_CHECK();
3639 rc = CFGMR3InsertNode(pLunL0, "Config/", NULL); RC_CHECK();
3640
3641 PCFGMNODE pLunL1, pLunL2;
3642 rc = CFGMR3InsertNode (pLunL0, "AttachedDriver/", &pLunL1); RC_CHECK();
3643 rc = CFGMR3InsertNode (pLunL1, "Config/", &pLunL2); RC_CHECK();
3644 rc = CFGMR3InsertString(pLunL1, "Driver", pszDriver); RC_CHECK();
3645
3646 rc = CFGMR3InsertString(pLunL2, "AudioDriver", pszDriver); RC_CHECK();
3647
3648 } while (0);
3649
3650 if (RT_SUCCESS(rc))
3651 rc = ichac97AttachInternal(pThis, uLUN, 0 /* fFlags */, NULL /* ppDrv */);
3652
3653 LogFunc(("pThis=%p, uLUN=%u, pszDriver=%s, rc=%Rrc\n", pThis, uLUN, pszDriver, rc));
3654
3655#undef RC_CHECK
3656
3657 return rc;
3658}
3659
3660/**
3661 * @interface_method_impl{PDMDEVREG,pfnConstruct}
3662 */
3663static DECLCALLBACK(int) ichac97Construct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
3664{
3665 RT_NOREF(iInstance);
3666 PAC97STATE pThis = PDMINS_2_DATA(pDevIns, PAC97STATE);
3667
3668 /* NB: This must be done *before* any possible failure (and running the destructor). */
3669 RTListInit(&pThis->lstDrv);
3670
3671 Assert(iInstance == 0);
3672 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
3673
3674 /*
3675 * Validations.
3676 */
3677 if (!CFGMR3AreValuesValid(pCfg,
3678 "Codec\0"
3679 "TimerHz\0"))
3680 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
3681 N_("Invalid configuration for the AC'97 device"));
3682
3683 /*
3684 * Read config data.
3685 */
3686 char szCodec[20];
3687 int rc = CFGMR3QueryStringDef(pCfg, "Codec", &szCodec[0], sizeof(szCodec), "STAC9700");
3688 if (RT_FAILURE(rc))
3689 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
3690 N_("AC'97 configuration error: Querying \"Codec\" as string failed"));
3691
3692#ifndef VBOX_WITH_AUDIO_AC97_CALLBACKS
3693 uint16_t uTimerHz;
3694 rc = CFGMR3QueryU16Def(pCfg, "TimerHz", &uTimerHz, AC97_TIMER_HZ /* Default value, if not set. */);
3695 if (RT_FAILURE(rc))
3696 return PDMDEV_SET_ERROR(pDevIns, rc,
3697 N_("AC'97 configuration error: failed to read Hertz (Hz) rate as unsigned integer"));
3698#endif
3699
3700 /*
3701 * The AD1980 codec (with corresponding PCI subsystem vendor ID) is whitelisted
3702 * in the Linux kernel; Linux makes no attempt to measure the data rate and assumes
3703 * 48 kHz rate, which is exactly what we need. Same goes for AD1981B.
3704 */
3705 if (!strcmp(szCodec, "STAC9700"))
3706 pThis->uCodecModel = AC97_CODEC_STAC9700;
3707 else if (!strcmp(szCodec, "AD1980"))
3708 pThis->uCodecModel = AC97_CODEC_AD1980;
3709 else if (!strcmp(szCodec, "AD1981B"))
3710 pThis->uCodecModel = AC97_CODEC_AD1981B;
3711 else
3712 {
3713 return PDMDevHlpVMSetError(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES, RT_SRC_POS,
3714 N_("AC'97 configuration error: The \"Codec\" value \"%s\" is unsupported"),
3715 szCodec);
3716 }
3717
3718 /*
3719 * Use an own critical section for the device instead of the default
3720 * one provided by PDM. This allows fine-grained locking in combination
3721 * with TM when timer-specific stuff is being called in e.g. the MMIO handlers.
3722 */
3723 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->CritSect, RT_SRC_POS, "AC'97");
3724 AssertRCReturn(rc, rc);
3725
3726 rc = PDMDevHlpSetDeviceCritSect(pDevIns, PDMDevHlpCritSectGetNop(pDevIns));
3727 AssertRCReturn(rc, rc);
3728
3729 /*
3730 * Initialize data (most of it anyway).
3731 */
3732 pThis->pDevInsR3 = pDevIns;
3733 /* IBase */
3734 pThis->IBase.pfnQueryInterface = ichac97QueryInterface;
3735
3736 /* PCI Device */
3737 PCIDevSetVendorId (&pThis->PciDev, 0x8086); /* 00 ro - intel. */ Assert(pThis->PciDev.abConfig[0x00] == 0x86); Assert(pThis->PciDev.abConfig[0x01] == 0x80);
3738 PCIDevSetDeviceId (&pThis->PciDev, 0x2415); /* 02 ro - 82801 / 82801aa(?). */ Assert(pThis->PciDev.abConfig[0x02] == 0x15); Assert(pThis->PciDev.abConfig[0x03] == 0x24);
3739 PCIDevSetCommand (&pThis->PciDev, 0x0000); /* 04 rw,ro - pcicmd. */ Assert(pThis->PciDev.abConfig[0x04] == 0x00); Assert(pThis->PciDev.abConfig[0x05] == 0x00);
3740 PCIDevSetStatus (&pThis->PciDev, VBOX_PCI_STATUS_DEVSEL_MEDIUM | VBOX_PCI_STATUS_FAST_BACK); /* 06 rwc?,ro? - pcists. */ Assert(pThis->PciDev.abConfig[0x06] == 0x80); Assert(pThis->PciDev.abConfig[0x07] == 0x02);
3741 PCIDevSetRevisionId (&pThis->PciDev, 0x01); /* 08 ro - rid. */ Assert(pThis->PciDev.abConfig[0x08] == 0x01);
3742 PCIDevSetClassProg (&pThis->PciDev, 0x00); /* 09 ro - pi. */ Assert(pThis->PciDev.abConfig[0x09] == 0x00);
3743 PCIDevSetClassSub (&pThis->PciDev, 0x01); /* 0a ro - scc; 01 == Audio. */ Assert(pThis->PciDev.abConfig[0x0a] == 0x01);
3744 PCIDevSetClassBase (&pThis->PciDev, 0x04); /* 0b ro - bcc; 04 == multimedia. */ Assert(pThis->PciDev.abConfig[0x0b] == 0x04);
3745 PCIDevSetHeaderType (&pThis->PciDev, 0x00); /* 0e ro - headtyp. */ Assert(pThis->PciDev.abConfig[0x0e] == 0x00);
3746 PCIDevSetBaseAddress (&pThis->PciDev, 0, /* 10 rw - nambar - native audio mixer base. */
3747 true /* fIoSpace */, false /* fPrefetchable */, false /* f64Bit */, 0x00000000); Assert(pThis->PciDev.abConfig[0x10] == 0x01); Assert(pThis->PciDev.abConfig[0x11] == 0x00); Assert(pThis->PciDev.abConfig[0x12] == 0x00); Assert(pThis->PciDev.abConfig[0x13] == 0x00);
3748 PCIDevSetBaseAddress (&pThis->PciDev, 1, /* 14 rw - nabmbar - native audio bus mastering. */
3749 true /* fIoSpace */, false /* fPrefetchable */, false /* f64Bit */, 0x00000000); Assert(pThis->PciDev.abConfig[0x14] == 0x01); Assert(pThis->PciDev.abConfig[0x15] == 0x00); Assert(pThis->PciDev.abConfig[0x16] == 0x00); Assert(pThis->PciDev.abConfig[0x17] == 0x00);
3750 PCIDevSetInterruptLine (&pThis->PciDev, 0x00); /* 3c rw. */ Assert(pThis->PciDev.abConfig[0x3c] == 0x00);
3751 PCIDevSetInterruptPin (&pThis->PciDev, 0x01); /* 3d ro - INTA#. */ Assert(pThis->PciDev.abConfig[0x3d] == 0x01);
3752
3753 if (pThis->uCodecModel == AC97_CODEC_AD1980)
3754 {
3755 PCIDevSetSubSystemVendorId(&pThis->PciDev, 0x1028); /* 2c ro - Dell.) */
3756 PCIDevSetSubSystemId (&pThis->PciDev, 0x0177); /* 2e ro. */
3757 }
3758 else if (pThis->uCodecModel == AC97_CODEC_AD1981B)
3759 {
3760 PCIDevSetSubSystemVendorId(&pThis->PciDev, 0x1028); /* 2c ro - Dell.) */
3761 PCIDevSetSubSystemId (&pThis->PciDev, 0x01ad); /* 2e ro. */
3762 }
3763 else
3764 {
3765 PCIDevSetSubSystemVendorId(&pThis->PciDev, 0x8086); /* 2c ro - Intel.) */
3766 PCIDevSetSubSystemId (&pThis->PciDev, 0x0000); /* 2e ro. */
3767 }
3768
3769 /*
3770 * Register the PCI device, it's I/O regions, the timer and the
3771 * saved state item.
3772 */
3773 rc = PDMDevHlpPCIRegister(pDevIns, &pThis->PciDev);
3774 if (RT_FAILURE(rc))
3775 return rc;
3776
3777 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 0, 256, PCI_ADDRESS_SPACE_IO, ichac97IOPortMap);
3778 if (RT_FAILURE(rc))
3779 return rc;
3780
3781 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 1, 64, PCI_ADDRESS_SPACE_IO, ichac97IOPortMap);
3782 if (RT_FAILURE(rc))
3783 return rc;
3784
3785 rc = PDMDevHlpSSMRegister(pDevIns, AC97_SSM_VERSION, sizeof(*pThis), ichac97SaveExec, ichac97LoadExec);
3786 if (RT_FAILURE(rc))
3787 return rc;
3788
3789#ifdef VBOX_WITH_AUDIO_AC97_ASYNC_IO
3790 LogRel(("AC97: Asynchronous I/O enabled\n"));
3791#endif
3792
3793 /*
3794 * Attach driver.
3795 */
3796 uint8_t uLUN;
3797 for (uLUN = 0; uLUN < UINT8_MAX; ++uLUN)
3798 {
3799 LogFunc(("Trying to attach driver for LUN #%RU8 ...\n", uLUN));
3800 rc = ichac97AttachInternal(pThis, uLUN, 0 /* fFlags */, NULL /* ppDrv */);
3801 if (RT_FAILURE(rc))
3802 {
3803 if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
3804 rc = VINF_SUCCESS;
3805 else if (rc == VERR_AUDIO_BACKEND_INIT_FAILED)
3806 {
3807 ichac97ReattachInternal(pThis, NULL /* pDrv */, uLUN, "NullAudio");
3808 PDMDevHlpVMSetRuntimeError(pDevIns, 0 /*fFlags*/, "HostAudioNotResponding",
3809 N_("Host audio backend initialization has failed. Selecting the NULL audio backend "
3810 "with the consequence that no sound is audible"));
3811 /* Attaching to the NULL audio backend will never fail. */
3812 rc = VINF_SUCCESS;
3813 }
3814 break;
3815 }
3816 }
3817
3818 LogFunc(("cLUNs=%RU8, rc=%Rrc\n", uLUN, rc));
3819
3820 if (RT_SUCCESS(rc))
3821 {
3822 rc = AudioMixerCreate("AC'97 Mixer", 0 /* uFlags */, &pThis->pMixer);
3823 if (RT_SUCCESS(rc))
3824 {
3825 rc = AudioMixerCreateSink(pThis->pMixer, "[Recording] Line In", AUDMIXSINKDIR_INPUT, &pThis->pSinkLineIn);
3826 AssertRC(rc);
3827 rc = AudioMixerCreateSink(pThis->pMixer, "[Recording] Microphone In", AUDMIXSINKDIR_INPUT, &pThis->pSinkMicIn);
3828 AssertRC(rc);
3829 rc = AudioMixerCreateSink(pThis->pMixer, "[Playback] PCM Output", AUDMIXSINKDIR_OUTPUT, &pThis->pSinkOut);
3830 AssertRC(rc);
3831 }
3832 }
3833
3834 if (RT_SUCCESS(rc))
3835 {
3836 /*
3837 * Create all hardware streams.
3838 */
3839 rc = ichac97StreamCreate(pThis, &pThis->StreamLineIn, AC97SOUNDSOURCE_PI_INDEX);
3840 if (RT_SUCCESS(rc))
3841 {
3842 rc = ichac97StreamCreate(pThis, &pThis->StreamMicIn, AC97SOUNDSOURCE_MC_INDEX);
3843 if (RT_SUCCESS(rc))
3844 rc = ichac97StreamCreate(pThis, &pThis->StreamOut, AC97SOUNDSOURCE_PO_INDEX);
3845 }
3846
3847#ifdef VBOX_WITH_AUDIO_AC97_ONETIME_INIT
3848 PAC97DRIVER pDrv;
3849 RTListForEach(&pThis->lstDrv, pDrv, AC97DRIVER, Node)
3850 {
3851 /*
3852 * Only primary drivers are critical for the VM to run. Everything else
3853 * might not worth showing an own error message box in the GUI.
3854 */
3855 if (!(pDrv->fFlags & PDMAUDIODRVFLAGS_PRIMARY))
3856 continue;
3857
3858 PPDMIAUDIOCONNECTOR pCon = pDrv->pConnector;
3859 AssertPtr(pCon);
3860
3861 bool fValidLineIn = AudioMixerStreamIsValid(pDrv->LineIn.pMixStrm);
3862 bool fValidMicIn = AudioMixerStreamIsValid(pDrv->MicIn.pMixStrm);
3863 bool fValidOut = AudioMixerStreamIsValid(pDrv->Out.pMixStrm);
3864
3865 if ( !fValidLineIn
3866 && !fValidMicIn
3867 && !fValidOut)
3868 {
3869 LogRel(("AC97: Falling back to NULL backend (no sound audible)\n"));
3870
3871 ichac97Reset(pDevIns);
3872 ichac97ReattachInternal(pThis, pDrv, pDrv->uLUN, "NullAudio");
3873
3874 PDMDevHlpVMSetRuntimeError(pDevIns, 0 /*fFlags*/, "HostAudioNotResponding",
3875 N_("No audio devices could be opened. Selecting the NULL audio backend "
3876 "with the consequence that no sound is audible"));
3877 }
3878 else
3879 {
3880 bool fWarn = false;
3881
3882 PDMAUDIOBACKENDCFG backendCfg;
3883 int rc2 = pCon->pfnGetConfig(pCon, &backendCfg);
3884 if (RT_SUCCESS(rc2))
3885 {
3886 if (backendCfg.cMaxStreamsIn)
3887 {
3888 /* If the audio backend supports two or more input streams at once,
3889 * warn if one of our two inputs (microphone-in and line-in) failed to initialize. */
3890 if (backendCfg.cMaxStreamsIn >= 2)
3891 fWarn = !fValidLineIn || !fValidMicIn;
3892 /* If the audio backend only supports one input stream at once (e.g. pure ALSA, and
3893 * *not* ALSA via PulseAudio plugin!), only warn if both of our inputs failed to initialize.
3894 * One of the two simply is not in use then. */
3895 else if (backendCfg.cMaxStreamsIn == 1)
3896 fWarn = !fValidLineIn && !fValidMicIn;
3897 /* Don't warn if our backend is not able of supporting any input streams at all. */
3898 }
3899
3900 if ( !fWarn
3901 && backendCfg.cMaxStreamsOut)
3902 {
3903 fWarn = !fValidOut;
3904 }
3905 }
3906 else
3907 {
3908 LogRel(("AC97: Unable to retrieve audio backend configuration for LUN #%RU8, rc=%Rrc\n", pDrv->uLUN, rc2));
3909 fWarn = true;
3910 }
3911
3912 if (fWarn)
3913 {
3914 char szMissingStreams[255] = "";
3915 size_t len = 0;
3916 if (!fValidLineIn)
3917 {
3918 LogRel(("AC97: WARNING: Unable to open PCM line input for LUN #%RU8!\n", pDrv->uLUN));
3919 len = RTStrPrintf(szMissingStreams, sizeof(szMissingStreams), "PCM Input");
3920 }
3921 if (!fValidMicIn)
3922 {
3923 LogRel(("AC97: WARNING: Unable to open PCM microphone input for LUN #%RU8!\n", pDrv->uLUN));
3924 len += RTStrPrintf(szMissingStreams + len,
3925 sizeof(szMissingStreams) - len, len ? ", PCM Microphone" : "PCM Microphone");
3926 }
3927 if (!fValidOut)
3928 {
3929 LogRel(("AC97: WARNING: Unable to open PCM output for LUN #%RU8!\n", pDrv->uLUN));
3930 len += RTStrPrintf(szMissingStreams + len,
3931 sizeof(szMissingStreams) - len, len ? ", PCM Output" : "PCM Output");
3932 }
3933
3934 PDMDevHlpVMSetRuntimeError(pDevIns, 0 /*fFlags*/, "HostAudioNotResponding",
3935 N_("Some AC'97 audio streams (%s) could not be opened. Guest applications generating audio "
3936 "output or depending on audio input may hang. Make sure your host audio device "
3937 "is working properly. Check the logfile for error messages of the audio "
3938 "subsystem"), szMissingStreams);
3939 }
3940 }
3941 }
3942#endif /* VBOX_WITH_AUDIO_AC97_ONETIME_INIT */
3943 }
3944
3945 if (RT_SUCCESS(rc))
3946 ichac97Reset(pDevIns);
3947
3948#ifndef VBOX_WITH_AUDIO_AC97_CALLBACKS
3949 if (RT_SUCCESS(rc))
3950 {
3951 /* Create the emulation timer.
3952 *
3953 * Note: Use TMCLOCK_VIRTUAL_SYNC here, as the guest's AC'97 driver
3954 * relies on exact (virtual) DMA timing and uses DMA Position Buffers
3955 * instead of the LPIB registers.
3956 */
3957 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL_SYNC, ichac97Timer, pThis,
3958 TMTIMER_FLAGS_NO_CRIT_SECT, "AC'97 Timer", &pThis->pTimer);
3959 AssertRCReturn(rc, rc);
3960
3961 /* Use our own critcal section for the device timer.
3962 * That way we can control more fine-grained when to lock what. */
3963 rc = TMR3TimerSetCritSect(pThis->pTimer, &pThis->CritSect);
3964 AssertRCReturn(rc, rc);
3965
3966 pThis->cTimerTicks = TMTimerGetFreq(pThis->pTimer) / uTimerHz;
3967 pThis->uTimerTS = TMTimerGet(pThis->pTimer);
3968 LogFunc(("Timer ticks=%RU64 (%RU16 Hz)\n", pThis->cTimerTicks, uTimerHz));
3969 }
3970#else /* !VBOX_WITH_AUDIO_AC97_CALLBACKS */
3971 if (RT_SUCCESS(rc))
3972 {
3973 PAC97DRIVER pDrv;
3974 RTListForEach(&pThis->lstDrv, pDrv, AC97DRIVER, Node)
3975 {
3976 /* Only register primary driver.
3977 * The device emulation does the output multiplexing then. */
3978 if (!(pDrv->fFlags & PDMAUDIODRVFLAGS_PRIMARY))
3979 continue;
3980
3981 PDMAUDIOCBRECORD AudioCallbacks[2];
3982
3983 AC97CALLBACKCTX Ctx = { pThis, pDrv };
3984
3985 AudioCallbacks[0].enmType = PDMAUDIOCALLBACKTYPE_INPUT;
3986 AudioCallbacks[0].pfnCallback = ac97CallbackInput;
3987 AudioCallbacks[0].pvCtx = &Ctx;
3988 AudioCallbacks[0].cbCtx = sizeof(AC97CALLBACKCTX);
3989
3990 AudioCallbacks[1].enmType = PDMAUDIOCALLBACKTYPE_OUTPUT;
3991 AudioCallbacks[1].pfnCallback = ac97CallbackOutput;
3992 AudioCallbacks[1].pvCtx = &Ctx;
3993 AudioCallbacks[1].cbCtx = sizeof(AC97CALLBACKCTX);
3994
3995 rc = pDrv->pConnector->pfnRegisterCallbacks(pDrv->pConnector, AudioCallbacks, RT_ELEMENTS(AudioCallbacks));
3996 if (RT_FAILURE(rc))
3997 break;
3998 }
3999 }
4000#endif /* VBOX_WITH_AUDIO_AC97_CALLBACKS */
4001
4002#ifdef VBOX_WITH_STATISTICS
4003 if (RT_SUCCESS(rc))
4004 {
4005 /*
4006 * Register statistics.
4007 */
4008 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTimer, STAMTYPE_PROFILE, "/Devices/AC97/Timer", STAMUNIT_TICKS_PER_CALL, "Profiling ichac97Timer.");
4009 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIn, STAMTYPE_PROFILE, "/Devices/AC97/Input", STAMUNIT_TICKS_PER_CALL, "Profiling input.");
4010 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatOut, STAMTYPE_PROFILE, "/Devices/AC97/Output", STAMUNIT_TICKS_PER_CALL, "Profiling output.");
4011 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatBytesRead, STAMTYPE_COUNTER, "/Devices/AC97/BytesRead" , STAMUNIT_BYTES, "Bytes read from AC97 emulation.");
4012 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatBytesWritten, STAMTYPE_COUNTER, "/Devices/AC97/BytesWritten", STAMUNIT_BYTES, "Bytes written to AC97 emulation.");
4013 }
4014#endif
4015
4016#ifdef VBOX_AUDIO_DEBUG_DUMP_PCM_DATA
4017 RTFileDelete(VBOX_AUDIO_DEBUG_DUMP_PCM_DATA_PATH "ac97DMARead.pcm");
4018 RTFileDelete(VBOX_AUDIO_DEBUG_DUMP_PCM_DATA_PATH "ac97DMAWrite.pcm");
4019 RTFileDelete(VBOX_AUDIO_DEBUG_DUMP_PCM_DATA_PATH "ac97StreamRead.pcm");
4020 RTFileDelete(VBOX_AUDIO_DEBUG_DUMP_PCM_DATA_PATH "ac97StreamWrite.pcm");
4021#endif
4022
4023 LogFlowFuncLeaveRC(rc);
4024 return rc;
4025}
4026
4027/**
4028 * The device registration structure.
4029 */
4030const PDMDEVREG g_DeviceICHAC97 =
4031{
4032 /* u32Version */
4033 PDM_DEVREG_VERSION,
4034 /* szName */
4035 "ichac97",
4036 /* szRCMod */
4037 "",
4038 /* szR0Mod */
4039 "",
4040 /* pszDescription */
4041 "ICH AC'97 Audio Controller",
4042 /* fFlags */
4043 PDM_DEVREG_FLAGS_DEFAULT_BITS,
4044 /* fClass */
4045 PDM_DEVREG_CLASS_AUDIO,
4046 /* cMaxInstances */
4047 1,
4048 /* cbInstance */
4049 sizeof(AC97STATE),
4050 /* pfnConstruct */
4051 ichac97Construct,
4052 /* pfnDestruct */
4053 ichac97Destruct,
4054 /* pfnRelocate */
4055 NULL,
4056 /* pfnMemSetup */
4057 NULL,
4058 /* pfnPowerOn */
4059 NULL,
4060 /* pfnReset */
4061 ichac97Reset,
4062 /* pfnSuspend */
4063 NULL,
4064 /* pfnResume */
4065 NULL,
4066 /* pfnAttach */
4067 ichac97Attach,
4068 /* pfnDetach */
4069 ichac97Detach,
4070 /* pfnQueryInterface. */
4071 NULL,
4072 /* pfnInitComplete */
4073 NULL,
4074 /* pfnPowerOff */
4075 ichac97PowerOff,
4076 /* pfnSoftReset */
4077 NULL,
4078 /* u32VersionEnd */
4079 PDM_DEVREG_VERSION
4080};
4081
4082#endif /* !IN_RING3 */
4083#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
4084
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