target-arm queue:
* Emulate the SMMUv3 (IOMMU); one will be created in the 'virt' board
    if the commandline includes "-machine iommu=smmuv3"
  * target/arm: Implement v8M VLLDM and VLSTM
  * hw/arm: Don't fail qtest due to missing SD card in -nodefaults mode
  * Some fixes to silence Coverity false-positives
  * arm: boot: set boot_info starting from first_cpu
    (fixes a technical bug not visible in practice)
  * hw/net/smc91c111: Convert away from old_mmio
  * hw/usb/tusb6010: Convert away from old_mmio
  * hw/char/cmsdk-apb-uart.c: Accept more input after character read
  * target/arm: Make MPUIR write-ignored on OMAP, StrongARM
  * hw/arm/virt: Add linux,pci-domain property
 -----BEGIN PGP SIGNATURE-----
 Version: GnuPG v1
 
 iQIcBAABCAAGBQJa7J5pAAoJEDwlJe0UNgzeHRUP/j8P4+Ftg8KStJvBiycrorVz
 EwHfkb5fRpHBH8WcjV85dLmk6AAmf9yvFowaxyU+H1Qa2xcchnIysCFXqfjkr0dN
 vloP7+mPTlb4Qhnfpa9KAVbxple91p5OBZCRBwIqTyqdTrZr5Oqw3c0fPGgVH4ze
 f+Oo6sWYQyhvpDmGE2PNLEo6VA3m8TOSOt19RpFW72WbEScAbw7Nart6WRGmzr2r
 BDizPgdQ55OmmPxv24jqy4HxDkpBQUuua74RcTQiQLmbHd2GYvEMnW4nhlntBsNo
 9TRH6xJ+7UqC+zpyE6AEJ8JtQwA47dfZfKqm1hI9AtYDCU0H220J2LHUjUfs/g3G
 2MYoF8XStPpmfBo7WHNP7gdQqOhEbMWrkHlsZ6c4waKhK/aYTLTwy37iPpxH3/iI
 MlceyexY/swUKDhssY1mcMaCszCzdYR534d1ZMZGFjrYcAF2VQcfVtPgCtAv1Icd
 FfjKzpFB0IQmFtkrNLxBoifxM3VvIvAXqktDWJuOdYa2G4MqpPktMHzFrV9f//Xn
 09P0VR46BaGnDyls4k5hiC1gyulVG5CAXUfZeBR8xsy1JrdbNNyp6Jc5ekY68Pa+
 oVDxyLWEYnzBqqtkujTIYVwvnx9KL+iW05mb5fOJ21SH9dt9a5YhFCbrZfB8roKz
 LrWbefXIP2b/4kLl8viJ
 =ZjhH
 -----END PGP SIGNATURE-----
Merge remote-tracking branch 'remotes/pmaydell/tags/pull-target-arm-20180504-1' into staging
target-arm queue:
 * Emulate the SMMUv3 (IOMMU); one will be created in the 'virt' board
   if the commandline includes "-machine iommu=smmuv3"
 * target/arm: Implement v8M VLLDM and VLSTM
 * hw/arm: Don't fail qtest due to missing SD card in -nodefaults mode
 * Some fixes to silence Coverity false-positives
 * arm: boot: set boot_info starting from first_cpu
   (fixes a technical bug not visible in practice)
 * hw/net/smc91c111: Convert away from old_mmio
 * hw/usb/tusb6010: Convert away from old_mmio
 * hw/char/cmsdk-apb-uart.c: Accept more input after character read
 * target/arm: Make MPUIR write-ignored on OMAP, StrongARM
 * hw/arm/virt: Add linux,pci-domain property
# gpg: Signature made Fri 04 May 2018 18:54:49 BST
# gpg:                using RSA key 3C2525ED14360CDE
# gpg: Good signature from "Peter Maydell <peter.maydell@linaro.org>"
# gpg:                 aka "Peter Maydell <pmaydell@gmail.com>"
# gpg:                 aka "Peter Maydell <pmaydell@chiark.greenend.org.uk>"
# Primary key fingerprint: E1A5 C593 CD41 9DE2 8E83  15CF 3C25 25ED 1436 0CDE
* remotes/pmaydell/tags/pull-target-arm-20180504-1: (24 commits)
  hw/arm/virt: Introduce the iommu option
  hw/arm/virt-acpi-build: Add smmuv3 node in IORT table
  hw/arm/virt: Add SMMUv3 to the virt board
  target/arm/kvm: Translate the MSI doorbell in kvm_arch_fixup_msi_route
  hw/arm/smmuv3: Abort on vfio or vhost case
  hw/arm/smmuv3: Implement translate callback
  hw/arm/smmuv3: Event queue recording helper
  hw/arm/smmuv3: Implement MMIO write operations
  hw/arm/smmuv3: Queue helpers
  hw/arm/smmuv3: Wired IRQ and GERROR helpers
  hw/arm/smmuv3: Skeleton
  hw/arm/smmu-common: VMSAv8-64 page table walk
  hw/arm/smmu-common: IOMMU memory region and address space setup
  hw/arm/smmu-common: smmu base device and datatypes
  target/arm: Implement v8M VLLDM and VLSTM
  hw/arm: Don't fail qtest due to missing SD card in -nodefaults mode
  target/arm: Tidy condition in disas_simd_two_reg_misc
  target/arm: Tidy conditions in handle_vec_simd_shri
  arm: boot: set boot_info starting from first_cpu
  hw/net/smc91c111: Convert away from old_mmio
  ...
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
			
			
This commit is contained in:
		
						commit
						853f8ca13c
					
				| @ -8,3 +8,4 @@ CONFIG_DDC=y | |||||||
| CONFIG_DPCD=y | CONFIG_DPCD=y | ||||||
| CONFIG_XLNX_ZYNQMP=y | CONFIG_XLNX_ZYNQMP=y | ||||||
| CONFIG_XLNX_ZYNQMP_ARM=y | CONFIG_XLNX_ZYNQMP_ARM=y | ||||||
|  | CONFIG_ARM_SMMUV3=y | ||||||
|  | |||||||
| @ -35,3 +35,4 @@ obj-$(CONFIG_MPS2) += mps2-tz.o | |||||||
| obj-$(CONFIG_MSF2) += msf2-soc.o msf2-som.o | obj-$(CONFIG_MSF2) += msf2-soc.o msf2-som.o | ||||||
| obj-$(CONFIG_IOTKIT) += iotkit.o | obj-$(CONFIG_IOTKIT) += iotkit.o | ||||||
| obj-$(CONFIG_FSL_IMX7) += fsl-imx7.o mcimx7d-sabre.o | obj-$(CONFIG_FSL_IMX7) += fsl-imx7.o mcimx7d-sabre.o | ||||||
|  | obj-$(CONFIG_ARM_SMMUV3) += smmu-common.o smmuv3.o | ||||||
|  | |||||||
| @ -1170,7 +1170,7 @@ static void arm_load_kernel_notify(Notifier *notifier, void *data) | |||||||
|     } |     } | ||||||
|     info->is_linux = is_linux; |     info->is_linux = is_linux; | ||||||
| 
 | 
 | ||||||
|     for (cs = CPU(cpu); cs; cs = CPU_NEXT(cs)) { |     for (cs = first_cpu; cs; cs = CPU_NEXT(cs)) { | ||||||
|         ARM_CPU(cs)->env.boot_info = info; |         ARM_CPU(cs)->env.boot_info = info; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -30,6 +30,7 @@ | |||||||
| #include "hw/arm/soc_dma.h" | #include "hw/arm/soc_dma.h" | ||||||
| #include "sysemu/block-backend.h" | #include "sysemu/block-backend.h" | ||||||
| #include "sysemu/blockdev.h" | #include "sysemu/blockdev.h" | ||||||
|  | #include "sysemu/qtest.h" | ||||||
| #include "qemu/range.h" | #include "qemu/range.h" | ||||||
| #include "hw/sysbus.h" | #include "hw/sysbus.h" | ||||||
| #include "qemu/cutils.h" | #include "qemu/cutils.h" | ||||||
| @ -3987,12 +3988,11 @@ struct omap_mpu_state_s *omap310_mpu_init(MemoryRegion *system_memory, | |||||||
|                                 omap_findclk(s, "dpll3")); |                                 omap_findclk(s, "dpll3")); | ||||||
| 
 | 
 | ||||||
|     dinfo = drive_get(IF_SD, 0, 0); |     dinfo = drive_get(IF_SD, 0, 0); | ||||||
|     if (!dinfo) { |     if (!dinfo && !qtest_enabled()) { | ||||||
|         error_report("missing SecureDigital device"); |         warn_report("missing SecureDigital device"); | ||||||
|         exit(1); |  | ||||||
|     } |     } | ||||||
|     s->mmc = omap_mmc_init(0xfffb7800, system_memory, |     s->mmc = omap_mmc_init(0xfffb7800, system_memory, | ||||||
|                            blk_by_legacy_dinfo(dinfo), |                            dinfo ? blk_by_legacy_dinfo(dinfo) : NULL, | ||||||
|                            qdev_get_gpio_in(s->ih[1], OMAP_INT_OQN), |                            qdev_get_gpio_in(s->ih[1], OMAP_INT_OQN), | ||||||
|                            &s->drq[OMAP_DMA_MMC_TX], |                            &s->drq[OMAP_DMA_MMC_TX], | ||||||
|                     omap_findclk(s, "mmc_ck")); |                     omap_findclk(s, "mmc_ck")); | ||||||
|  | |||||||
| @ -25,6 +25,7 @@ | |||||||
| #include "cpu.h" | #include "cpu.h" | ||||||
| #include "sysemu/block-backend.h" | #include "sysemu/block-backend.h" | ||||||
| #include "sysemu/blockdev.h" | #include "sysemu/blockdev.h" | ||||||
|  | #include "sysemu/qtest.h" | ||||||
| #include "hw/boards.h" | #include "hw/boards.h" | ||||||
| #include "hw/hw.h" | #include "hw/hw.h" | ||||||
| #include "hw/arm/arm.h" | #include "hw/arm/arm.h" | ||||||
| @ -2486,12 +2487,11 @@ struct omap_mpu_state_s *omap2420_mpu_init(MemoryRegion *sysmem, | |||||||
|                              s->drq[OMAP24XX_DMA_GPMC]); |                              s->drq[OMAP24XX_DMA_GPMC]); | ||||||
| 
 | 
 | ||||||
|     dinfo = drive_get(IF_SD, 0, 0); |     dinfo = drive_get(IF_SD, 0, 0); | ||||||
|     if (!dinfo) { |     if (!dinfo && !qtest_enabled()) { | ||||||
|         error_report("missing SecureDigital device"); |         warn_report("missing SecureDigital device"); | ||||||
|         exit(1); |  | ||||||
|     } |     } | ||||||
|     s->mmc = omap2_mmc_init(omap_l4tao(s->l4, 9), |     s->mmc = omap2_mmc_init(omap_l4tao(s->l4, 9), | ||||||
|                     blk_by_legacy_dinfo(dinfo), |                     dinfo ? blk_by_legacy_dinfo(dinfo) : NULL, | ||||||
|                     qdev_get_gpio_in(s->ih[0], OMAP_INT_24XX_MMC_IRQ), |                     qdev_get_gpio_in(s->ih[0], OMAP_INT_24XX_MMC_IRQ), | ||||||
|                     &s->drq[OMAP24XX_DMA_MMC1_TX], |                     &s->drq[OMAP24XX_DMA_MMC1_TX], | ||||||
|                     omap_findclk(s, "mmc_fclk"), omap_findclk(s, "mmc_iclk")); |                     omap_findclk(s, "mmc_fclk"), omap_findclk(s, "mmc_iclk")); | ||||||
|  | |||||||
| @ -21,6 +21,7 @@ | |||||||
| #include "chardev/char-fe.h" | #include "chardev/char-fe.h" | ||||||
| #include "sysemu/block-backend.h" | #include "sysemu/block-backend.h" | ||||||
| #include "sysemu/blockdev.h" | #include "sysemu/blockdev.h" | ||||||
|  | #include "sysemu/qtest.h" | ||||||
| #include "qemu/cutils.h" | #include "qemu/cutils.h" | ||||||
| 
 | 
 | ||||||
