diff options
author | Dave Hansen <dave@linux.vnet.ibm.com> | 2008-08-11 13:01:49 -0400 |
---|---|---|
committer | Avi Kivity <avi@qumranet.com> | 2008-10-15 04:15:18 -0400 |
commit | 6ad18fba05228fb1d47cdbc0339fe8b3fca1ca26 (patch) | |
tree | 4b64607dad75aa55dd397784d469d03244f0dfe7 | |
parent | b772ff362ec6b821c8a5227a3355e263f917bfad (diff) |
KVM: Reduce stack usage in kvm_pv_mmu_op()
We're in a hot path. We can't use kmalloc() because
it might impact performance. So, we just stick the buffer that
we need into the kvm_vcpu_arch structure. This is used very
often, so it is not really a waste.
We also have to move the buffer structure's definition to the
arch-specific x86 kvm header.
Signed-off-by: Dave Hansen <dave@linux.vnet.ibm.com>
Signed-off-by: Avi Kivity <avi@qumranet.com>
-rw-r--r-- | arch/x86/kvm/mmu.c | 23 | ||||
-rw-r--r-- | include/asm-x86/kvm_host.h | 10 |
2 files changed, 18 insertions, 15 deletions
diff --git a/arch/x86/kvm/mmu.c b/arch/x86/kvm/mmu.c index c3afbfe6b0c1..171bcea1be21 100644 --- a/arch/x86/kvm/mmu.c +++ b/arch/x86/kvm/mmu.c | |||
@@ -135,13 +135,6 @@ module_param(dbg, bool, 0644); | |||
135 | #define ACC_USER_MASK PT_USER_MASK | 135 | #define ACC_USER_MASK PT_USER_MASK |
136 | #define ACC_ALL (ACC_EXEC_MASK | ACC_WRITE_MASK | ACC_USER_MASK) | 136 | #define ACC_ALL (ACC_EXEC_MASK | ACC_WRITE_MASK | ACC_USER_MASK) |
137 | 137 | ||
138 | struct kvm_pv_mmu_op_buffer { | ||
139 | void *ptr; | ||
140 | unsigned len; | ||
141 | unsigned processed; | ||
142 | char buf[512] __aligned(sizeof(long)); | ||
143 | }; | ||
144 | |||
145 | struct kvm_rmap_desc { | 138 | struct kvm_rmap_desc { |
146 | u64 *shadow_ptes[RMAP_EXT]; | 139 | u64 *shadow_ptes[RMAP_EXT]; |
147 | struct kvm_rmap_desc *more; | 140 | struct kvm_rmap_desc *more; |
@@ -2292,18 +2285,18 @@ int kvm_pv_mmu_op(struct kvm_vcpu *vcpu, unsigned long bytes, | |||
2292 | gpa_t addr, unsigned long *ret) | 2285 | gpa_t addr, unsigned long *ret) |
2293 | { | 2286 | { |
2294 | int r; | 2287 | int r; |
2295 | struct kvm_pv_mmu_op_buffer buffer; | 2288 | struct kvm_pv_mmu_op_buffer *buffer = &vcpu->arch.mmu_op_buffer; |
2296 | 2289 | ||
2297 | buffer.ptr = buffer.buf; | 2290 | buffer->ptr = buffer->buf; |
2298 | buffer.len = min_t(unsigned long, bytes, sizeof buffer.buf); | 2291 | buffer->len = min_t(unsigned long, bytes, sizeof buffer->buf); |
2299 | buffer.processed = 0; | 2292 | buffer->processed = 0; |
2300 | 2293 | ||
2301 | r = kvm_read_guest(vcpu->kvm, addr, buffer.buf, buffer.len); | 2294 | r = kvm_read_guest(vcpu->kvm, addr, buffer->buf, buffer->len); |
2302 | if (r) | 2295 | if (r) |
2303 | goto out; | 2296 | goto out; |
2304 | 2297 | ||
2305 | while (buffer.len) { | 2298 | while (buffer->len) { |
2306 | r = kvm_pv_mmu_op_one(vcpu, &buffer); | 2299 | r = kvm_pv_mmu_op_one(vcpu, buffer); |
2307 | if (r < 0) | 2300 | if (r < 0) |
2308 | goto out; | 2301 | goto out; |
2309 | if (r == 0) | 2302 | if (r == 0) |
@@ -2312,7 +2305,7 @@ int kvm_pv_mmu_op(struct kvm_vcpu *vcpu, unsigned long bytes, | |||
2312 | 2305 | ||
2313 | r = 1; | 2306 | r = 1; |
2314 | out: | 2307 | out: |
2315 | *ret = buffer.processed; | 2308 | *ret = buffer->processed; |
2316 | return r; | 2309 | return r; |
2317 | } | 2310 | } |
2318 | 2311 | ||
diff --git a/include/asm-x86/kvm_host.h b/include/asm-x86/kvm_host.h index 99dddfcecf60..9cb4b4dae5c6 100644 --- a/include/asm-x86/kvm_host.h +++ b/include/asm-x86/kvm_host.h | |||
@@ -201,6 +201,13 @@ struct kvm_mmu_page { | |||
201 | }; | 201 | }; |
202 | }; | 202 | }; |
203 | 203 | ||
204 | struct kvm_pv_mmu_op_buffer { | ||
205 | void *ptr; | ||
206 | unsigned len; | ||
207 | unsigned processed; | ||
208 | char buf[512] __aligned(sizeof(long)); | ||
209 | }; | ||
210 | |||
204 | /* | 211 | /* |
205 | * x86 supports 3 paging modes (4-level 64-bit, 3-level 64-bit, and 2-level | 212 | * x86 supports 3 paging modes (4-level 64-bit, 3-level 64-bit, and 2-level |
206 | * 32-bit). The kvm_mmu structure abstracts the details of the current mmu | 213 | * 32-bit). The kvm_mmu structure abstracts the details of the current mmu |
@@ -248,6 +255,9 @@ struct kvm_vcpu_arch { | |||
248 | bool tpr_access_reporting; | 255 | bool tpr_access_reporting; |
249 | 256 | ||
250 | struct kvm_mmu mmu; | 257 | struct kvm_mmu mmu; |
258 | /* only needed in kvm_pv_mmu_op() path, but it's hot so | ||
259 | * put it here to avoid allocation */ | ||
260 | struct kvm_pv_mmu_op_buffer mmu_op_buffer; | ||
251 | 261 | ||
252 | struct kvm_mmu_memory_cache mmu_pte_chain_cache; | 262 | struct kvm_mmu_memory_cache mmu_pte_chain_cache; |
253 | struct kvm_mmu_memory_cache mmu_rmap_desc_cache; | 263 | struct kvm_mmu_memory_cache mmu_rmap_desc_cache; |