Blame view

include/asm-powerpc/time.h 5.67 KB
f2783c150   Paul Mackerras   powerpc: Merge ti...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
  /*
   * Common time prototypes and such for all ppc machines.
   *
   * Written by Cort Dougan (cort@cs.nmt.edu) to merge
   * Paul Mackerras' version and mine for PReP and Pmac.
   *
   * This program is free software; you can redistribute it and/or
   * modify it under the terms of the GNU General Public License
   * as published by the Free Software Foundation; either version
   * 2 of the License, or (at your option) any later version.
   */
  
  #ifndef __POWERPC_TIME_H
  #define __POWERPC_TIME_H
  
  #ifdef __KERNEL__
f2783c150   Paul Mackerras   powerpc: Merge ti...
17
18
19
20
  #include <linux/types.h>
  #include <linux/percpu.h>
  
  #include <asm/processor.h>
9b47569a9   Stephen Rothwell   [POWERPC] update ...
21
  #ifdef CONFIG_PPC_ISERIES
f2783c150   Paul Mackerras   powerpc: Merge ti...
22
  #include <asm/paca.h>
9b47569a9   Stephen Rothwell   [POWERPC] update ...
23
  #include <asm/firmware.h>
1da440378   Kelly Daly   merge filename an...
24
  #include <asm/iseries/hv_call.h>
f2783c150   Paul Mackerras   powerpc: Merge ti...
25
26
27
28
29
30
31
32
  #endif
  
  /* time.c */
  extern unsigned long tb_ticks_per_jiffy;
  extern unsigned long tb_ticks_per_usec;
  extern unsigned long tb_ticks_per_sec;
  extern u64 tb_to_xs;
  extern unsigned      tb_to_us;
f2783c150   Paul Mackerras   powerpc: Merge ti...
33
34
35
36
37
38
39
  
  struct rtc_time;
  extern void to_tm(int tim, struct rtc_time * tm);
  extern time_t last_rtc_update;
  
  extern void generic_calibrate_decr(void);
  extern void wakeup_decrementer(void);
c6622f63d   Paul Mackerras   powerpc: Implemen...
40
  extern void snapshot_timebase(void);
f2783c150   Paul Mackerras   powerpc: Merge ti...
41

f2a0bd375   Vitaly Bordug   [POWERPC] 8xx: po...
42
  extern void set_dec_cpu6(unsigned int val);
f2783c150   Paul Mackerras   powerpc: Merge ti...
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
  /* Some sane defaults: 125 MHz timebase, 1GHz processor */
  extern unsigned long ppc_proc_freq;
  #define DEFAULT_PROC_FREQ	(DEFAULT_TB_FREQ * 8)
  extern unsigned long ppc_tb_freq;
  #define DEFAULT_TB_FREQ		125000000UL
  
  /*
   * By putting all of this stuff into a single struct we 
   * reduce the number of cache lines touched by do_gettimeofday.
   * Both by collecting all of the data in one cache line and
   * by touching only one TOC entry on ppc64.
   */
  struct gettimeofday_vars {
  	u64 tb_to_xs;
  	u64 stamp_xsec;
  	u64 tb_orig_stamp;
  };
  
  struct gettimeofday_struct {
  	unsigned long tb_ticks_per_sec;
  	struct gettimeofday_vars vars[2];
  	struct gettimeofday_vars * volatile varp;
  	unsigned      var_idx;
  	unsigned      tb_to_us;
  };
  
  struct div_result {
  	u64 result_high;
  	u64 result_low;
  };
  
  /* Accessor functions for the timebase (RTC on 601) registers. */
  /* If one day CONFIG_POWER is added just define __USE_RTC as 1 */
  #ifdef CONFIG_6xx
5d14a18d5   Paul Mackerras   powerpc: Fix some...
77
  #define __USE_RTC()	(!cpu_has_feature(CPU_FTR_USE_TB))
f2783c150   Paul Mackerras   powerpc: Merge ti...
78
79
80
  #else
  #define __USE_RTC()	0
  #endif
859deea94   Benjamin Herrenschmidt   [POWERPC] Cell ti...
81
82
83
84
85
86
  #ifdef CONFIG_PPC64
  
  /* For compatibility, get_tbl() is defined as get_tb() on ppc64 */
  #define get_tbl		get_tb
  
  #else
