diff options
| -rw-r--r-- | arch/arm/xen/grant-table.c | 5 | ||||
| -rw-r--r-- | arch/x86/xen/grant-table.c | 148 | ||||
| -rw-r--r-- | drivers/xen/grant-table.c | 9 | ||||
| -rw-r--r-- | include/xen/grant_table.h | 1 |
4 files changed, 105 insertions, 58 deletions
diff --git a/arch/arm/xen/grant-table.c b/arch/arm/xen/grant-table.c index 859a9bb002d5..91cf08ba1e95 100644 --- a/arch/arm/xen/grant-table.c +++ b/arch/arm/xen/grant-table.c | |||
| @@ -51,3 +51,8 @@ int arch_gnttab_map_status(uint64_t *frames, unsigned long nr_gframes, | |||
| 51 | { | 51 | { |
| 52 | return -ENOSYS; | 52 | return -ENOSYS; |
| 53 | } | 53 | } |
| 54 | |||
| 55 | int arch_gnttab_init(unsigned long nr_shared, unsigned long nr_status) | ||
| 56 | { | ||
| 57 | return 0; | ||
| 58 | } | ||
diff --git a/arch/x86/xen/grant-table.c b/arch/x86/xen/grant-table.c index c98583588580..ebfa9b2c871d 100644 --- a/arch/x86/xen/grant-table.c +++ b/arch/x86/xen/grant-table.c | |||
| @@ -36,99 +36,133 @@ | |||
| 36 | 36 | ||
| 37 | #include <linux/sched.h> | 37 | #include <linux/sched.h> |
| 38 | #include <linux/mm.h> | 38 | #include <linux/mm.h> |
| 39 | #include <linux/slab.h> | ||
| 39 | #include <linux/vmalloc.h> | 40 | #include <linux/vmalloc.h> |
| 40 | 41 | ||
| 41 | #include <xen/interface/xen.h> | 42 | #include <xen/interface/xen.h> |
| 42 | #include <xen/page.h> | 43 | #include <xen/page.h> |
| 43 | #include <xen/grant_table.h> | 44 | #include <xen/grant_table.h> |
| 45 | #include <xen/xen.h> | ||
| 44 | 46 | ||
| 45 | #include <asm/pgtable.h> | 47 | #include <asm/pgtable.h> |
| 46 | 48 | ||
| 47 | static int map_pte_fn(pte_t *pte, struct page *pmd_page, | 49 | static struct gnttab_vm_area { |
| 48 | unsigned long addr, void *data) | 50 | struct vm_struct *area; |
| 51 | pte_t **ptes; | ||
| 52 | } gnttab_shared_vm_area, gnttab_status_vm_area; | ||
| 53 | |||
| 54 | int arch_gnttab_map_shared(unsigned long *frames, unsigned long nr_gframes, | ||
| 55 | unsigned long max_nr_gframes, | ||
| 56 | void **__shared) | ||
| 49 | { | 57 | { |
| 50 | unsigned long **frames = (unsigned long **)data; | 58 | void *shared = *__shared; |
| 59 | unsigned long addr; | ||
| 60 | unsigned long i; | ||
| 51 | 61 | ||
| 52 | set_pte_at(&init_mm, addr, pte, mfn_pte((*frames)[0], PAGE_KERNEL)); | 62 | if (shared == NULL) |
| 53 | (*frames)++; | 63 | *__shared = shared = gnttab_shared_vm_area.area->addr; |
| 54 | return 0; | ||
| 55 | } | ||
| 56 | 64 | ||
| 57 | /* | 65 | addr = (unsigned long)shared; |
| 58 | * This function is used to map shared frames to store grant status. It is | 66 | |
| 59 | * different from map_pte_fn above, the frames type here is uint64_t. | 67 | for (i = 0; i < nr_gframes; i++) { |
| 60 | */ | 68 | set_pte_at(&init_mm, addr, gnttab_shared_vm_area.ptes[i], |
| 61 | static int map_pte_fn_status(pte_t *pte, struct page *pmd_page, | 69 | mfn_pte(frames[i], PAGE_KERNEL)); |
| 62 | unsigned long addr, void *data) | 70 | addr += PAGE_SIZE; |
| 63 | { | 71 | } |
| 64 | uint64_t **frames = (uint64_t **)data; | ||
| 65 | 72 | ||
| 66 | set_pte_at(&init_mm, addr, pte, mfn_pte((*frames)[0], PAGE_KERNEL)); | ||
| 67 | (*frames)++; | ||
| 68 | return 0; | 73 | return 0; |
| 69 | } | 74 | } |
| 70 | 75 | ||
| 71 | static int unmap_pte_fn(pte_t *pte, struct page *pmd_page, | 76 | int arch_gnttab_map_status(uint64_t *frames, unsigned long nr_gframes, |
| 72 | unsigned long addr, void *data) | 77 | unsigned long max_nr_gframes, |
| 78 | grant_status_t **__shared) | ||
| 73 | { | 79 | { |
| 80 | grant_status_t *shared = *__shared; | ||
| 81 | unsigned long addr; | ||
| 82 | unsigned long i; | ||
| 83 | |||
| 84 | if (shared == NULL) | ||
| 85 | *__shared = shared = gnttab_status_vm_area.area->addr; | ||
| 86 | |||
| 87 | addr = (unsigned long)shared; | ||
| 88 | |||
| 89 | for (i = 0; i < nr_gframes; i++) { | ||
| 90 | set_pte_at(&init_mm, addr, gnttab_status_vm_area.ptes[i], | ||
| 91 | mfn_pte(frames[i], PAGE_KERNEL)); | ||
| 92 | addr += PAGE_SIZE; | ||
| 93 | } | ||
| 74 | 94 | ||
| 75 | set_pte_at(&init_mm, addr, pte, __pte(0)); | ||
| 76 | return 0; | 95 | return 0; |
| 77 | } | 96 | } |
| 78 | 97 | ||
| 79 | int arch_gnttab_map_shared(unsigned long *frames, unsigned long nr_gframes, | 98 | void arch_gnttab_unmap(void *shared, unsigned long nr_gframes) |
| 80 | unsigned long max_nr_gframes, | ||
| 81 | void **__shared) | ||
| 82 | { | 99 | { |
| 83 | int rc; | 100 | pte_t **ptes; |
| 84 | void *shared = *__shared; | 101 | unsigned long addr; |
| 102 | unsigned long i; | ||
| 85 | 103 | ||
| 86 | if (shared == NULL) { | 104 | if (shared == gnttab_status_vm_area.area->addr) |
| 87 | struct vm_struct *area = | 105 | ptes = gnttab_status_vm_area.ptes; |
| 88 | alloc_vm_area(PAGE_SIZE * max_nr_gframes, NULL); | 106 | else |
| 89 | BUG_ON(area == NULL); | 107 | ptes = gnttab_shared_vm_area.ptes; |
| 90 | shared = area->addr; | ||
| 91 | *__shared = shared; | ||
| 92 | } | ||
| 93 | 108 | ||
| 94 | rc = apply_to_page_range(&init_mm, (unsigned long)shared, | 109 | addr = (unsigned long)shared; |
| 95 | PAGE_SIZE * nr_gframes, | 110 | |
| 96 | map_pte_fn, &frames); | 111 | for (i = 0; i < nr_gframes; i++) { |
| 97 | return rc; | 112 | set_pte_at(&init_mm, addr, ptes[i], __pte(0)); |
| 113 | addr += PAGE_SIZE; | ||
| 114 | } | ||
| 98 | } | 115 | } |
| 99 | 116 | ||
| 100 | int arch_gnttab_map_status(uint64_t *frames, unsigned long nr_gframes, | 117 | static int arch_gnttab_valloc(struct gnttab_vm_area *area, unsigned nr_frames) |
| 101 | unsigned long max_nr_gframes, | ||
| 102 | grant_status_t **__shared) | ||
| 103 | { | 118 | { |
| 104 | int rc; | 119 | area->ptes = kmalloc(sizeof(pte_t *) * nr_frames, GFP_KERNEL); |
| 105 | grant_status_t *shared = *__shared; | 120 | if (area->ptes == NULL) |
| 121 | return -ENOMEM; | ||
| 106 | 122 | ||
| 107 | if (shared == NULL) { | 123 | area->area = alloc_vm_area(PAGE_SIZE * nr_frames, area->ptes); |
| 108 | /* No need to pass in PTE as we are going to do it | 124 | if (area->area == NULL) { |
| 109 | * in apply_to_page_range anyhow. */ | 125 | kfree(area->ptes); |
| 110 | struct vm_struct *area = | 126 | return -ENOMEM; |
| 111 | alloc_vm_area(PAGE_SIZE * max_nr_gframes, NULL); | ||
| 112 | BUG_ON(area == NULL); | ||
| 113 | shared = area->addr; | ||
| 114 | *__shared = shared; | ||
| 115 | } | 127 | } |
| 116 | 128 | ||
| 117 | rc = apply_to_page_range(&init_mm, (unsigned long)shared, | 129 | return 0; |
| 118 | PAGE_SIZE * nr_gframes, | ||
| 119 | map_pte_fn_status, &frames); | ||
| 120 | return rc; | ||
| 121 | } | 130 | } |
| 122 | 131 | ||
| 123 | void arch_gnttab_unmap(void *shared, unsigned long nr_gframes) | 132 | static void arch_gnttab_vfree(struct gnttab_vm_area *area) |
| 133 | { | ||
| 134 | free_vm_area(area->area); | ||
| 135 | kfree(area->ptes); | ||
| 136 | } | ||
| 137 | |||
| 138 | int arch_gnttab_init(unsigned long nr_shared, unsigned long nr_status) | ||
| 124 | { | 139 | { |
| 125 | apply_to_page_range(&init_mm, (unsigned long)shared, | 140 | int ret; |
| 126 | PAGE_SIZE * nr_gframes, unmap_pte_fn, NULL); | 141 | |
| 142 | if (!xen_pv_domain()) | ||
| 143 | return 0; | ||
| 144 | |||
| 145 | ret = arch_gnttab_valloc(&gnttab_shared_vm_area, nr_shared); | ||
| 146 | if (ret < 0) | ||
| 147 | return ret; | ||
| 148 | |||
| 149 | /* | ||
| 150 | * Always allocate the space for the status frames in case | ||
| 151 | * we're migrated to a host with V2 support. | ||
| 152 | */ | ||
| 153 | ret = arch_gnttab_valloc(&gnttab_status_vm_area, nr_status); | ||
| 154 | if (ret < 0) | ||
| 155 | goto err; | ||
| 156 | |||
| 157 | return 0; | ||
| 158 | err: | ||
| 159 | arch_gnttab_vfree(&gnttab_shared_vm_area); | ||
| 160 | return -ENOMEM; | ||
| 127 | } | 161 | } |
| 162 | |||
| 128 | #ifdef CONFIG_XEN_PVH | 163 | #ifdef CONFIG_XEN_PVH |
| 129 | #include <xen/balloon.h> | 164 | #include <xen/balloon.h> |
| 130 | #include <xen/events.h> | 165 | #include <xen/events.h> |
| 131 | #include <xen/xen.h> | ||
| 132 | #include <linux/slab.h> | 166 | #include <linux/slab.h> |
| 133 | static int __init xlated_setup_gnttab_pages(void) | 167 | static int __init xlated_setup_gnttab_pages(void) |
| 134 | { | 168 | { |
diff --git a/drivers/xen/grant-table.c b/drivers/xen/grant-table.c index 5d4de88fe5b8..eeba7544f0cd 100644 --- a/drivers/xen/grant-table.c +++ b/drivers/xen/grant-table.c | |||
| @@ -1195,18 +1195,20 @@ static int gnttab_expand(unsigned int req_entries) | |||
| 1195 | int gnttab_init(void) | 1195 | int gnttab_init(void) |
| 1196 | { | 1196 | { |
| 1197 | int i; | 1197 | int i; |
| 1198 | unsigned long max_nr_grant_frames; | ||
| 1198 | unsigned int max_nr_glist_frames, nr_glist_frames; | 1199 | unsigned int max_nr_glist_frames, nr_glist_frames; |
| 1199 | unsigned int nr_init_grefs; | 1200 | unsigned int nr_init_grefs; |
| 1200 | int ret; | 1201 | int ret; |
| 1201 | 1202 | ||
| 1202 | gnttab_request_version(); | 1203 | gnttab_request_version(); |
| 1204 | max_nr_grant_frames = gnttab_max_grant_frames(); | ||
| 1203 | nr_grant_frames = 1; | 1205 | nr_grant_frames = 1; |
| 1204 | 1206 | ||
| 1205 | /* Determine the maximum number of frames required for the | 1207 | /* Determine the maximum number of frames required for the |
| 1206 | * grant reference free list on the current hypervisor. | 1208 | * grant reference free list on the current hypervisor. |
| 1207 | */ | 1209 | */ |
| 1208 | BUG_ON(grefs_per_grant_frame == 0); | 1210 | BUG_ON(grefs_per_grant_frame == 0); |
| 1209 | max_nr_glist_frames = (gnttab_max_grant_frames() * | 1211 | max_nr_glist_frames = (max_nr_grant_frames * |
| 1210 | grefs_per_grant_frame / RPP); | 1212 | grefs_per_grant_frame / RPP); |
| 1211 | 1213 | ||
| 1212 | gnttab_list = kmalloc(max_nr_glist_frames * sizeof(grant_ref_t *), | 1214 | gnttab_list = kmalloc(max_nr_glist_frames * sizeof(grant_ref_t *), |
| @@ -1223,6 +1225,11 @@ int gnttab_init(void) | |||
| 1223 | } | 1225 | } |
| 1224 | } | 1226 | } |
| 1225 | 1227 | ||
| 1228 | ret = arch_gnttab_init(max_nr_grant_frames, | ||
| 1229 | nr_status_frames(max_nr_grant_frames)); | ||
| 1230 | if (ret < 0) | ||
| 1231 | goto ini_nomem; | ||
| 1232 | |||
| 1226 | if (gnttab_setup() < 0) { | 1233 | if (gnttab_setup() < 0) { |
| 1227 | ret = -ENODEV; | 1234 | ret = -ENODEV; |
| 1228 | goto ini_nomem; | 1235 | goto ini_nomem; |
diff --git a/include/xen/grant_table.h b/include/xen/grant_table.h index a5af2a26d94f..5c1aba154b64 100644 --- a/include/xen/grant_table.h +++ b/include/xen/grant_table.h | |||
| @@ -170,6 +170,7 @@ gnttab_set_unmap_op(struct gnttab_unmap_grant_ref *unmap, phys_addr_t addr, | |||
| 170 | unmap->dev_bus_addr = 0; | 170 | unmap->dev_bus_addr = 0; |
| 171 | } | 171 | } |
| 172 | 172 | ||
| 173 | int arch_gnttab_init(unsigned long nr_shared, unsigned long nr_status); | ||
| 173 | int arch_gnttab_map_shared(xen_pfn_t *frames, unsigned long nr_gframes, | 174 | int arch_gnttab_map_shared(xen_pfn_t *frames, unsigned long nr_gframes, |
| 174 | unsigned long max_nr_gframes, | 175 | unsigned long max_nr_gframes, |
| 175 | void **__shared); | 176 | void **__shared); |
