1 | /* $Id: DevHDACommon.h 70964 2018-02-11 21:25:29Z vboxsync $ */
|
---|
2 | /** @file
|
---|
3 | * DevHDACommon.h - Shared HDA device defines / functions.
|
---|
4 | */
|
---|
5 |
|
---|
6 | /*
|
---|
7 | * Copyright (C) 2016-2018 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 | #ifndef DEV_HDA_COMMON_H
|
---|
19 | #define DEV_HDA_COMMON_H
|
---|
20 |
|
---|
21 | #include "AudioMixer.h"
|
---|
22 |
|
---|
23 | /** See 302349 p 6.2. */
|
---|
24 | typedef struct HDAREGDESC
|
---|
25 | {
|
---|
26 | /** Register offset in the register space. */
|
---|
27 | uint32_t offset;
|
---|
28 | /** Size in bytes. Registers of size > 4 are in fact tables. */
|
---|
29 | uint32_t size;
|
---|
30 | /** Readable bits. */
|
---|
31 | uint32_t readable;
|
---|
32 | /** Writable bits. */
|
---|
33 | uint32_t writable;
|
---|
34 | /** Register descriptor (RD) flags of type HDA_RD_FLAG_.
|
---|
35 | * These are used to specify the handling (read/write)
|
---|
36 | * policy of the register. */
|
---|
37 | uint32_t fFlags;
|
---|
38 | /** Read callback. */
|
---|
39 | int (*pfnRead)(PHDASTATE pThis, uint32_t iReg, uint32_t *pu32Value);
|
---|
40 | /** Write callback. */
|
---|
41 | int (*pfnWrite)(PHDASTATE pThis, uint32_t iReg, uint32_t u32Value);
|
---|
42 | /** Index into the register storage array. */
|
---|
43 | uint32_t mem_idx;
|
---|
44 | /** Abbreviated name. */
|
---|
45 | const char *abbrev;
|
---|
46 | /** Descripton. */
|
---|
47 | const char *desc;
|
---|
48 | } HDAREGDESC, *PHDAREGDESC;
|
---|
49 |
|
---|
50 | /**
|
---|
51 | * HDA register aliases (HDA spec 3.3.45).
|
---|
52 | * @remarks Sorted by offReg.
|
---|
53 | */
|
---|
54 | typedef struct HDAREGALIAS
|
---|
55 | {
|
---|
56 | /** The alias register offset. */
|
---|
57 | uint32_t offReg;
|
---|
58 | /** The register index. */
|
---|
59 | int idxAlias;
|
---|
60 | } HDAREGALIAS, *PHDAREGALIAS;
|
---|
61 |
|
---|
62 | /**
|
---|
63 | * At the moment we support 4 input + 4 output streams max, which is 8 in total.
|
---|
64 | * Bidirectional streams are currently *not* supported.
|
---|
65 | *
|
---|
66 | * Note: When changing any of those values, be prepared for some saved state
|
---|
67 | * fixups / trouble!
|
---|
68 | */
|
---|
69 | #define HDA_MAX_SDI 4
|
---|
70 | #define HDA_MAX_SDO 4
|
---|
71 | #define HDA_MAX_STREAMS (HDA_MAX_SDI + HDA_MAX_SDO)
|
---|
72 | AssertCompile(HDA_MAX_SDI <= HDA_MAX_SDO);
|
---|
73 |
|
---|
74 | /** Number of general registers. */
|
---|
75 | #define HDA_NUM_GENERAL_REGS 34
|
---|
76 | /** Number of total registers in the HDA's register map. */
|
---|
77 | #define HDA_NUM_REGS (HDA_NUM_GENERAL_REGS + (HDA_MAX_STREAMS * 10 /* Each stream descriptor has 10 registers */))
|
---|
78 | /** Total number of stream tags (channels). Index 0 is reserved / invalid. */
|
---|
79 | #define HDA_MAX_TAGS 16
|
---|
80 |
|
---|
81 | /**
|
---|
82 | * ICH6 datasheet defines limits for FIFOS registers (18.2.39).
|
---|
83 | * Formula: size - 1
|
---|
84 | * Other values not listed are not supported.
|
---|
85 | */
|
---|
86 | /** Maximum FIFO size (in bytes). */
|
---|
87 | #define HDA_FIFO_MAX 256
|
---|
88 |
|
---|
89 | /** Default timer frequency (in Hz).
|
---|
90 | *
|
---|
91 | * Lowering this value can ask for trouble, as backends then can run
|
---|
92 | * into data underruns. */
|
---|
93 | #define HDA_TIMER_HZ_DEFAULT 100
|
---|
94 |
|
---|
95 | /** Default position adjustment (in audio samples).
|
---|
96 | *
|
---|
97 | * For snd_hda_intel (Linux guests), the first BDL entry always is being used as
|
---|
98 | * so-called BDL adjustment, which can vary, and is being used for chipsets which
|
---|
99 | * misbehave and/or are incorrectly implemented.
|
---|
100 | *
|
---|
101 | * The BDL adjustment entry *always* has the IOC (Interrupt on Completion) bit set.
|
---|
102 | *
|
---|
103 | * For Intel Baytrail / Braswell implementations the BDL default adjustment is 32 frames, whereas
|
---|
104 | * for ICH / PCH it's only one (1) frame.
|
---|
105 | *
|
---|
106 | * See default_bdl_pos_adj() and snd_hdac_stream_setup_periods() for more information.
|
---|
107 | *
|
---|
108 | * By default we apply some simple heuristics in hdaStreamInit().
|
---|
109 | */
|
---|
110 | #define HDA_POS_ADJUST_DEFAULT 0
|
---|
111 |
|
---|
112 | /** HDA's (fixed) audio frame size in bytes.
|
---|
113 | * We only support 16-bit stereo frames at the moment. */
|
---|
114 | #define HDA_FRAME_SIZE 4
|
---|
115 |
|
---|
116 | /** Offset of the SD0 register map. */
|
---|
117 | #define HDA_REG_DESC_SD0_BASE 0x80
|
---|
118 |
|
---|
119 | /** Turn a short global register name into an memory index and a stringized name. */
|
---|
120 | #define HDA_REG_IDX(abbrev) HDA_MEM_IND_NAME(abbrev), #abbrev
|
---|
121 |
|
---|
122 | /** Turns a short stream register name into an memory index and a stringized name. */
|
---|
123 | #define HDA_REG_IDX_STRM(reg, suff) HDA_MEM_IND_NAME(reg ## suff), #reg #suff
|
---|
124 |
|
---|
125 | /** Same as above for a register *not* stored in memory. */
|
---|
126 | #define HDA_REG_IDX_NOMEM(abbrev) 0, #abbrev
|
---|
127 |
|
---|
128 | extern const HDAREGDESC g_aHdaRegMap[HDA_NUM_REGS];
|
---|
129 |
|
---|
130 | /**
|
---|
131 | * NB: Register values stored in memory (au32Regs[]) are indexed through
|
---|
132 | * the HDA_RMX_xxx macros (also HDA_MEM_IND_NAME()). On the other hand, the
|
---|
133 | * register descriptors in g_aHdaRegMap[] are indexed through the
|
---|
134 | * HDA_REG_xxx macros (also HDA_REG_IND_NAME()).
|
---|
135 | *
|
---|
136 | * The au32Regs[] layout is kept unchanged for saved state
|
---|
137 | * compatibility.
|
---|
138 | */
|
---|
139 |
|
---|
140 | /* Registers */
|
---|
141 | #define HDA_REG_IND_NAME(x) HDA_REG_##x
|
---|
142 | #define HDA_MEM_IND_NAME(x) HDA_RMX_##x
|
---|
143 | #define HDA_REG_IND(pThis, x) ((pThis)->au32Regs[g_aHdaRegMap[x].mem_idx])
|
---|
144 | #define HDA_REG(pThis, x) (HDA_REG_IND((pThis), HDA_REG_IND_NAME(x)))
|
---|
145 |
|
---|
146 |
|
---|
147 | #define HDA_REG_GCAP 0 /* Range 0x00 - 0x01 */
|
---|
148 | #define HDA_RMX_GCAP 0
|
---|
149 | /**
|
---|
150 | * GCAP HDASpec 3.3.2 This macro encodes the following information about HDA in a compact manner:
|
---|
151 | *
|
---|
152 | * oss (15:12) - Number of output streams supported.
|
---|
153 | * iss (11:8) - Number of input streams supported.
|
---|
154 | * bss (7:3) - Number of bidirectional streams supported.
|
---|
155 | * bds (2:1) - Number of serial data out (SDO) signals supported.
|
---|
156 | * b64sup (0) - 64 bit addressing supported.
|
---|
157 | */
|
---|
158 | #define HDA_MAKE_GCAP(oss, iss, bss, bds, b64sup) \
|
---|
159 | ( (((oss) & 0xF) << 12) \
|
---|
160 | | (((iss) & 0xF) << 8) \
|
---|
161 | | (((bss) & 0x1F) << 3) \
|
---|
162 | | (((bds) & 0x3) << 2) \
|
---|
163 | | ((b64sup) & 1))
|
---|
164 |
|
---|
165 | #define HDA_REG_VMIN 1 /* 0x02 */
|
---|
166 | #define HDA_RMX_VMIN 1
|
---|
167 |
|
---|
168 | #define HDA_REG_VMAJ 2 /* 0x03 */
|
---|
169 | #define HDA_RMX_VMAJ 2
|
---|
170 |
|
---|
171 | #define HDA_REG_OUTPAY 3 /* 0x04-0x05 */
|
---|
172 | #define HDA_RMX_OUTPAY 3
|
---|
173 |
|
---|
174 | #define HDA_REG_INPAY 4 /* 0x06-0x07 */
|
---|
175 | #define HDA_RMX_INPAY 4
|
---|
176 |
|
---|
177 | #define HDA_REG_GCTL 5 /* 0x08-0x0B */
|
---|
178 | #define HDA_RMX_GCTL 5
|
---|
179 | #define HDA_GCTL_UNSOL RT_BIT(8) /* Accept Unsolicited Response Enable */
|
---|
180 | #define HDA_GCTL_FCNTRL RT_BIT(1) /* Flush Control */
|
---|
181 | #define HDA_GCTL_CRST RT_BIT(0) /* Controller Reset */
|
---|
182 |
|
---|
183 | #define HDA_REG_WAKEEN 6 /* 0x0C */
|
---|
184 | #define HDA_RMX_WAKEEN 6
|
---|
185 |
|
---|
186 | #define HDA_REG_STATESTS 7 /* 0x0E */
|
---|
187 | #define HDA_RMX_STATESTS 7
|
---|
188 | #define HDA_STATESTS_SCSF_MASK 0x7 /* State Change Status Flags (6.2.8). */
|
---|
189 |
|
---|
190 | #define HDA_REG_GSTS 8 /* 0x10-0x11*/
|
---|
191 | #define HDA_RMX_GSTS 8
|
---|
192 | #define HDA_GSTS_FSTS RT_BIT(1) /* Flush Status */
|
---|
193 |
|
---|
194 | #define HDA_REG_OUTSTRMPAY 9 /* 0x18 */
|
---|
195 | #define HDA_RMX_OUTSTRMPAY 112
|
---|
196 |
|
---|
197 | #define HDA_REG_INSTRMPAY 10 /* 0x1a */
|
---|
198 | #define HDA_RMX_INSTRMPAY 113
|
---|
199 |
|
---|
200 | #define HDA_REG_INTCTL 11 /* 0x20 */
|
---|
201 | #define HDA_RMX_INTCTL 9
|
---|
202 | #define HDA_INTCTL_GIE RT_BIT(31) /* Global Interrupt Enable */
|
---|
203 | #define HDA_INTCTL_CIE RT_BIT(30) /* Controller Interrupt Enable */
|
---|
204 | /** Bits 0-29 correspond to streams 0-29. */
|
---|
205 | #define HDA_STRMINT_MASK 0xFF /* Streams 0-7 implemented. Applies to INTCTL and INTSTS. */
|
---|
206 |
|
---|
207 | #define HDA_REG_INTSTS 12 /* 0x24 */
|
---|
208 | #define HDA_RMX_INTSTS 10
|
---|
209 | #define HDA_INTSTS_GIS RT_BIT(31) /* Global Interrupt Status */
|
---|
210 | #define HDA_INTSTS_CIS RT_BIT(30) /* Controller Interrupt Status */
|
---|
211 |
|
---|
212 | #define HDA_REG_WALCLK 13 /* 0x30 */
|
---|
213 | /**NB: HDA_RMX_WALCLK is not defined because the register is not stored in memory. */
|
---|
214 |
|
---|
215 | /**
|
---|
216 | * Note: The HDA specification defines a SSYNC register at offset 0x38. The
|
---|
217 | * ICH6/ICH9 datahseet defines SSYNC at offset 0x34. The Linux HDA driver matches
|
---|
218 | * the datasheet.
|
---|
219 | */
|
---|
220 | #define HDA_REG_SSYNC 14 /* 0x34 */
|
---|
221 | #define HDA_RMX_SSYNC 12
|
---|
222 |
|
---|
223 | #define HDA_REG_CORBLBASE 15 /* 0x40 */
|
---|
224 | #define HDA_RMX_CORBLBASE 13
|
---|
225 |
|
---|
226 | #define HDA_REG_CORBUBASE 16 /* 0x44 */
|
---|
227 | #define HDA_RMX_CORBUBASE 14
|
---|
228 |
|
---|
229 | #define HDA_REG_CORBWP 17 /* 0x48 */
|
---|
230 | #define HDA_RMX_CORBWP 15
|
---|
231 |
|
---|
232 | #define HDA_REG_CORBRP 18 /* 0x4A */
|
---|
233 | #define HDA_RMX_CORBRP 16
|
---|
234 | #define HDA_CORBRP_RST RT_BIT(15) /* CORB Read Pointer Reset */
|
---|
235 |
|
---|
236 | #define HDA_REG_CORBCTL 19 /* 0x4C */
|
---|
237 | #define HDA_RMX_CORBCTL 17
|
---|
238 | #define HDA_CORBCTL_DMA RT_BIT(1) /* Enable CORB DMA Engine */
|
---|
239 | #define HDA_CORBCTL_CMEIE RT_BIT(0) /* CORB Memory Error Interrupt Enable */
|
---|
240 |
|
---|
241 | #define HDA_REG_CORBSTS 20 /* 0x4D */
|
---|
242 | #define HDA_RMX_CORBSTS 18
|
---|
243 |
|
---|
244 | #define HDA_REG_CORBSIZE 21 /* 0x4E */
|
---|
245 | #define HDA_RMX_CORBSIZE 19
|
---|
246 | #define HDA_CORBSIZE_SZ_CAP 0xF0
|
---|
247 | #define HDA_CORBSIZE_SZ 0x3
|
---|
248 |
|
---|
249 | /** Number of CORB buffer entries. */
|
---|
250 | #define HDA_CORB_SIZE 256
|
---|
251 | /** CORB element size (in bytes). */
|
---|
252 | #define HDA_CORB_ELEMENT_SIZE 4
|
---|
253 | /** Number of RIRB buffer entries. */
|
---|
254 | #define HDA_RIRB_SIZE 256
|
---|
255 | /** RIRB element size (in bytes). */
|
---|
256 | #define HDA_RIRB_ELEMENT_SIZE 8
|
---|
257 |
|
---|
258 | #define HDA_REG_RIRBLBASE 22 /* 0x50 */
|
---|
259 | #define HDA_RMX_RIRBLBASE 20
|
---|
260 |
|
---|
261 | #define HDA_REG_RIRBUBASE 23 /* 0x54 */
|
---|
262 | #define HDA_RMX_RIRBUBASE 21
|
---|
263 |
|
---|
264 | #define HDA_REG_RIRBWP 24 /* 0x58 */
|
---|
265 | #define HDA_RMX_RIRBWP 22
|
---|
266 | #define HDA_RIRBWP_RST RT_BIT(15) /* RIRB Write Pointer Reset */
|
---|
267 |
|
---|
268 | #define HDA_REG_RINTCNT 25 /* 0x5A */
|
---|
269 | #define HDA_RMX_RINTCNT 23
|
---|
270 |
|
---|
271 | /** Maximum number of Response Interrupts. */
|
---|
272 | #define HDA_MAX_RINTCNT 256
|
---|
273 |
|
---|
274 | #define HDA_REG_RIRBCTL 26 /* 0x5C */
|
---|
275 | #define HDA_RMX_RIRBCTL 24
|
---|
276 | #define HDA_RIRBCTL_ROIC RT_BIT(2) /* Response Overrun Interrupt Control */
|
---|
277 | #define HDA_RIRBCTL_RDMAEN RT_BIT(1) /* RIRB DMA Enable */
|
---|
278 | #define HDA_RIRBCTL_RINTCTL RT_BIT(0) /* Response Interrupt Control */
|
---|
279 |
|
---|
280 | #define HDA_REG_RIRBSTS 27 /* 0x5D */
|
---|
281 | #define HDA_RMX_RIRBSTS 25
|
---|
282 | #define HDA_RIRBSTS_RIRBOIS RT_BIT(2) /* Response Overrun Interrupt Status */
|
---|
283 | #define HDA_RIRBSTS_RINTFL RT_BIT(0) /* Response Interrupt Flag */
|
---|
284 |
|
---|
285 | #define HDA_REG_RIRBSIZE 28 /* 0x5E */
|
---|
286 | #define HDA_RMX_RIRBSIZE 26
|
---|
287 |
|
---|
288 | #define HDA_REG_IC 29 /* 0x60 */
|
---|
289 | #define HDA_RMX_IC 27
|
---|
290 |
|
---|
291 | #define HDA_REG_IR 30 /* 0x64 */
|
---|
292 | #define HDA_RMX_IR 28
|
---|
293 |
|
---|
294 | #define HDA_REG_IRS 31 /* 0x68 */
|
---|
295 | #define HDA_RMX_IRS 29
|
---|
296 | #define HDA_IRS_IRV RT_BIT(1) /* Immediate Result Valid */
|
---|
297 | #define HDA_IRS_ICB RT_BIT(0) /* Immediate Command Busy */
|
---|
298 |
|
---|
299 | #define HDA_REG_DPLBASE 32 /* 0x70 */
|
---|
300 | #define HDA_RMX_DPLBASE 30
|
---|
301 |
|
---|
302 | #define HDA_REG_DPUBASE 33 /* 0x74 */
|
---|
303 | #define HDA_RMX_DPUBASE 31
|
---|
304 |
|
---|
305 | #define DPBASE_ADDR_MASK (~(uint64_t)0x7f)
|
---|
306 |
|
---|
307 | #define HDA_STREAM_REG_DEF(name, num) (HDA_REG_SD##num##name)
|
---|
308 | #define HDA_STREAM_RMX_DEF(name, num) (HDA_RMX_SD##num##name)
|
---|
309 | /** Note: sdnum here _MUST_ be stream reg number [0,7]. */
|
---|
310 | #define HDA_STREAM_REG(pThis, name, sdnum) (HDA_REG_IND((pThis), HDA_REG_SD0##name + (sdnum) * 10))
|
---|
311 |
|
---|
312 | #define HDA_SD_NUM_FROM_REG(pThis, func, reg) ((reg - HDA_STREAM_REG_DEF(func, 0)) / 10)
|
---|
313 |
|
---|
314 | /** @todo Condense marcos! */
|
---|
315 |
|
---|
316 | #define HDA_REG_SD0CTL HDA_NUM_GENERAL_REGS /* 0x80; other streams offset by 0x20 */
|
---|
317 | #define HDA_RMX_SD0CTL 32
|
---|
318 | #define HDA_RMX_SD1CTL (HDA_STREAM_RMX_DEF(CTL, 0) + 10)
|
---|
319 | #define HDA_RMX_SD2CTL (HDA_STREAM_RMX_DEF(CTL, 0) + 20)
|
---|
320 | #define HDA_RMX_SD3CTL (HDA_STREAM_RMX_DEF(CTL, 0) + 30)
|
---|
321 | #define HDA_RMX_SD4CTL (HDA_STREAM_RMX_DEF(CTL, 0) + 40)
|
---|
322 | #define HDA_RMX_SD5CTL (HDA_STREAM_RMX_DEF(CTL, 0) + 50)
|
---|
323 | #define HDA_RMX_SD6CTL (HDA_STREAM_RMX_DEF(CTL, 0) + 60)
|
---|
324 | #define HDA_RMX_SD7CTL (HDA_STREAM_RMX_DEF(CTL, 0) + 70)
|
---|
325 |
|
---|
326 | #define HDA_SDCTL_NUM_MASK 0xF
|
---|
327 | #define HDA_SDCTL_NUM_SHIFT 20
|
---|
328 | #define HDA_SDCTL_DIR RT_BIT(19) /* Direction (Bidirectional streams only!) */
|
---|
329 | #define HDA_SDCTL_TP RT_BIT(18) /* Traffic Priority (PCI Express) */
|
---|
330 | #define HDA_SDCTL_STRIPE_MASK 0x3
|
---|
331 | #define HDA_SDCTL_STRIPE_SHIFT 16
|
---|
332 | #define HDA_SDCTL_DEIE RT_BIT(4) /* Descriptor Error Interrupt Enable */
|
---|
333 | #define HDA_SDCTL_FEIE RT_BIT(3) /* FIFO Error Interrupt Enable */
|
---|
334 | #define HDA_SDCTL_IOCE RT_BIT(2) /* Interrupt On Completion Enable */
|
---|
335 | #define HDA_SDCTL_RUN RT_BIT(1) /* Stream Run */
|
---|
336 | #define HDA_SDCTL_SRST RT_BIT(0) /* Stream Reset */
|
---|
337 |
|
---|
338 | #define HDA_REG_SD0STS 35 /* 0x83; other streams offset by 0x20 */
|
---|
339 | #define HDA_RMX_SD0STS 33
|
---|
340 | #define HDA_RMX_SD1STS (HDA_STREAM_RMX_DEF(STS, 0) + 10)
|
---|
341 | #define HDA_RMX_SD2STS (HDA_STREAM_RMX_DEF(STS, 0) + 20)
|
---|
342 | #define HDA_RMX_SD3STS (HDA_STREAM_RMX_DEF(STS, 0) + 30)
|
---|
343 | #define HDA_RMX_SD4STS (HDA_STREAM_RMX_DEF(STS, 0) + 40)
|
---|
344 | #define HDA_RMX_SD5STS (HDA_STREAM_RMX_DEF(STS, 0) + 50)
|
---|
345 | #define HDA_RMX_SD6STS (HDA_STREAM_RMX_DEF(STS, 0) + 60)
|
---|
346 | #define HDA_RMX_SD7STS (HDA_STREAM_RMX_DEF(STS, 0) + 70)
|
---|
347 |
|
---|
348 | #define HDA_SDSTS_FIFORDY RT_BIT(5) /* FIFO Ready */
|
---|
349 | #define HDA_SDSTS_DESE RT_BIT(4) /* Descriptor Error */
|
---|
350 | #define HDA_SDSTS_FIFOE RT_BIT(3) /* FIFO Error */
|
---|
351 | #define HDA_SDSTS_BCIS RT_BIT(2) /* Buffer Completion Interrupt Status */
|
---|
352 |
|
---|
353 | #define HDA_REG_SD0LPIB 36 /* 0x84; other streams offset by 0x20 */
|
---|
354 | #define HDA_REG_SD1LPIB (HDA_STREAM_REG_DEF(LPIB, 0) + 10) /* 0xA4 */
|
---|
355 | #define HDA_REG_SD2LPIB (HDA_STREAM_REG_DEF(LPIB, 0) + 20) /* 0xC4 */
|
---|
356 | #define HDA_REG_SD3LPIB (HDA_STREAM_REG_DEF(LPIB, 0) + 30) /* 0xE4 */
|
---|
357 | #define HDA_REG_SD4LPIB (HDA_STREAM_REG_DEF(LPIB, 0) + 40) /* 0x104 */
|
---|
358 | #define HDA_REG_SD5LPIB (HDA_STREAM_REG_DEF(LPIB, 0) + 50) /* 0x124 */
|
---|
359 | #define HDA_REG_SD6LPIB (HDA_STREAM_REG_DEF(LPIB, 0) + 60) /* 0x144 */
|
---|
360 | #define HDA_REG_SD7LPIB (HDA_STREAM_REG_DEF(LPIB, 0) + 70) /* 0x164 */
|
---|
361 | #define HDA_RMX_SD0LPIB 34
|
---|
362 | #define HDA_RMX_SD1LPIB (HDA_STREAM_RMX_DEF(LPIB, 0) + 10)
|
---|
363 | #define HDA_RMX_SD2LPIB (HDA_STREAM_RMX_DEF(LPIB, 0) + 20)
|
---|
364 | #define HDA_RMX_SD3LPIB (HDA_STREAM_RMX_DEF(LPIB, 0) + 30)
|
---|
365 | #define HDA_RMX_SD4LPIB (HDA_STREAM_RMX_DEF(LPIB, 0) + 40)
|
---|
366 | #define HDA_RMX_SD5LPIB (HDA_STREAM_RMX_DEF(LPIB, 0) + 50)
|
---|
367 | #define HDA_RMX_SD6LPIB (HDA_STREAM_RMX_DEF(LPIB, 0) + 60)
|
---|
368 | #define HDA_RMX_SD7LPIB (HDA_STREAM_RMX_DEF(LPIB, 0) + 70)
|
---|
369 |
|
---|
370 | #define HDA_REG_SD0CBL 37 /* 0x88; other streams offset by 0x20 */
|
---|
371 | #define HDA_RMX_SD0CBL 35
|
---|
372 | #define HDA_RMX_SD1CBL (HDA_STREAM_RMX_DEF(CBL, 0) + 10)
|
---|
373 | #define HDA_RMX_SD2CBL (HDA_STREAM_RMX_DEF(CBL, 0) + 20)
|
---|
374 | #define HDA_RMX_SD3CBL (HDA_STREAM_RMX_DEF(CBL, 0) + 30)
|
---|
375 | #define HDA_RMX_SD4CBL (HDA_STREAM_RMX_DEF(CBL, 0) + 40)
|
---|
376 | #define HDA_RMX_SD5CBL (HDA_STREAM_RMX_DEF(CBL, 0) + 50)
|
---|
377 | #define HDA_RMX_SD6CBL (HDA_STREAM_RMX_DEF(CBL, 0) + 60)
|
---|
378 | #define HDA_RMX_SD7CBL (HDA_STREAM_RMX_DEF(CBL, 0) + 70)
|
---|
379 |
|
---|
380 | #define HDA_REG_SD0LVI 38 /* 0x8C; other streams offset by 0x20 */
|
---|
381 | #define HDA_RMX_SD0LVI 36
|
---|
382 | #define HDA_RMX_SD1LVI (HDA_STREAM_RMX_DEF(LVI, 0) + 10)
|
---|
383 | #define HDA_RMX_SD2LVI (HDA_STREAM_RMX_DEF(LVI, 0) + 20)
|
---|
384 | #define HDA_RMX_SD3LVI (HDA_STREAM_RMX_DEF(LVI, 0) + 30)
|
---|
385 | #define HDA_RMX_SD4LVI (HDA_STREAM_RMX_DEF(LVI, 0) + 40)
|
---|
386 | #define HDA_RMX_SD5LVI (HDA_STREAM_RMX_DEF(LVI, 0) + 50)
|
---|
387 | #define HDA_RMX_SD6LVI (HDA_STREAM_RMX_DEF(LVI, 0) + 60)
|
---|
388 | #define HDA_RMX_SD7LVI (HDA_STREAM_RMX_DEF(LVI, 0) + 70)
|
---|
389 |
|
---|
390 | #define HDA_REG_SD0FIFOW 39 /* 0x8E; other streams offset by 0x20 */
|
---|
391 | #define HDA_RMX_SD0FIFOW 37
|
---|
392 | #define HDA_RMX_SD1FIFOW (HDA_STREAM_RMX_DEF(FIFOW, 0) + 10)
|
---|
393 | #define HDA_RMX_SD2FIFOW (HDA_STREAM_RMX_DEF(FIFOW, 0) + 20)
|
---|
394 | #define HDA_RMX_SD3FIFOW (HDA_STREAM_RMX_DEF(FIFOW, 0) + 30)
|
---|
395 | #define HDA_RMX_SD4FIFOW (HDA_STREAM_RMX_DEF(FIFOW, 0) + 40)
|
---|
396 | #define HDA_RMX_SD5FIFOW (HDA_STREAM_RMX_DEF(FIFOW, 0) + 50)
|
---|
397 | #define HDA_RMX_SD6FIFOW (HDA_STREAM_RMX_DEF(FIFOW, 0) + 60)
|
---|
398 | #define HDA_RMX_SD7FIFOW (HDA_STREAM_RMX_DEF(FIFOW, 0) + 70)
|
---|
399 |
|
---|
400 | /*
|
---|
401 | * ICH6 datasheet defined limits for FIFOW values (18.2.38).
|
---|
402 | */
|
---|
403 | #define HDA_SDFIFOW_8B 0x2
|
---|
404 | #define HDA_SDFIFOW_16B 0x3
|
---|
405 | #define HDA_SDFIFOW_32B 0x4
|
---|
406 |
|
---|
407 | #define HDA_REG_SD0FIFOS 40 /* 0x90; other streams offset by 0x20 */
|
---|
408 | #define HDA_RMX_SD0FIFOS 38
|
---|
409 | #define HDA_RMX_SD1FIFOS (HDA_STREAM_RMX_DEF(FIFOS, 0) + 10)
|
---|
410 | #define HDA_RMX_SD2FIFOS (HDA_STREAM_RMX_DEF(FIFOS, 0) + 20)
|
---|
411 | #define HDA_RMX_SD3FIFOS (HDA_STREAM_RMX_DEF(FIFOS, 0) + 30)
|
---|
412 | #define HDA_RMX_SD4FIFOS (HDA_STREAM_RMX_DEF(FIFOS, 0) + 40)
|
---|
413 | #define HDA_RMX_SD5FIFOS (HDA_STREAM_RMX_DEF(FIFOS, 0) + 50)
|
---|
414 | #define HDA_RMX_SD6FIFOS (HDA_STREAM_RMX_DEF(FIFOS, 0) + 60)
|
---|
415 | #define HDA_RMX_SD7FIFOS (HDA_STREAM_RMX_DEF(FIFOS, 0) + 70)
|
---|
416 |
|
---|
417 | #define HDA_SDIFIFO_120B 0x77 /* 8-, 16-, 20-, 24-, 32-bit Input Streams */
|
---|
418 | #define HDA_SDIFIFO_160B 0x9F /* 20-, 24-bit Input Streams Streams */
|
---|
419 |
|
---|
420 | #define HDA_SDOFIFO_16B 0x0F /* 8-, 16-, 20-, 24-, 32-bit Output Streams */
|
---|
421 | #define HDA_SDOFIFO_32B 0x1F /* 8-, 16-, 20-, 24-, 32-bit Output Streams */
|
---|
422 | #define HDA_SDOFIFO_64B 0x3F /* 8-, 16-, 20-, 24-, 32-bit Output Streams */
|
---|
423 | #define HDA_SDOFIFO_128B 0x7F /* 8-, 16-, 20-, 24-, 32-bit Output Streams */
|
---|
424 | #define HDA_SDOFIFO_192B 0xBF /* 8-, 16-, 20-, 24-, 32-bit Output Streams */
|
---|
425 | #define HDA_SDOFIFO_256B 0xFF /* 20-, 24-bit Output Streams */
|
---|
426 |
|
---|
427 | #define HDA_REG_SD0FMT 41 /* 0x92; other streams offset by 0x20 */
|
---|
428 | #define HDA_RMX_SD0FMT 39
|
---|
429 | #define HDA_RMX_SD1FMT (HDA_STREAM_RMX_DEF(FMT, 0) + 10)
|
---|
430 | #define HDA_RMX_SD2FMT (HDA_STREAM_RMX_DEF(FMT, 0) + 20)
|
---|
431 | #define HDA_RMX_SD3FMT (HDA_STREAM_RMX_DEF(FMT, 0) + 30)
|
---|
432 | #define HDA_RMX_SD4FMT (HDA_STREAM_RMX_DEF(FMT, 0) + 40)
|
---|
433 | #define HDA_RMX_SD5FMT (HDA_STREAM_RMX_DEF(FMT, 0) + 50)
|
---|
434 | #define HDA_RMX_SD6FMT (HDA_STREAM_RMX_DEF(FMT, 0) + 60)
|
---|
435 | #define HDA_RMX_SD7FMT (HDA_STREAM_RMX_DEF(FMT, 0) + 70)
|
---|
436 |
|
---|
437 | #define HDA_REG_SD0BDPL 42 /* 0x98; other streams offset by 0x20 */
|
---|
438 | #define HDA_RMX_SD0BDPL 40
|
---|
439 | #define HDA_RMX_SD1BDPL (HDA_STREAM_RMX_DEF(BDPL, 0) + 10)
|
---|
440 | #define HDA_RMX_SD2BDPL (HDA_STREAM_RMX_DEF(BDPL, 0) + 20)
|
---|
441 | #define HDA_RMX_SD3BDPL (HDA_STREAM_RMX_DEF(BDPL, 0) + 30)
|
---|
442 | #define HDA_RMX_SD4BDPL (HDA_STREAM_RMX_DEF(BDPL, 0) + 40)
|
---|
443 | #define HDA_RMX_SD5BDPL (HDA_STREAM_RMX_DEF(BDPL, 0) + 50)
|
---|
444 | #define HDA_RMX_SD6BDPL (HDA_STREAM_RMX_DEF(BDPL, 0) + 60)
|
---|
445 | #define HDA_RMX_SD7BDPL (HDA_STREAM_RMX_DEF(BDPL, 0) + 70)
|
---|
446 |
|
---|
447 | #define HDA_REG_SD0BDPU 43 /* 0x9C; other streams offset by 0x20 */
|
---|
448 | #define HDA_RMX_SD0BDPU 41
|
---|
449 | #define HDA_RMX_SD1BDPU (HDA_STREAM_RMX_DEF(BDPU, 0) + 10)
|
---|
450 | #define HDA_RMX_SD2BDPU (HDA_STREAM_RMX_DEF(BDPU, 0) + 20)
|
---|
451 | #define HDA_RMX_SD3BDPU (HDA_STREAM_RMX_DEF(BDPU, 0) + 30)
|
---|
452 | #define HDA_RMX_SD4BDPU (HDA_STREAM_RMX_DEF(BDPU, 0) + 40)
|
---|
453 | #define HDA_RMX_SD5BDPU (HDA_STREAM_RMX_DEF(BDPU, 0) + 50)
|
---|
454 | #define HDA_RMX_SD6BDPU (HDA_STREAM_RMX_DEF(BDPU, 0) + 60)
|
---|
455 | #define HDA_RMX_SD7BDPU (HDA_STREAM_RMX_DEF(BDPU, 0) + 70)
|
---|
456 |
|
---|
457 | #define HDA_CODEC_CAD_SHIFT 28
|
---|
458 | /** Encodes the (required) LUN into a codec command. */
|
---|
459 | #define HDA_CODEC_CMD(cmd, lun) ((cmd) | (lun << HDA_CODEC_CAD_SHIFT))
|
---|
460 |
|
---|
461 | #define HDA_SDFMT_NON_PCM_SHIFT 15
|
---|
462 | #define HDA_SDFMT_NON_PCM_MASK 0x1
|
---|
463 | #define HDA_SDFMT_BASE_RATE_SHIFT 14
|
---|
464 | #define HDA_SDFMT_BASE_RATE_MASK 0x1
|
---|
465 | #define HDA_SDFMT_MULT_SHIFT 11
|
---|
466 | #define HDA_SDFMT_MULT_MASK 0x7
|
---|
467 | #define HDA_SDFMT_DIV_SHIFT 8
|
---|
468 | #define HDA_SDFMT_DIV_MASK 0x7
|
---|
469 | #define HDA_SDFMT_BITS_SHIFT 4
|
---|
470 | #define HDA_SDFMT_BITS_MASK 0x7
|
---|
471 | #define HDA_SDFMT_CHANNELS_MASK 0xF
|
---|
472 |
|
---|
473 | #define HDA_SDFMT_TYPE RT_BIT(15)
|
---|
474 | #define HDA_SDFMT_TYPE_PCM (0)
|
---|
475 | #define HDA_SDFMT_TYPE_NON_PCM (1)
|
---|
476 |
|
---|
477 | #define HDA_SDFMT_BASE RT_BIT(14)
|
---|
478 | #define HDA_SDFMT_BASE_48KHZ (0)
|
---|
479 | #define HDA_SDFMT_BASE_44KHZ (1)
|
---|
480 |
|
---|
481 | #define HDA_SDFMT_MULT_1X (0)
|
---|
482 | #define HDA_SDFMT_MULT_2X (1)
|
---|
483 | #define HDA_SDFMT_MULT_3X (2)
|
---|
484 | #define HDA_SDFMT_MULT_4X (3)
|
---|
485 |
|
---|
486 | #define HDA_SDFMT_DIV_1X (0)
|
---|
487 | #define HDA_SDFMT_DIV_2X (1)
|
---|
488 | #define HDA_SDFMT_DIV_3X (2)
|
---|
489 | #define HDA_SDFMT_DIV_4X (3)
|
---|
490 | #define HDA_SDFMT_DIV_5X (4)
|
---|
491 | #define HDA_SDFMT_DIV_6X (5)
|
---|
492 | #define HDA_SDFMT_DIV_7X (6)
|
---|
493 | #define HDA_SDFMT_DIV_8X (7)
|
---|
494 |
|
---|
495 | #define HDA_SDFMT_8_BIT (0)
|
---|
496 | #define HDA_SDFMT_16_BIT (1)
|
---|
497 | #define HDA_SDFMT_20_BIT (2)
|
---|
498 | #define HDA_SDFMT_24_BIT (3)
|
---|
499 | #define HDA_SDFMT_32_BIT (4)
|
---|
500 |
|
---|
501 | #define HDA_SDFMT_CHAN_MONO (0)
|
---|
502 | #define HDA_SDFMT_CHAN_STEREO (1)
|
---|
503 |
|
---|
504 | /** Emits a SDnFMT register format.
|
---|
505 | * Also being used in the codec's converter format. */
|
---|
506 | #define HDA_SDFMT_MAKE(_afNonPCM, _aBaseRate, _aMult, _aDiv, _aBits, _aChan) \
|
---|
507 | ( (((_afNonPCM) & HDA_SDFMT_NON_PCM_MASK) << HDA_SDFMT_NON_PCM_SHIFT) \
|
---|
508 | | (((_aBaseRate) & HDA_SDFMT_BASE_RATE_MASK) << HDA_SDFMT_BASE_RATE_SHIFT) \
|
---|
509 | | (((_aMult) & HDA_SDFMT_MULT_MASK) << HDA_SDFMT_MULT_SHIFT) \
|
---|
510 | | (((_aDiv) & HDA_SDFMT_DIV_MASK) << HDA_SDFMT_DIV_SHIFT) \
|
---|
511 | | (((_aBits) & HDA_SDFMT_BITS_MASK) << HDA_SDFMT_BITS_SHIFT) \
|
---|
512 | | ( (_aChan) & HDA_SDFMT_CHANNELS_MASK))
|
---|
513 |
|
---|
514 | /** Interrupt on completion (IOC) flag. */
|
---|
515 | #define HDA_BDLE_FLAG_IOC RT_BIT(0)
|
---|
516 |
|
---|
517 | /*********************************************************************************************************************************
|
---|
518 | * Prototypes *
|
---|
519 | *********************************************************************************************************************************/
|
---|
520 |
|
---|
521 | /** The HDA controller. */
|
---|
522 | typedef struct HDASTATE *PHDASTATE;
|
---|
523 | /** The HDA stream. */
|
---|
524 | typedef struct HDASTREAM *PHDASTREAM;
|
---|
525 |
|
---|
526 | typedef struct HDAMIXERSINK *PHDAMIXERSINK;
|
---|
527 |
|
---|
528 |
|
---|
529 | /**
|
---|
530 | * Internal state of a Buffer Descriptor List Entry (BDLE),
|
---|
531 | * needed to keep track of the data needed for the actual device
|
---|
532 | * emulation.
|
---|
533 | */
|
---|
534 | typedef struct HDABDLESTATE
|
---|
535 | {
|
---|
536 | /** Own index within the BDL (Buffer Descriptor List). */
|
---|
537 | uint32_t u32BDLIndex;
|
---|
538 | /** Number of bytes below the stream's FIFO watermark (SDFIFOW).
|
---|
539 | * Used to check if we need fill up the FIFO again. */
|
---|
540 | uint32_t cbBelowFIFOW;
|
---|
541 | /** Current offset in DMA buffer (in bytes).*/
|
---|
542 | uint32_t u32BufOff;
|
---|
543 | uint32_t Padding;
|
---|
544 | } HDABDLESTATE, *PHDABDLESTATE;
|
---|
545 |
|
---|
546 | /**
|
---|
547 | * BDL description structure.
|
---|
548 | * Do not touch this, as this must match to the HDA specs.
|
---|
549 | */
|
---|
550 | typedef struct HDABDLEDESC
|
---|
551 | {
|
---|
552 | /** Starting address of the actual buffer. Must be 128-bit aligned. */
|
---|
553 | uint64_t u64BufAdr;
|
---|
554 | /** Size of the actual buffer (in bytes). */
|
---|
555 | uint32_t u32BufSize;
|
---|
556 | /** Bit 0: Interrupt on completion; the controller will generate
|
---|
557 | * an interrupt when the last byte of the buffer has been
|
---|
558 | * fetched by the DMA engine.
|
---|
559 | *
|
---|
560 | * Rest is reserved for further use and must be 0. */
|
---|
561 | uint32_t fFlags;
|
---|
562 | } HDABDLEDESC, *PHDABDLEDESC;
|
---|
563 | AssertCompileSize(HDABDLEDESC, 16); /* Always 16 byte. Also must be aligned on 128-byte boundary. */
|
---|
564 |
|
---|
565 | /**
|
---|
566 | * Buffer Descriptor List Entry (BDLE) (3.6.3).
|
---|
567 | */
|
---|
568 | typedef struct HDABDLE
|
---|
569 | {
|
---|
570 | /** The actual BDL description. */
|
---|
571 | HDABDLEDESC Desc;
|
---|
572 | /** Internal state of this BDLE.
|
---|
573 | * Not part of the actual BDLE registers. */
|
---|
574 | HDABDLESTATE State;
|
---|
575 | } HDABDLE, *PHDABDLE;
|
---|
576 |
|
---|
577 | /** @name Object lookup functions.
|
---|
578 | * @{
|
---|
579 | */
|
---|
580 | PHDAMIXERSINK hdaGetDefaultSink(PHDASTATE pThis, uint8_t uSD);
|
---|
581 | PDMAUDIODIR hdaGetDirFromSD(uint8_t uSD);
|
---|
582 | PHDASTREAM hdaGetStreamFromSD(PHDASTATE pThis, uint8_t uSD);
|
---|
583 | PHDASTREAM hdaGetStreamFromSink(PHDASTATE pThis, PHDAMIXERSINK pSink);
|
---|
584 | /** @} */
|
---|
585 |
|
---|
586 | /** @name Interrupt functions.
|
---|
587 | * @{
|
---|
588 | */
|
---|
589 | #ifdef DEBUG
|
---|
590 | int hdaProcessInterrupt(PHDASTATE pThis, const char *pszSource);
|
---|
591 | #else
|
---|
592 | int hdaProcessInterrupt(PHDASTATE pThis);
|
---|
593 | #endif
|
---|
594 | /** @} */
|
---|
595 |
|
---|
596 | /** @name Wall clock (WALCLK) functions.
|
---|
597 | * @{
|
---|
598 | */
|
---|
599 | uint64_t hdaWalClkGetCurrent(PHDASTATE pThis);
|
---|
600 | #ifdef IN_RING3
|
---|
601 | bool hdaWalClkSet(PHDASTATE pThis, uint64_t u64WalClk, bool fForce);
|
---|
602 | #endif
|
---|
603 | /** @} */
|
---|
604 |
|
---|
605 | /** @name DMA utility functions.
|
---|
606 | * @{
|
---|
607 | */
|
---|
608 | int hdaDMARead(PHDASTATE pThis, PHDASTREAM pStream, void *pvBuf, uint32_t cbBuf, uint32_t *pcbRead);
|
---|
609 | int hdaDMAWrite(PHDASTATE pThis, PHDASTREAM pStream, const void *pvBuf, uint32_t cbBuf, uint32_t *pcbWritten);
|
---|
610 | /** @} */
|
---|
611 |
|
---|
612 | /** @name Register functions.
|
---|
613 | * @{
|
---|
614 | */
|
---|
615 | uint32_t hdaGetINTSTS(PHDASTATE pThis);
|
---|
616 | #ifdef IN_RING3
|
---|
617 | int hdaSDFMTToPCMProps(uint32_t u32SDFMT, PPDMAUDIOPCMPROPS pProps);
|
---|
618 | #endif /* IN_RING3 */
|
---|
619 | /** @} */
|
---|
620 |
|
---|
621 | /** @name BDLE (Buffer Descriptor List Entry) functions.
|
---|
622 | * @{
|
---|
623 | */
|
---|
624 | #ifdef IN_RING3
|
---|
625 | # ifdef LOG_ENABLED
|
---|
626 | void hdaBDLEDumpAll(PHDASTATE pThis, uint64_t u64BDLBase, uint16_t cBDLE);
|
---|
627 | # endif
|
---|
628 | int hdaBDLEFetch(PHDASTATE pThis, PHDABDLE pBDLE, uint64_t u64BaseDMA, uint16_t u16Entry);
|
---|
629 | bool hdaBDLEIsComplete(PHDABDLE pBDLE);
|
---|
630 | bool hdaBDLENeedsInterrupt(PHDABDLE pBDLE);
|
---|
631 | #endif /* IN_RING3 */
|
---|
632 | /** @} */
|
---|
633 |
|
---|
634 | /** @name Device timer functions.
|
---|
635 | * @{
|
---|
636 | */
|
---|
637 | #ifdef IN_RING3
|
---|
638 | bool hdaTimerSet(PHDASTATE pThis, PHDASTREAM pStream, uint64_t u64Expire, bool fForce);
|
---|
639 | #endif /* IN_RING3 */
|
---|
640 | /** @} */
|
---|
641 |
|
---|
642 | #endif /* !DEV_HDA_H_COMMON */
|
---|
643 |
|
---|