1 /* us3_cpufreq.c: UltraSPARC-III cpu frequency support
 
   3  * Copyright (C) 2003 David S. Miller (davem@redhat.com)
 
   5  * Many thanks to Dominik Brodowski for fixing up the cpufreq
 
   6  * infrastructure in order to make this driver easier to implement.
 
   9 #include <linux/kernel.h>
 
  10 #include <linux/module.h>
 
  11 #include <linux/sched.h>
 
  12 #include <linux/smp.h>
 
  13 #include <linux/cpufreq.h>
 
  14 #include <linux/threads.h>
 
  15 #include <linux/slab.h>
 
  16 #include <linux/init.h>
 
  19 #include <asm/timer.h>
 
  21 static struct cpufreq_driver *cpufreq_us3_driver;
 
  23 struct us3_freq_percpu_info {
 
  24         struct cpufreq_frequency_table table[4];
 
  27 /* Indexed by cpu number. */
 
  28 static struct us3_freq_percpu_info *us3_freq_table;
 
  30 /* UltraSPARC-III has three dividers: 1, 2, and 32.  These are controlled
 
  31  * in the Safari config register.
 
  33 #define SAFARI_CFG_DIV_1        0x0000000000000000UL
 
  34 #define SAFARI_CFG_DIV_2        0x0000000040000000UL
 
  35 #define SAFARI_CFG_DIV_32       0x0000000080000000UL
 
  36 #define SAFARI_CFG_DIV_MASK     0x00000000C0000000UL
 
  38 static unsigned long read_safari_cfg(void)
 
  42         __asm__ __volatile__("ldxa      [%%g0] %1, %0"
 
  44                              : "i" (ASI_SAFARI_CONFIG));
 
  48 static void write_safari_cfg(unsigned long val)
 
  50         __asm__ __volatile__("stxa      %0, [%%g0] %1\n\t"
 
  53                              : "r" (val), "i" (ASI_SAFARI_CONFIG)
 
  57 static unsigned long get_current_freq(unsigned int cpu, unsigned long safari_cfg)
 
  59         unsigned long clock_tick = sparc64_get_clock_tick(cpu) / 1000;
 
  62         switch (safari_cfg & SAFARI_CFG_DIV_MASK) {
 
  63         case SAFARI_CFG_DIV_1:
 
  66         case SAFARI_CFG_DIV_2:
 
  69         case SAFARI_CFG_DIV_32:
 
  70                 ret = clock_tick / 32;
 
  79 static unsigned int us3_freq_get(unsigned int cpu)
 
  81         cpumask_t cpus_allowed;
 
  88         cpus_allowed = current->cpus_allowed;
 
  89         set_cpus_allowed(current, cpumask_of_cpu(cpu));
 
  91         reg = read_safari_cfg();
 
  92         ret = get_current_freq(cpu, reg);
 
  94         set_cpus_allowed(current, cpus_allowed);
 
  99 static void us3_set_cpu_divider_index(unsigned int cpu, unsigned int index)
 
 101         unsigned long new_bits, new_freq, reg;
 
 102         cpumask_t cpus_allowed;
 
 103         struct cpufreq_freqs freqs;
 
 105         if (!cpu_online(cpu))
 
 108         cpus_allowed = current->cpus_allowed;
 
 109         set_cpus_allowed(current, cpumask_of_cpu(cpu));
 
 111         new_freq = sparc64_get_clock_tick(cpu) / 1000;
 
 114                 new_bits = SAFARI_CFG_DIV_1;
 
 118                 new_bits = SAFARI_CFG_DIV_2;
 
 122                 new_bits = SAFARI_CFG_DIV_32;
 
 130         reg = read_safari_cfg();
 
 132         freqs.old = get_current_freq(cpu, reg);
 
 133         freqs.new = new_freq;
 
 135         cpufreq_notify_transition(&freqs, CPUFREQ_PRECHANGE);
 
 137         reg &= ~SAFARI_CFG_DIV_MASK;
 
 139         write_safari_cfg(reg);
 
 141         cpufreq_notify_transition(&freqs, CPUFREQ_POSTCHANGE);
 
 143         set_cpus_allowed(current, cpus_allowed);
 
 146 static int us3_freq_target(struct cpufreq_policy *policy,
 
 147                           unsigned int target_freq,
 
 148                           unsigned int relation)
 
 150         unsigned int new_index = 0;
 
 152         if (cpufreq_frequency_table_target(policy,
 
 153                                            &us3_freq_table[policy->cpu].table[0],
 
 159         us3_set_cpu_divider_index(policy->cpu, new_index);
 
 164 static int us3_freq_verify(struct cpufreq_policy *policy)
 
 166         return cpufreq_frequency_table_verify(policy,
 
 167                                               &us3_freq_table[policy->cpu].table[0]);
 
 170 static int __init us3_freq_cpu_init(struct cpufreq_policy *policy)
 
 172         unsigned int cpu = policy->cpu;
 
 173         unsigned long clock_tick = sparc64_get_clock_tick(cpu) / 1000;
 
 174         struct cpufreq_frequency_table *table =
 
 175                 &us3_freq_table[cpu].table[0];
 
 178         table[0].frequency = clock_tick / 1;
 
 180         table[1].frequency = clock_tick / 2;
 
 182         table[2].frequency = clock_tick / 32;
 
 184         table[3].frequency = CPUFREQ_TABLE_END;
 
 186         policy->governor = CPUFREQ_DEFAULT_GOVERNOR;
 
 187         policy->cpuinfo.transition_latency = 0;
 
 188         policy->cur = clock_tick;
 
 190         return cpufreq_frequency_table_cpuinfo(policy, table);
 
 193 static int us3_freq_cpu_exit(struct cpufreq_policy *policy)
 
 195         if (cpufreq_us3_driver)
 
 196                 us3_set_cpu_divider_index(policy->cpu, 0);
 
 201 static int __init us3_freq_init(void)
 
 203         unsigned long manuf, impl, ver;
 
 206         __asm__("rdpr %%ver, %0" : "=r" (ver));
 
 207         manuf = ((ver >> 48) & 0xffff);
 
 208         impl  = ((ver >> 32) & 0xffff);
 
 210         if (manuf == CHEETAH_MANUF &&
 
 211             (impl == CHEETAH_IMPL ||
 
 212              impl == CHEETAH_PLUS_IMPL ||
 
 213              impl == JAGUAR_IMPL ||
 
 214              impl == PANTHER_IMPL)) {
 
 215                 struct cpufreq_driver *driver;
 
 218                 driver = kmalloc(sizeof(struct cpufreq_driver), GFP_KERNEL);
 
 221                 memset(driver, 0, sizeof(*driver));
 
 223                 us3_freq_table = kmalloc(
 
 224                         (NR_CPUS * sizeof(struct us3_freq_percpu_info)),
 
 229                 memset(us3_freq_table, 0,
 
 230                        (NR_CPUS * sizeof(struct us3_freq_percpu_info)));
 
 232                 driver->init = us3_freq_cpu_init;
 
 233                 driver->verify = us3_freq_verify;
 
 234                 driver->target = us3_freq_target;
 
 235                 driver->get = us3_freq_get;
 
 236                 driver->exit = us3_freq_cpu_exit;
 
 237                 driver->owner = THIS_MODULE,
 
 238                 strcpy(driver->name, "UltraSPARC-III");
 
 240                 cpufreq_us3_driver = driver;
 
 241                 ret = cpufreq_register_driver(driver);
 
 250                         cpufreq_us3_driver = NULL;
 
 252                 kfree(us3_freq_table);
 
 253                 us3_freq_table = NULL;
 
 260 static void __exit us3_freq_exit(void)
 
 262         if (cpufreq_us3_driver) {
 
 263                 cpufreq_unregister_driver(cpufreq_us3_driver);
 
 264                 kfree(cpufreq_us3_driver);
 
 265                 cpufreq_us3_driver = NULL;
 
 266                 kfree(us3_freq_table);
 
 267                 us3_freq_table = NULL;
 
 271 MODULE_AUTHOR("David S. Miller <davem@redhat.com>");
 
 272 MODULE_DESCRIPTION("cpufreq driver for UltraSPARC-III");
 
 273 MODULE_LICENSE("GPL");
 
 275 module_init(us3_freq_init);
 
 276 module_exit(us3_freq_exit);