sc520_freq.c 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. * sc520_freq.c: cpufreq driver for the AMD Elan sc520
  4. *
  5. * Copyright (C) 2005 Sean Young <[email protected]>
  6. *
  7. * Based on elanfreq.c
  8. *
  9. * 2005-03-30: - initial revision
  10. */
  11. #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  12. #include <linux/kernel.h>
  13. #include <linux/module.h>
  14. #include <linux/init.h>
  15. #include <linux/delay.h>
  16. #include <linux/cpufreq.h>
  17. #include <linux/timex.h>
  18. #include <linux/io.h>
  19. #include <asm/cpu_device_id.h>
  20. #include <asm/msr.h>
  21. #define MMCR_BASE 0xfffef000 /* The default base address */
  22. #define OFFS_CPUCTL 0x2 /* CPU Control Register */
  23. static __u8 __iomem *cpuctl;
  24. static struct cpufreq_frequency_table sc520_freq_table[] = {
  25. {0, 0x01, 100000},
  26. {0, 0x02, 133000},
  27. {0, 0, CPUFREQ_TABLE_END},
  28. };
  29. static unsigned int sc520_freq_get_cpu_frequency(unsigned int cpu)
  30. {
  31. u8 clockspeed_reg = *cpuctl;
  32. switch (clockspeed_reg & 0x03) {
  33. default:
  34. pr_err("error: cpuctl register has unexpected value %02x\n",
  35. clockspeed_reg);
  36. fallthrough;
  37. case 0x01:
  38. return 100000;
  39. case 0x02:
  40. return 133000;
  41. }
  42. }
  43. static int sc520_freq_target(struct cpufreq_policy *policy, unsigned int state)
  44. {
  45. u8 clockspeed_reg;
  46. local_irq_disable();
  47. clockspeed_reg = *cpuctl & ~0x03;
  48. *cpuctl = clockspeed_reg | sc520_freq_table[state].driver_data;
  49. local_irq_enable();
  50. return 0;
  51. }
  52. /*
  53. * Module init and exit code
  54. */
  55. static int sc520_freq_cpu_init(struct cpufreq_policy *policy)
  56. {
  57. struct cpuinfo_x86 *c = &cpu_data(0);
  58. /* capability check */
  59. if (c->x86_vendor != X86_VENDOR_AMD ||
  60. c->x86 != 4 || c->x86_model != 9)
  61. return -ENODEV;
  62. /* cpuinfo and default policy values */
  63. policy->cpuinfo.transition_latency = 1000000; /* 1ms */
  64. policy->freq_table = sc520_freq_table;
  65. return 0;
  66. }
  67. static struct cpufreq_driver sc520_freq_driver = {
  68. .get = sc520_freq_get_cpu_frequency,
  69. .verify = cpufreq_generic_frequency_table_verify,
  70. .target_index = sc520_freq_target,
  71. .init = sc520_freq_cpu_init,
  72. .name = "sc520_freq",
  73. .attr = cpufreq_generic_attr,
  74. };
  75. static const struct x86_cpu_id sc520_ids[] = {
  76. X86_MATCH_VENDOR_FAM_MODEL(AMD, 4, 9, NULL),
  77. {}
  78. };
  79. MODULE_DEVICE_TABLE(x86cpu, sc520_ids);
  80. static int __init sc520_freq_init(void)
  81. {
  82. int err;
  83. if (!x86_match_cpu(sc520_ids))
  84. return -ENODEV;
  85. cpuctl = ioremap((unsigned long)(MMCR_BASE + OFFS_CPUCTL), 1);
  86. if (!cpuctl) {
  87. pr_err("sc520_freq: error: failed to remap memory\n");
  88. return -ENOMEM;
  89. }
  90. err = cpufreq_register_driver(&sc520_freq_driver);
  91. if (err)
  92. iounmap(cpuctl);
  93. return err;
  94. }
  95. static void __exit sc520_freq_exit(void)
  96. {
  97. cpufreq_unregister_driver(&sc520_freq_driver);
  98. iounmap(cpuctl);
  99. }
  100. MODULE_LICENSE("GPL");
  101. MODULE_AUTHOR("Sean Young <[email protected]>");
  102. MODULE_DESCRIPTION("cpufreq driver for AMD's Elan sc520 CPU");
  103. module_init(sc520_freq_init);
  104. module_exit(sc520_freq_exit);