diff options
author | Markus Metzger <markus.t.metzger@intel.com> | 2009-04-03 10:43:37 -0400 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2009-04-07 07:36:16 -0400 |
commit | 38f801129ad07b9afa7f9bd3779f61b805416d8c (patch) | |
tree | 2a9fba8e7abfece1378e14472650e2e1cfb9403e /arch/x86/kernel/ds.c | |
parent | 8d99b3ac2726e5edd97ad147fa5c1f2acb63a745 (diff) |
x86, bts: fix race between per-task and per-cpu branch tracing
Per-task branch tracing installs a debug store context with the traced
task. This immediately results in the branch trace control bits to be
cleared for the next context switch of that task, if not set before.
Either per-cpu or per-task tracing are allowed at the same time.
An active per-cpu tracing would be disabled even if the per-task tracing
request is rejected and the task debug store context removed.
Check the tracing type (per-cpu or per-task) before installing a task
debug store context.
Signed-off-by: Markus Metzger <markus.t.metzger@intel.com>
Cc: roland@redhat.com
Cc: eranian@googlemail.com
Cc: oleg@redhat.com
Cc: juan.villacis@intel.com
Cc: ak@linux.jf.intel.com
LKML-Reference: <20090403144552.856000000@intel.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'arch/x86/kernel/ds.c')
-rw-r--r-- | arch/x86/kernel/ds.c | 72 |
1 files changed, 41 insertions, 31 deletions
diff --git a/arch/x86/kernel/ds.c b/arch/x86/kernel/ds.c index 5cd137ab2672..f03f117eff8c 100644 --- a/arch/x86/kernel/ds.c +++ b/arch/x86/kernel/ds.c | |||
@@ -193,12 +193,28 @@ static DEFINE_SPINLOCK(ds_lock); | |||
193 | */ | 193 | */ |
194 | static atomic_t tracers = ATOMIC_INIT(0); | 194 | static atomic_t tracers = ATOMIC_INIT(0); |
195 | 195 | ||
196 | static inline void get_tracer(struct task_struct *task) | 196 | static inline int get_tracer(struct task_struct *task) |
197 | { | 197 | { |
198 | if (task) | 198 | int error; |
199 | |||
200 | spin_lock_irq(&ds_lock); | ||
201 | |||
202 | if (task) { | ||
203 | error = -EPERM; | ||
204 | if (atomic_read(&tracers) < 0) | ||
205 | goto out; | ||
199 | atomic_inc(&tracers); | 206 | atomic_inc(&tracers); |
200 | else | 207 | } else { |
208 | error = -EPERM; | ||
209 | if (atomic_read(&tracers) > 0) | ||
210 | goto out; | ||
201 | atomic_dec(&tracers); | 211 | atomic_dec(&tracers); |
212 | } | ||
213 | |||
214 | error = 0; | ||
215 | out: | ||
216 | spin_unlock_irq(&ds_lock); | ||
217 | return error; | ||
202 | } | 218 | } |
203 | 219 | ||
204 | static inline void put_tracer(struct task_struct *task) | 220 | static inline void put_tracer(struct task_struct *task) |
@@ -209,14 +225,6 @@ static inline void put_tracer(struct task_struct *task) | |||
209 | atomic_inc(&tracers); | 225 | atomic_inc(&tracers); |
210 | } | 226 | } |
211 | 227 | ||
212 | static inline int check_tracer(struct task_struct *task) | ||
213 | { | ||
214 | return task ? | ||
215 | (atomic_read(&tracers) >= 0) : | ||
216 | (atomic_read(&tracers) <= 0); | ||
217 | } | ||
218 | |||
219 | |||
220 | /* | 228 | /* |
221 | * The DS context is either attached to a thread or to a cpu: | 229 | * The DS context is either attached to a thread or to a cpu: |
222 | * - in the former case, the thread_struct contains a pointer to the | 230 | * - in the former case, the thread_struct contains a pointer to the |
@@ -677,6 +685,10 @@ struct bts_tracer *ds_request_bts(struct task_struct *task, | |||
677 | if (ovfl) | 685 | if (ovfl) |
678 | goto out; | 686 | goto out; |
679 | 687 | ||
688 | error = get_tracer(task); | ||
689 | if (error < 0) | ||
690 | goto out; | ||
691 | |||
680 | /* | 692 | /* |
681 | * Per-cpu tracing is typically requested using smp_call_function(). | 693 | * Per-cpu tracing is typically requested using smp_call_function(). |
682 | * We must not sleep. | 694 | * We must not sleep. |
@@ -684,7 +696,7 @@ struct bts_tracer *ds_request_bts(struct task_struct *task, | |||
684 | error = -ENOMEM; | 696 | error = -ENOMEM; |
685 | tracer = kzalloc(sizeof(*tracer), GFP_ATOMIC); | 697 | tracer = kzalloc(sizeof(*tracer), GFP_ATOMIC); |
686 | if (!tracer) | 698 | if (!tracer) |
687 | goto out; | 699 | goto out_put_tracer; |
688 | tracer->ovfl = ovfl; | 700 | tracer->ovfl = ovfl; |
689 | 701 | ||
690 | error = ds_request(&tracer->ds, &tracer->trace.ds, | 702 | error = ds_request(&tracer->ds, &tracer->trace.ds, |
@@ -696,13 +708,8 @@ struct bts_tracer *ds_request_bts(struct task_struct *task, | |||
696 | spin_lock_irqsave(&ds_lock, irq); | 708 | spin_lock_irqsave(&ds_lock, irq); |
697 | 709 | ||
698 | error = -EPERM; | 710 | error = -EPERM; |
699 | if (!check_tracer(task)) | ||
700 | goto out_unlock; | ||
701 | get_tracer(task); | ||
702 | |||
703 | error = -EPERM; | ||
704 | if (tracer->ds.context->bts_master) | 711 | if (tracer->ds.context->bts_master) |
705 | goto out_put_tracer; | 712 | goto out_unlock; |
706 | tracer->ds.context->bts_master = tracer; | 713 | tracer->ds.context->bts_master = tracer; |
707 | 714 | ||
708 | spin_unlock_irqrestore(&ds_lock, irq); | 715 | spin_unlock_irqrestore(&ds_lock, irq); |
@@ -716,13 +723,13 @@ struct bts_tracer *ds_request_bts(struct task_struct *task, | |||
716 | 723 | ||
717 | return tracer; | 724 | return tracer; |
718 | 725 | ||
719 | out_put_tracer: | ||
720 | put_tracer(task); | ||
721 | out_unlock: | 726 | out_unlock: |
722 | spin_unlock_irqrestore(&ds_lock, irq); | 727 | spin_unlock_irqrestore(&ds_lock, irq); |
723 | ds_put_context(tracer->ds.context); | 728 | ds_put_context(tracer->ds.context); |
724 | out_tracer: | 729 | out_tracer: |
725 | kfree(tracer); | 730 | kfree(tracer); |
731 | out_put_tracer: | ||
732 | put_tracer(task); | ||
726 | out: | 733 | out: |
727 | return ERR_PTR(error); | 734 | return ERR_PTR(error); |
728 | } | 735 | } |
@@ -741,6 +748,10 @@ struct pebs_tracer *ds_request_pebs(struct task_struct *task, | |||
741 | if (ovfl) | 748 | if (ovfl) |
742 | goto out; | 749 | goto out; |
743 | 750 | ||
751 | error = get_tracer(task); | ||
752 | if (error < 0) | ||
753 | goto out; | ||
754 | |||
744 | /* | 755 | /* |
745 | * Per-cpu tracing is typically requested using smp_call_function(). | 756 | * Per-cpu tracing is typically requested using smp_call_function(). |
746 | * We must not sleep. | 757 | * We must not sleep. |
@@ -748,7 +759,7 @@ struct pebs_tracer *ds_request_pebs(struct task_struct *task, | |||
748 | error = -ENOMEM; | 759 | error = -ENOMEM; |
749 | tracer = kzalloc(sizeof(*tracer), GFP_ATOMIC); | 760 | tracer = kzalloc(sizeof(*tracer), GFP_ATOMIC); |
750 | if (!tracer) | 761 | if (!tracer) |
751 | goto out; | 762 | goto out_put_tracer; |
752 | tracer->ovfl = ovfl; | 763 | tracer->ovfl = ovfl; |
753 | 764 | ||
754 | error = ds_request(&tracer->ds, &tracer->trace.ds, | 765 | error = ds_request(&tracer->ds, &tracer->trace.ds, |
@@ -759,13 +770,8 @@ struct pebs_tracer *ds_request_pebs(struct task_struct *task, | |||
759 | spin_lock_irqsave(&ds_lock, irq); | 770 | spin_lock_irqsave(&ds_lock, irq); |
760 | 771 | ||
761 | error = -EPERM; | 772 | error = -EPERM; |
762 | if (!check_tracer(task)) | ||
763 | goto out_unlock; | ||
764 | get_tracer(task); | ||
765 | |||
766 | error = -EPERM; | ||
767 | if (tracer->ds.context->pebs_master) | 773 | if (tracer->ds.context->pebs_master) |
768 | goto out_put_tracer; | 774 | goto out_unlock; |
769 | tracer->ds.context->pebs_master = tracer; | 775 | tracer->ds.context->pebs_master = tracer; |
770 | 776 | ||
771 | spin_unlock_irqrestore(&ds_lock, irq); | 777 | spin_unlock_irqrestore(&ds_lock, irq); |
@@ -775,13 +781,13 @@ struct pebs_tracer *ds_request_pebs(struct task_struct *task, | |||
775 | 781 | ||
776 | return tracer; | 782 | return tracer; |
777 | 783 | ||
778 | out_put_tracer: | ||
779 | put_tracer(task); | ||
780 | out_unlock: | 784 | out_unlock: |
781 | spin_unlock_irqrestore(&ds_lock, irq); | 785 | spin_unlock_irqrestore(&ds_lock, irq); |
782 | ds_put_context(tracer->ds.context); | 786 | ds_put_context(tracer->ds.context); |
783 | out_tracer: | 787 | out_tracer: |
784 | kfree(tracer); | 788 | kfree(tracer); |
789 | out_put_tracer: | ||
790 | put_tracer(task); | ||
785 | out: | 791 | out: |
786 | return ERR_PTR(error); | 792 | return ERR_PTR(error); |
787 | } | 793 | } |
@@ -804,8 +810,8 @@ void ds_release_bts(struct bts_tracer *tracer) | |||
804 | if (task && (task != current)) | 810 | if (task && (task != current)) |
805 | wait_task_context_switch(task); | 811 | wait_task_context_switch(task); |
806 | 812 | ||
807 | put_tracer(task); | ||
808 | ds_put_context(tracer->ds.context); | 813 | ds_put_context(tracer->ds.context); |
814 | put_tracer(task); | ||
809 | 815 | ||
810 | kfree(tracer); | 816 | kfree(tracer); |
811 | } | 817 | } |
@@ -861,16 +867,20 @@ void ds_resume_bts(struct bts_tracer *tracer) | |||
861 | 867 | ||
862 | void ds_release_pebs(struct pebs_tracer *tracer) | 868 | void ds_release_pebs(struct pebs_tracer *tracer) |
863 | { | 869 | { |
870 | struct task_struct *task; | ||
871 | |||
864 | if (!tracer) | 872 | if (!tracer) |
865 | return; | 873 | return; |
866 | 874 | ||
875 | task = tracer->ds.context->task; | ||
876 | |||
867 | ds_suspend_pebs(tracer); | 877 | ds_suspend_pebs(tracer); |
868 | 878 | ||
869 | WARN_ON_ONCE(tracer->ds.context->pebs_master != tracer); | 879 | WARN_ON_ONCE(tracer->ds.context->pebs_master != tracer); |
870 | tracer->ds.context->pebs_master = NULL; | 880 | tracer->ds.context->pebs_master = NULL; |
871 | 881 | ||
872 | put_tracer(tracer->ds.context->task); | ||
873 | ds_put_context(tracer->ds.context); | 882 | ds_put_context(tracer->ds.context); |
883 | put_tracer(task); | ||
874 | 884 | ||
875 | kfree(tracer); | 885 | kfree(tracer); |
876 | } | 886 | } |