f2783c150   Paul Mackerras   powerpc: Merge ti...
87
88
  static inline unsigned long get_tbl(void)
  {
f2783c150   Paul Mackerras   powerpc: Merge ti...
89
  #if defined(CONFIG_403GCX)
859deea94   Benjamin Herrenschmidt   [POWERPC] Cell ti...
90
  	unsigned long tbl;
f2783c150   Paul Mackerras   powerpc: Merge ti...
91
  	asm volatile("mfspr %0, 0x3dd" : "=r" (tbl));
859deea94   Benjamin Herrenschmidt   [POWERPC] Cell ti...
92
  	return tbl;
f2783c150   Paul Mackerras   powerpc: Merge ti...
93
  #else
859deea94   Benjamin Herrenschmidt   [POWERPC] Cell ti...
94
  	return mftbl();
f2783c150   Paul Mackerras   powerpc: Merge ti...
95
  #endif
f2783c150   Paul Mackerras   powerpc: Merge ti...
96
97
98
99
  }
  
  static inline unsigned int get_tbu(void)
  {
859deea94   Benjamin Herrenschmidt   [POWERPC] Cell ti...
100
  #ifdef CONFIG_403GCX
f2783c150   Paul Mackerras   powerpc: Merge ti...
101
  	unsigned int tbu;
f2783c150   Paul Mackerras   powerpc: Merge ti...
102
  	asm volatile("mfspr %0, 0x3dc" : "=r" (tbu));
859deea94   Benjamin Herrenschmidt   [POWERPC] Cell ti...
103
  	return tbu;
f2783c150   Paul Mackerras   powerpc: Merge ti...
104
  #else
859deea94   Benjamin Herrenschmidt   [POWERPC] Cell ti...
105
  	return mftbu();
f2783c150   Paul Mackerras   powerpc: Merge ti...
106
  #endif
f2783c150   Paul Mackerras   powerpc: Merge ti...
107
  }
859deea94   Benjamin Herrenschmidt   [POWERPC] Cell ti...
108
  #endif /* !CONFIG_PPC64 */
f2783c150   Paul Mackerras   powerpc: Merge ti...
109
110
111
112
113
114
115
116
  
  static inline unsigned int get_rtcl(void)
  {
  	unsigned int rtcl;
  
  	asm volatile("mfrtcl %0" : "=r" (rtcl));
  	return rtcl;
  }
96c445076   Paul Mackerras   powerpc: Fix time...
117
118
119
120
121
122
123
124
125
126
  static inline u64 get_rtc(void)
  {
  	unsigned int hi, lo, hi2;
  
  	do {
  		asm volatile("mfrtcu %0; mfrtcl %1; mfrtcu %2"
  			     : "=r" (hi), "=r" (lo), "=r" (hi2));
  	} while (hi2 != hi);
  	return (u64)hi * 1000000000 + lo;
  }
f2783c150   Paul Mackerras   powerpc: Merge ti...
127
128
129
130
131
  #ifdef CONFIG_PPC64
  static inline u64 get_tb(void)
  {
  	return mftb();
  }
859deea94   Benjamin Herrenschmidt   [POWERPC] Cell ti...
132
  #else /* CONFIG_PPC64 */
f2783c150   Paul Mackerras   powerpc: Merge ti...
133
134
135
136
137
138
139
140
141
142
143
144
  static inline u64 get_tb(void)
  {
  	unsigned int tbhi, tblo, tbhi2;
  
  	do {
  		tbhi = get_tbu();
  		tblo = get_tbl();
  		tbhi2 = get_tbu();
  	} while (tbhi != tbhi2);
  
  	return ((u64)tbhi << 32) | tblo;
  }
859deea94   Benjamin Herrenschmidt   [POWERPC] Cell ti...
145
  #endif /* !CONFIG_PPC64 */
