1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
|
/*
* Copyright (c) 2015 Jordan Hargrave <jordan_hargrave@hotmail.com>
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#ifndef _DEV_ACPI_DMARREG_H_
#define _DEV_ACPI_DMARREG_H_
/*#define IOMMU_DEBUG*/
#define VTD_STRIDE_MASK 0x1FF
#define VTD_STRIDE_SIZE 9
#define VTD_PAGE_SIZE 4096
#define VTD_PAGE_MASK 0xFFF
#define VTD_PTE_MASK 0x0000FFFFFFFFF000LL
#define VTD_LEVEL0 12
#define VTD_LEVEL1 21
#define VTD_LEVEL2 30 /* Minimum level supported */
#define VTD_LEVEL3 39 /* Also supported */
#define VTD_LEVEL4 48
#define VTD_LEVEL5 57
#define _xbit(x,y) (((x)>> (y)) & 1)
#define _xfld(x,y) (uint32_t)(((x)>> y##_SHIFT) & y##_MASK)
#define VTD_AWTOLEVEL(x) (((x) - 30) / VTD_STRIDE_SIZE)
#define VTD_LEVELTOAW(x) (((x) * VTD_STRIDE_SIZE) + 30)
#define DMAR_VER_REG 0x00 /* 32:Arch version supported by this IOMMU */
#define DMAR_RTADDR_REG 0x20 /* 64:Root entry table */
#define DMAR_FEDATA_REG 0x3c /* 32:Fault event interrupt data register */
#define DMAR_FEADDR_REG 0x40 /* 32:Fault event interrupt addr register */
#define DMAR_FEUADDR_REG 0x44 /* 32:Upper address register */
#define DMAR_AFLOG_REG 0x58 /* 64:Advanced Fault control */
#define DMAR_PMEN_REG 0x64 /* 32:Enable Protected Memory Region */
#define DMAR_PLMBASE_REG 0x68 /* 32:PMRR Low addr */
#define DMAR_PLMLIMIT_REG 0x6c /* 32:PMRR low limit */
#define DMAR_PHMBASE_REG 0x70 /* 64:pmrr high base addr */
#define DMAR_PHMLIMIT_REG 0x78 /* 64:pmrr high limit */
#define DMAR_ICS_REG 0x9C /* 32:Invalidation complete status register */
#define DMAR_IECTL_REG 0xa0 /* 32:Invalidation event control register */
#define DMAR_IEDATA_REG 0xa4 /* 32:Invalidation event data register */
#define DMAR_IEADDR_REG 0xa8 /* 32:Invalidation event address register */
#define DMAR_IEUADDR_REG 0xac /* 32:Invalidation event upper address register */
#define DMAR_IRTA_REG 0xb8 /* 64:Interrupt remapping table addr register */
#define DMAR_CAP_REG 0x08 /* 64:Hardware supported capabilities */
#define CAP_PI (1LL << 59)
#define CAP_FL1GP (1LL << 56)
#define CAP_DRD (1LL << 55)
#define CAP_DWD (1LL << 54)
#define CAP_MAMV_MASK 0x3F
#define CAP_MAMV_SHIFT 48LL
#define cap_mamv(x) _xfld(x,CAP_MAMV)
#define CAP_NFR_MASK 0xFF
#define CAP_NFR_SHIFT 40LL
#define cap_nfr(x) (_xfld(x,CAP_NFR) + 1)
#define CAP_PSI (1LL << 39)
#define CAP_SLLPS_MASK 0xF
#define CAP_SLLPS_SHIFT 34LL
#define cap_sllps(x) _xfld(x,CAP_SLLPS)
#define CAP_FRO_MASK 0x3FF
#define CAP_FRO_SHIFT 24LL
#define cap_fro(x) (_xfld(x,CAP_FRO) * 16)
#define CAP_ZLR (1LL << 22)
#define CAP_MGAW_MASK 0x3F
#define CAP_MGAW_SHIFT 16LL
#define cap_mgaw(x) (_xfld(x,CAP_MGAW) + 1)
#define CAP_SAGAW_MASK 0x1F
#define CAP_SAGAW_SHIFT 8LL
#define cap_sagaw(x) _xfld(x,CAP_SAGAW)
#define CAP_CM (1LL << 7)
#define CAP_PHMR (1LL << 6)
#define CAP_PLMR (1LL << 5)
#define CAP_RWBF (1LL << 4)
#define CAP_AFL (1LL << 3)
#define CAP_ND_MASK 0x7
#define CAP_ND_SHIFT 0x00
#define cap_nd(x) (16 << (((x) & CAP_ND_MASK) << 1))
#define DMAR_ECAP_REG 0x10 /* 64:Extended capabilities supported */
#define ECAP_PSS_MASK 0x1F
#define ECAP_PSS_SHIFT 35
#define ECAP_EAFS (1LL << 34)
#define ECAP_NWFS (1LL << 33)
#define ECAP_SRS (1LL << 31)
#define ECAP_ERS (1LL << 30)
#define ECAP_PRS (1LL << 29)
#define ECAP_PASID (1LL << 28)
#define ECAP_DIS (1LL << 27)
#define ECAP_NEST (1LL << 26)
#define ECAP_MTS (1LL << 25)
#define ECAP_ECS (1LL << 24)
#define ECAP_MHMV_MASK 0xF
#define ECAP_MHMV_SHIFT 0x20
#define ecap_mhmv(x) _xfld(x,ECAP_MHMV)
#define ECAP_IRO_MASK 0x3FF /* IOTLB Register */
#define ECAP_IRO_SHIFT 0x8
#define ecap_iro(x) (_xfld(x,ECAP_IRO) * 16)
#define ECAP_SC (1LL << 7) /* Snoop Control */
#define ECAP_PT (1LL << 6) /* HW Passthru */
#define ECAP_EIM (1LL << 4)
#define ECAP_IR (1LL << 3) /* Interrupt remap */
#define ECAP_DT (1LL << 2) /* Device IOTLB */
#define ECAP_QI (1LL << 1) /* Queued Invalidation */
#define ECAP_C (1LL << 0) /* Coherent cache */
#define DMAR_GCMD_REG 0x18 /* 32:Global command register */
#define GCMD_TE (1LL << 31)
#define GCMD_SRTP (1LL << 30)
#define GCMD_SFL (1LL << 29)
#define GCMD_EAFL (1LL << 28)
#define GCMD_WBF (1LL << 27)
#define GCMD_QIE (1LL << 26)
#define GCMD_IRE (1LL << 25)
#define GCMD_SIRTP (1LL << 24)
#define GCMD_CFI (1LL << 23)
#define DMAR_GSTS_REG 0x1c /* 32:Global status register */
#define GSTS_TES (1LL << 31)
#define GSTS_RTPS (1LL << 30)
#define GSTS_FLS (1LL << 29)
#define GSTS_AFLS (1LL << 28)
#define GSTS_WBFS (1LL << 27)
#define GSTS_QIES (1LL << 26)
#define GSTS_IRES (1LL << 25)
#define GSTS_IRTPS (1LL << 24)
#define GSTS_CFIS (1LL << 23)
#define DMAR_CCMD_REG 0x28 /* 64:Context command reg */
#define CCMD_ICC (1LL << 63)
#define CCMD_CIRG_MASK 0x3
#define CCMD_CIRG_SHIFT 61
#define CCMD_CIRG(x) ((uint64_t)(x) << CCMD_CIRG_SHIFT)
#define CCMD_CAIG_MASK 0x3
#define CCMD_CAIG_SHIFT 59
#define CCMD_FM_MASK 0x3
#define CCMD_FM_SHIFT 32
#define CCMD_FM(x) (((uint64_t)(x) << CCMD_FM_SHIFT))
#define CCMD_SID_MASK 0xFFFF
#define CCMD_SID_SHIFT 8
#define CCMD_SID(x) (((x) << CCMD_SID_SHIFT))
#define CCMD_DID_MASK 0xFFFF
#define CCMD_DID_SHIFT 0
#define CCMD_DID(x) (((x) << CCMD_DID_SHIFT))
#define CIG_GLOBAL CCMD_CIRG(CTX_GLOBAL)
#define CIG_DOMAIN CCMD_CIRG(CTX_DOMAIN)
#define CIG_DEVICE CCMD_CIRG(CTX_DEVICE)
#define DMAR_FSTS_REG 0x34 /* 32:Fault Status register */
#define FSTS_FRI_MASK 0xFF
#define FSTS_FRI_SHIFT 8
#define FSTS_PRO (1LL << 7)
#define FSTS_ITE (1LL << 6)
#define FSTS_ICE (1LL << 5)
#define FSTS_IQE (1LL << 4)
#define FSTS_APF (1LL << 3)
#define FSTS_APO (1LL << 2)
#define FSTS_PPF (1LL << 1)
#define FSTS_PFO (1LL << 0)
#define DMAR_FECTL_REG 0x38 /* 32:Fault control register */
#define FECTL_IM (1LL << 31)
#define FECTL_IP (1LL << 30)
#define FRCD_HI_F (1LL << (127-64))
#define FRCD_HI_T (1LL << (126-64))
#define FRCD_HI_AT_MASK 0x3
#define FRCD_HI_AT_SHIFT (124-64)
#define FRCD_HI_PV_MASK 0xFFFFF
#define FRCD_HI_PV_SHIFT (104-64)
#define FRCD_HI_FR_MASK 0xFF
#define FRCD_HI_FR_SHIFT (96-64)
#define FRCD_HI_PP (1LL << (95-64))
#define FRCD_HI_SID_MASK 0xFF
#define FRCD_HI_SID_SHIFT 0
#define FRCD_HI_BUS_SHIFT 8
#define FRCD_HI_BUS_MASK 0xFF
#define FRCD_HI_DEV_SHIFT 3
#define FRCD_HI_DEV_MASK 0x1F
#define FRCD_HI_FUN_SHIFT 0
#define FRCD_HI_FUN_MASK 0x7
#define DMAR_IOTLB_REG(x) (ecap_iro((x)->ecap) + 8)
#define DMAR_IVA_REG(x) (ecap_iro((x)->ecap) + 0)
#define DMAR_FRIH_REG(x,i) (cap_fro((x)->cap) + 16*(i) + 8)
#define DMAR_FRIL_REG(x,i) (cap_fro((x)->cap) + 16*(i) + 0)
#define IOTLB_IVT (1LL << 63)
#define IOTLB_IIRG_MASK 0x3
#define IOTLB_IIRG_SHIFT 60
#define IOTLB_IIRG(x) ((uint64_t)(x) << IOTLB_IIRG_SHIFT)
#define IOTLB_IAIG_MASK 0x3
#define IOTLB_IAIG_SHIFT 57
#define IOTLB_DR (1LL << 49)
#define IOTLB_DW (1LL << 48)
#define IOTLB_DID_MASK 0xFFFF
#define IOTLB_DID_SHIFT 32
#define IOTLB_DID(x) ((uint64_t)(x) << IOTLB_DID_SHIFT)
#define IIG_GLOBAL IOTLB_IIRG(IOTLB_GLOBAL)
#define IIG_DOMAIN IOTLB_IIRG(IOTLB_DOMAIN)
#define IIG_PAGE IOTLB_IIRG(IOTLB_PAGE)
#define DMAR_IQH_REG 0x80 /* 64:Invalidation queue head register */
#define DMAR_IQT_REG 0x88 /* 64:Invalidation queue tail register */
#define DMAR_IQA_REG 0x90 /* 64:Invalidation queue addr register */
#define IQA_QS_256 0 /* 256 entries */
#define IQA_QS_512 1 /* 512 */
#define IQA_QS_1K 2 /* 1024 */
#define IQA_QS_2K 3 /* 2048 */
#define IQA_QS_4K 4 /* 4096 */
#define IQA_QS_8K 5 /* 8192 */
#define IQA_QS_16K 6 /* 16384 */
#define IQA_QS_32K 7 /* 32768 */
/* Read-Modify-Write helpers */
static inline void iommu_rmw32(void *ov, uint32_t mask, uint32_t shift, uint32_t nv)
{
*(uint32_t *)ov &= ~(mask << shift);
*(uint32_t *)ov |= (nv & mask) << shift;
}
static inline void iommu_rmw64(void *ov, uint32_t mask, uint32_t shift, uint64_t nv)
{
*(uint64_t *)ov &= ~(mask << shift);
*(uint64_t *)ov |= (nv & mask) << shift;
}
/*
* Root Entry: one per bus (256 x 128 bit = 4k)
* 0 = Present
* 1:11 = Reserved
* 12:HAW-1 = Context Table Pointer
* HAW:63 = Reserved
* 64:127 = Reserved
*/
#define ROOT_P (1L << 0)
struct root_entry {
uint64_t lo;
uint64_t hi;
};
/* Check if root entry is valid */
static inline bool
root_entry_is_valid(struct root_entry *re)
{
return (re->lo & ROOT_P);
}
/*
* Context Entry: one per devfn (256 x 128 bit = 4k)
* 0 = Present
* 1 = Fault Processing Disable
* 2:3 = Translation Type
* 4:11 = Reserved
* 12:63 = Second Level Page Translation
* 64:66 = Address Width (# PTE levels)
* 67:70 = Ignore
* 71 = Reserved
* 72:87 = Domain ID
* 88:127 = Reserved
*/
#define CTX_P (1L << 0)
#define CTX_FPD (1L << 1)
#define CTX_T_MASK 0x3
#define CTX_T_SHIFT 2
enum {
CTX_T_MULTI,
CTX_T_IOTLB,
CTX_T_PASSTHRU
};
#define CTX_H_AW_MASK 0x7
#define CTX_H_AW_SHIFT 0
#define CTX_H_USER_MASK 0xF
#define CTX_H_USER_SHIFT 3
#define CTX_H_DID_MASK 0xFFFF
#define CTX_H_DID_SHIFT 8
struct context_entry {
uint64_t lo;
uint64_t hi;
};
/* Set fault processing enable/disable */
static inline void
context_set_fpd(struct context_entry *ce, int enable)
{
ce->lo &= ~CTX_FPD;
if (enable)
ce->lo |= CTX_FPD;
}
/* Set context entry present */
static inline void
context_set_present(struct context_entry *ce)
{
ce->lo |= CTX_P;
}
/* Set Second Level Page Table Entry PA */
static inline void
context_set_slpte(struct context_entry *ce, paddr_t slpte)
{
ce->lo &= VTD_PAGE_MASK;
ce->lo |= (slpte & ~VTD_PAGE_MASK);
}
/* Set translation type */
static inline void
context_set_translation_type(struct context_entry *ce, int tt)
{
ce->lo &= ~(CTX_T_MASK << CTX_T_SHIFT);
ce->lo |= ((tt & CTX_T_MASK) << CTX_T_SHIFT);
}
/* Set Address Width (# of Page Table levels) */
static inline void
context_set_address_width(struct context_entry *ce, int lvl)
{
ce->hi &= ~(CTX_H_AW_MASK << CTX_H_AW_SHIFT);
ce->hi |= ((lvl & CTX_H_AW_MASK) << CTX_H_AW_SHIFT);
}
/* Set domain ID */
static inline void
context_set_domain_id(struct context_entry *ce, int did)
{
ce->hi &= ~(CTX_H_DID_MASK << CTX_H_DID_SHIFT);
ce->hi |= ((did & CTX_H_DID_MASK) << CTX_H_DID_SHIFT);
}
/* Get Second Level Page Table PA */
static inline uint64_t
context_pte(struct context_entry *ce)
{
return (ce->lo & ~VTD_PAGE_MASK);
}
/* Get translation type */
static inline int
context_translation_type(struct context_entry *ce)
{
return (ce->lo >> CTX_T_SHIFT) & CTX_T_MASK;
}
/* Get domain ID */
static inline int
context_domain_id(struct context_entry *ce)
{
return (ce->hi >> CTX_H_DID_SHIFT) & CTX_H_DID_MASK;
}
/* Get Address Width */
static inline int
context_address_width(struct context_entry *ce)
{
return VTD_LEVELTOAW((ce->hi >> CTX_H_AW_SHIFT) & CTX_H_AW_MASK);
}
/* Check if context entry is valid */
static inline bool
context_entry_is_valid(struct context_entry *ce)
{
return (ce->lo & CTX_P);
}
/* User-available bits in context entry */
static inline int
context_user(struct context_entry *ce)
{
return (ce->hi >> CTX_H_USER_SHIFT) & CTX_H_USER_MASK;
}
static inline void
context_set_user(struct context_entry *ce, int v)
{
ce->hi &= ~(CTX_H_USER_MASK << CTX_H_USER_SHIFT);
ce->hi |= ((v & CTX_H_USER_MASK) << CTX_H_USER_SHIFT);
}
/*
* Fault entry
* 0..HAW-1 = Fault address
* HAW:63 = Reserved
* 64:71 = Source ID
* 96:103 = Fault Reason
* 104:123 = PV
* 124:125 = Address Translation type
* 126 = Type (0 = Read, 1 = Write)
* 127 = Fault bit
*/
struct fault_entry
{
uint64_t lo;
uint64_t hi;
};
/* PTE Entry: 512 x 64-bit = 4k */
#define PTE_P (1L << 0)
#define PTE_R 0x00
#define PTE_W (1L << 1)
#define PTE_US (1L << 2)
#define PTE_PWT (1L << 3)
#define PTE_PCD (1L << 4)
#define PTE_A (1L << 5)
#define PTE_D (1L << 6)
#define PTE_PAT (1L << 7)
#define PTE_G (1L << 8)
#define PTE_EA (1L << 10)
#define PTE_XD (1LL << 63)
/* PDE Level entry */
#define PTE_PS (1L << 7)
/* PDPE Level entry */
/* ----------------------------------------------------------------
* 5555555444444444333333333222222222111111111000000000------------
* [PML4 ->] PDPE.1GB
* [PML4 ->] PDPE.PDE -> PDE.2MB
* [PML4 ->] PDPE.PDE -> PDE -> PTE
* GAW0 = (12.20) (PTE)
* GAW1 = (21.29) (PDE)
* GAW2 = (30.38) (PDPE)
* GAW3 = (39.47) (PML4)
* GAW4 = (48.57) (n/a)
* GAW5 = (58.63) (n/a)
*/
struct pte_entry {
uint64_t val;
};
/*
* Queued Invalidation entry
* 0:3 = 01h
* 4:5 = Granularity
* 6:15 = Reserved
* 16:31 = Domain ID
* 32:47 = Source ID
* 48:49 = FM
*/
/* Invalidate Context Entry */
#define QI_CTX_DID_MASK 0xFFFF
#define QI_CTX_DID_SHIFT 16
#define QI_CTX_SID_MASK 0xFFFF
#define QI_CTX_SID_SHIFT 32
#define QI_CTX_FM_MASK 0x3
#define QI_CTX_FM_SHIFT 48
#define QI_CTX_IG_MASK 0x3
#define QI_CTX_IG_SHIFT 4
#define QI_CTX_DID(x) (((uint64_t)(x) << QI_CTX_DID_SHIFT))
#define QI_CTX_SID(x) (((uint64_t)(x) << QI_CTX_SID_SHIFT))
#define QI_CTX_FM(x) (((uint64_t)(x) << QI_CTX_FM_SHIFT))
#define QI_CTX_IG_GLOBAL (CTX_GLOBAL << QI_CTX_IG_SHIFT)
#define QI_CTX_IG_DOMAIN (CTX_DOMAIN << QI_CTX_IG_SHIFT)
#define QI_CTX_IG_DEVICE (CTX_DEVICE << QI_CTX_IG_SHIFT)
/* Invalidate IOTLB Entry */
#define QI_IOTLB_DID_MASK 0xFFFF
#define QI_IOTLB_DID_SHIFT 16
#define QI_IOTLB_IG_MASK 0x3
#define QI_IOTLB_IG_SHIFT 4
#define QI_IOTLB_DR (1LL << 6)
#define QI_IOTLB_DW (1LL << 5)
#define QI_IOTLB_DID(x) (((uint64_t)(x) << QI_IOTLB_DID_SHIFT))
#define QI_IOTLB_IG_GLOBAL (1 << QI_IOTLB_IG_SHIFT)
#define QI_IOTLB_IG_DOMAIN (2 << QI_IOTLB_IG_SHIFT)
#define QI_IOTLB_IG_PAGE (3 << QI_IOTLB_IG_SHIFT)
/* QI Commands */
#define QI_CTX 0x1
#define QI_IOTLB 0x2
#define QI_DEVTLB 0x3
#define QI_INTR 0x4
#define QI_WAIT 0x5
#define QI_EXTTLB 0x6
#define QI_PAS 0x7
#define QI_EXTDEV 0x8
struct qi_entry {
uint64_t lo;
uint64_t hi;
};
enum {
CTX_GLOBAL = 1,
CTX_DOMAIN,
CTX_DEVICE,
IOTLB_GLOBAL = 1,
IOTLB_DOMAIN,
IOTLB_PAGE,
};
enum {
VTD_FAULT_ROOT_P = 0x1, /* P field in root entry is 0 */
VTD_FAULT_CTX_P = 0x2, /* P field in context entry is 0 */
VTD_FAULT_CTX_INVAL = 0x3, /* context AW/TT/SLPPTR invalid */
VTD_FAULT_LIMIT = 0x4, /* Address is outside of MGAW */
VTD_FAULT_WRITE = 0x5, /* Address-translation fault, non-writable */
VTD_FAULT_READ = 0x6, /* Address-translation fault, non-readable */
VTD_FAULT_PTE_INVAL = 0x7, /* page table hw access error */
VTD_FAULT_ROOT_INVAL = 0x8, /* root table hw access error */
VTD_FAULT_CTX_TBL_INVAL = 0x9, /* context entry hw access error */
VTD_FAULT_ROOT_RESERVED = 0xa, /* non-zero reserved field in root entry */
VTD_FAULT_CTX_RESERVED = 0xb, /* non-zero reserved field in context entry */
VTD_FAULT_PTE_RESERVED = 0xc, /* non-zero reserved field in paging entry */
VTD_FAULT_CTX_TT = 0xd, /* invalid translation type */
};
#endif
void acpidmar_pci_hook(pci_chipset_tag_t, struct pci_attach_args *);
void dmar_ptmap(bus_dma_tag_t, bus_addr_t);
void acpidmar_sw(int);
#define __EXTRACT(v,m) (((v) >> m##_SHIFT) & m##_MASK)
|