diff options
author | Ingo Molnar <mingo@elte.hu> | 2008-07-31 06:39:15 -0400 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2008-07-31 06:39:15 -0400 |
commit | eac4345be6d17541039791f15f173d0426423df1 (patch) | |
tree | 1101a0a9bdbac146b7316eefbfc97b8a6917fc55 /arch/x86/xen | |
parent | 5fbf24659b75356e2142e1f1b88f67b34cbc3e75 (diff) | |
parent | d5de8841355a48f7f634a04507185eaf1f9755e3 (diff) |
Merge branch 'x86/spinlocks' into x86/xen
Diffstat (limited to 'arch/x86/xen')
-rw-r--r-- | arch/x86/xen/Makefile | 8 | ||||
-rw-r--r-- | arch/x86/xen/smp.c | 167 | ||||
-rw-r--r-- | arch/x86/xen/spinlock.c | 183 | ||||
-rw-r--r-- | arch/x86/xen/xen-ops.h | 3 |
4 files changed, 193 insertions, 168 deletions
diff --git a/arch/x86/xen/Makefile b/arch/x86/xen/Makefile index 59c1e539aed2..5bfee243cf9a 100644 --- a/arch/x86/xen/Makefile +++ b/arch/x86/xen/Makefile | |||
@@ -1,4 +1,10 @@ | |||
1 | ifdef CONFIG_FTRACE | ||
2 | # Do not profile debug and lowlevel utilities | ||
3 | CFLAGS_REMOVE_spinlock.o = -pg | ||
4 | CFLAGS_REMOVE_time.o = -pg | ||
5 | endif | ||
6 | |||
1 | obj-y := enlighten.o setup.o multicalls.o mmu.o \ | 7 | obj-y := enlighten.o setup.o multicalls.o mmu.o \ |
2 | time.o xen-asm_$(BITS).o grant-table.o suspend.o | 8 | time.o xen-asm_$(BITS).o grant-table.o suspend.o |
3 | 9 | ||
4 | obj-$(CONFIG_SMP) += smp.o | 10 | obj-$(CONFIG_SMP) += smp.o spinlock.o |
diff --git a/arch/x86/xen/smp.c b/arch/x86/xen/smp.c index d8faf79a0a1d..baca7f2fbd8a 100644 --- a/arch/x86/xen/smp.c +++ b/arch/x86/xen/smp.c | |||
@@ -15,7 +15,6 @@ | |||
15 | * This does not handle HOTPLUG_CPU yet. | 15 | * This does not handle HOTPLUG_CPU yet. |
16 | */ | 16 | */ |
17 | #include <linux/sched.h> | 17 | #include <linux/sched.h> |
18 | #include <linux/kernel_stat.h> | ||
19 | #include <linux/err.h> | 18 | #include <linux/err.h> |
20 | #include <linux/smp.h> | 19 | #include <linux/smp.h> |
21 | 20 | ||
@@ -36,8 +35,6 @@ | |||
36 | #include "xen-ops.h" | 35 | #include "xen-ops.h" |
37 | #include "mmu.h" | 36 | #include "mmu.h" |
38 | 37 | ||
39 | static void __cpuinit xen_init_lock_cpu(int cpu); | ||
40 | |||
41 | cpumask_t xen_cpu_initialized_map; | 38 | cpumask_t xen_cpu_initialized_map; |
42 | 39 | ||
43 | static DEFINE_PER_CPU(int, resched_irq); | 40 | static DEFINE_PER_CPU(int, resched_irq); |
@@ -419,170 +416,6 @@ static irqreturn_t xen_call_function_single_interrupt(int irq, void *dev_id) | |||
419 | return IRQ_HANDLED; | 416 | return IRQ_HANDLED; |
420 | } | 417 | } |
421 | 418 | ||
422 | struct xen_spinlock { | ||
423 | unsigned char lock; /* 0 -> free; 1 -> locked */ | ||
424 | unsigned short spinners; /* count of waiting cpus */ | ||
425 | }; | ||
426 | |||
427 | static int xen_spin_is_locked(struct raw_spinlock *lock) | ||
428 | { | ||
429 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
430 | |||
431 | return xl->lock != 0; | ||
432 | } | ||
433 | |||
434 | static int xen_spin_is_contended(struct raw_spinlock *lock) | ||
435 | { | ||
436 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
437 | |||
438 | /* Not strictly true; this is only the count of contended | ||
439 | lock-takers entering the slow path. */ | ||
440 | return xl->spinners != 0; | ||
441 | } | ||
442 | |||
443 | static int xen_spin_trylock(struct raw_spinlock *lock) | ||
444 | { | ||
445 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
446 | u8 old = 1; | ||
447 | |||
448 | asm("xchgb %b0,%1" | ||
449 | : "+q" (old), "+m" (xl->lock) : : "memory"); | ||
450 | |||
451 | return old == 0; | ||
452 | } | ||
453 | |||
454 | static DEFINE_PER_CPU(int, lock_kicker_irq) = -1; | ||
455 | static DEFINE_PER_CPU(struct xen_spinlock *, lock_spinners); | ||
456 | |||
457 | static inline void spinning_lock(struct xen_spinlock *xl) | ||
458 | { | ||
459 | __get_cpu_var(lock_spinners) = xl; | ||
460 | wmb(); /* set lock of interest before count */ | ||
461 | asm(LOCK_PREFIX " incw %0" | ||
462 | : "+m" (xl->spinners) : : "memory"); | ||
463 | } | ||
464 | |||
465 | static inline void unspinning_lock(struct xen_spinlock *xl) | ||
466 | { | ||
467 | asm(LOCK_PREFIX " decw %0" | ||
468 | : "+m" (xl->spinners) : : "memory"); | ||
469 | wmb(); /* decrement count before clearing lock */ | ||
470 | __get_cpu_var(lock_spinners) = NULL; | ||
471 | } | ||
472 | |||
473 | static noinline int xen_spin_lock_slow(struct raw_spinlock *lock) | ||
474 | { | ||
475 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
476 | int irq = __get_cpu_var(lock_kicker_irq); | ||
477 | int ret; | ||
478 | |||
479 | /* If kicker interrupts not initialized yet, just spin */ | ||
480 | if (irq == -1) | ||
481 | return 0; | ||
482 | |||
483 | /* announce we're spinning */ | ||
484 | spinning_lock(xl); | ||
485 | |||
486 | /* clear pending */ | ||
487 | xen_clear_irq_pending(irq); | ||
488 | |||
489 | /* check again make sure it didn't become free while | ||
490 | we weren't looking */ | ||
491 | ret = xen_spin_trylock(lock); | ||
492 | if (ret) | ||
493 | goto out; | ||
494 | |||
495 | /* block until irq becomes pending */ | ||
496 | xen_poll_irq(irq); | ||
497 | kstat_this_cpu.irqs[irq]++; | ||
498 | |||
499 | out: | ||
500 | unspinning_lock(xl); | ||
501 | return ret; | ||
502 | } | ||
503 | |||
504 | static void xen_spin_lock(struct raw_spinlock *lock) | ||
505 | { | ||
506 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
507 | int timeout; | ||
508 | u8 oldval; | ||
509 | |||
510 | do { | ||
511 | timeout = 1 << 10; | ||
512 | |||
513 | asm("1: xchgb %1,%0\n" | ||
514 | " testb %1,%1\n" | ||
515 | " jz 3f\n" | ||
516 | "2: rep;nop\n" | ||
517 | " cmpb $0,%0\n" | ||
518 | " je 1b\n" | ||
519 | " dec %2\n" | ||
520 | " jnz 2b\n" | ||
521 | "3:\n" | ||
522 | : "+m" (xl->lock), "=q" (oldval), "+r" (timeout) | ||
523 | : "1" (1) | ||
524 | : "memory"); | ||
525 | |||
526 | } while (unlikely(oldval != 0 && !xen_spin_lock_slow(lock))); | ||
527 | } | ||
528 | |||
529 | static noinline void xen_spin_unlock_slow(struct xen_spinlock *xl) | ||
530 | { | ||
531 | int cpu; | ||
532 | |||
533 | for_each_online_cpu(cpu) { | ||
534 | /* XXX should mix up next cpu selection */ | ||
535 | if (per_cpu(lock_spinners, cpu) == xl) { | ||
536 | xen_send_IPI_one(cpu, XEN_SPIN_UNLOCK_VECTOR); | ||
537 | break; | ||
538 | } | ||
539 | } | ||
540 | } | ||
541 | |||
542 | static void xen_spin_unlock(struct raw_spinlock *lock) | ||
543 | { | ||
544 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
545 | |||
546 | smp_wmb(); /* make sure no writes get moved after unlock */ | ||
547 | xl->lock = 0; /* release lock */ | ||
548 | |||
549 | /* make sure unlock happens before kick */ | ||
550 | barrier(); | ||
551 | |||
552 | if (unlikely(xl->spinners)) | ||
553 | xen_spin_unlock_slow(xl); | ||
554 | } | ||
555 | |||
556 | static __cpuinit void xen_init_lock_cpu(int cpu) | ||
557 | { | ||
558 | int irq; | ||
559 | const char *name; | ||
560 | |||
561 | name = kasprintf(GFP_KERNEL, "spinlock%d", cpu); | ||
562 | irq = bind_ipi_to_irqhandler(XEN_SPIN_UNLOCK_VECTOR, | ||
563 | cpu, | ||
564 | xen_reschedule_interrupt, | ||
565 | IRQF_DISABLED|IRQF_PERCPU|IRQF_NOBALANCING, | ||
566 | name, | ||
567 | NULL); | ||
568 | |||
569 | if (irq >= 0) { | ||
570 | disable_irq(irq); /* make sure it's never delivered */ | ||
571 | per_cpu(lock_kicker_irq, cpu) = irq; | ||
572 | } | ||
573 | |||
574 | printk("cpu %d spinlock event irq %d\n", cpu, irq); | ||
575 | } | ||
576 | |||
577 | static void __init xen_init_spinlocks(void) | ||
578 | { | ||
579 | pv_lock_ops.spin_is_locked = xen_spin_is_locked; | ||
580 | pv_lock_ops.spin_is_contended = xen_spin_is_contended; | ||
581 | pv_lock_ops.spin_lock = xen_spin_lock; | ||
582 | pv_lock_ops.spin_trylock = xen_spin_trylock; | ||
583 | pv_lock_ops.spin_unlock = xen_spin_unlock; | ||
584 | } | ||
585 | |||
586 | static const struct smp_ops xen_smp_ops __initdata = { | 419 | static const struct smp_ops xen_smp_ops __initdata = { |
587 | .smp_prepare_boot_cpu = xen_smp_prepare_boot_cpu, | 420 | .smp_prepare_boot_cpu = xen_smp_prepare_boot_cpu, |
588 | .smp_prepare_cpus = xen_smp_prepare_cpus, | 421 | .smp_prepare_cpus = xen_smp_prepare_cpus, |
diff --git a/arch/x86/xen/spinlock.c b/arch/x86/xen/spinlock.c new file mode 100644 index 000000000000..8dc4d31da67f --- /dev/null +++ b/arch/x86/xen/spinlock.c | |||
@@ -0,0 +1,183 @@ | |||
1 | /* | ||
2 | * Split spinlock implementation out into its own file, so it can be | ||
3 | * compiled in a FTRACE-compatible way. | ||
4 | */ | ||
5 | #include <linux/kernel_stat.h> | ||
6 | #include <linux/spinlock.h> | ||
7 | |||
8 | #include <asm/paravirt.h> | ||
9 | |||
10 | #include <xen/interface/xen.h> | ||
11 | #include <xen/events.h> | ||
12 | |||
13 | #include "xen-ops.h" | ||
14 | |||
15 | struct xen_spinlock { | ||
16 | unsigned char lock; /* 0 -> free; 1 -> locked */ | ||
17 | unsigned short spinners; /* count of waiting cpus */ | ||
18 | }; | ||
19 | |||
20 | static int xen_spin_is_locked(struct raw_spinlock *lock) | ||
21 | { | ||
22 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
23 | |||
24 | return xl->lock != 0; | ||
25 | } | ||
26 | |||
27 | static int xen_spin_is_contended(struct raw_spinlock *lock) | ||
28 | { | ||
29 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
30 | |||
31 | /* Not strictly true; this is only the count of contended | ||
32 | lock-takers entering the slow path. */ | ||
33 | return xl->spinners != 0; | ||
34 | } | ||
35 | |||
36 | static int xen_spin_trylock(struct raw_spinlock *lock) | ||
37 | { | ||
38 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
39 | u8 old = 1; | ||
40 | |||
41 | asm("xchgb %b0,%1" | ||
42 | : "+q" (old), "+m" (xl->lock) : : "memory"); | ||
43 | |||
44 | return old == 0; | ||
45 | } | ||
46 | |||
47 | static DEFINE_PER_CPU(int, lock_kicker_irq) = -1; | ||
48 | static DEFINE_PER_CPU(struct xen_spinlock *, lock_spinners); | ||
49 | |||
50 | static inline void spinning_lock(struct xen_spinlock *xl) | ||
51 | { | ||
52 | __get_cpu_var(lock_spinners) = xl; | ||
53 | wmb(); /* set lock of interest before count */ | ||
54 | asm(LOCK_PREFIX " incw %0" | ||
55 | : "+m" (xl->spinners) : : "memory"); | ||
56 | } | ||
57 | |||
58 | static inline void unspinning_lock(struct xen_spinlock *xl) | ||
59 | { | ||
60 | asm(LOCK_PREFIX " decw %0" | ||
61 | : "+m" (xl->spinners) : : "memory"); | ||
62 | wmb(); /* decrement count before clearing lock */ | ||
63 | __get_cpu_var(lock_spinners) = NULL; | ||
64 | } | ||
65 | |||
66 | static noinline int xen_spin_lock_slow(struct raw_spinlock *lock) | ||
67 | { | ||
68 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
69 | int irq = __get_cpu_var(lock_kicker_irq); | ||
70 | int ret; | ||
71 | |||
72 | /* If kicker interrupts not initialized yet, just spin */ | ||
73 | if (irq == -1) | ||
74 | return 0; | ||
75 | |||
76 | /* announce we're spinning */ | ||
77 | spinning_lock(xl); | ||
78 | |||
79 | /* clear pending */ | ||
80 | xen_clear_irq_pending(irq); | ||
81 | |||
82 | /* check again make sure it didn't become free while | ||
83 | we weren't looking */ | ||
84 | ret = xen_spin_trylock(lock); | ||
85 | if (ret) | ||
86 | goto out; | ||
87 | |||
88 | /* block until irq becomes pending */ | ||
89 | xen_poll_irq(irq); | ||
90 | kstat_this_cpu.irqs[irq]++; | ||
91 | |||
92 | out: | ||
93 | unspinning_lock(xl); | ||
94 | return ret; | ||
95 | } | ||
96 | |||
97 | static void xen_spin_lock(struct raw_spinlock *lock) | ||
98 | { | ||
99 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
100 | int timeout; | ||
101 | u8 oldval; | ||
102 | |||
103 | do { | ||
104 | timeout = 1 << 10; | ||
105 | |||
106 | asm("1: xchgb %1,%0\n" | ||
107 | " testb %1,%1\n" | ||
108 | " jz 3f\n" | ||
109 | "2: rep;nop\n" | ||
110 | " cmpb $0,%0\n" | ||
111 | " je 1b\n" | ||
112 | " dec %2\n" | ||
113 | " jnz 2b\n" | ||
114 | "3:\n" | ||
115 | : "+m" (xl->lock), "=q" (oldval), "+r" (timeout) | ||
116 | : "1" (1) | ||
117 | : "memory"); | ||
118 | |||
119 | } while (unlikely(oldval != 0 && !xen_spin_lock_slow(lock))); | ||
120 | } | ||
121 | |||
122 | static noinline void xen_spin_unlock_slow(struct xen_spinlock *xl) | ||
123 | { | ||
124 | int cpu; | ||
125 | |||
126 | for_each_online_cpu(cpu) { | ||
127 | /* XXX should mix up next cpu selection */ | ||
128 | if (per_cpu(lock_spinners, cpu) == xl) { | ||
129 | xen_send_IPI_one(cpu, XEN_SPIN_UNLOCK_VECTOR); | ||
130 | break; | ||
131 | } | ||
132 | } | ||
133 | } | ||
134 | |||
135 | static void xen_spin_unlock(struct raw_spinlock *lock) | ||
136 | { | ||
137 | struct xen_spinlock *xl = (struct xen_spinlock *)lock; | ||
138 | |||
139 | smp_wmb(); /* make sure no writes get moved after unlock */ | ||
140 | xl->lock = 0; /* release lock */ | ||
141 | |||
142 | /* make sure unlock happens before kick */ | ||
143 | barrier(); | ||
144 | |||
145 | if (unlikely(xl->spinners)) | ||
146 | xen_spin_unlock_slow(xl); | ||
147 | } | ||
148 | |||
149 | static irqreturn_t dummy_handler(int irq, void *dev_id) | ||
150 | { | ||
151 | BUG(); | ||
152 | return IRQ_HANDLED; | ||
153 | } | ||
154 | |||
155 | void __cpuinit xen_init_lock_cpu(int cpu) | ||
156 | { | ||
157 | int irq; | ||
158 | const char *name; | ||
159 | |||
160 | name = kasprintf(GFP_KERNEL, "spinlock%d", cpu); | ||
161 | irq = bind_ipi_to_irqhandler(XEN_SPIN_UNLOCK_VECTOR, | ||
162 | cpu, | ||
163 | dummy_handler, | ||
164 | IRQF_DISABLED|IRQF_PERCPU|IRQF_NOBALANCING, | ||
165 | name, | ||
166 | NULL); | ||
167 | |||
168 | if (irq >= 0) { | ||
169 | disable_irq(irq); /* make sure it's never delivered */ | ||
170 | per_cpu(lock_kicker_irq, cpu) = irq; | ||
171 | } | ||
172 | |||
173 | printk("cpu %d spinlock event irq %d\n", cpu, irq); | ||
174 | } | ||
175 | |||
176 | void __init xen_init_spinlocks(void) | ||
177 | { | ||
178 | pv_lock_ops.spin_is_locked = xen_spin_is_locked; | ||
179 | pv_lock_ops.spin_is_contended = xen_spin_is_contended; | ||
180 | pv_lock_ops.spin_lock = xen_spin_lock; | ||
181 | pv_lock_ops.spin_trylock = xen_spin_trylock; | ||
182 | pv_lock_ops.spin_unlock = xen_spin_unlock; | ||
183 | } | ||
diff --git a/arch/x86/xen/xen-ops.h b/arch/x86/xen/xen-ops.h index dd3c23152a2e..8847fb34f17e 100644 --- a/arch/x86/xen/xen-ops.h +++ b/arch/x86/xen/xen-ops.h | |||
@@ -50,6 +50,9 @@ void __init xen_setup_vcpu_info_placement(void); | |||
50 | #ifdef CONFIG_SMP | 50 | #ifdef CONFIG_SMP |
51 | void xen_smp_init(void); | 51 | void xen_smp_init(void); |
52 | 52 | ||
53 | void __init xen_init_spinlocks(void); | ||
54 | __cpuinit void xen_init_lock_cpu(int cpu); | ||
55 | |||
53 | extern cpumask_t xen_cpu_initialized_map; | 56 | extern cpumask_t xen_cpu_initialized_map; |
54 | #else | 57 | #else |
55 | static inline void xen_smp_init(void) {} | 58 | static inline void xen_smp_init(void) {} |