f2783c150   Paul Mackerras   powerpc: Merge ti...
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
  
  static inline void set_tb(unsigned int upper, unsigned int lower)
  {
  	mtspr(SPRN_TBWL, 0);
  	mtspr(SPRN_TBWU, upper);
  	mtspr(SPRN_TBWL, lower);
  }
  
  /* Accessor functions for the decrementer register.
   * The 4xx doesn't even have a decrementer.  I tried to use the
   * generic timer interrupt code, which seems OK, with the 4xx PIT
   * in auto-reload mode.  The problem is PIT stops counting when it
   * hits zero.  If it would wrap, we could use it just like a decrementer.
   */
  static inline unsigned int get_dec(void)
  {
  #if defined(CONFIG_40x)
  	return (mfspr(SPRN_PIT));
  #else
  	return (mfspr(SPRN_DEC));
  #endif
  }
  
  static inline void set_dec(int val)
  {
  #if defined(CONFIG_40x)
  	return;		/* Have to let it auto-reload */
  #elif defined(CONFIG_8xx_CPU6)
  	set_dec_cpu6(val);
  #else
  #ifdef CONFIG_PPC_ISERIES
f2783c150   Paul Mackerras   powerpc: Merge ti...
177
  	int cur_dec;
9b47569a9   Stephen Rothwell   [POWERPC] update ...
178
179
  	if (firmware_has_feature(FW_FEATURE_ISERIES) &&
  			get_lppaca()->shared_proc) {
3356bb9f7   David Gibson   [PATCH] powerpc: ...
180
  		get_lppaca()->virtual_decr = val;
f2783c150   Paul Mackerras   powerpc: Merge ti...
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
  		cur_dec = get_dec();
  		if (cur_dec > val)
  			HvCall_setVirtualDecr();
  	} else
  #endif
  		mtspr(SPRN_DEC, val);
  #endif /* not 40x or 8xx_CPU6 */
  }
  
  static inline unsigned long tb_ticks_since(unsigned long tstamp)
  {
  	if (__USE_RTC()) {
  		int delta = get_rtcl() - (unsigned int) tstamp;
  		return delta < 0 ? delta + 1000000000 : delta;
  	}
  	return get_tbl() - tstamp;
  }
  
  #define mulhwu(x,y) \
  ({unsigned z; asm ("mulhwu %0,%1,%2" : "=r" (z) : "r" (x), "r" (y)); z;})
  
  #ifdef CONFIG_PPC64
  #define mulhdu(x,y) \
  ({unsigned long z; asm ("mulhdu %0,%1,%2" : "=r" (z) : "r" (x), "r" (y)); z;})
  #else
  extern u64 mulhdu(u64, u64);
  #endif
a5b518ed3   Paul Mackerras   ppc64/powerpc: Fi...
208
209
210
211
212
  extern void smp_space_timers(unsigned int);
  
  extern unsigned mulhwu_scale_factor(unsigned, unsigned);
  extern void div128_by_32(u64 dividend_high, u64 dividend_low,
  			 unsigned divisor, struct div_result *dr);
f2783c150   Paul Mackerras   powerpc: Merge ti...
213
214
215
216
217
218
219
220
  
  /* Used to store Processor Utilization register (purr) values */
  
  struct cpu_usage {
          u64 current_tb;  /* Holds the current purr register values */
  };
  
  DECLARE_PER_CPU(struct cpu_usage, cpu_usage_array);
c6622f63d   Paul Mackerras   powerpc: Implemen...
221
222
223
224
225
  #ifdef CONFIG_VIRT_CPU_ACCOUNTING
  extern void account_process_vtime(struct task_struct *tsk);
  #else
  #define account_process_vtime(tsk)		do { } while (0)
  #endif
4cefebb1b   Michael Neuling   [POWERPC] Fix sto...
226
  #if defined(CONFIG_VIRT_CPU_ACCOUNTING)
c6622f63d   Paul Mackerras   powerpc: Implemen...
227
228
229
230
231
232
  extern void calculate_steal_time(void);
  extern void snapshot_timebases(void);
  #else
  #define calculate_steal_time()			do { } while (0)
  #define snapshot_timebases()			do { } while (0)
  #endif
71712b455   Tony Breeds   [POWERPC] Move iS...
233
  extern void iSeries_time_init_early(void);
f2783c150   Paul Mackerras   powerpc: Merge ti...
234
  #endif /* __KERNEL__ */
7a69af63e   Kim Phillips   [POWERPC] Add pow...
235
  #endif /* __POWERPC_TIME_H */