diff options
author | Ingo Molnar <mingo@elte.hu> | 2007-01-11 02:15:38 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.osdl.org> | 2007-01-11 21:18:21 -0500 |
commit | 07031e14c1127fc7e1a5b98dfcc59f434e025104 (patch) | |
tree | be4f545e674c529abb0f51c8b87e1f7137c9acb6 /kernel | |
parent | e3881a6816b45668df60a426e5c3431ece1539a7 (diff) |
[PATCH] KVM: add VM-exit profiling
This adds the profile=kvm boot option, which enables KVM to profile VM
exits.
Use: "readprofile -m ./System.map | sort -n" to see the resulting
output:
[...]
18246 serial_out 148.3415
18945 native_flush_tlb 378.9000
23618 serial_in 212.7748
29279 __spin_unlock_irq 622.9574
43447 native_apic_write 2068.9048
52702 enable_8259A_irq 742.2817
54250 vgacon_scroll 89.3740
67394 ide_inb 6126.7273
79514 copy_page_range 98.1654
84868 do_wp_page 86.6000
140266 pit_read 783.6089
151436 ide_outb 25239.3333
152668 native_io_delay 21809.7143
174783 mask_and_ack_8259A 783.7803
362404 native_set_pte_at 36240.4000
1688747 total 0.5009
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Acked-by: Avi Kivity <avi@qumranet.com>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'kernel')
-rw-r--r-- | kernel/profile.c | 14 |
1 files changed, 14 insertions, 0 deletions
diff --git a/kernel/profile.c b/kernel/profile.c index 11550b2290b6..a6574a18514e 100644 --- a/kernel/profile.c +++ b/kernel/profile.c | |||
@@ -40,7 +40,10 @@ int (*timer_hook)(struct pt_regs *) __read_mostly; | |||
40 | 40 | ||
41 | static atomic_t *prof_buffer; | 41 | static atomic_t *prof_buffer; |
42 | static unsigned long prof_len, prof_shift; | 42 | static unsigned long prof_len, prof_shift; |
43 | |||
43 | int prof_on __read_mostly; | 44 | int prof_on __read_mostly; |
45 | EXPORT_SYMBOL_GPL(prof_on); | ||
46 | |||
44 | static cpumask_t prof_cpu_mask = CPU_MASK_ALL; | 47 | static cpumask_t prof_cpu_mask = CPU_MASK_ALL; |
45 | #ifdef CONFIG_SMP | 48 | #ifdef CONFIG_SMP |
46 | static DEFINE_PER_CPU(struct profile_hit *[2], cpu_profile_hits); | 49 | static DEFINE_PER_CPU(struct profile_hit *[2], cpu_profile_hits); |
@@ -52,6 +55,7 @@ static int __init profile_setup(char * str) | |||
52 | { | 55 | { |
53 | static char __initdata schedstr[] = "schedule"; | 56 | static char __initdata schedstr[] = "schedule"; |
54 | static char __initdata sleepstr[] = "sleep"; | 57 | static char __initdata sleepstr[] = "sleep"; |
58 | static char __initdata kvmstr[] = "kvm"; | ||
55 | int par; | 59 | int par; |
56 | 60 | ||
57 | if (!strncmp(str, sleepstr, strlen(sleepstr))) { | 61 | if (!strncmp(str, sleepstr, strlen(sleepstr))) { |
@@ -72,6 +76,15 @@ static int __init profile_setup(char * str) | |||
72 | printk(KERN_INFO | 76 | printk(KERN_INFO |
73 | "kernel schedule profiling enabled (shift: %ld)\n", | 77 | "kernel schedule profiling enabled (shift: %ld)\n", |
74 | prof_shift); | 78 | prof_shift); |
79 | } else if (!strncmp(str, kvmstr, strlen(kvmstr))) { | ||
80 | prof_on = KVM_PROFILING; | ||
81 | if (str[strlen(kvmstr)] == ',') | ||
82 | str += strlen(kvmstr) + 1; | ||
83 | if (get_option(&str, &par)) | ||
84 | prof_shift = par; | ||
85 | printk(KERN_INFO | ||
86 | "kernel KVM profiling enabled (shift: %ld)\n", | ||
87 | prof_shift); | ||
75 | } else if (get_option(&str, &par)) { | 88 | } else if (get_option(&str, &par)) { |
76 | prof_shift = par; | 89 | prof_shift = par; |
77 | prof_on = CPU_PROFILING; | 90 | prof_on = CPU_PROFILING; |
@@ -318,6 +331,7 @@ out: | |||
318 | local_irq_restore(flags); | 331 | local_irq_restore(flags); |
319 | put_cpu(); | 332 | put_cpu(); |
320 | } | 333 | } |
334 | EXPORT_SYMBOL_GPL(profile_hits); | ||
321 | 335 | ||
322 | static int __devinit profile_cpu_callback(struct notifier_block *info, | 336 | static int __devinit profile_cpu_callback(struct notifier_block *info, |
323 | unsigned long action, void *__cpu) | 337 | unsigned long action, void *__cpu) |