Rev 600 | Go to most recent revision | Details | Last modification | View Log | RSS feed
Rev | Author | Line No. | Line |
---|---|---|---|
582 | mauro | 1 | /* |
2 | * Cyrix MediaGX and NatSemi Geode Suspend Modulation |
||
3 | * (C) 2002 Zwane Mwaikambo <zwane@commfireservices.com> |
||
4 | * (C) 2002 Hiroshi Miura <miura@da-cha.org> |
||
5 | * All Rights Reserved |
||
6 | * |
||
7 | * This program is free software; you can redistribute it and/or |
||
8 | * modify it under the terms of the GNU General Public License |
||
9 | * version 2 as published by the Free Software Foundation |
||
10 | * |
||
11 | * The author(s) of this software shall not be held liable for damages |
||
12 | * of any nature resulting due to the use of this software. This |
||
13 | * software is provided AS-IS with no warranties. |
||
14 | * |
||
15 | * Theoritical note: |
||
16 | * |
||
17 | * (see Geode(tm) CS5530 manual (rev.4.1) page.56) |
||
18 | * |
||
19 | * CPU frequency control on NatSemi Geode GX1/GXLV processor and CS55x0 |
||
20 | * are based on Suspend Moduration. |
||
21 | * |
||
22 | * Suspend Modulation works by asserting and de-asserting the SUSP# pin |
||
23 | * to CPU(GX1/GXLV) for configurable durations. When asserting SUSP# |
||
24 | * the CPU enters an idle state. GX1 stops its core clock when SUSP# is |
||
25 | * asserted then power consumption is reduced. |
||
26 | * |
||
27 | * Suspend Modulation's OFF/ON duration are configurable |
||
28 | * with 'Suspend Modulation OFF Count Register' |
||
29 | * and 'Suspend Modulation ON Count Register'. |
||
30 | * These registers are 8bit counters that represent the number of |
||
31 | * 32us intervals which the SUSP# pin is asserted/de-asserted to the |
||
32 | * processor. |
||
33 | * |
||
34 | * These counters define a ratio which is the effective frequency |
||
35 | * of operation of the system. |
||
36 | * |
||
37 | * On Count |
||
38 | * F_eff = Fgx * ---------------------- |
||
39 | * On Count + Off Count |
||
40 | * |
||
41 | * 0 <= On Count, Off Count <= 255 |
||
42 | * |
||
43 | * From these limits, we can get register values |
||
44 | * |
||
45 | * on_duration + off_duration <= MAX_DURATION |
||
46 | * off_duration = on_duration * (stock_freq - freq) / freq |
||
47 | * |
||
48 | * on_duration = (freq * DURATION) / stock_freq |
||
49 | * off_duration = DURATION - on_duration |
||
50 | * |
||
51 | * |
||
52 | *--------------------------------------------------------------------------- |
||
53 | * |
||
54 | * ChangeLog: |
||
55 | * Dec. 11, 2002 Hiroshi Miura <miura@da-cha.org> |
||
56 | * - rewrite for Cyrix MediaGX Cx5510/5520 and |
||
57 | * NatSemi Geode Cs5530(A). |
||
58 | * |
||
59 | * Jul. ??, 2002 Zwane Mwaikambo <zwane@commfireservices.com> |
||
60 | * - cs5530_mod patch for 2.4.19-rc1. |
||
61 | * |
||
62 | *--------------------------------------------------------------------------- |
||
63 | * |
||
64 | * Todo |
||
65 | * Test on machines with 5510, 5530, 5530A |
||
66 | */ |
||
67 | |||
68 | /************************************************************************ |
||
69 | * Suspend Modulation - Definitions * |
||
70 | ************************************************************************/ |
||
71 | |||
72 | #include <linuxcomp.h> |
||
73 | |||
74 | #include <linux/kernel.h> |
||
75 | #include <linux/module.h> |
||
76 | #include <linux/sched.h> |
||
77 | #include <linux/init.h> |
||
78 | #include <linux/smp.h> |
||
79 | #include <linux/cpufreq.h> |
||
80 | #include <linux/pci.h> |
||
81 | #include <asm/processor.h> |
||
82 | #include <asm/errno.h> |
||
83 | |||
84 | extern struct cpuinfo_x86 new_cpu_data; |
||
85 | extern unsigned long cpu_khz; |
||
86 | |||
87 | /* PCI config registers, all at F0 */ |
||
88 | #define PCI_PMER1 0x80 /* power management enable register 1 */ |
||
89 | #define PCI_PMER2 0x81 /* power management enable register 2 */ |
||
90 | #define PCI_PMER3 0x82 /* power management enable register 3 */ |
||
91 | #define PCI_IRQTC 0x8c /* irq speedup timer counter register:typical 2 to 4ms */ |
||
92 | #define PCI_VIDTC 0x8d /* video speedup timer counter register: typical 50 to 100ms */ |
||
93 | #define PCI_MODOFF 0x94 /* suspend modulation OFF counter register, 1 = 32us */ |
||
94 | #define PCI_MODON 0x95 /* suspend modulation ON counter register */ |
||
95 | #define PCI_SUSCFG 0x96 /* suspend configuration register */ |
||
96 | |||
97 | /* PMER1 bits */ |
||
98 | #define GPM (1<<0) /* global power management */ |
||
99 | #define GIT (1<<1) /* globally enable PM device idle timers */ |
||
100 | #define GTR (1<<2) /* globally enable IO traps */ |
||
101 | #define IRQ_SPDUP (1<<3) /* disable clock throttle during interrupt handling */ |
||
102 | #define VID_SPDUP (1<<4) /* disable clock throttle during vga video handling */ |
||
103 | |||
104 | /* SUSCFG bits */ |
||
105 | #define SUSMOD (1<<0) /* enable/disable suspend modulation */ |
||
106 | /* the belows support only with cs5530 (after rev.1.2)/cs5530A */ |
||
107 | #define SMISPDUP (1<<1) /* select how SMI re-enable suspend modulation: */ |
||
108 | /* IRQTC timer or read SMI speedup disable reg.(F1BAR[08-09h]) */ |
||
109 | #define SUSCFG (1<<2) /* enable powering down a GXLV processor. "Special 3Volt Suspend" mode */ |
||
110 | /* the belows support only with cs5530A */ |
||
111 | #define PWRSVE_ISA (1<<3) /* stop ISA clock */ |
||
112 | #define PWRSVE (1<<4) /* active idle */ |
||
113 | |||
114 | struct gxfreq_params { |
||
115 | u8 on_duration; |
||
116 | u8 off_duration; |
||
117 | u8 pci_suscfg; |
||
118 | u8 pci_pmer1; |
||
119 | u8 pci_pmer2; |
||
120 | u8 pci_rev; |
||
121 | struct pci_dev *cs55x0; |
||
122 | }; |
||
123 | |||
124 | static struct gxfreq_params *gx_params; |
||
125 | static int stock_freq; |
||
126 | |||
127 | /* PCI bus clock - defaults to 30.000 if cpu_khz is not available */ |
||
128 | static int pci_busclk = 0; |
||
129 | MODULE_PARM(pci_busclk, "i"); |
||
130 | |||
131 | /* maximum duration for which the cpu may be suspended |
||
132 | * (32us * MAX_DURATION). If no parameter is given, this defaults |
||
133 | * to 255. |
||
134 | * Note that this leads to a maximum of 8 ms(!) where the CPU clock |
||
135 | * is suspended -- processing power is just 0.39% of what it used to be, |
||
136 | * though. 781.25 kHz(!) for a 200 MHz processor -- wow. */ |
||
137 | static int max_duration = 255; |
||
138 | MODULE_PARM(max_duration, "i"); |
||
139 | |||
140 | /* For the default policy, we want at least some processing power |
||
141 | * - let's say 5%. (min = maxfreq / POLICY_MIN_DIV) |
||
142 | */ |
||
143 | #define POLICY_MIN_DIV 20 |
||
144 | |||
145 | |||
146 | /* DEBUG |
||
147 | * Define it if you want verbose debug output |
||
148 | */ |
||
149 | |||
150 | #define SUSPMOD_DEBUG 1 |
||
151 | |||
152 | #ifdef SUSPMOD_DEBUG |
||
153 | #define dprintk(msg...) printk(KERN_DEBUG "cpufreq:" msg) |
||
154 | #else |
||
155 | #define dprintk(msg...) do { } while(0) |
||
156 | #endif |
||
157 | |||
158 | /** |
||
159 | * we can detect a core multipiler from dir0_lsb |
||
160 | * from GX1 datasheet p.56, |
||
161 | * MULT[3:0]: |
||
162 | * 0000 = SYSCLK multiplied by 4 (test only) |
||
163 | * 0001 = SYSCLK multiplied by 10 |
||
164 | * 0010 = SYSCLK multiplied by 4 |
||
165 | * 0011 = SYSCLK multiplied by 6 |
||
166 | * 0100 = SYSCLK multiplied by 9 |
||
167 | * 0101 = SYSCLK multiplied by 5 |
||
168 | * 0110 = SYSCLK multiplied by 7 |
||
169 | * 0111 = SYSCLK multiplied by 8 |
||
170 | * of 33.3MHz |
||
171 | **/ |
||
172 | static int gx_freq_mult[16] = { |
||
173 | 4, 10, 4, 6, 9, 5, 7, 8, |
||
174 | 0, 0, 0, 0, 0, 0, 0, 0 |
||
175 | }; |
||
176 | |||
177 | |||
178 | /**************************************************************** |
||
179 | * Low Level chipset interface * |
||
180 | ****************************************************************/ |
||
181 | static struct pci_device_id gx_chipset_tbl[] __initdata = { |
||
182 | { PCI_VENDOR_ID_CYRIX, PCI_DEVICE_ID_CYRIX_5530_LEGACY, PCI_ANY_ID, PCI_ANY_ID }, |
||
183 | { PCI_VENDOR_ID_CYRIX, PCI_DEVICE_ID_CYRIX_5520, PCI_ANY_ID, PCI_ANY_ID }, |
||
184 | { PCI_VENDOR_ID_CYRIX, PCI_DEVICE_ID_CYRIX_5510, PCI_ANY_ID, PCI_ANY_ID }, |
||
185 | { 0, }, |
||
186 | }; |
||
187 | |||
188 | /** |
||
189 | * gx_detect_chipset: |
||
190 | * |
||
191 | **/ |
||
192 | static __init struct pci_dev *gx_detect_chipset(void) |
||
193 | { |
||
194 | struct pci_dev *gx_pci = NULL; |
||
195 | |||
196 | /* check if CPU is a MediaGX or a Geode. */ |
||
197 | if ((new_cpu_data.x86_vendor != X86_VENDOR_NSC) && |
||
198 | (new_cpu_data.x86_vendor != X86_VENDOR_CYRIX)) { |
||
199 | dprintk(KERN_INFO "gx-suspmod: error: no MediaGX/Geode processor found!\n"); |
||
200 | return NULL; |
||
201 | } |
||
202 | |||
203 | /* detect which companion chip is used */ |
||
204 | while ((gx_pci = pci_find_device(PCI_ANY_ID, PCI_ANY_ID, gx_pci)) != NULL) { |
||
205 | if ((pci_match_device (gx_chipset_tbl, gx_pci)) != NULL) { |
||
206 | return gx_pci; |
||
207 | } |
||
208 | } |
||
209 | |||
210 | dprintk(KERN_INFO "gx-suspmod: error: no supported chipset found!\n"); |
||
211 | return NULL; |
||
212 | } |
||
213 | |||
214 | /** |
||
215 | * gx_get_cpuspeed: |
||
216 | * |
||
217 | * Finds out at which efficient frequency the Cyrix MediaGX/NatSemi Geode CPU runs. |
||
218 | */ |
||
219 | static int gx_get_cpuspeed(void) |
||
220 | { |
||
221 | if ((gx_params->pci_suscfg & SUSMOD) == 0) |
||
222 | return stock_freq; |
||
223 | |||
224 | return (stock_freq * gx_params->on_duration) |
||
225 | / (gx_params->on_duration + gx_params->off_duration); |
||
226 | } |
||
227 | |||
228 | /** |
||
229 | * gx_validate_speed: |
||
230 | * determine current cpu speed |
||
231 | * |
||
232 | **/ |
||
233 | |||
234 | static unsigned int gx_validate_speed(unsigned int khz, u8 *on_duration, u8 *off_duration) |
||
235 | { |
||
236 | unsigned int i; |
||
237 | u8 tmp_on, tmp_off; |
||
238 | int old_tmp_freq = stock_freq; |
||
239 | int tmp_freq; |
||
240 | |||
241 | *on_duration=1; |
||
242 | *off_duration=0; |
||
243 | |||
244 | for (i=max_duration; i>0; i--) { |
||
245 | tmp_on = ((khz * i) / stock_freq) & 0xff; |
||
246 | tmp_off = i - tmp_on; |
||
247 | tmp_freq = (stock_freq * tmp_on) / i; |
||
248 | /* if this relation is closer to khz, use this. If it's equal, |
||
249 | * prefer it, too - lower latency */ |
||
250 | if (abs(tmp_freq - khz) <= abs(old_tmp_freq - khz)) { |
||
251 | *on_duration = tmp_on; |
||
252 | *off_duration = tmp_off; |
||
253 | old_tmp_freq = tmp_freq; |
||
254 | } |
||
255 | } |
||
256 | |||
257 | return old_tmp_freq; |
||
258 | } |
||
259 | |||
260 | |||
261 | /** |
||
262 | * gx_set_cpuspeed: |
||
263 | * set cpu speed in khz. |
||
264 | **/ |
||
265 | |||
266 | static void gx_set_cpuspeed(unsigned int khz) |
||
267 | { |
||
268 | u8 suscfg, pmer1; |
||
269 | unsigned int new_khz; |
||
270 | unsigned long flags; |
||
271 | struct cpufreq_freqs freqs; |
||
272 | |||
273 | |||
274 | freqs.cpu = 0; |
||
275 | freqs.old = gx_get_cpuspeed(); |
||
276 | |||
277 | new_khz = gx_validate_speed(khz, &gx_params->on_duration, &gx_params->off_duration); |
||
278 | |||
279 | freqs.new = new_khz; |
||
280 | |||
281 | if (new_khz == stock_freq) { /* if new khz == 100% of CPU speed, it is special case */ |
||
282 | local_irq_save(flags); |
||
283 | //!!!cpufreq_notify_transition(&freqs, CPUFREQ_PRECHANGE); |
||
284 | pci_write_config_byte(gx_params->cs55x0, PCI_SUSCFG, (gx_params->pci_suscfg & ~(SUSMOD))); |
||
285 | pci_read_config_byte(gx_params->cs55x0, PCI_SUSCFG, &(gx_params->pci_suscfg)); |
||
286 | local_irq_restore(flags); |
||
287 | dprintk("suspend modulation disabled: cpu runs 100 percent speed.\n"); |
||
288 | //!!!cpufreq_notify_transition(&freqs, CPUFREQ_POSTCHANGE); |
||
289 | return; |
||
290 | } |
||
291 | |||
292 | //!!!cpufreq_notify_transition(&freqs, CPUFREQ_PRECHANGE); |
||
293 | |||
294 | local_irq_save(flags); |
||
295 | switch (gx_params->cs55x0->device) { |
||
296 | case PCI_DEVICE_ID_CYRIX_5530_LEGACY: |
||
297 | pmer1 = gx_params->pci_pmer1 | IRQ_SPDUP | VID_SPDUP; |
||
298 | /* FIXME: need to test other values -- Zwane,Miura */ |
||
299 | pci_write_config_byte(gx_params->cs55x0, PCI_IRQTC, 4); /* typical 2 to 4ms */ |
||
300 | pci_write_config_byte(gx_params->cs55x0, PCI_VIDTC, 100);/* typical 50 to 100ms */ |
||
301 | pci_write_config_byte(gx_params->cs55x0, PCI_PMER1, pmer1); |
||
302 | |||
303 | if (gx_params->pci_rev < 0x10) { /* CS5530(rev 1.2, 1.3) */ |
||
304 | suscfg = gx_params->pci_suscfg | SUSMOD; |
||
305 | } else { /* CS5530A,B.. */ |
||
306 | suscfg = gx_params->pci_suscfg | SUSMOD | PWRSVE; |
||
307 | } |
||
308 | break; |
||
309 | case PCI_DEVICE_ID_CYRIX_5520: |
||
310 | case PCI_DEVICE_ID_CYRIX_5510: |
||
311 | suscfg = gx_params->pci_suscfg | SUSMOD; |
||
312 | break; |
||
313 | default: |
||
314 | local_irq_restore(flags); |
||
315 | dprintk("fatal: try to set unknown chipset.\n"); |
||
316 | return; |
||
317 | } |
||
318 | |||
319 | pci_write_config_byte(gx_params->cs55x0, PCI_MODOFF, gx_params->off_duration); |
||
320 | pci_write_config_byte(gx_params->cs55x0, PCI_MODON, gx_params->on_duration); |
||
321 | |||
322 | pci_write_config_byte(gx_params->cs55x0, PCI_SUSCFG, suscfg); |
||
323 | pci_read_config_byte(gx_params->cs55x0, PCI_SUSCFG, &suscfg); |
||
324 | |||
325 | local_irq_restore(flags); |
||
326 | |||
327 | gx_params->pci_suscfg = suscfg; |
||
328 | |||
329 | //!!!cpufreq_notify_transition(&freqs, CPUFREQ_POSTCHANGE); |
||
330 | |||
331 | dprintk("suspend modulation w/ duration of ON:%d us, OFF:%d us\n", |
||
332 | gx_params->on_duration * 32, gx_params->off_duration * 32); |
||
333 | dprintk("suspend modulation w/ clock speed: %d kHz.\n", freqs.new); |
||
334 | } |
||
335 | |||
336 | /**************************************************************** |
||
337 | * High level functions * |
||
338 | ****************************************************************/ |
||
339 | |||
340 | /* |
||
341 | * cpufreq_gx_verify: test if frequency range is valid |
||
342 | * |
||
343 | * This function checks if a given frequency range in kHz is valid |
||
344 | * for the hardware supported by the driver. |
||
345 | */ |
||
346 | |||
347 | static int cpufreq_gx_verify(struct cpufreq_policy *policy) |
||
348 | { |
||
349 | unsigned int tmp_freq = 0; |
||
350 | u8 tmp1, tmp2; |
||
351 | |||
352 | if (!stock_freq || !policy) |
||
353 | return -EINVAL; |
||
354 | |||
355 | policy->cpu = 0; |
||
356 | cpufreq_verify_within_limits(policy, (stock_freq / max_duration), stock_freq); |
||
357 | |||
358 | /* it needs to be assured that at least one supported frequency is |
||
359 | * within policy->min and policy->max. If it is not, policy->max |
||
360 | * needs to be increased until one freuqency is supported. |
||
361 | * policy->min may not be decreased, though. This way we guarantee a |
||
362 | * specific processing capacity. |
||
363 | */ |
||
364 | tmp_freq = gx_validate_speed(policy->min, &tmp1, &tmp2); |
||
365 | if (tmp_freq < policy->min) |
||
366 | tmp_freq += stock_freq / max_duration; |
||
367 | policy->min = tmp_freq; |
||
368 | if (policy->min > policy->max) |
||
369 | policy->max = tmp_freq; |
||
370 | tmp_freq = gx_validate_speed(policy->max, &tmp1, &tmp2); |
||
371 | if (tmp_freq > policy->max) |
||
372 | tmp_freq -= stock_freq / max_duration; |
||
373 | policy->max = tmp_freq; |
||
374 | if (policy->max < policy->min) |
||
375 | policy->max = policy->min; |
||
376 | cpufreq_verify_within_limits(policy, (stock_freq / max_duration), stock_freq); |
||
377 | |||
378 | return 0; |
||
379 | } |
||
380 | |||
381 | /* |
||
382 | * cpufreq_gx_target: |
||
383 | * |
||
384 | */ |
||
385 | static int cpufreq_gx_target(struct cpufreq_policy *policy, |
||
386 | unsigned int target_freq, |
||
387 | unsigned int relation) |
||
388 | { |
||
389 | u8 tmp1, tmp2; |
||
390 | unsigned int tmp_freq; |
||
391 | |||
392 | if (!stock_freq || !policy) |
||
393 | return -EINVAL; |
||
394 | |||
395 | policy->cpu = 0; |
||
396 | |||
397 | tmp_freq = gx_validate_speed(target_freq, &tmp1, &tmp2); |
||
398 | while (tmp_freq < policy->min) { |
||
399 | tmp_freq += stock_freq / max_duration; |
||
400 | tmp_freq = gx_validate_speed(tmp_freq, &tmp1, &tmp2); |
||
401 | } |
||
402 | while (tmp_freq > policy->max) { |
||
403 | tmp_freq -= stock_freq / max_duration; |
||
404 | tmp_freq = gx_validate_speed(tmp_freq, &tmp1, &tmp2); |
||
405 | } |
||
406 | |||
407 | gx_set_cpuspeed(tmp_freq); |
||
408 | |||
409 | return 0; |
||
410 | } |
||
411 | |||
412 | static int cpufreq_gx_cpu_init(struct cpufreq_policy *policy) |
||
413 | { |
||
414 | int maxfreq, curfreq; |
||
415 | |||
416 | if (!policy || policy->cpu != 0) |
||
417 | return -ENODEV; |
||
418 | |||
419 | /* determine maximum frequency */ |
||
420 | if (pci_busclk) { |
||
421 | maxfreq = pci_busclk * gx_freq_mult[getCx86(CX86_DIR1) & 0x0f]; |
||
422 | } else if (cpu_khz) { |
||
423 | maxfreq = cpu_khz; |
||
424 | } else { |
||
425 | maxfreq = 30000 * gx_freq_mult[getCx86(CX86_DIR1) & 0x0f]; |
||
426 | } |
||
427 | stock_freq = maxfreq; |
||
428 | curfreq = gx_get_cpuspeed(); |
||
429 | |||
430 | dprintk("cpu max frequency is %d.\n", maxfreq); |
||
431 | dprintk("cpu current frequency is %dkHz.\n",curfreq); |
||
432 | |||
433 | /* setup basic struct for cpufreq API */ |
||
434 | policy->cpu = 0; |
||
435 | |||
436 | if (max_duration < POLICY_MIN_DIV) |
||
437 | policy->min = maxfreq / max_duration; |
||
438 | else |
||
439 | policy->min = maxfreq / POLICY_MIN_DIV; |
||
440 | policy->max = maxfreq; |
||
441 | policy->cur = curfreq; |
||
442 | policy->governor = 0; //!!!CPUFREQ_DEFAULT_GOVERNOR; |
||
443 | policy->cpuinfo.min_freq = maxfreq / max_duration; |
||
444 | policy->cpuinfo.max_freq = maxfreq; |
||
445 | policy->cpuinfo.transition_latency = CPUFREQ_ETERNAL; |
||
446 | |||
447 | return 0; |
||
448 | } |
||
449 | |||
450 | /* |
||
451 | * cpufreq_gx_init: |
||
452 | * MediaGX/Geode GX initialize cpufreq driver |
||
453 | */ |
||
454 | static struct cpufreq_driver gx_suspmod_driver = { |
||
455 | .verify = cpufreq_gx_verify, |
||
456 | .target = cpufreq_gx_target, |
||
457 | .init = cpufreq_gx_cpu_init, |
||
458 | .name = "gx-suspmod", |
||
459 | .owner = THIS_MODULE, |
||
460 | }; |
||
461 | |||
462 | /*static*/ int __init cpufreq_gx_init(void) |
||
463 | { |
||
464 | int ret; |
||
465 | struct gxfreq_params *params; |
||
466 | struct pci_dev *gx_pci; |
||
467 | u32 class_rev; |
||
468 | |||
469 | /* Test if we have the right hardware */ |
||
470 | if ((gx_pci = gx_detect_chipset()) == NULL) |
||
471 | return -ENODEV; |
||
472 | |||
473 | /* check whether module parameters are sane */ |
||
474 | if (max_duration > 0xff) |
||
475 | max_duration = 0xff; |
||
476 | |||
477 | dprintk("geode suspend modulation available.\n"); |
||
478 | |||
479 | params = kmalloc(sizeof(struct gxfreq_params), GFP_KERNEL); |
||
480 | if (params == NULL) |
||
481 | return -ENOMEM; |
||
482 | memset(params, 0, sizeof(struct gxfreq_params)); |
||
483 | |||
484 | params->cs55x0 = gx_pci; |
||
485 | gx_params = params; |
||
486 | |||
487 | /* keep cs55x0 configurations */ |
||
488 | pci_read_config_byte(params->cs55x0, PCI_SUSCFG, &(params->pci_suscfg)); |
||
489 | pci_read_config_byte(params->cs55x0, PCI_PMER1, &(params->pci_pmer1)); |
||
490 | pci_read_config_byte(params->cs55x0, PCI_PMER2, &(params->pci_pmer2)); |
||
491 | pci_read_config_byte(params->cs55x0, PCI_MODON, &(params->on_duration)); |
||
492 | pci_read_config_byte(params->cs55x0, PCI_MODOFF, &(params->off_duration)); |
||
493 | pci_read_config_dword(params->cs55x0, PCI_CLASS_REVISION, &class_rev); |
||
494 | params->pci_rev = class_rev && 0xff; |
||
495 | |||
496 | if ((ret = cpufreq_register_driver(&gx_suspmod_driver))) { |
||
497 | kfree(params); |
||
498 | return ret; /* register error! */ |
||
499 | } |
||
500 | |||
501 | return 0; |
||
502 | } |
||
503 | |||
504 | /*static*/ void __exit cpufreq_gx_exit(void) |
||
505 | { |
||
506 | cpufreq_unregister_driver(&gx_suspmod_driver); |
||
507 | kfree(gx_params); |
||
508 | } |
||
509 | |||
510 | MODULE_AUTHOR ("Hiroshi Miura <miura@da-cha.org>"); |
||
511 | MODULE_DESCRIPTION ("Cpufreq driver for Cyrix MediaGX and NatSemi Geode"); |
||
512 | MODULE_LICENSE ("GPL"); |
||
513 | |||
514 | module_init(cpufreq_gx_init); |
||
515 | module_exit(cpufreq_gx_exit); |
||
516 |