vtd.c 4.9 KB
Newer Older
B
Ben-Ami Yassour 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38
/*
 * Copyright (c) 2006, Intel Corporation.
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms and conditions of the GNU General Public License,
 * version 2, as published by the Free Software Foundation.
 *
 * This program is distributed in the hope 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, write to the Free Software Foundation, Inc., 59 Temple
 * Place - Suite 330, Boston, MA 02111-1307 USA.
 *
 * Copyright (C) 2006-2008 Intel Corporation
 * Copyright IBM Corporation, 2008
 * Author: Allen M. Kay <allen.m.kay@intel.com>
 * Author: Weidong Han <weidong.han@intel.com>
 * Author: Ben-Ami Yassour <benami@il.ibm.com>
 */

#include <linux/list.h>
#include <linux/kvm_host.h>
#include <linux/pci.h>
#include <linux/dmar.h>
#include <linux/intel-iommu.h>

static int kvm_iommu_unmap_memslots(struct kvm *kvm);
static void kvm_iommu_put_pages(struct kvm *kvm,
				gfn_t base_gfn, unsigned long npages);

int kvm_iommu_map_pages(struct kvm *kvm,
			gfn_t base_gfn, unsigned long npages)
{
	gfn_t gfn = base_gfn;
	pfn_t pfn;
39
	int i, r = 0;
B
Ben-Ami Yassour 已提交
40 41 42 43 44 45 46 47
	struct dmar_domain *domain = kvm->arch.intel_iommu_domain;

	/* check if iommu exists and in use */
	if (!domain)
		return 0;

	for (i = 0; i < npages; i++) {
		/* check if already mapped */
W
Weidong Han 已提交
48 49
		if (intel_iommu_iova_to_phys(domain,
					     gfn_to_gpa(gfn)))
B
Ben-Ami Yassour 已提交
50 51 52
			continue;

		pfn = gfn_to_pfn(kvm, gfn);
W
Weidong Han 已提交
53 54 55 56 57
		r = intel_iommu_map_address(domain,
					    gfn_to_gpa(gfn),
					    pfn_to_hpa(pfn),
					    PAGE_SIZE,
					    DMA_PTE_READ | DMA_PTE_WRITE);
58
		if (r) {
W
Weidong Han 已提交
59
			printk(KERN_ERR "kvm_iommu_map_address:"
60
			       "iommu failed to map pfn=%lx\n", pfn);
B
Ben-Ami Yassour 已提交
61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86
			goto unmap_pages;
		}
		gfn++;
	}
	return 0;

unmap_pages:
	kvm_iommu_put_pages(kvm, base_gfn, i);
	return r;
}

static int kvm_iommu_map_memslots(struct kvm *kvm)
{
	int i, r;

	down_read(&kvm->slots_lock);
	for (i = 0; i < kvm->nmemslots; i++) {
		r = kvm_iommu_map_pages(kvm, kvm->memslots[i].base_gfn,
					kvm->memslots[i].npages);
		if (r)
			break;
	}
	up_read(&kvm->slots_lock);
	return r;
}

W
Weidong Han 已提交
87 88
int kvm_assign_device(struct kvm *kvm,
		      struct kvm_assigned_dev_kernel *assigned_dev)
B
Ben-Ami Yassour 已提交
89 90
{
	struct pci_dev *pdev = NULL;
W
Weidong Han 已提交
91
	struct dmar_domain *domain = kvm->arch.intel_iommu_domain;
B
Ben-Ami Yassour 已提交
92 93
	int r;

W
Weidong Han 已提交
94 95 96 97 98 99
	/* check if iommu exists and in use */
	if (!domain)
		return 0;

	pdev = assigned_dev->dev;
	if (pdev == NULL)
B
Ben-Ami Yassour 已提交
100
		return -ENODEV;
W
Weidong Han 已提交
101 102 103 104 105 106 107 108

	r = intel_iommu_attach_device(domain, pdev);
	if (r) {
		printk(KERN_ERR "assign device %x:%x.%x failed",
			pdev->bus->number,
			PCI_SLOT(pdev->devfn),
			PCI_FUNC(pdev->devfn));
		return r;
B
Ben-Ami Yassour 已提交
109 110
	}

W
Weidong Han 已提交
111 112 113 114
	printk(KERN_DEBUG "assign device: host bdf = %x:%x:%x\n",
		assigned_dev->host_busnr,
		PCI_SLOT(assigned_dev->host_devfn),
		PCI_FUNC(assigned_dev->host_devfn));
B
Ben-Ami Yassour 已提交
115

W
Weidong Han 已提交
116 117
	return 0;
}
B
Ben-Ami Yassour 已提交
118

