Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 2877
b: refs/heads/master
c: 8a9e1b0
h: refs/heads/master
i:
  2875: 39c2988
v: v3
  • Loading branch information
Venkatesh Pallipadi authored and Linus Torvalds committed Jun 23, 2005
1 parent 5a72bde commit 7f4767b
Show file tree
Hide file tree
Showing 11 changed files with 131 additions and 1 deletion.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: 0f8e2d62fa04441cd12c08ce521e84e5bd3f8a46
refs/heads/master: 8a9e1b0f564615bd92ba50162623e25c2904e564
9 changes: 9 additions & 0 deletions trunk/arch/i386/kernel/timers/common.c
Original file line number Diff line number Diff line change
Expand Up @@ -139,6 +139,15 @@ unsigned long __init calibrate_tsc_hpet(unsigned long *tsc_hpet_quotient_ptr)
}
#endif


unsigned long read_timer_tsc(void)
{
unsigned long retval;
rdtscl(retval);
return retval;
}


/* calculate cpu_khz */
void init_cpu_khz(void)
{
Expand Down
9 changes: 9 additions & 0 deletions trunk/arch/i386/kernel/timers/timer.c
Original file line number Diff line number Diff line change
Expand Up @@ -64,3 +64,12 @@ struct timer_opts* __init select_timer(void)
panic("select_timer: Cannot find a suitable timer\n");
return NULL;
}

int read_current_timer(unsigned long *timer_val)
{
if (cur_timer->read_timer) {
*timer_val = cur_timer->read_timer();
return 0;
}
return -1;
}
1 change: 1 addition & 0 deletions trunk/arch/i386/kernel/timers/timer_hpet.c
Original file line number Diff line number Diff line change
Expand Up @@ -186,6 +186,7 @@ static struct timer_opts timer_hpet = {
.get_offset = get_offset_hpet,
.monotonic_clock = monotonic_clock_hpet,
.delay = delay_hpet,
.read_timer = read_timer_tsc,
};

