/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (the "License"). You may not use this file except in compliance * with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright (c) 1991-2000 by Sun Microsystems, Inc. * All rights reserved. */ /* * Copyright 2012 Garrett D'Amore . All rights reserved. */ #ifndef _SYS_IOMMU_H #define _SYS_IOMMU_H #if defined(_KERNEL) && !defined(_ASM) #include #include #include #endif /* defined(_KERNEL) && !defined(_ASM) */ #ifdef __cplusplus extern "C" { #endif #ifndef _ASM /* constants for DVMA */ /* * It takes an 8byte TSB entry to map i an 8k page, so the conversion * from tsb size to dvma mapping is to multiply by 1000 or 0x400 * left shift by 10 does this */ #define IOMMU_TSB_TO_RNG 0xa #define IOMMU_TSB_SIZE_8M 0x2000 #define IOMMU_TSB_SIZE_16M 0x4000 #define IOMMU_TSB_SIZE_32M 0x8000 #define IOMMU_TSB_SIZE_64M 0x10000 #define IOMMU_TSB_SIZE_128M 0x20000 #define IOMMU_TSB_SIZE_256M 0x40000 #define IOMMU_TSB_SIZE_512M 0x80000 #define IOMMU_TSB_SIZE_1G 0x100000 #define IOMMU_PAGESIZE 0x2000 /* 8k page */ #define IOMMU_PAGEMASK 0x1fff #define IOMMU_PAGEOFFSET (IOMMU_PAGESIZE - 1) #define IOMMU_N_TTES (IOMMU_DVMA_RANGE/IOMMU_PAGESIZE) #define IOMMU_TSB_TBL_SIZE (IOMMU_N_TTES << 3) /* 8B for each entry */ #define IOMMU_PAGESHIFT 13 #define OFF_IOMMU_CTRL_REG 0x2400 #define IOMMU_CTRL_REG_SIZE (NATURAL_REG_SIZE) #define OFF_TSB_BASE_ADDR 0x2408 #define TSB_BASE_ADDR_SIZE (NATURAL_REG_SIZE) #define OFF_IOMMU_FLUSH_REG 0x2410 #define IOMMU_FLUSH_REG (NATURAL_REG_SIZE) #define OFF_IOMMU_TLB_TAG 0x4580 #define OFF_IOMMU_TLB_DATA 0x4600 #define TSB_SIZE 3 /* 64M of DVMA */ #define TSB_SIZE_SHIFT 16 #define IOMMU_TLB_ENTRIES 16 #define IOMMU_DISABLE 0 /* Turns off the IOMMU */ #define IOMMU_ENABLE 1 /* Turns on the IOMMU */ #define IOMMU_TLB_VALID 0x40000000ull #define IOMMU_DIAG_ENABLE 0x2ull /* * Bit positions in the TLB entries */ #define IOMMU_TLBTAG_WRITABLE (1 << 21) #define IOMMU_TLBTAB_STREAM (1 << 20) #define IOMMU_TLBTAG_SIZE (1 << 19) #define IOMMU_TLBTAG_VA_MASK 0x7ffff /* 19-bit vpn */ #define IOMMU_TLBTAG_VA_SHIFT 13 #define IOMMU_TLBDATA_VALID (1 << 30) #define IOMMU_TLBDATA_LOCAL (1 << 29) #define IOMMU_TLBDATA_CACHEABLE (1 << 28) #define IOMMU_TLBDATA_PA_MASK 0xfffffff /* 28-bit ppn */ #define IOMMU_TLBDATA_PA_SHIFT 13 /* * define IOPTEs */ #define IOTTE_PFN_MSK 0x1ffffffe000ull #define IOTTE_CACHE 0x10ull #define IOTTE_WRITE 0x2ull #define IOTTE_STREAM 0x1000000000000000ull #define IOTTE_INTRA 0x800000000000000ull #define IOTTE_64K_PAGE 0x2000000000000000ull #endif /* _ASM */ #define IOTTE_VALID 0x8000000000000000ull #define IOTTE_PFN_SHIFT 13 /* * IOMMU pages to bytes, and back (with and without rounding) */ #define iommu_ptob(x) ((x) << IOMMU_PAGESHIFT) #define iommu_btop(x) (((ioaddr_t)(x)) >> IOMMU_PAGESHIFT) #define iommu_btopr(x) \ ((((ioaddr_t)(x) + IOMMU_PAGEOFFSET) >> IOMMU_PAGESHIFT)) #if defined(_KERNEL) && !defined(_ASM) /* sbus nexus private dma mapping structure. */ struct dma_impl_priv { ddi_dma_impl_t mp; struct sbus_soft_state *softsp; volatile int sync_flag; uint64_t phys_sync_flag; }; extern int iommu_init(struct sbus_soft_state *, caddr_t); extern int iommu_resume_init(struct sbus_soft_state *); extern int iommu_dma_mctl(dev_info_t *, dev_info_t *, ddi_dma_handle_t, enum ddi_dma_ctlops, off_t *, size_t *, caddr_t *, uint_t); extern int iommu_dma_allochdl(dev_info_t *, dev_info_t *, ddi_dma_attr_t *, int (*waitfp)(caddr_t), caddr_t arg, ddi_dma_handle_t *); extern int iommu_dma_freehdl(dev_info_t *, dev_info_t *, ddi_dma_handle_t); extern int iommu_dma_bindhdl(dev_info_t *, dev_info_t *, ddi_dma_handle_t, struct ddi_dma_req *, ddi_dma_cookie_t *, uint_t *); extern int iommu_dma_unbindhdl(dev_info_t *, dev_info_t *, ddi_dma_handle_t); extern int iommu_dma_flush(dev_info_t *, dev_info_t *, ddi_dma_handle_t, off_t, size_t, uint_t); extern int iommu_dma_win(dev_info_t *, dev_info_t *, ddi_dma_handle_t, uint_t, off_t *, size_t *, ddi_dma_cookie_t *, uint_t *); extern void iommu_dvma_kaddr_load(ddi_dma_handle_t h, caddr_t a, uint_t len, uint_t index, ddi_dma_cookie_t *cp); extern void iommu_dvma_unload(ddi_dma_handle_t h, uint_t objindex, uint_t view); extern void iommu_dvma_sync(ddi_dma_handle_t h, uint_t objindex, uint_t view); #endif /* _KERNEL && !_ASM */ #ifdef __cplusplus } #endif #endif /* _SYS_IOMMU_H */