diff options
author | Heiko Carstens <heiko.carstens@de.ibm.com> | 2007-02-21 04:55:15 -0500 |
---|---|---|
committer | Martin Schwidefsky <schwidefsky@de.ibm.com> | 2007-02-21 04:55:15 -0500 |
commit | 6c732de2d3673e28c6a976c98ee6ba4d197a919a (patch) | |
tree | 759b9e1d5a1a8ac33dee9ad5eb670ff99cb2d5ea | |
parent | 118bcd31b309d12638f67729d5d96d4974750249 (diff) |
[S390] etr: Add barrier() to etr_sync_cpu_start().
Force reading of *in_sync in while loop. Loops where the content that
is checked for is changed by a different cpu always should have some
sort of barrier() semantics.
Otherwise this might lead to very subtle bugs.
Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
-rw-r--r-- | arch/s390/kernel/time.c | 10 |
1 files changed, 9 insertions, 1 deletions
diff --git a/arch/s390/kernel/time.c b/arch/s390/kernel/time.c index ee9fd7b85928..e1ad464b6f20 100644 --- a/arch/s390/kernel/time.c +++ b/arch/s390/kernel/time.c | |||
@@ -747,6 +747,7 @@ static void etr_adjust_time(unsigned long long clock, unsigned long long delay) | |||
747 | } | 747 | } |
748 | } | 748 | } |
749 | 749 | ||
750 | #ifdef CONFIG_SMP | ||
750 | static void etr_sync_cpu_start(void *dummy) | 751 | static void etr_sync_cpu_start(void *dummy) |
751 | { | 752 | { |
752 | int *in_sync = dummy; | 753 | int *in_sync = dummy; |
@@ -758,8 +759,14 @@ static void etr_sync_cpu_start(void *dummy) | |||
758 | * __udelay will stop the cpu on an enabled wait psw until the | 759 | * __udelay will stop the cpu on an enabled wait psw until the |
759 | * TOD is running again. | 760 | * TOD is running again. |
760 | */ | 761 | */ |
761 | while (*in_sync == 0) | 762 | while (*in_sync == 0) { |
762 | __udelay(1); | 763 | __udelay(1); |
764 | /* | ||
765 | * A different cpu changes *in_sync. Therefore use | ||
766 | * barrier() to force memory access. | ||
767 | */ | ||
768 | barrier(); | ||
769 | } | ||
763 | if (*in_sync != 1) | 770 | if (*in_sync != 1) |
764 | /* Didn't work. Clear per-cpu in sync bit again. */ | 771 | /* Didn't work. Clear per-cpu in sync bit again. */ |
765 | etr_disable_sync_clock(NULL); | 772 | etr_disable_sync_clock(NULL); |
@@ -773,6 +780,7 @@ static void etr_sync_cpu_start(void *dummy) | |||
773 | static void etr_sync_cpu_end(void *dummy) | 780 | static void etr_sync_cpu_end(void *dummy) |
774 | { | 781 | { |
775 | } | 782 | } |
783 | #endif /* CONFIG_SMP */ | ||
776 | 784 | ||
777 | /* | 785 | /* |
778 | * Sync the TOD clock using the port refered to by aibp. This port | 786 | * Sync the TOD clock using the port refered to by aibp. This port |