struct init_timer_opts __initdata timer_hpet_init = {
Expand Down
1 change: 1 addition & 0 deletions trunk/arch/i386/kernel/timers/timer_pm.c
Original file line number Diff line number Diff line change
Expand Up @@ -246,6 +246,7 @@ static struct timer_opts timer_pmtmr = {
.get_offset = get_offset_pmtmr,
.monotonic_clock = monotonic_clock_pmtmr,
.delay = delay_pmtmr,
.read_timer = read_timer_tsc,
};

struct init_timer_opts __initdata timer_pmtmr_init = {
Expand Down
1 change: 1 addition & 0 deletions trunk/arch/i386/kernel/timers/timer_tsc.c
Original file line number Diff line number Diff line change
Expand Up @@ -572,6 +572,7 @@ static struct timer_opts timer_tsc = {
.get_offset = get_offset_tsc,
.monotonic_clock = monotonic_clock_tsc,
.delay = delay_tsc,
.read_timer = read_timer_tsc,
};

struct init_timer_opts __initdata timer_tsc_init = {
Expand Down
7 changes: 7 additions & 0 deletions trunk/arch/x86_64/lib/delay.c
Original file line number Diff line number Diff line change
Expand Up @@ -12,13 +12,20 @@
#include <linux/sched.h>
#include <linux/delay.h>
#include <asm/delay.h>
#include <asm/msr.h>

#ifdef CONFIG_SMP
#include <asm/smp.h>
#endif

int x86_udelay_tsc = 0; /* Delay via TSC */

int read_current_timer(unsigned long *timer_value)
{
rdtscll(*timer_value);
return 0;
}

void __delay(unsigned long loops)
{
unsigned bclock, now;
Expand Down
2 changes: 2 additions & 0 deletions trunk/include/asm-i386/timer.h
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ struct timer_opts {
unsigned long (*get_offset)(void);
unsigned long long (*monotonic_clock)(void);
void (*delay)(unsigned long);
unsigned long (*read_timer)(void);
};

struct init_timer_opts {
Expand Down Expand Up @@ -52,6 +53,7 @@ extern struct init_timer_opts timer_cyclone_init;
#endif

extern unsigned long calibrate_tsc(void);
extern unsigned long read_timer_tsc(void);
extern void init_cpu_khz(void);
extern int recalibrate_cpu_khz(void);
#ifdef CONFIG_HPET_TIMER
Expand Down
3 changes: 3 additions & 0 deletions trunk/include/asm-i386/timex.h
Original file line number Diff line number Diff line change
Expand Up @@ -49,4 +49,7 @@ static inline cycles_t get_cycles (void)

extern unsigned long cpu_khz;

extern int read_current_timer(unsigned long *timer_value);
#define ARCH_HAS_READ_CURRENT_TIMER 1

#endif
3 changes: 3 additions & 0 deletions trunk/include/asm-x86_64/timex.h
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,9 @@ static inline cycles_t get_cycles (void)

extern unsigned int cpu_khz;

extern int read_current_timer(unsigned long *timer_value);
#define ARCH_HAS_READ_CURRENT_TIMER 1

extern struct vxtime_data vxtime;

#endif
94 changes: 94 additions & 0 deletions trunk/init/calibrate.c
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,8 @@
#include <linux/delay.h>
#include <linux/init.h>

#include <asm/timex.h>

static unsigned long preset_lpj;
static int __init lpj_setup(char *str)
{
Expand All @@ -17,6 +19,92 @@ static int __init lpj_setup(char *str)

__setup("lpj=", lpj_setup);

#ifdef ARCH_HAS_READ_CURRENT_TIMER

/* This routine uses the read_current_timer() routine and gets the
* loops per jiffy directly, instead of guessing it using delay().
* Also, this code tries to handle non-maskable asynchronous events
* (like SMIs)
*/
#define DELAY_CALIBRATION_TICKS ((HZ < 100) ? 1 : (HZ/100))
#define MAX_DIRECT_CALIBRATION_RETRIES 5

static unsigned long __devinit calibrate_delay_direct(void)
{
unsigned long pre_start, start, post_start;
unsigned long pre_end, end, post_end;
unsigned long start_jiffies;
unsigned long tsc_rate_min, tsc_rate_max;
unsigned long good_tsc_sum = 0;
unsigned long good_tsc_count = 0;
int i;

if (read_current_timer(&pre_start) < 0 )
return 0;

/*
* A simple loop like
* while ( jiffies < start_jiffies+1)
* start = read_current_timer();
* will not do. As we don't really know whether jiffy switch
* happened first or timer_value was read first. And some asynchronous
* event can happen between these two events introducing errors in lpj.
*
* So, we do
* 1. pre_start <- When we are sure that jiffy switch hasn't happened
* 2. check jiffy switch
* 3. start <- timer value before or after jiffy switch
* 4. post_start <- When we are sure that jiffy switch has happened
*
* Note, we don't know anything about order of 2 and 3.
* Now, by looking at post_start and pre_start difference, we can
* check whether any asynchronous event happened or not
*/

for (i = 0; i < MAX_DIRECT_CALIBRATION_RETRIES; i++) {
pre_start = 0;
read_current_timer(&start);
start_jiffies = jiffies;
while (jiffies <= (start_jiffies + 1)) {
pre_start = start;
read_current_timer(&start);
}
read_current_timer(&post_start);

pre_end = 0;
end = post_start;
while (jiffies <=
(start_jiffies + 1 + DELAY_CALIBRATION_TICKS)) {
pre_end = end;
read_current_timer(&end);
}
read_current_timer(&post_end);

tsc_rate_max = (post_end - pre_start) / DELAY_CALIBRATION_TICKS;
tsc_rate_min = (pre_end - post_start) / DELAY_CALIBRATION_TICKS;

/*
* If the upper limit and lower limit of the tsc_rate is
* >= 12.5% apart, redo calibration.
*/
if (pre_start != 0 && pre_end != 0 &&
(tsc_rate_max - tsc_rate_min) < (tsc_rate_max >> 3)) {
good_tsc_count++;
good_tsc_sum += tsc_rate_max;
}
}

if (good_tsc_count)
return (good_tsc_sum/good_tsc_count);

printk(KERN_WARNING "calibrate_delay_direct() failed to get a good "
"estimate for loops_per_jiffy.\nProbably due to long platform interrupts. Consider using \"lpj=\" boot option.\n");
return 0;
}
#else
static unsigned long __devinit calibrate_delay_direct(void) {return 0;}
#endif

/*
* This is the number of bits of precision for the loops_per_jiffy. Each
* bit takes on average 1.5/HZ seconds. This (like the original) is a little
Expand All @@ -35,6 +123,12 @@ void __devinit calibrate_delay(void)
"%lu.%02lu BogoMIPS preset\n",
loops_per_jiffy/(500000/HZ),
(loops_per_jiffy/(5000/HZ)) % 100);
} else if ((loops_per_jiffy = calibrate_delay_direct()) != 0) {
printk("Calibrating delay using timer specific routine.. ");
printk("%lu.%02lu BogoMIPS (lpj=%lu)\n",
loops_per_jiffy/(500000/HZ),
(loops_per_jiffy/(5000/HZ)) % 100,
loops_per_jiffy);
} else {
loops_per_jiffy = (1<<12);

Expand Down

0 comments on commit 7f4767b

Please sign in to comment.