W
Weidong Han 已提交
119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142
int kvm_deassign_device(struct kvm *kvm,
			struct kvm_assigned_dev_kernel *assigned_dev)
{
	struct dmar_domain *domain = kvm->arch.intel_iommu_domain;
	struct pci_dev *pdev = NULL;

	/* check if iommu exists and in use */
	if (!domain)
		return 0;

	pdev = assigned_dev->dev;
	if (pdev == NULL)
		return -ENODEV;

	intel_iommu_detach_device(domain, pdev);

	printk(KERN_DEBUG "deassign device: host bdf = %x:%x:%x\n",
		assigned_dev->host_busnr,
		PCI_SLOT(assigned_dev->host_devfn),
		PCI_FUNC(assigned_dev->host_devfn));

	return 0;
}

W
Weidong Han 已提交
143 144 145 146 147 148
int kvm_iommu_map_guest(struct kvm *kvm)
{
	int r;

	if (!intel_iommu_found()) {
		printk(KERN_ERR "%s: intel iommu not found\n", __func__);
B
Ben-Ami Yassour 已提交
149 150 151
		return -ENODEV;
	}

W
Weidong Han 已提交
152
	kvm->arch.intel_iommu_domain = intel_iommu_alloc_domain();
B
Ben-Ami Yassour 已提交
153
	if (!kvm->arch.intel_iommu_domain)
W
Weidong Han 已提交
154
		return -ENOMEM;
B
Ben-Ami Yassour 已提交
155 156 157 158 159 160 161 162 163 164 165 166 167

	r = kvm_iommu_map_memslots(kvm);
	if (r)
		goto out_unmap;

	return 0;

out_unmap:
	kvm_iommu_unmap_memslots(kvm);
	return r;
}

static void kvm_iommu_put_pages(struct kvm *kvm,
W
Weidong Han 已提交
168
				gfn_t base_gfn, unsigned long npages)
B
Ben-Ami Yassour 已提交
169 170 171 172
{
	gfn_t gfn = base_gfn;
	pfn_t pfn;
	struct dmar_domain *domain = kvm->arch.intel_iommu_domain;
W
Weidong Han 已提交
173 174 175 176 177 178
	unsigned long i;
	u64 phys;

	/* check if iommu exists and in use */
	if (!domain)
		return;
B
Ben-Ami Yassour 已提交
179 180

	for (i = 0; i < npages; i++) {
W
Weidong Han 已提交
181 182 183
		phys = intel_iommu_iova_to_phys(domain,
						gfn_to_gpa(gfn));
		pfn = phys >> PAGE_SHIFT;
B
Ben-Ami Yassour 已提交
184 185 186
		kvm_release_pfn_clean(pfn);
		gfn++;
	}
W
Weidong Han 已提交
187 188 189 190

	intel_iommu_unmap_address(domain,
				  gfn_to_gpa(base_gfn),
				  PAGE_SIZE * npages);
B
Ben-Ami Yassour 已提交
191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214
}

static int kvm_iommu_unmap_memslots(struct kvm *kvm)
{
	int i;
	down_read(&kvm->slots_lock);
	for (i = 0; i < kvm->nmemslots; i++) {
		kvm_iommu_put_pages(kvm, kvm->memslots[i].base_gfn,
				    kvm->memslots[i].npages);
	}
	up_read(&kvm->slots_lock);

	return 0;
}

int kvm_iommu_unmap_guest(struct kvm *kvm)
{
	struct dmar_domain *domain = kvm->arch.intel_iommu_domain;

	/* check if iommu exists and in use */
	if (!domain)
		return 0;

	kvm_iommu_unmap_memslots(kvm);
W
Weidong Han 已提交
215
	intel_iommu_free_domain(domain);
B
Ben-Ami Yassour 已提交
216 217
	return 0;
}