| static struct { | static struct { | ||||||
| @ -2095,12 +2096,11 @@ PXA2xxState *pxa270_init(MemoryRegion *address_space, | |||||||
|     s->gpio = pxa2xx_gpio_init(0x40e00000, s->cpu, s->pic, 121); |     s->gpio = pxa2xx_gpio_init(0x40e00000, s->cpu, s->pic, 121); | ||||||
| 
 | 
 | ||||||
|     dinfo = drive_get(IF_SD, 0, 0); |     dinfo = drive_get(IF_SD, 0, 0); | ||||||
|     if (!dinfo) { |     if (!dinfo && !qtest_enabled()) { | ||||||
|         error_report("missing SecureDigital device"); |         warn_report("missing SecureDigital device"); | ||||||
|         exit(1); |  | ||||||
|     } |     } | ||||||
|     s->mmc = pxa2xx_mmci_init(address_space, 0x41100000, |     s->mmc = pxa2xx_mmci_init(address_space, 0x41100000, | ||||||
|                     blk_by_legacy_dinfo(dinfo), |                     dinfo ? blk_by_legacy_dinfo(dinfo) : NULL, | ||||||
|                     qdev_get_gpio_in(s->pic, PXA2XX_PIC_MMC), |                     qdev_get_gpio_in(s->pic, PXA2XX_PIC_MMC), | ||||||
|                     qdev_get_gpio_in(s->dma, PXA2XX_RX_RQ_MMCI), |                     qdev_get_gpio_in(s->dma, PXA2XX_RX_RQ_MMCI), | ||||||
|                     qdev_get_gpio_in(s->dma, PXA2XX_TX_RQ_MMCI)); |                     qdev_get_gpio_in(s->dma, PXA2XX_TX_RQ_MMCI)); | ||||||
| @ -2220,12 +2220,11 @@ PXA2xxState *pxa255_init(MemoryRegion *address_space, unsigned int sdram_size) | |||||||
|     s->gpio = pxa2xx_gpio_init(0x40e00000, s->cpu, s->pic, 85); |     s->gpio = pxa2xx_gpio_init(0x40e00000, s->cpu, s->pic, 85); | ||||||
| 
 | 
 | ||||||
|     dinfo = drive_get(IF_SD, 0, 0); |     dinfo = drive_get(IF_SD, 0, 0); | ||||||
|     if (!dinfo) { |     if (!dinfo && !qtest_enabled()) { | ||||||
|         error_report("missing SecureDigital device"); |         warn_report("missing SecureDigital device"); | ||||||
|         exit(1); |  | ||||||
|     } |     } | ||||||
|     s->mmc = pxa2xx_mmci_init(address_space, 0x41100000, |     s->mmc = pxa2xx_mmci_init(address_space, 0x41100000, | ||||||
|                     blk_by_legacy_dinfo(dinfo), |                     dinfo ? blk_by_legacy_dinfo(dinfo) : NULL, | ||||||
|                     qdev_get_gpio_in(s->pic, PXA2XX_PIC_MMC), |                     qdev_get_gpio_in(s->pic, PXA2XX_PIC_MMC), | ||||||
|                     qdev_get_gpio_in(s->dma, PXA2XX_RX_RQ_MMCI), |                     qdev_get_gpio_in(s->dma, PXA2XX_RX_RQ_MMCI), | ||||||
|                     qdev_get_gpio_in(s->dma, PXA2XX_TX_RQ_MMCI)); |                     qdev_get_gpio_in(s->dma, PXA2XX_TX_RQ_MMCI)); | ||||||
|  | |||||||
							
								
								
									
										372
									
								
								hw/arm/smmu-common.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										372
									
								
								hw/arm/smmu-common.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,372 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2014-2016 Broadcom Corporation | ||||||
|  |  * Copyright (c) 2017 Red Hat, Inc. | ||||||
|  |  * Written by Prem Mallappa, Eric Auger | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License version 2 as | ||||||
|  |  * published by the Free Software Foundation. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * Author: Prem Mallappa <pmallapp@broadcom.com> | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include "qemu/osdep.h" | ||||||
|  | #include "sysemu/sysemu.h" | ||||||
|  | #include "exec/address-spaces.h" | ||||||
|  | #include "trace.h" | ||||||
|  | #include "exec/target_page.h" | ||||||
|  | #include "qom/cpu.h" | ||||||
|  | #include "hw/qdev-properties.h" | ||||||
|  | #include "qapi/error.h" | ||||||
|  | 
 | ||||||
|  | #include "qemu/error-report.h" | ||||||
|  | #include "hw/arm/smmu-common.h" | ||||||
|  | #include "smmu-internal.h" | ||||||
|  | 
 | ||||||
|  | /* VMSAv8-64 Translation */ | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * get_pte - Get the content of a page table entry located at | ||||||
|  |  * @base_addr[@index] | ||||||
|  |  */ | ||||||
|  | static int get_pte(dma_addr_t baseaddr, uint32_t index, uint64_t *pte, | ||||||
|  |                    SMMUPTWEventInfo *info) | ||||||
|  | { | ||||||
|  |     int ret; | ||||||
|  |     dma_addr_t addr = baseaddr + index * sizeof(*pte); | ||||||
|  | 
 | ||||||
|  |     /* TODO: guarantee 64-bit single-copy atomicity */ | ||||||
|  |     ret = dma_memory_read(&address_space_memory, addr, | ||||||
|  |                           (uint8_t *)pte, sizeof(*pte)); | ||||||
|  | 
 | ||||||
|  |     if (ret != MEMTX_OK) { | ||||||
|  |         info->type = SMMU_PTW_ERR_WALK_EABT; | ||||||
|  |         info->addr = addr; | ||||||
|  |         return -EINVAL; | ||||||
|  |     } | ||||||
|  |     trace_smmu_get_pte(baseaddr, index, addr, *pte); | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* VMSAv8-64 Translation Table Format Descriptor Decoding */ | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * get_page_pte_address - returns the L3 descriptor output address, | ||||||
|  |  * ie. the page frame | ||||||
|  |  * ARM ARM spec: Figure D4-17 VMSAv8-64 level 3 descriptor format | ||||||
|  |  */ | ||||||
|  | static inline hwaddr get_page_pte_address(uint64_t pte, int granule_sz) | ||||||
|  | { | ||||||
|  |     return PTE_ADDRESS(pte, granule_sz); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * get_table_pte_address - return table descriptor output address, | ||||||
|  |  * ie. address of next level table | ||||||
|  |  * ARM ARM Figure D4-16 VMSAv8-64 level0, level1, and level 2 descriptor formats | ||||||
|  |  */ | ||||||
|  | static inline hwaddr get_table_pte_address(uint64_t pte, int granule_sz) | ||||||
|  | { | ||||||
|  |     return PTE_ADDRESS(pte, granule_sz); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * get_block_pte_address - return block descriptor output address and block size | ||||||
|  |  * ARM ARM Figure D4-16 VMSAv8-64 level0, level1, and level 2 descriptor formats | ||||||
|  |  */ | ||||||
|  | static inline hwaddr get_block_pte_address(uint64_t pte, int level, | ||||||
|  |                                            int granule_sz, uint64_t *bsz) | ||||||
|  | { | ||||||
|  |     int n = (granule_sz - 3) * (4 - level) + 3; | ||||||
|  | 
 | ||||||
|  |     *bsz = 1 << n; | ||||||
|  |     return PTE_ADDRESS(pte, n); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | SMMUTransTableInfo *select_tt(SMMUTransCfg *cfg, dma_addr_t iova) | ||||||
|  | { | ||||||
|  |     bool tbi = extract64(iova, 55, 1) ? TBI1(cfg->tbi) : TBI0(cfg->tbi); | ||||||
|  |     uint8_t tbi_byte = tbi * 8; | ||||||
|  | 
 | ||||||
|  |     if (cfg->tt[0].tsz && | ||||||
|  |         !extract64(iova, 64 - cfg->tt[0].tsz, cfg->tt[0].tsz - tbi_byte)) { | ||||||
|  |         /* there is a ttbr0 region and we are in it (high bits all zero) */ | ||||||
|  |         return &cfg->tt[0]; | ||||||
|  |     } else if (cfg->tt[1].tsz && | ||||||
|  |            !extract64(iova, 64 - cfg->tt[1].tsz, cfg->tt[1].tsz - tbi_byte)) { | ||||||
|  |         /* there is a ttbr1 region and we are in it (high bits all one) */ | ||||||
|  |         return &cfg->tt[1]; | ||||||
|  |     } else if (!cfg->tt[0].tsz) { | ||||||
|  |         /* ttbr0 region is "everything not in the ttbr1 region" */ | ||||||
|  |         return &cfg->tt[0]; | ||||||
|  |     } else if (!cfg->tt[1].tsz) { | ||||||
|  |         /* ttbr1 region is "everything not in the ttbr0 region" */ | ||||||
|  |         return &cfg->tt[1]; | ||||||
|  |     } | ||||||
|  |     /* in the gap between the two regions, this is a Translation fault */ | ||||||
|  |     return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * smmu_ptw_64 - VMSAv8-64 Walk of the page tables for a given IOVA | ||||||
|  |  * @cfg: translation config | ||||||
|  |  * @iova: iova to translate | ||||||
|  |  * @perm: access type | ||||||
|  |  * @tlbe: IOMMUTLBEntry (out) | ||||||
|  |  * @info: handle to an error info | ||||||
|  |  * | ||||||
|  |  * Return 0 on success, < 0 on error. In case of error, @info is filled | ||||||
|  |  * and tlbe->perm is set to IOMMU_NONE. | ||||||
|  |  * Upon success, @tlbe is filled with translated_addr and entry | ||||||
|  |  * permission rights. | ||||||
|  |  */ | ||||||
|  | static int smmu_ptw_64(SMMUTransCfg *cfg, | ||||||
|  |                        dma_addr_t iova, IOMMUAccessFlags perm, | ||||||
|  |                        IOMMUTLBEntry *tlbe, SMMUPTWEventInfo *info) | ||||||
|  | { | ||||||
|  |     dma_addr_t baseaddr, indexmask; | ||||||
|  |     int stage = cfg->stage; | ||||||
|  |     SMMUTransTableInfo *tt = select_tt(cfg, iova); | ||||||
|  |     uint8_t level, granule_sz, inputsize, stride; | ||||||
|  | 
 | ||||||
|  |     if (!tt || tt->disabled) { | ||||||
|  |         info->type = SMMU_PTW_ERR_TRANSLATION; | ||||||
|  |         goto error; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     granule_sz = tt->granule_sz; | ||||||
|  |     stride = granule_sz - 3; | ||||||
|  |     inputsize = 64 - tt->tsz; | ||||||
|  |     level = 4 - (inputsize - 4) / stride; | ||||||
|  |     indexmask = (1ULL << (inputsize - (stride * (4 - level)))) - 1; | ||||||
|  |     baseaddr = extract64(tt->ttb, 0, 48); | ||||||
|  |     baseaddr &= ~indexmask; | ||||||
|  | 
 | ||||||
|  |     tlbe->iova = iova; | ||||||
|  |     tlbe->addr_mask = (1 << granule_sz) - 1; | ||||||
|  | 
 | ||||||
|  |     while (level <= 3) { | ||||||
|  |         uint64_t subpage_size = 1ULL << level_shift(level, granule_sz); | ||||||
|  |         uint64_t mask = subpage_size - 1; | ||||||
|  |         uint32_t offset = iova_level_offset(iova, inputsize, level, granule_sz); | ||||||
|  |         uint64_t pte; | ||||||
|  |         dma_addr_t pte_addr = baseaddr + offset * sizeof(pte); | ||||||
|  |         uint8_t ap; | ||||||
|  | 
 | ||||||
|  |         if (get_pte(baseaddr, offset, &pte, info)) { | ||||||
|  |                 goto error; | ||||||
|  |         } | ||||||
|  |         trace_smmu_ptw_level(level, iova, subpage_size, | ||||||
|  |                              baseaddr, offset, pte); | ||||||
|  | 
 | ||||||
|  |         if (is_invalid_pte(pte) || is_reserved_pte(pte, level)) { | ||||||
|  |             trace_smmu_ptw_invalid_pte(stage, level, baseaddr, | ||||||
|  |                                        pte_addr, offset, pte); | ||||||
|  |             info->type = SMMU_PTW_ERR_TRANSLATION; | ||||||
|  |             goto error; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (is_page_pte(pte, level)) { | ||||||
|  |             uint64_t gpa = get_page_pte_address(pte, granule_sz); | ||||||
|  | 
 | ||||||
|  |             ap = PTE_AP(pte); | ||||||
|  |             if (is_permission_fault(ap, perm)) { | ||||||
|  |                 info->type = SMMU_PTW_ERR_PERMISSION; | ||||||
|  |                 goto error; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             tlbe->translated_addr = gpa + (iova & mask); | ||||||
|  |             tlbe->perm = PTE_AP_TO_PERM(ap); | ||||||
|  |             trace_smmu_ptw_page_pte(stage, level, iova, | ||||||
|  |                                     baseaddr, pte_addr, pte, gpa); | ||||||
|  |             return 0; | ||||||
|  |         } | ||||||
|  |         if (is_block_pte(pte, level)) { | ||||||
|  |             uint64_t block_size; | ||||||
|  |             hwaddr gpa = get_block_pte_address(pte, level, granule_sz, | ||||||
|  |                                                &block_size); | ||||||
|  | 
 | ||||||
|  |             ap = PTE_AP(pte); | ||||||
|  |             if (is_permission_fault(ap, perm)) { | ||||||
|  |                 info->type = SMMU_PTW_ERR_PERMISSION; | ||||||
|  |                 goto error; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             trace_smmu_ptw_block_pte(stage, level, baseaddr, | ||||||
|  |                                      pte_addr, pte, iova, gpa, | ||||||
|  |                                      block_size >> 20); | ||||||
|  | 
 | ||||||
|  |             tlbe->translated_addr = gpa + (iova & mask); | ||||||
|  |             tlbe->perm = PTE_AP_TO_PERM(ap); | ||||||
|  |             return 0; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /* table pte */ | ||||||
|  |         ap = PTE_APTABLE(pte); | ||||||
|  | 
 | ||||||
|  |         if (is_permission_fault(ap, perm)) { | ||||||
|  |             info->type = SMMU_PTW_ERR_PERMISSION; | ||||||
|  |             goto error; | ||||||
|  |         } | ||||||
|  |         baseaddr = get_table_pte_address(pte, granule_sz); | ||||||
|  |         level++; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     info->type = SMMU_PTW_ERR_TRANSLATION; | ||||||
|  | 
 | ||||||
|  | error: | ||||||
|  |     tlbe->perm = IOMMU_NONE; | ||||||
|  |     return -EINVAL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * smmu_ptw - Walk the page tables for an IOVA, according to @cfg | ||||||
|  |  * | ||||||
|  |  * @cfg: translation configuration | ||||||
|  |  * @iova: iova to translate | ||||||
|  |  * @perm: tentative access type | ||||||
|  |  * @tlbe: returned entry | ||||||
|  |  * @info: ptw event handle | ||||||
|  |  * | ||||||
|  |  * return 0 on success | ||||||
|  |  */ | ||||||
|  | inline int smmu_ptw(SMMUTransCfg *cfg, dma_addr_t iova, IOMMUAccessFlags perm, | ||||||
|  |              IOMMUTLBEntry *tlbe, SMMUPTWEventInfo *info) | ||||||
|  | { | ||||||
|  |     if (!cfg->aa64) { | ||||||
|  |         /*
 | ||||||
|  |          * This code path is not entered as we check this while decoding | ||||||
|  |          * the configuration data in the derived SMMU model. | ||||||
|  |          */ | ||||||
|  |         g_assert_not_reached(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return smmu_ptw_64(cfg, iova, perm, tlbe, info); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * The bus number is used for lookup when SID based invalidation occurs. | ||||||
|  |  * In that case we lazily populate the SMMUPciBus array from the bus hash | ||||||
|  |  * table. At the time the SMMUPciBus is created (smmu_find_add_as), the bus | ||||||
|  |  * numbers may not be always initialized yet. | ||||||
|  |  */ | ||||||
|  | SMMUPciBus *smmu_find_smmu_pcibus(SMMUState *s, uint8_t bus_num) | ||||||
|  | { | ||||||
|  |     SMMUPciBus *smmu_pci_bus = s->smmu_pcibus_by_bus_num[bus_num]; | ||||||
|  | 
 | ||||||
|  |     if (!smmu_pci_bus) { | ||||||
|  |         GHashTableIter iter; | ||||||
|  | 
 | ||||||
|  |         g_hash_table_iter_init(&iter, s->smmu_pcibus_by_busptr); | ||||||
|  |         while (g_hash_table_iter_next(&iter, NULL, (void **)&smmu_pci_bus)) { | ||||||
|  |             if (pci_bus_num(smmu_pci_bus->bus) == bus_num) { | ||||||
|  |                 s->smmu_pcibus_by_bus_num[bus_num] = smmu_pci_bus; | ||||||
|  |                 return smmu_pci_bus; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return smmu_pci_bus; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static AddressSpace *smmu_find_add_as(PCIBus *bus, void *opaque, int devfn) | ||||||
|  | { | ||||||
|  |     SMMUState *s = opaque; | ||||||
|  |     SMMUPciBus *sbus = g_hash_table_lookup(s->smmu_pcibus_by_busptr, bus); | ||||||
|  |     SMMUDevice *sdev; | ||||||
|  | 
 | ||||||
|  |     if (!sbus) { | ||||||
|  |         sbus = g_malloc0(sizeof(SMMUPciBus) + | ||||||
|  |                          sizeof(SMMUDevice *) * SMMU_PCI_DEVFN_MAX); | ||||||
|  |         sbus->bus = bus; | ||||||
|  |         g_hash_table_insert(s->smmu_pcibus_by_busptr, bus, sbus); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     sdev = sbus->pbdev[devfn]; | ||||||
|  |     if (!sdev) { | ||||||
|  |         char *name = g_strdup_printf("%s-%d-%d", | ||||||
|  |                                      s->mrtypename, | ||||||
|  |                                      pci_bus_num(bus), devfn); | ||||||
|  |         sdev = sbus->pbdev[devfn] = g_new0(SMMUDevice, 1); | ||||||
|  | 
 | ||||||
|  |         sdev->smmu = s; | ||||||
|  |         sdev->bus = bus; | ||||||
|  |         sdev->devfn = devfn; | ||||||
|  | 
 | ||||||
|  |         memory_region_init_iommu(&sdev->iommu, sizeof(sdev->iommu), | ||||||
|  |                                  s->mrtypename, | ||||||
|  |                                  OBJECT(s), name, 1ULL << SMMU_MAX_VA_BITS); | ||||||
|  |         address_space_init(&sdev->as, | ||||||
|  |                            MEMORY_REGION(&sdev->iommu), name); | ||||||
|  |         trace_smmu_add_mr(name); | ||||||
|  |         g_free(name); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return &sdev->as; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void smmu_base_realize(DeviceState *dev, Error **errp) | ||||||
|  | { | ||||||
|  |     SMMUState *s = ARM_SMMU(dev); | ||||||
|  |     SMMUBaseClass *sbc = ARM_SMMU_GET_CLASS(dev); | ||||||
|  |     Error *local_err = NULL; | ||||||
|  | 
 | ||||||
|  |     sbc->parent_realize(dev, &local_err); | ||||||
|  |     if (local_err) { | ||||||
|  |         error_propagate(errp, local_err); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     s->smmu_pcibus_by_busptr = g_hash_table_new(NULL, NULL); | ||||||
|  | 
 | ||||||
|  |     if (s->primary_bus) { | ||||||
|  |         pci_setup_iommu(s->primary_bus, smmu_find_add_as, s); | ||||||
|  |     } else { | ||||||
|  |         error_setg(errp, "SMMU is not attached to any PCI bus!"); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void smmu_base_reset(DeviceState *dev) | ||||||
|  | { | ||||||
|  |     /* will be filled later on */ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static Property smmu_dev_properties[] = { | ||||||
|  |     DEFINE_PROP_UINT8("bus_num", SMMUState, bus_num, 0), | ||||||
|  |     DEFINE_PROP_LINK("primary-bus", SMMUState, primary_bus, "PCI", PCIBus *), | ||||||
|  |     DEFINE_PROP_END_OF_LIST(), | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void smmu_base_class_init(ObjectClass *klass, void *data) | ||||||
|  | { | ||||||
|  |     DeviceClass *dc = DEVICE_CLASS(klass); | ||||||
|  |     SMMUBaseClass *sbc = ARM_SMMU_CLASS(klass); | ||||||
|  | 
 | ||||||
|  |     dc->props = smmu_dev_properties; | ||||||
|  |     device_class_set_parent_realize(dc, smmu_base_realize, | ||||||
|  |                                     &sbc->parent_realize); | ||||||
|  |     dc->reset = smmu_base_reset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static const TypeInfo smmu_base_info = { | ||||||
|  |     .name          = TYPE_ARM_SMMU, | ||||||
|  |     .parent        = TYPE_SYS_BUS_DEVICE, | ||||||
|  |     .instance_size = sizeof(SMMUState), | ||||||
|  |     .class_data    = NULL, | ||||||
|  |     .class_size    = sizeof(SMMUBaseClass), | ||||||
|  |     .class_init    = smmu_base_class_init, | ||||||
|  |     .abstract      = true, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void smmu_base_register_types(void) | ||||||
|  | { | ||||||
|  |     type_register_static(&smmu_base_info); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type_init(smmu_base_register_types) | ||||||
|  | 
 | ||||||
							
								
								
									
										99
									
								
								hw/arm/smmu-internal.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								hw/arm/smmu-internal.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,99 @@ | |||||||
|  | /*
 | ||||||
|  |  * ARM SMMU support - Internal API | ||||||
|  |  * | ||||||
|  |  * Copyright (c) 2017 Red Hat, Inc. | ||||||
|  |  * Copyright (C) 2014-2016 Broadcom Corporation | ||||||
|  |  * Written by Prem Mallappa, Eric Auger | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License version 2 as | ||||||
|  |  * published by the Free Software Foundation. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef HW_ARM_SMMU_INTERNAL_H | ||||||
|  | #define HW_ARM_SMMU_INTERNAL_H | ||||||
|  | 
 | ||||||
|  | #define TBI0(tbi) ((tbi) & 0x1) | ||||||
|  | #define TBI1(tbi) ((tbi) & 0x2 >> 1) | ||||||
|  | 
 | ||||||
|  | /* PTE Manipulation */ | ||||||
|  | 
 | ||||||
|  | #define ARM_LPAE_PTE_TYPE_SHIFT         0 | ||||||
|  | #define ARM_LPAE_PTE_TYPE_MASK          0x3 | ||||||
|  | 
 | ||||||
|  | #define ARM_LPAE_PTE_TYPE_BLOCK         1 | ||||||
|  | #define ARM_LPAE_PTE_TYPE_TABLE         3 | ||||||
|  | 
 | ||||||
|  | #define ARM_LPAE_L3_PTE_TYPE_RESERVED   1 | ||||||
|  | #define ARM_LPAE_L3_PTE_TYPE_PAGE       3 | ||||||
|  | 
 | ||||||
|  | #define ARM_LPAE_PTE_VALID              (1 << 0) | ||||||
|  | 
 | ||||||
|  | #define PTE_ADDRESS(pte, shift) \ | ||||||
|  |     (extract64(pte, shift, 47 - shift + 1) << shift) | ||||||
|  | 
 | ||||||
|  | #define is_invalid_pte(pte) (!(pte & ARM_LPAE_PTE_VALID)) | ||||||
|  | 
 | ||||||
|  | #define is_reserved_pte(pte, level)                                      \ | ||||||
|  |     ((level == 3) &&                                                     \ | ||||||
|  |      ((pte & ARM_LPAE_PTE_TYPE_MASK) == ARM_LPAE_L3_PTE_TYPE_RESERVED)) | ||||||
|  | 
 | ||||||
|  | #define is_block_pte(pte, level)                                         \ | ||||||
|  |     ((level < 3) &&                                                      \ | ||||||
|  |      ((pte & ARM_LPAE_PTE_TYPE_MASK) == ARM_LPAE_PTE_TYPE_BLOCK)) | ||||||
|  | 
 | ||||||
|  | #define is_table_pte(pte, level)                                        \ | ||||||
|  |     ((level < 3) &&                                                     \ | ||||||
|  |      ((pte & ARM_LPAE_PTE_TYPE_MASK) == ARM_LPAE_PTE_TYPE_TABLE)) | ||||||
|  | 
 | ||||||
|  | #define is_page_pte(pte, level)                                         \ | ||||||
|  |     ((level == 3) &&                                                    \ | ||||||
|  |      ((pte & ARM_LPAE_PTE_TYPE_MASK) == ARM_LPAE_L3_PTE_TYPE_PAGE)) | ||||||
|  | 
 | ||||||
|  | /* access permissions */ | ||||||
|  | 
 | ||||||
|  | #define PTE_AP(pte) \ | ||||||
|  |     (extract64(pte, 6, 2)) | ||||||
|  | 
 | ||||||
|  | #define PTE_APTABLE(pte) \ | ||||||
|  |     (extract64(pte, 61, 2)) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * TODO: At the moment all transactions are considered as privileged (EL1) | ||||||
|  |  * as IOMMU translation callback does not pass user/priv attributes. | ||||||
|  |  */ | ||||||
|  | #define is_permission_fault(ap, perm) \ | ||||||
|  |     (((perm) & IOMMU_WO) && ((ap) & 0x2)) | ||||||
|  | 
 | ||||||
|  | #define PTE_AP_TO_PERM(ap) \ | ||||||
|  |     (IOMMU_ACCESS_FLAG(true, !((ap) & 0x2))) | ||||||
|  | 
 | ||||||
|  | /* Level Indexing */ | ||||||
|  | 
 | ||||||
|  | static inline int level_shift(int level, int granule_sz) | ||||||
|  | { | ||||||
|  |     return granule_sz + (3 - level) * (granule_sz - 3); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline uint64_t level_page_mask(int level, int granule_sz) | ||||||
|  | { | ||||||
|  |     return ~(MAKE_64BIT_MASK(0, level_shift(level, granule_sz))); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline | ||||||
|  | uint64_t iova_level_offset(uint64_t iova, int inputsize, | ||||||
|  |                            int level, int gsz) | ||||||
|  | { | ||||||
|  |     return ((iova & MAKE_64BIT_MASK(0, inputsize)) >> level_shift(level, gsz)) & | ||||||
|  |             MAKE_64BIT_MASK(0, gsz - 3); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
							
								
								
									
										621
									
								
								hw/arm/smmuv3-internal.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										621
									
								
								hw/arm/smmuv3-internal.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,621 @@ | |||||||
|  | /*
 | ||||||
|  |  * ARM SMMUv3 support - Internal API | ||||||
|  |  * | ||||||
|  |  * Copyright (C) 2014-2016 Broadcom Corporation | ||||||
|  |  * Copyright (c) 2017 Red Hat, Inc. | ||||||
|  |  * Written by Prem Mallappa, Eric Auger | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License version 2 as | ||||||
|  |  * published by the Free Software Foundation. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef HW_ARM_SMMU_V3_INTERNAL_H | ||||||
|  | #define HW_ARM_SMMU_V3_INTERNAL_H | ||||||
|  | 
 | ||||||
|  | #include "hw/arm/smmu-common.h" | ||||||
|  | 
 | ||||||
|  | /* MMIO Registers */ | ||||||
|  | 
 | ||||||
|  | REG32(IDR0,                0x0) | ||||||
|  |     FIELD(IDR0, S1P,         1 , 1) | ||||||
|  |     FIELD(IDR0, TTF,         2 , 2) | ||||||
|  |     FIELD(IDR0, COHACC,      4 , 1) | ||||||
|  |     FIELD(IDR0, ASID16,      12, 1) | ||||||
|  |     FIELD(IDR0, TTENDIAN,    21, 2) | ||||||
|  |     FIELD(IDR0, STALL_MODEL, 24, 2) | ||||||
|  |     FIELD(IDR0, TERM_MODEL,  26, 1) | ||||||
|  |     FIELD(IDR0, STLEVEL,     27, 2) | ||||||
|  | 
 | ||||||
|  | REG32(IDR1,                0x4) | ||||||
|  |     FIELD(IDR1, SIDSIZE,      0 , 6) | ||||||
|  |     FIELD(IDR1, EVENTQS,      16, 5) | ||||||
|  |     FIELD(IDR1, CMDQS,        21, 5) | ||||||
|  | 
 | ||||||
|  | #define SMMU_IDR1_SIDSIZE 16 | ||||||
|  | #define SMMU_CMDQS   19 | ||||||
|  | #define SMMU_EVENTQS 19 | ||||||
|  | 
 | ||||||
|  | REG32(IDR2,                0x8) | ||||||
|  | REG32(IDR3,                0xc) | ||||||
|  | REG32(IDR4,                0x10) | ||||||
|  | REG32(IDR5,                0x14) | ||||||
|  |      FIELD(IDR5, OAS,         0, 3); | ||||||
|  |      FIELD(IDR5, GRAN4K,      4, 1); | ||||||
|  |      FIELD(IDR5, GRAN16K,     5, 1); | ||||||
|  |      FIELD(IDR5, GRAN64K,     6, 1); | ||||||
|  | 
 | ||||||
|  | #define SMMU_IDR5_OAS 4 | ||||||
|  | 
 | ||||||
|  | REG32(IIDR,                0x1c) | ||||||
|  | REG32(CR0,                 0x20) | ||||||
|  |     FIELD(CR0, SMMU_ENABLE,   0, 1) | ||||||
|  |     FIELD(CR0, EVENTQEN,      2, 1) | ||||||
|  |     FIELD(CR0, CMDQEN,        3, 1) | ||||||
|  | 
 | ||||||
|  | #define SMMU_CR0_RESERVED 0xFFFFFC20 | ||||||
|  | 
 | ||||||
|  | REG32(CR0ACK,              0x24) | ||||||
|  | REG32(CR1,                 0x28) | ||||||
|  | REG32(CR2,                 0x2c) | ||||||
|  | REG32(STATUSR,             0x40) | ||||||
|  | REG32(IRQ_CTRL,            0x50) | ||||||
|  |     FIELD(IRQ_CTRL, GERROR_IRQEN,        0, 1) | ||||||
|  |     FIELD(IRQ_CTRL, PRI_IRQEN,           1, 1) | ||||||
|  |     FIELD(IRQ_CTRL, EVENTQ_IRQEN,        2, 1) | ||||||
|  | 
 | ||||||
|  | REG32(IRQ_CTRL_ACK,        0x54) | ||||||
|  | REG32(GERROR,              0x60) | ||||||
|  |     FIELD(GERROR, CMDQ_ERR,           0, 1) | ||||||
|  |     FIELD(GERROR, EVENTQ_ABT_ERR,     2, 1) | ||||||
|  |     FIELD(GERROR, PRIQ_ABT_ERR,       3, 1) | ||||||
|  |     FIELD(GERROR, MSI_CMDQ_ABT_ERR,   4, 1) | ||||||
|  |     FIELD(GERROR, MSI_EVENTQ_ABT_ERR, 5, 1) | ||||||
|  |     FIELD(GERROR, MSI_PRIQ_ABT_ERR,   6, 1) | ||||||
|  |     FIELD(GERROR, MSI_GERROR_ABT_ERR, 7, 1) | ||||||
|  |     FIELD(GERROR, MSI_SFM_ERR,        8, 1) | ||||||
|  | 
 | ||||||
|  | REG32(GERRORN,             0x64) | ||||||
|  | 
 | ||||||
|  | #define A_GERROR_IRQ_CFG0  0x68 /* 64b */ | ||||||
|  | REG32(GERROR_IRQ_CFG1, 0x70) | ||||||
|  | REG32(GERROR_IRQ_CFG2, 0x74) | ||||||
|  | 
 | ||||||
|  | #define A_STRTAB_BASE      0x80 /* 64b */ | ||||||
|  | 
 | ||||||
|  | #define SMMU_BASE_ADDR_MASK 0xffffffffffe0 | ||||||
|  | 
 | ||||||
|  | REG32(STRTAB_BASE_CFG,     0x88) | ||||||
|  |     FIELD(STRTAB_BASE_CFG, FMT,      16, 2) | ||||||
|  |     FIELD(STRTAB_BASE_CFG, SPLIT,    6 , 5) | ||||||
|  |     FIELD(STRTAB_BASE_CFG, LOG2SIZE, 0 , 6) | ||||||
|  | 
 | ||||||
|  | #define A_CMDQ_BASE        0x90 /* 64b */ | ||||||
|  | REG32(CMDQ_PROD,           0x98) | ||||||
|  | REG32(CMDQ_CONS,           0x9c) | ||||||
|  |     FIELD(CMDQ_CONS, ERR, 24, 7) | ||||||
|  | 
 | ||||||
|  | #define A_EVENTQ_BASE      0xa0 /* 64b */ | ||||||
|  | REG32(EVENTQ_PROD,         0xa8) | ||||||
|  | REG32(EVENTQ_CONS,         0xac) | ||||||
|  | 
 | ||||||
|  | #define A_EVENTQ_IRQ_CFG0  0xb0 /* 64b */ | ||||||
|  | REG32(EVENTQ_IRQ_CFG1,     0xb8) | ||||||
|  | REG32(EVENTQ_IRQ_CFG2,     0xbc) | ||||||
|  | 
 | ||||||
|  | #define A_IDREGS           0xfd0 | ||||||
|  | 
 | ||||||
|  | static inline int smmu_enabled(SMMUv3State *s) | ||||||
|  | { | ||||||
|  |     return FIELD_EX32(s->cr[0], CR0, SMMU_ENABLE); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Command Queue Entry */ | ||||||
|  | typedef struct Cmd { | ||||||
|  |     uint32_t word[4]; | ||||||
|  | } Cmd; | ||||||
|  | 
 | ||||||
|  | /* Event Queue Entry */ | ||||||
|  | typedef struct Evt  { | ||||||
|  |     uint32_t word[8]; | ||||||
|  | } Evt; | ||||||
|  | 
 | ||||||
|  | static inline uint32_t smmuv3_idreg(int regoffset) | ||||||
|  | { | ||||||
|  |     /*
 | ||||||
|  |      * Return the value of the Primecell/Corelink ID registers at the | ||||||
|  |      * specified offset from the first ID register. | ||||||
|  |      * These value indicate an ARM implementation of MMU600 p1 | ||||||
|  |      */ | ||||||
|  |     static const uint8_t smmuv3_ids[] = { | ||||||
|  |         0x04, 0, 0, 0, 0x84, 0xB4, 0xF0, 0x10, 0x0D, 0xF0, 0x05, 0xB1 | ||||||
|  |     }; | ||||||
|  |     return smmuv3_ids[regoffset / 4]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool smmuv3_eventq_irq_enabled(SMMUv3State *s) | ||||||
|  | { | ||||||
|  |     return FIELD_EX32(s->irq_ctrl, IRQ_CTRL, EVENTQ_IRQEN); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool smmuv3_gerror_irq_enabled(SMMUv3State *s) | ||||||
|  | { | ||||||
|  |     return FIELD_EX32(s->irq_ctrl, IRQ_CTRL, GERROR_IRQEN); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Queue Handling */ | ||||||
|  | 
 | ||||||
|  | #define Q_BASE(q)          ((q)->base & SMMU_BASE_ADDR_MASK) | ||||||
|  | #define WRAP_MASK(q)       (1 << (q)->log2size) | ||||||
|  | #define INDEX_MASK(q)      (((1 << (q)->log2size)) - 1) | ||||||
|  | #define WRAP_INDEX_MASK(q) ((1 << ((q)->log2size + 1)) - 1) | ||||||
|  | 
 | ||||||
|  | #define Q_CONS(q) ((q)->cons & INDEX_MASK(q)) | ||||||
|  | #define Q_PROD(q) ((q)->prod & INDEX_MASK(q)) | ||||||
|  | 
 | ||||||
|  | #define Q_CONS_ENTRY(q)  (Q_BASE(q) + (q)->entry_size * Q_CONS(q)) | ||||||
|  | #define Q_PROD_ENTRY(q)  (Q_BASE(q) + (q)->entry_size * Q_PROD(q)) | ||||||
|  | 
 | ||||||
|  | #define Q_CONS_WRAP(q) (((q)->cons & WRAP_MASK(q)) >> (q)->log2size) | ||||||
|  | #define Q_PROD_WRAP(q) (((q)->prod & WRAP_MASK(q)) >> (q)->log2size) | ||||||
|  | 
 | ||||||
|  | static inline bool smmuv3_q_full(SMMUQueue *q) | ||||||
|  | { | ||||||
|  |     return ((q->cons ^ q->prod) & WRAP_INDEX_MASK(q)) == WRAP_MASK(q); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool smmuv3_q_empty(SMMUQueue *q) | ||||||
|  | { | ||||||
|  |     return (q->cons & WRAP_INDEX_MASK(q)) == (q->prod & WRAP_INDEX_MASK(q)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void queue_prod_incr(SMMUQueue *q) | ||||||
|  | { | ||||||
|  |     q->prod = (q->prod + 1) & WRAP_INDEX_MASK(q); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void queue_cons_incr(SMMUQueue *q) | ||||||
|  | { | ||||||
|  |     /*
 | ||||||
|  |      * We have to use deposit for the CONS registers to preserve | ||||||
|  |      * the ERR field in the high bits. | ||||||
|  |      */ | ||||||
|  |     q->cons = deposit32(q->cons, 0, q->log2size + 1, q->cons + 1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool smmuv3_cmdq_enabled(SMMUv3State *s) | ||||||
|  | { | ||||||
|  |     return FIELD_EX32(s->cr[0], CR0, CMDQEN); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline bool smmuv3_eventq_enabled(SMMUv3State *s) | ||||||
|  | { | ||||||
|  |     return FIELD_EX32(s->cr[0], CR0, EVENTQEN); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void smmu_write_cmdq_err(SMMUv3State *s, uint32_t err_type) | ||||||
|  | { | ||||||
|  |     s->cmdq.cons = FIELD_DP32(s->cmdq.cons, CMDQ_CONS, ERR, err_type); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Commands */ | ||||||
|  | 
 | ||||||
|  | typedef enum SMMUCommandType { | ||||||
|  |     SMMU_CMD_NONE            = 0x00, | ||||||
|  |     SMMU_CMD_PREFETCH_CONFIG       , | ||||||
|  |     SMMU_CMD_PREFETCH_ADDR, | ||||||
|  |     SMMU_CMD_CFGI_STE, | ||||||
|  |     SMMU_CMD_CFGI_STE_RANGE, | ||||||
|  |     SMMU_CMD_CFGI_CD, | ||||||
|  |     SMMU_CMD_CFGI_CD_ALL, | ||||||
|  |     SMMU_CMD_CFGI_ALL, | ||||||
|  |     SMMU_CMD_TLBI_NH_ALL     = 0x10, | ||||||
|  |     SMMU_CMD_TLBI_NH_ASID, | ||||||
|  |     SMMU_CMD_TLBI_NH_VA, | ||||||
|  |     SMMU_CMD_TLBI_NH_VAA, | ||||||
|  |     SMMU_CMD_TLBI_EL3_ALL    = 0x18, | ||||||
|  |     SMMU_CMD_TLBI_EL3_VA     = 0x1a, | ||||||
|  |     SMMU_CMD_TLBI_EL2_ALL    = 0x20, | ||||||
|  |     SMMU_CMD_TLBI_EL2_ASID, | ||||||
|  |     SMMU_CMD_TLBI_EL2_VA, | ||||||
|  |     SMMU_CMD_TLBI_EL2_VAA, | ||||||
|  |     SMMU_CMD_TLBI_S12_VMALL  = 0x28, | ||||||
|  |     SMMU_CMD_TLBI_S2_IPA     = 0x2a, | ||||||
|  |     SMMU_CMD_TLBI_NSNH_ALL   = 0x30, | ||||||
|  |     SMMU_CMD_ATC_INV         = 0x40, | ||||||
|  |     SMMU_CMD_PRI_RESP, | ||||||
|  |     SMMU_CMD_RESUME          = 0x44, | ||||||
|  |     SMMU_CMD_STALL_TERM, | ||||||
|  |     SMMU_CMD_SYNC, | ||||||
|  | } SMMUCommandType; | ||||||
|  | 
 | ||||||
|  | static const char *cmd_stringify[] = { | ||||||
|  |     [SMMU_CMD_PREFETCH_CONFIG] = "SMMU_CMD_PREFETCH_CONFIG", | ||||||
|  |     [SMMU_CMD_PREFETCH_ADDR]   = "SMMU_CMD_PREFETCH_ADDR", | ||||||
|  |     [SMMU_CMD_CFGI_STE]        = "SMMU_CMD_CFGI_STE", | ||||||
|  |     [SMMU_CMD_CFGI_STE_RANGE]  = "SMMU_CMD_CFGI_STE_RANGE", | ||||||
|  |     [SMMU_CMD_CFGI_CD]         = "SMMU_CMD_CFGI_CD", | ||||||
|  |     [SMMU_CMD_CFGI_CD_ALL]     = "SMMU_CMD_CFGI_CD_ALL", | ||||||
|  |     [SMMU_CMD_CFGI_ALL]        = "SMMU_CMD_CFGI_ALL", | ||||||
|  |     [SMMU_CMD_TLBI_NH_ALL]     = "SMMU_CMD_TLBI_NH_ALL", | ||||||
|  |     [SMMU_CMD_TLBI_NH_ASID]    = "SMMU_CMD_TLBI_NH_ASID", | ||||||
|  |     [SMMU_CMD_TLBI_NH_VA]      = "SMMU_CMD_TLBI_NH_VA", | ||||||
|  |     [SMMU_CMD_TLBI_NH_VAA]     = "SMMU_CMD_TLBI_NH_VAA", | ||||||
|  |     [SMMU_CMD_TLBI_EL3_ALL]    = "SMMU_CMD_TLBI_EL3_ALL", | ||||||
|  |     [SMMU_CMD_TLBI_EL3_VA]     = "SMMU_CMD_TLBI_EL3_VA", | ||||||
|  |     [SMMU_CMD_TLBI_EL2_ALL]    = "SMMU_CMD_TLBI_EL2_ALL", | ||||||
|  |     [SMMU_CMD_TLBI_EL2_ASID]   = "SMMU_CMD_TLBI_EL2_ASID", | ||||||
|  |     [SMMU_CMD_TLBI_EL2_VA]     = "SMMU_CMD_TLBI_EL2_VA", | ||||||
|  |     [SMMU_CMD_TLBI_EL2_VAA]    = "SMMU_CMD_TLBI_EL2_VAA", | ||||||
|  |     [SMMU_CMD_TLBI_S12_VMALL]  = "SMMU_CMD_TLBI_S12_VMALL", | ||||||
|  |     [SMMU_CMD_TLBI_S2_IPA]     = "SMMU_CMD_TLBI_S2_IPA", | ||||||
|  |     [SMMU_CMD_TLBI_NSNH_ALL]   = "SMMU_CMD_TLBI_NSNH_ALL", | ||||||
|  |     [SMMU_CMD_ATC_INV]         = "SMMU_CMD_ATC_INV", | ||||||
|  |     [SMMU_CMD_PRI_RESP]        = "SMMU_CMD_PRI_RESP", | ||||||
|  |     [SMMU_CMD_RESUME]          = "SMMU_CMD_RESUME", | ||||||
|  |     [SMMU_CMD_STALL_TERM]      = "SMMU_CMD_STALL_TERM", | ||||||
|  |     [SMMU_CMD_SYNC]            = "SMMU_CMD_SYNC", | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static inline const char *smmu_cmd_string(SMMUCommandType type) | ||||||
|  | { | ||||||
|  |     if (type > SMMU_CMD_NONE && type < ARRAY_SIZE(cmd_stringify)) { | ||||||
|  |         return cmd_stringify[type] ? cmd_stringify[type] : "UNKNOWN"; | ||||||
|  |     } else { | ||||||
|  |         return "INVALID"; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* CMDQ fields */ | ||||||
|  | 
 | ||||||
|  | typedef enum { | ||||||
|  |     SMMU_CERROR_NONE = 0, | ||||||
|  |     SMMU_CERROR_ILL, | ||||||
|  |     SMMU_CERROR_ABT, | ||||||
|  |     SMMU_CERROR_ATC_INV_SYNC, | ||||||
|  | } SMMUCmdError; | ||||||
|  | 
 | ||||||
|  | enum { /* Command completion notification */ | ||||||
|  |     CMD_SYNC_SIG_NONE, | ||||||
|  |     CMD_SYNC_SIG_IRQ, | ||||||
|  |     CMD_SYNC_SIG_SEV, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define CMD_TYPE(x)         extract32((x)->word[0], 0 , 8) | ||||||
|  | #define CMD_SSEC(x)         extract32((x)->word[0], 10, 1) | ||||||
|  | #define CMD_SSV(x)          extract32((x)->word[0], 11, 1) | ||||||
|  | #define CMD_RESUME_AC(x)    extract32((x)->word[0], 12, 1) | ||||||
|  | #define CMD_RESUME_AB(x)    extract32((x)->word[0], 13, 1) | ||||||
|  | #define CMD_SYNC_CS(x)      extract32((x)->word[0], 12, 2) | ||||||
|  | #define CMD_SSID(x)         extract32((x)->word[0], 12, 20) | ||||||
|  | #define CMD_SID(x)          ((x)->word[1]) | ||||||
|  | #define CMD_VMID(x)         extract32((x)->word[1], 0 , 16) | ||||||
|  | #define CMD_ASID(x)         extract32((x)->word[1], 16, 16) | ||||||
|  | #define CMD_RESUME_STAG(x)  extract32((x)->word[2], 0 , 16) | ||||||
|  | #define CMD_RESP(x)         extract32((x)->word[2], 11, 2) | ||||||
|  | #define CMD_LEAF(x)         extract32((x)->word[2], 0 , 1) | ||||||
|  | #define CMD_STE_RANGE(x)    extract32((x)->word[2], 0 , 5) | ||||||
|  | #define CMD_ADDR(x) ({                                        \ | ||||||
|  |             uint64_t high = (uint64_t)(x)->word[3];           \ | ||||||
|  |             uint64_t low = extract32((x)->word[2], 12, 20);    \ | ||||||
|  |             uint64_t addr = high << 32 | (low << 12);         \ | ||||||
|  |             addr;                                             \ | ||||||
|  |         }) | ||||||
|  | 
 | ||||||
|  | #define SMMU_FEATURE_2LVL_STE (1 << 0) | ||||||
|  | 
 | ||||||
|  | /* Events */ | ||||||
|  | 
 | ||||||
|  | typedef enum SMMUEventType { | ||||||
|  |     SMMU_EVT_OK                 = 0x00, | ||||||
|  |     SMMU_EVT_F_UUT                    , | ||||||
|  |     SMMU_EVT_C_BAD_STREAMID           , | ||||||
|  |     SMMU_EVT_F_STE_FETCH              , | ||||||
|  |     SMMU_EVT_C_BAD_STE                , | ||||||
|  |     SMMU_EVT_F_BAD_ATS_TREQ           , | ||||||
|  |     SMMU_EVT_F_STREAM_DISABLED        , | ||||||
|  |     SMMU_EVT_F_TRANS_FORBIDDEN        , | ||||||
|  |     SMMU_EVT_C_BAD_SUBSTREAMID        , | ||||||
|  |     SMMU_EVT_F_CD_FETCH               , | ||||||
|  |     SMMU_EVT_C_BAD_CD                 , | ||||||
|  |     SMMU_EVT_F_WALK_EABT              , | ||||||
|  |     SMMU_EVT_F_TRANSLATION      = 0x10, | ||||||
|  |     SMMU_EVT_F_ADDR_SIZE              , | ||||||
|  |     SMMU_EVT_F_ACCESS                 , | ||||||
|  |     SMMU_EVT_F_PERMISSION             , | ||||||
|  |     SMMU_EVT_F_TLB_CONFLICT     = 0x20, | ||||||
|  |     SMMU_EVT_F_CFG_CONFLICT           , | ||||||
|  |     SMMU_EVT_E_PAGE_REQ         = 0x24, | ||||||
|  | } SMMUEventType; | ||||||
|  | 
 | ||||||
|  | static const char *event_stringify[] = { | ||||||
|  |     [SMMU_EVT_OK]                       = "SMMU_EVT_OK", | ||||||
|  |     [SMMU_EVT_F_UUT]                    = "SMMU_EVT_F_UUT", | ||||||
|  |     [SMMU_EVT_C_BAD_STREAMID]           = "SMMU_EVT_C_BAD_STREAMID", | ||||||
|  |     [SMMU_EVT_F_STE_FETCH]              = "SMMU_EVT_F_STE_FETCH", | ||||||
|  |     [SMMU_EVT_C_BAD_STE]                = "SMMU_EVT_C_BAD_STE", | ||||||
|  |     [SMMU_EVT_F_BAD_ATS_TREQ]           = "SMMU_EVT_F_BAD_ATS_TREQ", | ||||||
|  |     [SMMU_EVT_F_STREAM_DISABLED]        = "SMMU_EVT_F_STREAM_DISABLED", | ||||||
|  |     [SMMU_EVT_F_TRANS_FORBIDDEN]        = "SMMU_EVT_F_TRANS_FORBIDDEN", | ||||||
|  |     [SMMU_EVT_C_BAD_SUBSTREAMID]        = "SMMU_EVT_C_BAD_SUBSTREAMID", | ||||||
|  |     [SMMU_EVT_F_CD_FETCH]               = "SMMU_EVT_F_CD_FETCH", | ||||||
|  |     [SMMU_EVT_C_BAD_CD]                 = "SMMU_EVT_C_BAD_CD", | ||||||
|  |     [SMMU_EVT_F_WALK_EABT]              = "SMMU_EVT_F_WALK_EABT", | ||||||
|  |     [SMMU_EVT_F_TRANSLATION]            = "SMMU_EVT_F_TRANSLATION", | ||||||
|  |     [SMMU_EVT_F_ADDR_SIZE]              = "SMMU_EVT_F_ADDR_SIZE", | ||||||
|  |     [SMMU_EVT_F_ACCESS]                 = "SMMU_EVT_F_ACCESS", | ||||||
|  |     [SMMU_EVT_F_PERMISSION]             = "SMMU_EVT_F_PERMISSION", | ||||||
|  |     [SMMU_EVT_F_TLB_CONFLICT]           = "SMMU_EVT_F_TLB_CONFLICT", | ||||||
|  |     [SMMU_EVT_F_CFG_CONFLICT]           = "SMMU_EVT_F_CFG_CONFLICT", | ||||||
|  |     [SMMU_EVT_E_PAGE_REQ]               = "SMMU_EVT_E_PAGE_REQ", | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static inline const char *smmu_event_string(SMMUEventType type) | ||||||
|  | { | ||||||
|  |     if (type < ARRAY_SIZE(event_stringify)) { | ||||||
|  |         return event_stringify[type] ? event_stringify[type] : "UNKNOWN"; | ||||||
|  |     } else { | ||||||
|  |         return "INVALID"; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*  Encode an event record */ | ||||||
|  | typedef struct SMMUEventInfo { | ||||||
|  |     SMMUEventType type; | ||||||
|  |     uint32_t sid; | ||||||
|  |     bool recorded; | ||||||
|  |     bool record_trans_faults; | ||||||
|  |     union { | ||||||
|  |         struct { | ||||||
|  |             uint32_t ssid; | ||||||
|  |             bool ssv; | ||||||
|  |             dma_addr_t addr; | ||||||
|  |             bool rnw; | ||||||
|  |             bool pnu; | ||||||
|  |             bool ind; | ||||||
|  |        } f_uut; | ||||||
|  |        struct SSIDInfo { | ||||||
|  |             uint32_t ssid; | ||||||
|  |             bool ssv; | ||||||
|  |        } c_bad_streamid; | ||||||
|  |        struct SSIDAddrInfo { | ||||||
|  |             uint32_t ssid; | ||||||
|  |             bool ssv; | ||||||
|  |             dma_addr_t addr; | ||||||
|  |        } f_ste_fetch; | ||||||
|  |        struct SSIDInfo c_bad_ste; | ||||||
|  |        struct { | ||||||
|  |             dma_addr_t addr; | ||||||
|  |             bool rnw; | ||||||
|  |        } f_transl_forbidden; | ||||||
|  |        struct { | ||||||
|  |             uint32_t ssid; | ||||||
|  |        } c_bad_substream; | ||||||
|  |        struct SSIDAddrInfo f_cd_fetch; | ||||||
|  |        struct SSIDInfo c_bad_cd; | ||||||
|  |        struct FullInfo { | ||||||
|  |             bool stall; | ||||||
|  |             uint16_t stag; | ||||||
|  |             uint32_t ssid; | ||||||
|  |             bool ssv; | ||||||
|  |             bool s2; | ||||||
|  |             dma_addr_t addr; | ||||||
|  |             bool rnw; | ||||||
|  |             bool pnu; | ||||||
|  |             bool ind; | ||||||
|  |             uint8_t class; | ||||||
|  |             dma_addr_t addr2; | ||||||
|  |        } f_walk_eabt; | ||||||
|  |        struct FullInfo f_translation; | ||||||
|  |        struct FullInfo f_addr_size; | ||||||
|  |        struct FullInfo f_access; | ||||||
|  |        struct FullInfo f_permission; | ||||||
|  |        struct SSIDInfo f_cfg_conflict; | ||||||
|  |        /**
 | ||||||
|  |         * not supported yet: | ||||||
|  |         * F_BAD_ATS_TREQ | ||||||
|  |         * F_BAD_ATS_TREQ | ||||||
|  |         * F_TLB_CONFLICT | ||||||
|  |         * E_PAGE_REQUEST | ||||||
|  |         * IMPDEF_EVENTn | ||||||
|  |         */ | ||||||
|  |     } u; | ||||||
|  | } SMMUEventInfo; | ||||||
|  | 
 | ||||||
|  | /* EVTQ fields */ | ||||||
|  | 
 | ||||||
|  | #define EVT_Q_OVERFLOW        (1 << 31) | ||||||
|  | 
 | ||||||
|  | #define EVT_SET_TYPE(x, v)              deposit32((x)->word[0], 0 , 8 , v) | ||||||
|  | #define EVT_SET_SSV(x, v)               deposit32((x)->word[0], 11, 1 , v) | ||||||
|  | #define EVT_SET_SSID(x, v)              deposit32((x)->word[0], 12, 20, v) | ||||||
|  | #define EVT_SET_SID(x, v)               ((x)->word[1] = v) | ||||||
|  | #define EVT_SET_STAG(x, v)              deposit32((x)->word[2], 0 , 16, v) | ||||||
|  | #define EVT_SET_STALL(x, v)             deposit32((x)->word[2], 31, 1 , v) | ||||||
|  | #define EVT_SET_PNU(x, v)               deposit32((x)->word[3], 1 , 1 , v) | ||||||
|  | #define EVT_SET_IND(x, v)               deposit32((x)->word[3], 2 , 1 , v) | ||||||
|  | #define EVT_SET_RNW(x, v)               deposit32((x)->word[3], 3 , 1 , v) | ||||||
|  | #define EVT_SET_S2(x, v)                deposit32((x)->word[3], 7 , 1 , v) | ||||||
|  | #define EVT_SET_CLASS(x, v)             deposit32((x)->word[3], 8 , 2 , v) | ||||||
|  | #define EVT_SET_ADDR(x, addr)                             \ | ||||||
|  |     do {                                                  \ | ||||||
|  |             (x)->word[5] = (uint32_t)(addr >> 32);        \ | ||||||
|  |             (x)->word[4] = (uint32_t)(addr & 0xffffffff); \ | ||||||
|  |     } while (0) | ||||||
|  | #define EVT_SET_ADDR2(x, addr)                            \ | ||||||
|  |     do {                                                  \ | ||||||
|  |             deposit32((x)->word[7], 3, 29, addr >> 16);   \ | ||||||
|  |             deposit32((x)->word[7], 0, 16, addr & 0xffff);\ | ||||||
|  |     } while (0) | ||||||
|  | 
 | ||||||
|  | void smmuv3_record_event(SMMUv3State *s, SMMUEventInfo *event); | ||||||
|  | 
 | ||||||
|  | /* Configuration Data */ | ||||||
|  | 
 | ||||||
|  | /* STE Level 1 Descriptor */ | ||||||
|  | typedef struct STEDesc { | ||||||
|  |     uint32_t word[2]; | ||||||
|  | } STEDesc; | ||||||
|  | 
 | ||||||
|  | /* CD Level 1 Descriptor */ | ||||||
|  | typedef struct CDDesc { | ||||||
|  |     uint32_t word[2]; | ||||||
|  | } CDDesc; | ||||||
|  | 
 | ||||||
|  | /* Stream Table Entry(STE) */ | ||||||
|  | typedef struct STE { | ||||||
|  |     uint32_t word[16]; | ||||||
|  | } STE; | ||||||
|  | 
 | ||||||
|  | /* Context Descriptor(CD) */ | ||||||
|  | typedef struct CD { | ||||||
|  |     uint32_t word[16]; | ||||||
|  | } CD; | ||||||
|  | 
 | ||||||
|  | /* STE fields */ | ||||||
|  | 
 | ||||||
|  | #define STE_VALID(x)   extract32((x)->word[0], 0, 1) | ||||||
|  | 
 | ||||||
|  | #define STE_CONFIG(x)  extract32((x)->word[0], 1, 3) | ||||||
|  | #define STE_CFG_S1_ENABLED(config) (config & 0x1) | ||||||
|  | #define STE_CFG_S2_ENABLED(config) (config & 0x2) | ||||||
|  | #define STE_CFG_ABORT(config)      (!(config & 0x4)) | ||||||
|  | #define STE_CFG_BYPASS(config)     (config == 0x4) | ||||||
|  | 
 | ||||||
|  | #define STE_S1FMT(x)       extract32((x)->word[0], 4 , 2) | ||||||
|  | #define STE_S1CDMAX(x)     extract32((x)->word[1], 27, 5) | ||||||
|  | #define STE_S1STALLD(x)    extract32((x)->word[2], 27, 1) | ||||||
|  | #define STE_EATS(x)        extract32((x)->word[2], 28, 2) | ||||||
|  | #define STE_STRW(x)        extract32((x)->word[2], 30, 2) | ||||||
|  | #define STE_S2VMID(x)      extract32((x)->word[4], 0 , 16) | ||||||
|  | #define STE_S2T0SZ(x)      extract32((x)->word[5], 0 , 6) | ||||||
|  | #define STE_S2SL0(x)       extract32((x)->word[5], 6 , 2) | ||||||
|  | #define STE_S2TG(x)        extract32((x)->word[5], 14, 2) | ||||||
|  | #define STE_S2PS(x)        extract32((x)->word[5], 16, 3) | ||||||
|  | #define STE_S2AA64(x)      extract32((x)->word[5], 19, 1) | ||||||
|  | #define STE_S2HD(x)        extract32((x)->word[5], 24, 1) | ||||||
|  | #define STE_S2HA(x)        extract32((x)->word[5], 25, 1) | ||||||
|  | #define STE_S2S(x)         extract32((x)->word[5], 26, 1) | ||||||
|  | #define STE_CTXPTR(x)                                           \ | ||||||
|  |     ({                                                          \ | ||||||
|  |         unsigned long addr;                                     \ | ||||||
|  |         addr = (uint64_t)extract32((x)->word[1], 0, 16) << 32;  \ | ||||||
|  |         addr |= (uint64_t)((x)->word[0] & 0xffffffc0);          \ | ||||||
|  |         addr;                                                   \ | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  | #define STE_S2TTB(x)                                            \ | ||||||
|  |     ({                                                          \ | ||||||
|  |         unsigned long addr;                                     \ | ||||||
|  |         addr = (uint64_t)extract32((x)->word[7], 0, 16) << 32;  \ | ||||||
|  |         addr |= (uint64_t)((x)->word[6] & 0xfffffff0);          \ | ||||||
|  |         addr;                                                   \ | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  | static inline int oas2bits(int oas_field) | ||||||
|  | { | ||||||
|  |     switch (oas_field) { | ||||||
|  |     case 0: | ||||||
|  |         return 32; | ||||||
|  |     case 1: | ||||||
|  |         return 36; | ||||||
|  |     case 2: | ||||||
|  |         return 40; | ||||||
|  |     case 3: | ||||||
|  |         return 42; | ||||||
|  |     case 4: | ||||||
|  |         return 44; | ||||||
|  |     case 5: | ||||||
|  |         return 48; | ||||||
|  |     } | ||||||
|  |     return -1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int pa_range(STE *ste) | ||||||
|  | { | ||||||
|  |     int oas_field = MIN(STE_S2PS(ste), SMMU_IDR5_OAS); | ||||||
|  | 
 | ||||||
|  |     if (!STE_S2AA64(ste)) { | ||||||
|  |         return 40; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return oas2bits(oas_field); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define MAX_PA(ste) ((1 << pa_range(ste)) - 1) | ||||||
|  | 
 | ||||||
|  | /* CD fields */ | ||||||
|  | 
 | ||||||
|  | #define CD_VALID(x)   extract32((x)->word[0], 30, 1) | ||||||
|  | #define CD_ASID(x)    extract32((x)->word[1], 16, 16) | ||||||
|  | #define CD_TTB(x, sel)                                      \ | ||||||
|  |     ({                                                      \ | ||||||
|  |         uint64_t hi, lo;                                    \ | ||||||
|  |         hi = extract32((x)->word[(sel) * 2 + 3], 0, 19);    \ | ||||||
|  |         hi <<= 32;                                          \ | ||||||
|  |         lo = (x)->word[(sel) * 2 + 2] & ~0xfULL;            \ | ||||||
|  |         hi | lo;                                            \ | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  | #define CD_TSZ(x, sel)   extract32((x)->word[0], (16 * (sel)) + 0, 6) | ||||||
|  | #define CD_TG(x, sel)    extract32((x)->word[0], (16 * (sel)) + 6, 2) | ||||||
|  | #define CD_EPD(x, sel)   extract32((x)->word[0], (16 * (sel)) + 14, 1) | ||||||
|  | #define CD_ENDI(x)       extract32((x)->word[0], 15, 1) | ||||||
|  | #define CD_IPS(x)        extract32((x)->word[1], 0 , 3) | ||||||
|  | #define CD_TBI(x)        extract32((x)->word[1], 6 , 2) | ||||||
|  | #define CD_HD(x)         extract32((x)->word[1], 10 , 1) | ||||||
|  | #define CD_HA(x)         extract32((x)->word[1], 11 , 1) | ||||||
|  | #define CD_S(x)          extract32((x)->word[1], 12, 1) | ||||||
|  | #define CD_R(x)          extract32((x)->word[1], 13, 1) | ||||||
|  | #define CD_A(x)          extract32((x)->word[1], 14, 1) | ||||||
|  | #define CD_AARCH64(x)    extract32((x)->word[1], 9 , 1) | ||||||
|  | 
 | ||||||
|  | #define CDM_VALID(x)    ((x)->word[0] & 0x1) | ||||||
|  | 
 | ||||||
|  | static inline int is_cd_valid(SMMUv3State *s, STE *ste, CD *cd) | ||||||
|  | { | ||||||
|  |     return CD_VALID(cd); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * tg2granule - Decodes the CD translation granule size field according | ||||||
|  |  * to the ttbr in use | ||||||
|  |  * @bits: TG0/1 fields | ||||||
|  |  * @ttbr: ttbr index in use | ||||||
|  |  */ | ||||||
|  | static inline int tg2granule(int bits, int ttbr) | ||||||
|  | { | ||||||
|  |     switch (bits) { | ||||||
|  |     case 0: | ||||||
|  |         return ttbr ? 0  : 12; | ||||||
|  |     case 1: | ||||||
|  |         return ttbr ? 14 : 16; | ||||||
|  |     case 2: | ||||||
|  |         return ttbr ? 12 : 14; | ||||||
|  |     case 3: | ||||||
|  |         return ttbr ? 16 :  0; | ||||||
|  |     default: | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline uint64_t l1std_l2ptr(STEDesc *desc) | ||||||
|  | { | ||||||
|  |     uint64_t hi, lo; | ||||||
|  | 
 | ||||||
|  |     hi = desc->word[1]; | ||||||
|  |     lo = desc->word[0] & ~0x1fULL; | ||||||
|  |     return hi << 32 | lo; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define L1STD_SPAN(stm) (extract32((stm)->word[0], 0, 4)) | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
							
								
								
									
										1191
									
								
								hw/arm/smmuv3.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1191
									
								
								hw/arm/smmuv3.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -2,3 +2,40 @@ | |||||||
| 
 | 
 | ||||||
| # hw/arm/virt-acpi-build.c | # hw/arm/virt-acpi-build.c | ||||||
| virt_acpi_setup(void) "No fw cfg or ACPI disabled. Bailing out." | virt_acpi_setup(void) "No fw cfg or ACPI disabled. Bailing out." | ||||||
|  | 
 | ||||||
|  | # hw/arm/smmu-common.c | ||||||
|  | smmu_add_mr(const char *name) "%s" | ||||||
|  | smmu_page_walk(int stage, uint64_t baseaddr, int first_level, uint64_t start, uint64_t end) "stage=%d, baseaddr=0x%"PRIx64", first level=%d, start=0x%"PRIx64", end=0x%"PRIx64 | ||||||
|  | smmu_lookup_table(int level, uint64_t baseaddr, int granule_sz, uint64_t start, uint64_t end, int flags, uint64_t subpage_size) "level=%d baseaddr=0x%"PRIx64" granule=%d, start=0x%"PRIx64" end=0x%"PRIx64" flags=%d subpage_size=0x%"PRIx64 | ||||||
|  | smmu_ptw_level(int level, uint64_t iova, size_t subpage_size, uint64_t baseaddr, uint32_t offset, uint64_t pte) "level=%d iova=0x%"PRIx64" subpage_sz=0x%zx baseaddr=0x%"PRIx64" offset=%d => pte=0x%"PRIx64 | ||||||
|  | smmu_ptw_invalid_pte(int stage, int level, uint64_t baseaddr, uint64_t pteaddr, uint32_t offset, uint64_t pte) "stage=%d level=%d base@=0x%"PRIx64" pte@=0x%"PRIx64" offset=%d pte=0x%"PRIx64 | ||||||
|  | smmu_ptw_page_pte(int stage, int level,  uint64_t iova, uint64_t baseaddr, uint64_t pteaddr, uint64_t pte, uint64_t address) "stage=%d level=%d iova=0x%"PRIx64" base@=0x%"PRIx64" pte@=0x%"PRIx64" pte=0x%"PRIx64" page address = 0x%"PRIx64 | ||||||
|  | smmu_ptw_block_pte(int stage, int level, uint64_t baseaddr, uint64_t pteaddr, uint64_t pte, uint64_t iova, uint64_t gpa, int bsize_mb) "stage=%d level=%d base@=0x%"PRIx64" pte@=0x%"PRIx64" pte=0x%"PRIx64" iova=0x%"PRIx64" block address = 0x%"PRIx64" block size = %d MiB" | ||||||
|  | smmu_get_pte(uint64_t baseaddr, int index, uint64_t pteaddr, uint64_t pte) "baseaddr=0x%"PRIx64" index=0x%x, pteaddr=0x%"PRIx64", pte=0x%"PRIx64 | ||||||
|  | 
 | ||||||
|  | #hw/arm/smmuv3.c | ||||||
|  | smmuv3_read_mmio(uint64_t addr, uint64_t val, unsigned size, uint32_t r) "addr: 0x%"PRIx64" val:0x%"PRIx64" size: 0x%x(%d)" | ||||||
|  | smmuv3_trigger_irq(int irq) "irq=%d" | ||||||
|  | smmuv3_write_gerror(uint32_t toggled, uint32_t gerror) "toggled=0x%x, new GERROR=0x%x" | ||||||
|  | smmuv3_write_gerrorn(uint32_t acked, uint32_t gerrorn) "acked=0x%x, new GERRORN=0x%x" | ||||||
|  | smmuv3_unhandled_cmd(uint32_t type) "Unhandled command type=%d" | ||||||
|  | smmuv3_cmdq_consume(uint32_t prod, uint32_t cons, uint8_t prod_wrap, uint8_t cons_wrap) "prod=%d cons=%d prod.wrap=%d cons.wrap=%d" | ||||||
|  | smmuv3_cmdq_opcode(const char *opcode) "<--- %s" | ||||||
|  | smmuv3_cmdq_consume_out(uint32_t prod, uint32_t cons, uint8_t prod_wrap, uint8_t cons_wrap) "prod:%d, cons:%d, prod_wrap:%d, cons_wrap:%d " | ||||||
|  | smmuv3_cmdq_consume_error(const char *cmd_name, uint8_t cmd_error) "Error on %s command execution: %d" | ||||||
|  | smmuv3_update(bool is_empty, uint32_t prod, uint32_t cons, uint8_t prod_wrap, uint8_t cons_wrap) "q empty:%d prod:%d cons:%d p.wrap:%d p.cons:%d" | ||||||
|  | smmuv3_update_check_cmd(int error) "cmdq not enabled or error :0x%x" | ||||||
|  | smmuv3_write_mmio(uint64_t addr, uint64_t val, unsigned size, uint32_t r) "addr: 0x%"PRIx64" val:0x%"PRIx64" size: 0x%x(%d)" | ||||||
|  | smmuv3_write_mmio_idr(uint64_t addr, uint64_t val) "write to RO/Unimpl reg 0x%"PRIx64" val64:0x%"PRIx64 | ||||||
|  | smmuv3_write_mmio_evtq_cons_bef_clear(uint32_t prod, uint32_t cons, uint8_t prod_wrap, uint8_t cons_wrap) "Before clearing interrupt prod:0x%x cons:0x%x prod.w:%d cons.w:%d" | ||||||
|  | smmuv3_write_mmio_evtq_cons_after_clear(uint32_t prod, uint32_t cons, uint8_t prod_wrap, uint8_t cons_wrap) "after clearing interrupt prod:0x%x cons:0x%x prod.w:%d cons.w:%d" | ||||||
|  | smmuv3_record_event(const char *type, uint32_t sid) "%s sid=%d" | ||||||
|  | smmuv3_find_ste(uint16_t sid, uint32_t features, uint16_t sid_split) "SID:0x%x features:0x%x, sid_split:0x%x" | ||||||
|  | smmuv3_find_ste_2lvl(uint64_t strtab_base, uint64_t l1ptr, int l1_ste_offset, uint64_t l2ptr, int l2_ste_offset, int max_l2_ste) "strtab_base:0x%"PRIx64" l1ptr:0x%"PRIx64" l1_off:0x%x, l2ptr:0x%"PRIx64" l2_off:0x%x max_l2_ste:%d" | ||||||
|  | smmuv3_get_ste(uint64_t addr) "STE addr: 0x%"PRIx64 | ||||||
|  | smmuv3_translate_bypass(const char *n, uint16_t sid, uint64_t addr, bool is_write) "%s sid=%d bypass iova:0x%"PRIx64" is_write=%d" | ||||||
|  | smmuv3_translate_in(uint16_t sid, int pci_bus_num, uint64_t strtab_base) "SID:0x%x bus:%d strtab_base:0x%"PRIx64 | ||||||
|  | smmuv3_get_cd(uint64_t addr) "CD addr: 0x%"PRIx64 | ||||||
|  | smmuv3_translate(const char *n, uint16_t sid, uint64_t iova, uint64_t translated, int perm) "%s sid=%d iova=0x%"PRIx64" translated=0x%"PRIx64" perm=0x%x" | ||||||
|  | smmuv3_decode_cd(uint32_t oas) "oas=%d" | ||||||
|  | smmuv3_decode_cd_tt(int i, uint32_t tsz, uint64_t ttb, uint32_t granule_sz) "TT[%d]:tsz:%d ttb:0x%"PRIx64" granule_sz:%d" | ||||||
|  | |||||||
| @ -393,19 +393,26 @@ build_rsdp(GArray *rsdp_table, BIOSLinker *linker, unsigned xsdt_tbl_offset) | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void | static void | ||||||
| build_iort(GArray *table_data, BIOSLinker *linker) | build_iort(GArray *table_data, BIOSLinker *linker, VirtMachineState *vms) | ||||||
| { | { | ||||||
|     int iort_start = table_data->len; |     int nb_nodes, iort_start = table_data->len; | ||||||
|     AcpiIortIdMapping *idmap; |     AcpiIortIdMapping *idmap; | ||||||
|     AcpiIortItsGroup *its; |     AcpiIortItsGroup *its; | ||||||
|     AcpiIortTable *iort; |     AcpiIortTable *iort; | ||||||
|     size_t node_size, iort_length; |     AcpiIortSmmu3 *smmu; | ||||||
|  |     size_t node_size, iort_length, smmu_offset = 0; | ||||||
|     AcpiIortRC *rc; |     AcpiIortRC *rc; | ||||||
| 
 | 
 | ||||||
|     iort = acpi_data_push(table_data, sizeof(*iort)); |     iort = acpi_data_push(table_data, sizeof(*iort)); | ||||||
| 
 | 
 | ||||||
|  |     if (vms->iommu == VIRT_IOMMU_SMMUV3) { | ||||||
|  |         nb_nodes = 3; /* RC, ITS, SMMUv3 */ | ||||||
|  |     } else { | ||||||
|  |         nb_nodes = 2; /* RC, ITS */ | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     iort_length = sizeof(*iort); |     iort_length = sizeof(*iort); | ||||||
|     iort->node_count = cpu_to_le32(2); /* RC and ITS nodes */ |     iort->node_count = cpu_to_le32(nb_nodes); | ||||||
|     iort->node_offset = cpu_to_le32(sizeof(*iort)); |     iort->node_offset = cpu_to_le32(sizeof(*iort)); | ||||||
| 
 | 
 | ||||||
|     /* ITS group node */ |     /* ITS group node */ | ||||||
| @ -418,6 +425,34 @@ build_iort(GArray *table_data, BIOSLinker *linker) | |||||||
|     its->its_count = cpu_to_le32(1); |     its->its_count = cpu_to_le32(1); | ||||||
|     its->identifiers[0] = 0; /* MADT translation_id */ |     its->identifiers[0] = 0; /* MADT translation_id */ | ||||||
| 
 | 
 | ||||||
|  |     if (vms->iommu == VIRT_IOMMU_SMMUV3) { | ||||||
|  |         int irq =  vms->irqmap[VIRT_SMMU]; | ||||||
|  | 
 | ||||||
|  |         /* SMMUv3 node */ | ||||||
|  |         smmu_offset = iort->node_offset + node_size; | ||||||
|  |         node_size = sizeof(*smmu) + sizeof(*idmap); | ||||||
|  |         iort_length += node_size; | ||||||
|  |         smmu = acpi_data_push(table_data, node_size); | ||||||
|  | 
 | ||||||
|  |         smmu->type = ACPI_IORT_NODE_SMMU_V3; | ||||||
|  |         smmu->length = cpu_to_le16(node_size); | ||||||
|  |         smmu->mapping_count = cpu_to_le32(1); | ||||||
|  |         smmu->mapping_offset = cpu_to_le32(sizeof(*smmu)); | ||||||
|  |         smmu->base_address = cpu_to_le64(vms->memmap[VIRT_SMMU].base); | ||||||
|  |         smmu->event_gsiv = cpu_to_le32(irq); | ||||||
|  |         smmu->pri_gsiv = cpu_to_le32(irq + 1); | ||||||
|  |         smmu->gerr_gsiv = cpu_to_le32(irq + 2); | ||||||
|  |         smmu->sync_gsiv = cpu_to_le32(irq + 3); | ||||||
|  | 
 | ||||||
|  |         /* Identity RID mapping covering the whole input RID range */ | ||||||
|  |         idmap = &smmu->id_mapping_array[0]; | ||||||
|  |         idmap->input_base = 0; | ||||||
|  |         idmap->id_count = cpu_to_le32(0xFFFF); | ||||||
|  |         idmap->output_base = 0; | ||||||
|  |         /* output IORT node is the ITS group node (the first node) */ | ||||||
|  |         idmap->output_reference = cpu_to_le32(iort->node_offset); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /* Root Complex Node */ |     /* Root Complex Node */ | ||||||
|     node_size = sizeof(*rc) + sizeof(*idmap); |     node_size = sizeof(*rc) + sizeof(*idmap); | ||||||
|     iort_length += node_size; |     iort_length += node_size; | ||||||
| @ -438,8 +473,14 @@ build_iort(GArray *table_data, BIOSLinker *linker) | |||||||
|     idmap->input_base = 0; |     idmap->input_base = 0; | ||||||
|     idmap->id_count = cpu_to_le32(0xFFFF); |     idmap->id_count = cpu_to_le32(0xFFFF); | ||||||
|     idmap->output_base = 0; |     idmap->output_base = 0; | ||||||
|     /* output IORT node is the ITS group node (the first node) */ | 
 | ||||||
|     idmap->output_reference = cpu_to_le32(iort->node_offset); |     if (vms->iommu == VIRT_IOMMU_SMMUV3) { | ||||||
|  |         /* output IORT node is the smmuv3 node */ | ||||||
|  |         idmap->output_reference = cpu_to_le32(smmu_offset); | ||||||
|  |     } else { | ||||||
|  |         /* output IORT node is the ITS group node (the first node) */ | ||||||
|  |         idmap->output_reference = cpu_to_le32(iort->node_offset); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     iort->length = cpu_to_le32(iort_length); |     iort->length = cpu_to_le32(iort_length); | ||||||
| 
 | 
 | ||||||
| @ -777,7 +818,7 @@ void virt_acpi_build(VirtMachineState *vms, AcpiBuildTables *tables) | |||||||
| 
 | 
 | ||||||
|     if (its_class_name() && !vmc->no_its) { |     if (its_class_name() && !vmc->no_its) { | ||||||
|         acpi_add_table(table_offsets, tables_blob); |         acpi_add_table(table_offsets, tables_blob); | ||||||
|         build_iort(tables_blob, tables->linker); |         build_iort(tables_blob, tables->linker, vms); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /* XSDT is pointed to by RSDP */ |     /* XSDT is pointed to by RSDP */ | ||||||
|  | |||||||
							
								
								
									
										101
									
								
								hw/arm/virt.c
									
									
									
									
									
								
							
							
						
						
									
										101
									
								
								hw/arm/virt.c
									
									
									
									
									
								
							| @ -58,6 +58,7 @@ | |||||||
| #include "hw/smbios/smbios.h" | #include "hw/smbios/smbios.h" | ||||||
| #include "qapi/visitor.h" | #include "qapi/visitor.h" | ||||||
| #include "standard-headers/linux/input.h" | #include "standard-headers/linux/input.h" | ||||||
|  | #include "hw/arm/smmuv3.h" | ||||||
| 
 | 
 | ||||||
| #define DEFINE_VIRT_MACHINE_LATEST(major, minor, latest) \ | #define DEFINE_VIRT_MACHINE_LATEST(major, minor, latest) \ | ||||||
|     static void virt_##major##_##minor##_class_init(ObjectClass *oc, \ |     static void virt_##major##_##minor##_class_init(ObjectClass *oc, \ | ||||||
| @ -141,6 +142,7 @@ static const MemMapEntry a15memmap[] = { | |||||||
|     [VIRT_FW_CFG] =             { 0x09020000, 0x00000018 }, |     [VIRT_FW_CFG] =             { 0x09020000, 0x00000018 }, | ||||||
|     [VIRT_GPIO] =               { 0x09030000, 0x00001000 }, |     [VIRT_GPIO] =               { 0x09030000, 0x00001000 }, | ||||||
|     [VIRT_SECURE_UART] =        { 0x09040000, 0x00001000 }, |     [VIRT_SECURE_UART] =        { 0x09040000, 0x00001000 }, | ||||||
|  |     [VIRT_SMMU] =               { 0x09050000, 0x00020000 }, | ||||||
|     [VIRT_MMIO] =               { 0x0a000000, 0x00000200 }, |     [VIRT_MMIO] =               { 0x0a000000, 0x00000200 }, | ||||||
|     /* ...repeating for a total of NUM_VIRTIO_TRANSPORTS, each of that size */ |     /* ...repeating for a total of NUM_VIRTIO_TRANSPORTS, each of that size */ | ||||||
|     [VIRT_PLATFORM_BUS] =       { 0x0c000000, 0x02000000 }, |     [VIRT_PLATFORM_BUS] =       { 0x0c000000, 0x02000000 }, | ||||||
| @ -161,6 +163,7 @@ static const int a15irqmap[] = { | |||||||
|     [VIRT_SECURE_UART] = 8, |     [VIRT_SECURE_UART] = 8, | ||||||
|     [VIRT_MMIO] = 16, /* ...to 16 + NUM_VIRTIO_TRANSPORTS - 1 */ |     [VIRT_MMIO] = 16, /* ...to 16 + NUM_VIRTIO_TRANSPORTS - 1 */ | ||||||
|     [VIRT_GIC_V2M] = 48, /* ...to 48 + NUM_GICV2M_SPIS - 1 */ |     [VIRT_GIC_V2M] = 48, /* ...to 48 + NUM_GICV2M_SPIS - 1 */ | ||||||
|  |     [VIRT_SMMU] = 74,    /* ...to 74 + NUM_SMMU_IRQS - 1 */ | ||||||
|     [VIRT_PLATFORM_BUS] = 112, /* ...to 112 + PLATFORM_BUS_NUM_IRQS -1 */ |     [VIRT_PLATFORM_BUS] = 112, /* ...to 112 + PLATFORM_BUS_NUM_IRQS -1 */ | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| @ -942,7 +945,57 @@ static void create_pcie_irq_map(const VirtMachineState *vms, | |||||||
|                            0x7           /* PCI irq */); |                            0x7           /* PCI irq */); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void create_pcie(const VirtMachineState *vms, qemu_irq *pic) | static void create_smmu(const VirtMachineState *vms, qemu_irq *pic, | ||||||
|  |                         PCIBus *bus) | ||||||
|  | { | ||||||
|  |     char *node; | ||||||
|  |     const char compat[] = "arm,smmu-v3"; | ||||||
|  |     int irq =  vms->irqmap[VIRT_SMMU]; | ||||||
|  |     int i; | ||||||
|  |     hwaddr base = vms->memmap[VIRT_SMMU].base; | ||||||
|  |     hwaddr size = vms->memmap[VIRT_SMMU].size; | ||||||
|  |     const char irq_names[] = "eventq\0priq\0cmdq-sync\0gerror"; | ||||||
|  |     DeviceState *dev; | ||||||
|  | 
 | ||||||
|  |     if (vms->iommu != VIRT_IOMMU_SMMUV3 || !vms->iommu_phandle) { | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     dev = qdev_create(NULL, "arm-smmuv3"); | ||||||
|  | 
 | ||||||
|  |     object_property_set_link(OBJECT(dev), OBJECT(bus), "primary-bus", | ||||||
|  |                              &error_abort); | ||||||
|  |     qdev_init_nofail(dev); | ||||||
|  |     sysbus_mmio_map(SYS_BUS_DEVICE(dev), 0, base); | ||||||
|  |     for (i = 0; i < NUM_SMMU_IRQS; i++) { | ||||||
|  |         sysbus_connect_irq(SYS_BUS_DEVICE(dev), i, pic[irq + i]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     node = g_strdup_printf("/smmuv3@%" PRIx64, base); | ||||||
|  |     qemu_fdt_add_subnode(vms->fdt, node); | ||||||
|  |     qemu_fdt_setprop(vms->fdt, node, "compatible", compat, sizeof(compat)); | ||||||
|  |     qemu_fdt_setprop_sized_cells(vms->fdt, node, "reg", 2, base, 2, size); | ||||||
|  | 
 | ||||||
|  |     qemu_fdt_setprop_cells(vms->fdt, node, "interrupts", | ||||||
|  |             GIC_FDT_IRQ_TYPE_SPI, irq    , GIC_FDT_IRQ_FLAGS_EDGE_LO_HI, | ||||||
|  |             GIC_FDT_IRQ_TYPE_SPI, irq + 1, GIC_FDT_IRQ_FLAGS_EDGE_LO_HI, | ||||||
|  |             GIC_FDT_IRQ_TYPE_SPI, irq + 2, GIC_FDT_IRQ_FLAGS_EDGE_LO_HI, | ||||||
|  |             GIC_FDT_IRQ_TYPE_SPI, irq + 3, GIC_FDT_IRQ_FLAGS_EDGE_LO_HI); | ||||||
|  | 
 | ||||||
|  |     qemu_fdt_setprop(vms->fdt, node, "interrupt-names", irq_names, | ||||||
|  |                      sizeof(irq_names)); | ||||||
|  | 
 | ||||||
|  |     qemu_fdt_setprop_cell(vms->fdt, node, "clocks", vms->clock_phandle); | ||||||
|  |     qemu_fdt_setprop_string(vms->fdt, node, "clock-names", "apb_pclk"); | ||||||
|  |     qemu_fdt_setprop(vms->fdt, node, "dma-coherent", NULL, 0); | ||||||
|  | 
 | ||||||
|  |     qemu_fdt_setprop_cell(vms->fdt, node, "#iommu-cells", 1); | ||||||
|  | 
 | ||||||
|  |     qemu_fdt_setprop_cell(vms->fdt, node, "phandle", vms->iommu_phandle); | ||||||
|  |     g_free(node); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void create_pcie(VirtMachineState *vms, qemu_irq *pic) | ||||||
| { | { | ||||||
|     hwaddr base_mmio = vms->memmap[VIRT_PCIE_MMIO].base; |     hwaddr base_mmio = vms->memmap[VIRT_PCIE_MMIO].base; | ||||||
|     hwaddr size_mmio = vms->memmap[VIRT_PCIE_MMIO].size; |     hwaddr size_mmio = vms->memmap[VIRT_PCIE_MMIO].size; | ||||||
| @ -1023,6 +1076,7 @@ static void create_pcie(const VirtMachineState *vms, qemu_irq *pic) | |||||||
|     qemu_fdt_setprop_string(vms->fdt, nodename, "device_type", "pci"); |     qemu_fdt_setprop_string(vms->fdt, nodename, "device_type", "pci"); | ||||||
|     qemu_fdt_setprop_cell(vms->fdt, nodename, "#address-cells", 3); |     qemu_fdt_setprop_cell(vms->fdt, nodename, "#address-cells", 3); | ||||||
|     qemu_fdt_setprop_cell(vms->fdt, nodename, "#size-cells", 2); |     qemu_fdt_setprop_cell(vms->fdt, nodename, "#size-cells", 2); | ||||||
|  |     qemu_fdt_setprop_cell(vms->fdt, nodename, "linux,pci-domain", 0); | ||||||
|     qemu_fdt_setprop_cells(vms->fdt, nodename, "bus-range", 0, |     qemu_fdt_setprop_cells(vms->fdt, nodename, "bus-range", 0, | ||||||
|                            nr_pcie_buses - 1); |                            nr_pcie_buses - 1); | ||||||
|     qemu_fdt_setprop(vms->fdt, nodename, "dma-coherent", NULL, 0); |     qemu_fdt_setprop(vms->fdt, nodename, "dma-coherent", NULL, 0); | ||||||
| @ -1055,6 +1109,15 @@ static void create_pcie(const VirtMachineState *vms, qemu_irq *pic) | |||||||
|     qemu_fdt_setprop_cell(vms->fdt, nodename, "#interrupt-cells", 1); |     qemu_fdt_setprop_cell(vms->fdt, nodename, "#interrupt-cells", 1); | ||||||
|     create_pcie_irq_map(vms, vms->gic_phandle, irq, nodename); |     create_pcie_irq_map(vms, vms->gic_phandle, irq, nodename); | ||||||
| 
 | 
 | ||||||
|  |     if (vms->iommu) { | ||||||
|  |         vms->iommu_phandle = qemu_fdt_alloc_phandle(vms->fdt); | ||||||
|  | 
 | ||||||
|  |         create_smmu(vms, pic, pci->bus); | ||||||
|  | 
 | ||||||
|  |         qemu_fdt_setprop_cells(vms->fdt, nodename, "iommu-map", | ||||||
|  |                                0x0, vms->iommu_phandle, 0x0, 0x10000); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     g_free(nodename); |     g_free(nodename); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -1498,6 +1561,34 @@ static void virt_set_gic_version(Object *obj, const char *value, Error **errp) | |||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static char *virt_get_iommu(Object *obj, Error **errp) | ||||||
|  | { | ||||||
|  |     VirtMachineState *vms = VIRT_MACHINE(obj); | ||||||
|  | 
 | ||||||
|  |     switch (vms->iommu) { | ||||||
|  |     case VIRT_IOMMU_NONE: | ||||||
|  |         return g_strdup("none"); | ||||||
|  |     case VIRT_IOMMU_SMMUV3: | ||||||
|  |         return g_strdup("smmuv3"); | ||||||
|  |     default: | ||||||
|  |         g_assert_not_reached(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void virt_set_iommu(Object *obj, const char *value, Error **errp) | ||||||
|  | { | ||||||
|  |     VirtMachineState *vms = VIRT_MACHINE(obj); | ||||||
|  | 
 | ||||||
|  |     if (!strcmp(value, "smmuv3")) { | ||||||
|  |         vms->iommu = VIRT_IOMMU_SMMUV3; | ||||||
|  |     } else if (!strcmp(value, "none")) { | ||||||
|  |         vms->iommu = VIRT_IOMMU_NONE; | ||||||
|  |     } else { | ||||||
|  |         error_setg(errp, "Invalid iommu value"); | ||||||
|  |         error_append_hint(errp, "Valid values are none, smmuv3.\n"); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static CpuInstanceProperties | static CpuInstanceProperties | ||||||
| virt_cpu_index_to_props(MachineState *ms, unsigned cpu_index) | virt_cpu_index_to_props(MachineState *ms, unsigned cpu_index) | ||||||
| { | { | ||||||
| @ -1630,6 +1721,14 @@ static void virt_2_12_instance_init(Object *obj) | |||||||
|                                         NULL); |                                         NULL); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /* Default disallows iommu instantiation */ | ||||||
|  |     vms->iommu = VIRT_IOMMU_NONE; | ||||||
|  |     object_property_add_str(obj, "iommu", virt_get_iommu, virt_set_iommu, NULL); | ||||||
|  |     object_property_set_description(obj, "iommu", | ||||||
|  |                                     "Set the IOMMU type. " | ||||||
|  |                                     "Valid values are none and smmuv3", | ||||||
|  |                                     NULL); | ||||||
|  | 
 | ||||||
|     vms->memmap = a15memmap; |     vms->memmap = a15memmap; | ||||||
|     vms->irqmap = a15irqmap; |     vms->irqmap = a15irqmap; | ||||||
| } | } | ||||||
|  | |||||||
| @ -157,6 +157,7 @@ static uint64_t uart_read(void *opaque, hwaddr offset, unsigned size) | |||||||
|         r = s->rxbuf; |         r = s->rxbuf; | ||||||
|         s->state &= ~R_STATE_RXFULL_MASK; |         s->state &= ~R_STATE_RXFULL_MASK; | ||||||
|         cmsdk_apb_uart_update(s); |         cmsdk_apb_uart_update(s); | ||||||
|  |         qemu_chr_fe_accept_input(&s->chr); | ||||||
|         break; |         break; | ||||||
|     case A_STATE: |     case A_STATE: | ||||||
|         r = s->state; |         r = s->state; | ||||||
|  | |||||||
| @ -625,37 +625,33 @@ static uint32_t smc91c111_readb(void *opaque, hwaddr offset) | |||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void smc91c111_writew(void *opaque, hwaddr offset, | static uint64_t smc91c111_readfn(void *opaque, hwaddr addr, unsigned size) | ||||||
|                              uint32_t value) |  | ||||||
| { | { | ||||||
|     smc91c111_writeb(opaque, offset, value & 0xff); |     int i; | ||||||
|     smc91c111_writeb(opaque, offset + 1, value >> 8); |     uint32_t val = 0; | ||||||
|  | 
 | ||||||
|  |     for (i = 0; i < size; i++) { | ||||||
|  |         val |= smc91c111_readb(opaque, addr + i) << (i * 8); | ||||||
|  |     } | ||||||
|  |     return val; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void smc91c111_writel(void *opaque, hwaddr offset, | static void smc91c111_writefn(void *opaque, hwaddr addr, | ||||||
|                              uint32_t value) |                                uint64_t value, unsigned size) | ||||||
| { | { | ||||||
|  |     int i = 0; | ||||||
|  | 
 | ||||||
|     /* 32-bit writes to offset 0xc only actually write to the bank select
 |     /* 32-bit writes to offset 0xc only actually write to the bank select
 | ||||||
|        register (offset 0xe)  */ |      * register (offset 0xe), so skip the first two bytes we would write. | ||||||
|     if (offset != 0xc) |      */ | ||||||
|         smc91c111_writew(opaque, offset, value & 0xffff); |     if (addr == 0xc && size == 4) { | ||||||
|     smc91c111_writew(opaque, offset + 2, value >> 16); |         i += 2; | ||||||
| } |     } | ||||||
| 
 | 
 | ||||||
| static uint32_t smc91c111_readw(void *opaque, hwaddr offset) |     for (; i < size; i++) { | ||||||
| { |         smc91c111_writeb(opaque, addr + i, | ||||||
|     uint32_t val; |                          extract32(value, i * 8, 8)); | ||||||
|     val = smc91c111_readb(opaque, offset); |     } | ||||||
|     val |= smc91c111_readb(opaque, offset + 1) << 8; |  | ||||||
|     return val; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static uint32_t smc91c111_readl(void *opaque, hwaddr offset) |  | ||||||
| { |  | ||||||
|     uint32_t val; |  | ||||||
|     val = smc91c111_readw(opaque, offset); |  | ||||||
|     val |= smc91c111_readw(opaque, offset + 2) << 16; |  | ||||||
|     return val; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int smc91c111_can_receive_nc(NetClientState *nc) | static int smc91c111_can_receive_nc(NetClientState *nc) | ||||||
| @ -747,10 +743,10 @@ static const MemoryRegionOps smc91c111_mem_ops = { | |||||||
|     /* The special case for 32 bit writes to 0xc means we can't just
 |     /* The special case for 32 bit writes to 0xc means we can't just
 | ||||||
|      * set .impl.min/max_access_size to 1, unfortunately |      * set .impl.min/max_access_size to 1, unfortunately | ||||||
|      */ |      */ | ||||||
|     .old_mmio = { |     .read = smc91c111_readfn, | ||||||
|         .read = { smc91c111_readb, smc91c111_readw, smc91c111_readl, }, |     .write = smc91c111_writefn, | ||||||
|         .write = { smc91c111_writeb, smc91c111_writew, smc91c111_writel, }, |     .valid.min_access_size = 1, | ||||||
|     }, |     .valid.max_access_size = 4, | ||||||
|     .endianness = DEVICE_NATIVE_ENDIAN, |     .endianness = DEVICE_NATIVE_ENDIAN, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -641,11 +641,43 @@ static void tusb_async_writew(void *opaque, hwaddr addr, | |||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static uint64_t tusb_async_readfn(void *opaque, hwaddr addr, unsigned size) | ||||||
|  | { | ||||||
|  |     switch (size) { | ||||||
|  |     case 1: | ||||||
|  |         return tusb_async_readb(opaque, addr); | ||||||
|  |     case 2: | ||||||
|  |         return tusb_async_readh(opaque, addr); | ||||||
|  |     case 4: | ||||||
|  |         return tusb_async_readw(opaque, addr); | ||||||
|  |     default: | ||||||
|  |         g_assert_not_reached(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void tusb_async_writefn(void *opaque, hwaddr addr, | ||||||
|  |                                uint64_t value, unsigned size) | ||||||
|  | { | ||||||
|  |     switch (size) { | ||||||
|  |     case 1: | ||||||
|  |         tusb_async_writeb(opaque, addr, value); | ||||||
|  |         break; | ||||||
|  |     case 2: | ||||||
|  |         tusb_async_writeh(opaque, addr, value); | ||||||
|  |         break; | ||||||
|  |     case 4: | ||||||
|  |         tusb_async_writew(opaque, addr, value); | ||||||
|  |         break; | ||||||
|  |     default: | ||||||
|  |         g_assert_not_reached(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static const MemoryRegionOps tusb_async_ops = { | static const MemoryRegionOps tusb_async_ops = { | ||||||
|     .old_mmio = { |     .read = tusb_async_readfn, | ||||||
|         .read = { tusb_async_readb, tusb_async_readh, tusb_async_readw, }, |     .write = tusb_async_writefn, | ||||||
|         .write =  { tusb_async_writeb, tusb_async_writeh, tusb_async_writew, }, |     .valid.min_access_size = 1, | ||||||
|     }, |     .valid.max_access_size = 4, | ||||||
|     .endianness = DEVICE_NATIVE_ENDIAN, |     .endianness = DEVICE_NATIVE_ENDIAN, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -628,6 +628,21 @@ struct AcpiIortItsGroup { | |||||||
| } QEMU_PACKED; | } QEMU_PACKED; | ||||||
| typedef struct AcpiIortItsGroup AcpiIortItsGroup; | typedef struct AcpiIortItsGroup AcpiIortItsGroup; | ||||||
| 
 | 
 | ||||||
|  | struct AcpiIortSmmu3 { | ||||||
|  |     ACPI_IORT_NODE_HEADER_DEF | ||||||
|  |     uint64_t base_address; | ||||||
|  |     uint32_t flags; | ||||||
|  |     uint32_t reserved2; | ||||||
|  |     uint64_t vatos_address; | ||||||
|  |     uint32_t model; | ||||||
|  |     uint32_t event_gsiv; | ||||||
|  |     uint32_t pri_gsiv; | ||||||
|  |     uint32_t gerr_gsiv; | ||||||
|  |     uint32_t sync_gsiv; | ||||||
|  |     AcpiIortIdMapping id_mapping_array[0]; | ||||||
|  | } QEMU_PACKED; | ||||||
|  | typedef struct AcpiIortSmmu3 AcpiIortSmmu3; | ||||||
|  | 
 | ||||||
| struct AcpiIortRC { | struct AcpiIortRC { | ||||||
|     ACPI_IORT_NODE_HEADER_DEF |     ACPI_IORT_NODE_HEADER_DEF | ||||||
|     AcpiIortMemoryAccess memory_properties; |     AcpiIortMemoryAccess memory_properties; | ||||||
|  | |||||||
							
								
								
									
										145
									
								
								include/hw/arm/smmu-common.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								include/hw/arm/smmu-common.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,145 @@ | |||||||
|  | /*
 | ||||||
|  |  * ARM SMMU Support | ||||||
|  |  * | ||||||
|  |  * Copyright (C) 2015-2016 Broadcom Corporation | ||||||
|  |  * Copyright (c) 2017 Red Hat, Inc. | ||||||
|  |  * Written by Prem Mallappa, Eric Auger | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License version 2 as | ||||||
|  |  * published by the Free Software Foundation. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef HW_ARM_SMMU_COMMON_H | ||||||
|  | #define HW_ARM_SMMU_COMMON_H | ||||||
|  | 
 | ||||||
|  | #include "hw/sysbus.h" | ||||||
|  | #include "hw/pci/pci.h" | ||||||
|  | 
 | ||||||
|  | #define SMMU_PCI_BUS_MAX      256 | ||||||
|  | #define SMMU_PCI_DEVFN_MAX    256 | ||||||
|  | 
 | ||||||
|  | #define SMMU_MAX_VA_BITS      48 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Page table walk error types | ||||||
|  |  */ | ||||||
|  | typedef enum { | ||||||
|  |     SMMU_PTW_ERR_NONE, | ||||||
|  |     SMMU_PTW_ERR_WALK_EABT,   /* Translation walk external abort */ | ||||||
|  |     SMMU_PTW_ERR_TRANSLATION, /* Translation fault */ | ||||||
|  |     SMMU_PTW_ERR_ADDR_SIZE,   /* Address Size fault */ | ||||||
|  |     SMMU_PTW_ERR_ACCESS,      /* Access fault */ | ||||||
|  |     SMMU_PTW_ERR_PERMISSION,  /* Permission fault */ | ||||||
|  | } SMMUPTWEventType; | ||||||
|  | 
 | ||||||
|  | typedef struct SMMUPTWEventInfo { | ||||||
|  |     SMMUPTWEventType type; | ||||||
|  |     dma_addr_t addr; /* fetched address that induced an abort, if any */ | ||||||
|  | } SMMUPTWEventInfo; | ||||||
|  | 
 | ||||||
|  | typedef struct SMMUTransTableInfo { | ||||||
|  |     bool disabled;             /* is the translation table disabled? */ | ||||||
|  |     uint64_t ttb;              /* TT base address */ | ||||||
|  |     uint8_t tsz;               /* input range, ie. 2^(64 -tsz)*/ | ||||||
|  |     uint8_t granule_sz;        /* granule page shift */ | ||||||
|  | } SMMUTransTableInfo; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Generic structure populated by derived SMMU devices | ||||||
|  |  * after decoding the configuration information and used as | ||||||
|  |  * input to the page table walk | ||||||
|  |  */ | ||||||
|  | typedef struct SMMUTransCfg { | ||||||
|  |     int stage;                 /* translation stage */ | ||||||
|  |     bool aa64;                 /* arch64 or aarch32 translation table */ | ||||||
|  |     bool disabled;             /* smmu is disabled */ | ||||||
|  |     bool bypassed;             /* translation is bypassed */ | ||||||
|  |     bool aborted;              /* translation is aborted */ | ||||||
|  |     uint64_t ttb;              /* TT base address */ | ||||||
|  |     uint8_t oas;               /* output address width */ | ||||||
|  |     uint8_t tbi;               /* Top Byte Ignore */ | ||||||
|  |     uint16_t asid; | ||||||
|  |     SMMUTransTableInfo tt[2]; | ||||||
|  | } SMMUTransCfg; | ||||||
|  | 
 | ||||||
|  | typedef struct SMMUDevice { | ||||||
|  |     void               *smmu; | ||||||
|  |     PCIBus             *bus; | ||||||
|  |     int                devfn; | ||||||
|  |     IOMMUMemoryRegion  iommu; | ||||||
|  |     AddressSpace       as; | ||||||
|  | } SMMUDevice; | ||||||
|  | 
 | ||||||
|  | typedef struct SMMUNotifierNode { | ||||||
|  |     SMMUDevice *sdev; | ||||||
|  |     QLIST_ENTRY(SMMUNotifierNode) next; | ||||||
|  | } SMMUNotifierNode; | ||||||
|  | 
 | ||||||
|  | typedef struct SMMUPciBus { | ||||||
|  |     PCIBus       *bus; | ||||||
|  |     SMMUDevice   *pbdev[0]; /* Parent array is sparse, so dynamically alloc */ | ||||||
|  | } SMMUPciBus; | ||||||
|  | 
 | ||||||
|  | typedef struct SMMUState { | ||||||
|  |     /* <private> */ | ||||||
|  |     SysBusDevice  dev; | ||||||
|  |     const char *mrtypename; | ||||||
|  |     MemoryRegion iomem; | ||||||
|  | 
 | ||||||
|  |     GHashTable *smmu_pcibus_by_busptr; | ||||||
|  |     GHashTable *configs; /* cache for configuration data */ | ||||||
|  |     GHashTable *iotlb; | ||||||
|  |     SMMUPciBus *smmu_pcibus_by_bus_num[SMMU_PCI_BUS_MAX]; | ||||||
|  |     PCIBus *pci_bus; | ||||||
|  |     QLIST_HEAD(, SMMUNotifierNode) notifiers_list; | ||||||
|  |     uint8_t bus_num; | ||||||
|  |     PCIBus *primary_bus; | ||||||
|  | } SMMUState; | ||||||
|  | 
 | ||||||
|  | typedef struct { | ||||||
|  |     /* <private> */ | ||||||
|  |     SysBusDeviceClass parent_class; | ||||||
|  | 
 | ||||||
|  |     /*< public >*/ | ||||||
|  | 
 | ||||||
|  |     DeviceRealize parent_realize; | ||||||
|  | 
 | ||||||
|  | } SMMUBaseClass; | ||||||
|  | 
 | ||||||
|  | #define TYPE_ARM_SMMU "arm-smmu" | ||||||
|  | #define ARM_SMMU(obj) OBJECT_CHECK(SMMUState, (obj), TYPE_ARM_SMMU) | ||||||
|  | #define ARM_SMMU_CLASS(klass)                                    \ | ||||||
|  |     OBJECT_CLASS_CHECK(SMMUBaseClass, (klass), TYPE_ARM_SMMU) | ||||||
|  | #define ARM_SMMU_GET_CLASS(obj)                              \ | ||||||
|  |     OBJECT_GET_CLASS(SMMUBaseClass, (obj), TYPE_ARM_SMMU) | ||||||
|  | 
 | ||||||
|  | /* Return the SMMUPciBus handle associated to a PCI bus number */ | ||||||
|  | SMMUPciBus *smmu_find_smmu_pcibus(SMMUState *s, uint8_t bus_num); | ||||||
|  | 
 | ||||||
|  | /* Return the stream ID of an SMMU device */ | ||||||
|  | static inline uint16_t smmu_get_sid(SMMUDevice *sdev) | ||||||
|  | { | ||||||
|  |     return PCI_BUILD_BDF(pci_bus_num(sdev->bus), sdev->devfn); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * smmu_ptw - Perform the page table walk for a given iova / access flags | ||||||
|  |  * pair, according to @cfg translation config | ||||||
|  |  */ | ||||||
|  | int smmu_ptw(SMMUTransCfg *cfg, dma_addr_t iova, IOMMUAccessFlags perm, | ||||||
|  |              IOMMUTLBEntry *tlbe, SMMUPTWEventInfo *info); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * select_tt - compute which translation table shall be used according to | ||||||
|  |  * the input iova and translation config and return the TT specific info | ||||||
|  |  */ | ||||||
|  | SMMUTransTableInfo *select_tt(SMMUTransCfg *cfg, dma_addr_t iova); | ||||||
|  | 
 | ||||||
|  | #endif  /* HW_ARM_SMMU_COMMON */ | ||||||
							
								
								
									
										87
									
								
								include/hw/arm/smmuv3.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								include/hw/arm/smmuv3.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,87 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2014-2016 Broadcom Corporation | ||||||
|  |  * Copyright (c) 2017 Red Hat, Inc. | ||||||
|  |  * Written by Prem Mallappa, Eric Auger | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License version 2 as | ||||||
|  |  * published by the Free Software Foundation. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License along | ||||||
|  |  * with this program; if not, see <http://www.gnu.org/licenses/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef HW_ARM_SMMUV3_H | ||||||
|  | #define HW_ARM_SMMUV3_H | ||||||
|  | 
 | ||||||
|  | #include "hw/arm/smmu-common.h" | ||||||
|  | #include "hw/registerfields.h" | ||||||
|  | 
 | ||||||
|  | #define TYPE_SMMUV3_IOMMU_MEMORY_REGION "smmuv3-iommu-memory-region" | ||||||
|  | 
 | ||||||
|  | typedef struct SMMUQueue { | ||||||
|  |      uint64_t base; /* base register */ | ||||||
|  |      uint32_t prod; | ||||||
|  |      uint32_t cons; | ||||||
|  |      uint8_t entry_size; | ||||||
|  |      uint8_t log2size; | ||||||
|  | } SMMUQueue; | ||||||
|  | 
 | ||||||
|  | typedef struct SMMUv3State { | ||||||
|  |     SMMUState     smmu_state; | ||||||
|  | 
 | ||||||
|  |     uint32_t features; | ||||||
|  |     uint8_t sid_size; | ||||||
|  |     uint8_t sid_split; | ||||||
|  | 
 | ||||||
|  |     uint32_t idr[6]; | ||||||
|  |     uint32_t iidr; | ||||||
|  |     uint32_t cr[3]; | ||||||
|  |     uint32_t cr0ack; | ||||||
|  |     uint32_t statusr; | ||||||
|  |     uint32_t irq_ctrl; | ||||||
|  |     uint32_t gerror; | ||||||
|  |     uint32_t gerrorn; | ||||||
|  |     uint64_t gerror_irq_cfg0; | ||||||
|  |     uint32_t gerror_irq_cfg1; | ||||||
|  |     uint32_t gerror_irq_cfg2; | ||||||
|  |     uint64_t strtab_base; | ||||||
|  |     uint32_t strtab_base_cfg; | ||||||
|  |     uint64_t eventq_irq_cfg0; | ||||||
|  |     uint32_t eventq_irq_cfg1; | ||||||
|  |     uint32_t eventq_irq_cfg2; | ||||||
|  | 
 | ||||||
|  |     SMMUQueue eventq, cmdq; | ||||||
|  | 
 | ||||||
|  |     qemu_irq     irq[4]; | ||||||
|  | } SMMUv3State; | ||||||
|  | 
 | ||||||
|  | typedef enum { | ||||||
|  |     SMMU_IRQ_EVTQ, | ||||||
|  |     SMMU_IRQ_PRIQ, | ||||||
|  |     SMMU_IRQ_CMD_SYNC, | ||||||
|  |     SMMU_IRQ_GERROR, | ||||||
|  | } SMMUIrq; | ||||||
|  | 
 | ||||||
|  | typedef struct { | ||||||
|  |     /*< private >*/ | ||||||
|  |     SMMUBaseClass smmu_base_class; | ||||||
|  |     /*< public >*/ | ||||||
|  | 
 | ||||||
|  |     DeviceRealize parent_realize; | ||||||
|  |     DeviceReset   parent_reset; | ||||||
|  | } SMMUv3Class; | ||||||
|  | 
 | ||||||
|  | #define TYPE_ARM_SMMUV3   "arm-smmuv3" | ||||||
|  | #define ARM_SMMUV3(obj) OBJECT_CHECK(SMMUv3State, (obj), TYPE_ARM_SMMUV3) | ||||||
|  | #define ARM_SMMUV3_CLASS(klass)                              \ | ||||||
|  |     OBJECT_CLASS_CHECK(SMMUv3Class, (klass), TYPE_ARM_SMMUV3) | ||||||
|  | #define ARM_SMMUV3_GET_CLASS(obj) \ | ||||||
|  |      OBJECT_GET_CLASS(SMMUv3Class, (obj), TYPE_ARM_SMMUV3) | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
| @ -38,6 +38,7 @@ | |||||||
| 
 | 
 | ||||||
| #define NUM_GICV2M_SPIS       64 | #define NUM_GICV2M_SPIS       64 | ||||||
| #define NUM_VIRTIO_TRANSPORTS 32 | #define NUM_VIRTIO_TRANSPORTS 32 | ||||||
|  | #define NUM_SMMU_IRQS          4 | ||||||
| 
 | 
 | ||||||
| #define ARCH_GICV3_MAINT_IRQ  9 | #define ARCH_GICV3_MAINT_IRQ  9 | ||||||
| 
 | 
 | ||||||
| @ -59,6 +60,7 @@ enum { | |||||||
|     VIRT_GIC_V2M, |     VIRT_GIC_V2M, | ||||||
|     VIRT_GIC_ITS, |     VIRT_GIC_ITS, | ||||||
|     VIRT_GIC_REDIST, |     VIRT_GIC_REDIST, | ||||||
|  |     VIRT_SMMU, | ||||||
|     VIRT_UART, |     VIRT_UART, | ||||||
|     VIRT_MMIO, |     VIRT_MMIO, | ||||||
|     VIRT_RTC, |     VIRT_RTC, | ||||||
| @ -74,6 +76,12 @@ enum { | |||||||
|     VIRT_SECURE_MEM, |     VIRT_SECURE_MEM, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | typedef enum VirtIOMMUType { | ||||||
|  |     VIRT_IOMMU_NONE, | ||||||
|  |     VIRT_IOMMU_SMMUV3, | ||||||
|  |     VIRT_IOMMU_VIRTIO, | ||||||
|  | } VirtIOMMUType; | ||||||
|  | 
 | ||||||
| typedef struct MemMapEntry { | typedef struct MemMapEntry { | ||||||
|     hwaddr base; |     hwaddr base; | ||||||
|     hwaddr size; |     hwaddr size; | ||||||
| @ -97,6 +105,7 @@ typedef struct { | |||||||
|     bool its; |     bool its; | ||||||
|     bool virt; |     bool virt; | ||||||
|     int32_t gic_version; |     int32_t gic_version; | ||||||
|  |     VirtIOMMUType iommu; | ||||||
|     struct arm_boot_info bootinfo; |     struct arm_boot_info bootinfo; | ||||||
|     const MemMapEntry *memmap; |     const MemMapEntry *memmap; | ||||||
|     const int *irqmap; |     const int *irqmap; | ||||||
| @ -106,6 +115,7 @@ typedef struct { | |||||||
|     uint32_t clock_phandle; |     uint32_t clock_phandle; | ||||||
|     uint32_t gic_phandle; |     uint32_t gic_phandle; | ||||||
|     uint32_t msi_phandle; |     uint32_t msi_phandle; | ||||||
|  |     uint32_t iommu_phandle; | ||||||
|     int psci_conduit; |     int psci_conduit; | ||||||
| } VirtMachineState; | } VirtMachineState; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -5347,7 +5347,7 @@ void register_cp_regs_for_features(ARMCPU *cpu) | |||||||
|             for (r = id_cp_reginfo; r->type != ARM_CP_SENTINEL; r++) { |             for (r = id_cp_reginfo; r->type != ARM_CP_SENTINEL; r++) { | ||||||
|                 r->access = PL1_RW; |                 r->access = PL1_RW; | ||||||
|             } |             } | ||||||
|             id_tlbtr_reginfo.access = PL1_RW; |             id_mpuir_reginfo.access = PL1_RW; | ||||||
|             id_tlbtr_reginfo.access = PL1_RW; |             id_tlbtr_reginfo.access = PL1_RW; | ||||||
|         } |         } | ||||||
|         if (arm_feature(env, ARM_FEATURE_V8)) { |         if (arm_feature(env, ARM_FEATURE_V8)) { | ||||||
|  | |||||||
| @ -20,8 +20,10 @@ | |||||||
| #include "sysemu/kvm.h" | #include "sysemu/kvm.h" | ||||||
| #include "kvm_arm.h" | #include "kvm_arm.h" | ||||||
| #include "cpu.h" | #include "cpu.h" | ||||||
|  | #include "trace.h" | ||||||
| #include "internals.h" | #include "internals.h" | ||||||
| #include "hw/arm/arm.h" | #include "hw/arm/arm.h" | ||||||
|  | #include "hw/pci/pci.h" | ||||||
| #include "exec/memattrs.h" | #include "exec/memattrs.h" | ||||||
| #include "exec/address-spaces.h" | #include "exec/address-spaces.h" | ||||||
| #include "hw/boards.h" | #include "hw/boards.h" | ||||||
| @ -649,7 +651,41 @@ int kvm_arm_vgic_probe(void) | |||||||
| int kvm_arch_fixup_msi_route(struct kvm_irq_routing_entry *route, | int kvm_arch_fixup_msi_route(struct kvm_irq_routing_entry *route, | ||||||
|                              uint64_t address, uint32_t data, PCIDevice *dev) |                              uint64_t address, uint32_t data, PCIDevice *dev) | ||||||
| { | { | ||||||
|     return 0; |     AddressSpace *as = pci_device_iommu_address_space(dev); | ||||||
|  |     hwaddr xlat, len, doorbell_gpa; | ||||||
|  |     MemoryRegionSection mrs; | ||||||
|  |     MemoryRegion *mr; | ||||||
|  |     int ret = 1; | ||||||
|  | 
 | ||||||
|  |     if (as == &address_space_memory) { | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /* MSI doorbell address is translated by an IOMMU */ | ||||||
|  | 
 | ||||||
|  |     rcu_read_lock(); | ||||||
|  |     mr = address_space_translate(as, address, &xlat, &len, true); | ||||||
|  |     if (!mr) { | ||||||
|  |         goto unlock; | ||||||
|  |     } | ||||||
|  |     mrs = memory_region_find(mr, xlat, 1); | ||||||
|  |     if (!mrs.mr) { | ||||||
|  |         goto unlock; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     doorbell_gpa = mrs.offset_within_address_space; | ||||||
|  |     memory_region_unref(mrs.mr); | ||||||
|  | 
 | ||||||
|  |     route->u.msi.address_lo = doorbell_gpa; | ||||||
|  |     route->u.msi.address_hi = doorbell_gpa >> 32; | ||||||
|  | 
 | ||||||
|  |     trace_kvm_arm_fixup_msi_route(address, doorbell_gpa); | ||||||
|  | 
 | ||||||
|  |     ret = 0; | ||||||
|  | 
 | ||||||
|  | unlock: | ||||||
|  |     rcu_read_unlock(); | ||||||
|  |     return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int kvm_arch_add_msi_route_post(struct kvm_irq_routing_entry *route, | int kvm_arch_add_msi_route_post(struct kvm_irq_routing_entry *route, | ||||||
|  | |||||||
| @ -8,3 +8,6 @@ arm_gt_tval_write(int timer, uint64_t value) "gt_tval_write: timer %d value 0x%" | |||||||
| arm_gt_ctl_write(int timer, uint64_t value) "gt_ctl_write: timer %d value 0x%" PRIx64 | arm_gt_ctl_write(int timer, uint64_t value) "gt_ctl_write: timer %d value 0x%" PRIx64 | ||||||
| arm_gt_imask_toggle(int timer, int irqstate) "gt_ctl_write: timer %d IMASK toggle, new irqstate %d" | arm_gt_imask_toggle(int timer, int irqstate) "gt_ctl_write: timer %d IMASK toggle, new irqstate %d" | ||||||
| arm_gt_cntvoff_write(uint64_t value) "gt_cntvoff_write: value 0x%" PRIx64 | arm_gt_cntvoff_write(uint64_t value) "gt_cntvoff_write: value 0x%" PRIx64 | ||||||
|  | 
 | ||||||
|  | # target/arm/kvm.c | ||||||
|  | kvm_arm_fixup_msi_route(uint64_t iova, uint64_t gpa) "MSI iova = 0x%"PRIx64" is translated into 0x%"PRIx64 | ||||||
|  | |||||||
| @ -9019,11 +9019,7 @@ static void handle_vec_simd_shri(DisasContext *s, bool is_q, bool is_u, | |||||||
|         unallocated_encoding(s); |         unallocated_encoding(s); | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
| 
 |     tcg_debug_assert(size <= 3); | ||||||
|     if (size > 3 && !is_q) { |  | ||||||
|         unallocated_encoding(s); |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|     if (!fp_access_check(s)) { |     if (!fp_access_check(s)) { | ||||||
|         return; |         return; | ||||||
| @ -11477,7 +11473,11 @@ static void disas_simd_two_reg_misc(DisasContext *s, uint32_t insn) | |||||||
|         /* All 64-bit element operations can be shared with scalar 2misc */ |         /* All 64-bit element operations can be shared with scalar 2misc */ | ||||||
|         int pass; |         int pass; | ||||||
| 
 | 
 | ||||||
|         for (pass = 0; pass < (is_q ? 2 : 1); pass++) { |         /* Coverity claims (size == 3 && !is_q) has been eliminated
 | ||||||
|  |          * from all paths leading to here. | ||||||
|  |          */ | ||||||
|  |         tcg_debug_assert(is_q); | ||||||
|  |         for (pass = 0; pass < 2; pass++) { | ||||||
|             TCGv_i64 tcg_op = tcg_temp_new_i64(); |             TCGv_i64 tcg_op = tcg_temp_new_i64(); | ||||||
|             TCGv_i64 tcg_res = tcg_temp_new_i64(); |             TCGv_i64 tcg_res = tcg_temp_new_i64(); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -10795,8 +10795,23 @@ static void disas_thumb2_insn(DisasContext *s, uint32_t insn) | |||||||
|         /* Coprocessor.  */ |         /* Coprocessor.  */ | ||||||
|         if (arm_dc_feature(s, ARM_FEATURE_M)) { |         if (arm_dc_feature(s, ARM_FEATURE_M)) { | ||||||
|             /* We don't currently implement M profile FP support,
 |             /* We don't currently implement M profile FP support,
 | ||||||
|              * so this entire space should give a NOCP fault. |              * so this entire space should give a NOCP fault, with | ||||||
|  |              * the exception of the v8M VLLDM and VLSTM insns, which | ||||||
|  |              * must be NOPs in Secure state and UNDEF in Nonsecure state. | ||||||
|              */ |              */ | ||||||
|  |             if (arm_dc_feature(s, ARM_FEATURE_V8) && | ||||||
|  |                 (insn & 0xffa00f00) == 0xec200a00) { | ||||||
|  |                 /* 0b1110_1100_0x1x_xxxx_xxxx_1010_xxxx_xxxx
 | ||||||
|  |                  *  - VLLDM, VLSTM | ||||||
|  |                  * We choose to UNDEF if the RAZ bits are non-zero. | ||||||
|  |                  */ | ||||||
|  |                 if (!s->v8m_secure || (insn & 0x0040f0ff)) { | ||||||
|  |                     goto illegal_op; | ||||||
|  |                 } | ||||||
|  |                 /* Just NOP since FP support is not implemented */ | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             /* All other insns: NOCP */ | ||||||
|             gen_exception_insn(s, 4, EXCP_NOCP, syn_uncategorized(), |             gen_exception_insn(s, 4, EXCP_NOCP, syn_uncategorized(), | ||||||
|                                default_exception_el(s)); |                                default_exception_el(s)); | ||||||
|             break; |             break; | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Peter Maydell
						Peter Maydell