18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * Copyright (c) 2006-2008 Simtec Electronics 48c2ecf20Sopenharmony_ci * http://armlinux.simtec.co.uk/ 58c2ecf20Sopenharmony_ci * Ben Dooks <ben@simtec.co.uk> 68c2ecf20Sopenharmony_ci * 78c2ecf20Sopenharmony_ci * S3C2410 CPU Frequency scaling 88c2ecf20Sopenharmony_ci*/ 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_ci#include <linux/init.h> 118c2ecf20Sopenharmony_ci#include <linux/module.h> 128c2ecf20Sopenharmony_ci#include <linux/interrupt.h> 138c2ecf20Sopenharmony_ci#include <linux/ioport.h> 148c2ecf20Sopenharmony_ci#include <linux/cpufreq.h> 158c2ecf20Sopenharmony_ci#include <linux/device.h> 168c2ecf20Sopenharmony_ci#include <linux/clk.h> 178c2ecf20Sopenharmony_ci#include <linux/err.h> 188c2ecf20Sopenharmony_ci#include <linux/io.h> 198c2ecf20Sopenharmony_ci#include <linux/soc/samsung/s3c-cpufreq-core.h> 208c2ecf20Sopenharmony_ci#include <linux/soc/samsung/s3c-pm.h> 218c2ecf20Sopenharmony_ci 228c2ecf20Sopenharmony_ci#include <asm/mach/arch.h> 238c2ecf20Sopenharmony_ci#include <asm/mach/map.h> 248c2ecf20Sopenharmony_ci 258c2ecf20Sopenharmony_ci#define S3C2410_CLKDIVN_PDIVN (1<<0) 268c2ecf20Sopenharmony_ci#define S3C2410_CLKDIVN_HDIVN (1<<1) 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ci/* Note, 2410A has an extra mode for 1:4:4 ratio, bit 2 of CLKDIV */ 298c2ecf20Sopenharmony_ci 308c2ecf20Sopenharmony_cistatic void s3c2410_cpufreq_setdivs(struct s3c_cpufreq_config *cfg) 318c2ecf20Sopenharmony_ci{ 328c2ecf20Sopenharmony_ci u32 clkdiv = 0; 338c2ecf20Sopenharmony_ci 348c2ecf20Sopenharmony_ci if (cfg->divs.h_divisor == 2) 358c2ecf20Sopenharmony_ci clkdiv |= S3C2410_CLKDIVN_HDIVN; 368c2ecf20Sopenharmony_ci 378c2ecf20Sopenharmony_ci if (cfg->divs.p_divisor != cfg->divs.h_divisor) 388c2ecf20Sopenharmony_ci clkdiv |= S3C2410_CLKDIVN_PDIVN; 398c2ecf20Sopenharmony_ci 408c2ecf20Sopenharmony_ci s3c24xx_write_clkdivn(clkdiv); 418c2ecf20Sopenharmony_ci} 428c2ecf20Sopenharmony_ci 438c2ecf20Sopenharmony_cistatic int s3c2410_cpufreq_calcdivs(struct s3c_cpufreq_config *cfg) 448c2ecf20Sopenharmony_ci{ 458c2ecf20Sopenharmony_ci unsigned long hclk, fclk, pclk; 468c2ecf20Sopenharmony_ci unsigned int hdiv, pdiv; 478c2ecf20Sopenharmony_ci unsigned long hclk_max; 488c2ecf20Sopenharmony_ci 498c2ecf20Sopenharmony_ci fclk = cfg->freq.fclk; 508c2ecf20Sopenharmony_ci hclk_max = cfg->max.hclk; 518c2ecf20Sopenharmony_ci 528c2ecf20Sopenharmony_ci cfg->freq.armclk = fclk; 538c2ecf20Sopenharmony_ci 548c2ecf20Sopenharmony_ci s3c_freq_dbg("%s: fclk is %lu, max hclk %lu\n", 558c2ecf20Sopenharmony_ci __func__, fclk, hclk_max); 568c2ecf20Sopenharmony_ci 578c2ecf20Sopenharmony_ci hdiv = (fclk > cfg->max.hclk) ? 2 : 1; 588c2ecf20Sopenharmony_ci hclk = fclk / hdiv; 598c2ecf20Sopenharmony_ci 608c2ecf20Sopenharmony_ci if (hclk > cfg->max.hclk) { 618c2ecf20Sopenharmony_ci s3c_freq_dbg("%s: hclk too big\n", __func__); 628c2ecf20Sopenharmony_ci return -EINVAL; 638c2ecf20Sopenharmony_ci } 648c2ecf20Sopenharmony_ci 658c2ecf20Sopenharmony_ci pdiv = (hclk > cfg->max.pclk) ? 2 : 1; 668c2ecf20Sopenharmony_ci pclk = hclk / pdiv; 678c2ecf20Sopenharmony_ci 688c2ecf20Sopenharmony_ci if (pclk > cfg->max.pclk) { 698c2ecf20Sopenharmony_ci s3c_freq_dbg("%s: pclk too big\n", __func__); 708c2ecf20Sopenharmony_ci return -EINVAL; 718c2ecf20Sopenharmony_ci } 728c2ecf20Sopenharmony_ci 738c2ecf20Sopenharmony_ci pdiv *= hdiv; 748c2ecf20Sopenharmony_ci 758c2ecf20Sopenharmony_ci /* record the result */ 768c2ecf20Sopenharmony_ci cfg->divs.p_divisor = pdiv; 778c2ecf20Sopenharmony_ci cfg->divs.h_divisor = hdiv; 788c2ecf20Sopenharmony_ci 798c2ecf20Sopenharmony_ci return 0; 808c2ecf20Sopenharmony_ci} 818c2ecf20Sopenharmony_ci 828c2ecf20Sopenharmony_cistatic struct s3c_cpufreq_info s3c2410_cpufreq_info = { 838c2ecf20Sopenharmony_ci .max = { 848c2ecf20Sopenharmony_ci .fclk = 200000000, 858c2ecf20Sopenharmony_ci .hclk = 100000000, 868c2ecf20Sopenharmony_ci .pclk = 50000000, 878c2ecf20Sopenharmony_ci }, 888c2ecf20Sopenharmony_ci 898c2ecf20Sopenharmony_ci /* transition latency is about 5ms worst-case, so 908c2ecf20Sopenharmony_ci * set 10ms to be sure */ 918c2ecf20Sopenharmony_ci .latency = 10000000, 928c2ecf20Sopenharmony_ci 938c2ecf20Sopenharmony_ci .locktime_m = 150, 948c2ecf20Sopenharmony_ci .locktime_u = 150, 958c2ecf20Sopenharmony_ci .locktime_bits = 12, 968c2ecf20Sopenharmony_ci 978c2ecf20Sopenharmony_ci .need_pll = 1, 988c2ecf20Sopenharmony_ci 998c2ecf20Sopenharmony_ci .name = "s3c2410", 1008c2ecf20Sopenharmony_ci .calc_iotiming = s3c2410_iotiming_calc, 1018c2ecf20Sopenharmony_ci .set_iotiming = s3c2410_iotiming_set, 1028c2ecf20Sopenharmony_ci .get_iotiming = s3c2410_iotiming_get, 1038c2ecf20Sopenharmony_ci 1048c2ecf20Sopenharmony_ci .set_fvco = s3c2410_set_fvco, 1058c2ecf20Sopenharmony_ci .set_refresh = s3c2410_cpufreq_setrefresh, 1068c2ecf20Sopenharmony_ci .set_divs = s3c2410_cpufreq_setdivs, 1078c2ecf20Sopenharmony_ci .calc_divs = s3c2410_cpufreq_calcdivs, 1088c2ecf20Sopenharmony_ci 1098c2ecf20Sopenharmony_ci .debug_io_show = s3c_cpufreq_debugfs_call(s3c2410_iotiming_debugfs), 1108c2ecf20Sopenharmony_ci}; 1118c2ecf20Sopenharmony_ci 1128c2ecf20Sopenharmony_cistatic int s3c2410_cpufreq_add(struct device *dev, 1138c2ecf20Sopenharmony_ci struct subsys_interface *sif) 1148c2ecf20Sopenharmony_ci{ 1158c2ecf20Sopenharmony_ci return s3c_cpufreq_register(&s3c2410_cpufreq_info); 1168c2ecf20Sopenharmony_ci} 1178c2ecf20Sopenharmony_ci 1188c2ecf20Sopenharmony_cistatic struct subsys_interface s3c2410_cpufreq_interface = { 1198c2ecf20Sopenharmony_ci .name = "s3c2410_cpufreq", 1208c2ecf20Sopenharmony_ci .subsys = &s3c2410_subsys, 1218c2ecf20Sopenharmony_ci .add_dev = s3c2410_cpufreq_add, 1228c2ecf20Sopenharmony_ci}; 1238c2ecf20Sopenharmony_ci 1248c2ecf20Sopenharmony_cistatic int __init s3c2410_cpufreq_init(void) 1258c2ecf20Sopenharmony_ci{ 1268c2ecf20Sopenharmony_ci return subsys_interface_register(&s3c2410_cpufreq_interface); 1278c2ecf20Sopenharmony_ci} 1288c2ecf20Sopenharmony_ciarch_initcall(s3c2410_cpufreq_init); 1298c2ecf20Sopenharmony_ci 1308c2ecf20Sopenharmony_cistatic int s3c2410a_cpufreq_add(struct device *dev, 1318c2ecf20Sopenharmony_ci struct subsys_interface *sif) 1328c2ecf20Sopenharmony_ci{ 1338c2ecf20Sopenharmony_ci /* alter the maximum freq settings for S3C2410A. If a board knows 1348c2ecf20Sopenharmony_ci * it only has a maximum of 200, then it should register its own 1358c2ecf20Sopenharmony_ci * limits. */ 1368c2ecf20Sopenharmony_ci 1378c2ecf20Sopenharmony_ci s3c2410_cpufreq_info.max.fclk = 266000000; 1388c2ecf20Sopenharmony_ci s3c2410_cpufreq_info.max.hclk = 133000000; 1398c2ecf20Sopenharmony_ci s3c2410_cpufreq_info.max.pclk = 66500000; 1408c2ecf20Sopenharmony_ci s3c2410_cpufreq_info.name = "s3c2410a"; 1418c2ecf20Sopenharmony_ci 1428c2ecf20Sopenharmony_ci return s3c2410_cpufreq_add(dev, sif); 1438c2ecf20Sopenharmony_ci} 1448c2ecf20Sopenharmony_ci 1458c2ecf20Sopenharmony_cistatic struct subsys_interface s3c2410a_cpufreq_interface = { 1468c2ecf20Sopenharmony_ci .name = "s3c2410a_cpufreq", 1478c2ecf20Sopenharmony_ci .subsys = &s3c2410a_subsys, 1488c2ecf20Sopenharmony_ci .add_dev = s3c2410a_cpufreq_add, 1498c2ecf20Sopenharmony_ci}; 1508c2ecf20Sopenharmony_ci 1518c2ecf20Sopenharmony_cistatic int __init s3c2410a_cpufreq_init(void) 1528c2ecf20Sopenharmony_ci{ 1538c2ecf20Sopenharmony_ci return subsys_interface_register(&s3c2410a_cpufreq_interface); 1548c2ecf20Sopenharmony_ci} 1558c2ecf20Sopenharmony_ciarch_initcall(s3c2410a_cpufreq_init); 156