diff options
Diffstat (limited to 'drivers/clocksource')
-rw-r--r-- | drivers/clocksource/acpi_pm.c | 60 |
1 files changed, 37 insertions, 23 deletions
diff --git a/drivers/clocksource/acpi_pm.c b/drivers/clocksource/acpi_pm.c index 5ca1d80de182..c20171078d1d 100644 --- a/drivers/clocksource/acpi_pm.c +++ b/drivers/clocksource/acpi_pm.c | |||
@@ -21,6 +21,7 @@ | |||
21 | #include <linux/errno.h> | 21 | #include <linux/errno.h> |
22 | #include <linux/init.h> | 22 | #include <linux/init.h> |
23 | #include <linux/pci.h> | 23 | #include <linux/pci.h> |
24 | #include <linux/delay.h> | ||
24 | #include <asm/io.h> | 25 | #include <asm/io.h> |
25 | 26 | ||
26 | /* | 27 | /* |
@@ -151,13 +152,13 @@ DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_SERVERWORKS, PCI_DEVICE_ID_SERVERWORKS_LE, | |||
151 | */ | 152 | */ |
152 | static int verify_pmtmr_rate(void) | 153 | static int verify_pmtmr_rate(void) |
153 | { | 154 | { |
154 | u32 value1, value2; | 155 | cycle_t value1, value2; |
155 | unsigned long count, delta; | 156 | unsigned long count, delta; |
156 | 157 | ||
157 | mach_prepare_counter(); | 158 | mach_prepare_counter(); |
158 | value1 = read_pmtmr(); | 159 | value1 = clocksource_acpi_pm.read(); |
159 | mach_countup(&count); | 160 | mach_countup(&count); |
160 | value2 = read_pmtmr(); | 161 | value2 = clocksource_acpi_pm.read(); |
161 | delta = (value2 - value1) & ACPI_PM_MASK; | 162 | delta = (value2 - value1) & ACPI_PM_MASK; |
162 | 163 | ||
163 | /* Check that the PMTMR delta is within 5% of what we expect */ | 164 | /* Check that the PMTMR delta is within 5% of what we expect */ |
@@ -175,10 +176,15 @@ static int verify_pmtmr_rate(void) | |||
175 | #define verify_pmtmr_rate() (0) | 176 | #define verify_pmtmr_rate() (0) |
176 | #endif | 177 | #endif |
177 | 178 | ||
179 | /* Number of monotonicity checks to perform during initialization */ | ||
180 | #define ACPI_PM_MONOTONICITY_CHECKS 10 | ||
181 | /* Number of reads we try to get two different values */ | ||
182 | #define ACPI_PM_READ_CHECKS 10000 | ||
183 | |||
178 | static int __init init_acpi_pm_clocksource(void) | 184 | static int __init init_acpi_pm_clocksource(void) |
179 | { | 185 | { |
180 | u32 value1, value2; | 186 | cycle_t value1, value2; |
181 | unsigned int i; | 187 | unsigned int i, j = 0; |
182 | 188 | ||
183 | if (!pmtmr_ioport) | 189 | if (!pmtmr_ioport) |
184 | return -ENODEV; | 190 | return -ENODEV; |
@@ -187,24 +193,29 @@ static int __init init_acpi_pm_clocksource(void) | |||
187 | clocksource_acpi_pm.shift); | 193 | clocksource_acpi_pm.shift); |
188 | 194 | ||
189 | /* "verify" this timing source: */ | 195 | /* "verify" this timing source: */ |
190 | value1 = read_pmtmr(); | 196 | for (j = 0; j < ACPI_PM_MONOTONICITY_CHECKS; j++) { |
191 | for (i = 0; i < 10000; i++) { | 197 | udelay(100 * j); |
192 | value2 = read_pmtmr(); | 198 | value1 = clocksource_acpi_pm.read(); |
193 | if (value2 == value1) | 199 | for (i = 0; i < ACPI_PM_READ_CHECKS; i++) { |
194 | continue; | 200 | value2 = clocksource_acpi_pm.read(); |
195 | if (value2 > value1) | 201 | if (value2 == value1) |
196 | goto pm_good; | 202 | continue; |
197 | if ((value2 < value1) && ((value2) < 0xFFF)) | 203 | if (value2 > value1) |
198 | goto pm_good; | 204 | break; |
199 | printk(KERN_INFO "PM-Timer had inconsistent results:" | 205 | if ((value2 < value1) && ((value2) < 0xFFF)) |
200 | " 0x%#x, 0x%#x - aborting.\n", value1, value2); | 206 | break; |
201 | return -EINVAL; | 207 | printk(KERN_INFO "PM-Timer had inconsistent results:" |
208 | " 0x%#llx, 0x%#llx - aborting.\n", | ||
209 | value1, value2); | ||
210 | return -EINVAL; | ||
211 | } | ||
212 | if (i == ACPI_PM_READ_CHECKS) { | ||
213 | printk(KERN_INFO "PM-Timer failed consistency check " | ||
214 | " (0x%#llx) - aborting.\n", value1); | ||
215 | return -ENODEV; | ||
216 | } | ||
202 | } | 217 | } |
203 | printk(KERN_INFO "PM-Timer had no reasonable result:" | ||
204 | " 0x%#x - aborting.\n", value1); | ||
205 | return -ENODEV; | ||
206 | 218 | ||
207 | pm_good: | ||
208 | if (verify_pmtmr_rate() != 0) | 219 | if (verify_pmtmr_rate() != 0) |
209 | return -ENODEV; | 220 | return -ENODEV; |
210 | 221 | ||
@@ -226,9 +237,12 @@ static int __init parse_pmtmr(char *arg) | |||
226 | 237 | ||
227 | if (strict_strtoul(arg, 16, &base)) | 238 | if (strict_strtoul(arg, 16, &base)) |
228 | return -EINVAL; | 239 | return -EINVAL; |
229 | 240 | #ifdef CONFIG_X86_64 | |
241 | if (base > UINT_MAX) | ||
242 | return -ERANGE; | ||
243 | #endif | ||
230 | printk(KERN_INFO "PMTMR IOPort override: 0x%04x -> 0x%04lx\n", | 244 | printk(KERN_INFO "PMTMR IOPort override: 0x%04x -> 0x%04lx\n", |
231 | (unsigned int)pmtmr_ioport, base); | 245 | pmtmr_ioport, base); |
232 | pmtmr_ioport = base; | 246 | pmtmr_ioport = base; |
233 | 247 | ||
234 | return 1; | 248 | return 1; |