2023-01-27 02:03:19 +03:00
// SPDX-License-Identifier: GPL-2.0
/*
* Qualcomm APCS clock controller driver
*
* Copyright ( c ) 2022 , Linaro Limited
* Author : Dmitry Baryshkov < dmitry . baryshkov @ linaro . org >
*/
# include <linux/bits.h>
2023-02-23 04:36:42 +03:00
# include <linux/bitfield.h>
2023-01-27 02:03:19 +03:00
# include <linux/clk-provider.h>
# include <linux/delay.h>
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/regmap.h>
# define APCS_AUX_OFFSET 0x50
# define APCS_AUX_DIV_MASK GENMASK(17, 16)
# define APCS_AUX_DIV_2 0x1
static int qcom_apcs_msm8996_clk_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
struct device * parent = dev - > parent ;
struct regmap * regmap ;
struct clk_hw * hw ;
unsigned int val ;
int ret = - ENODEV ;
regmap = dev_get_regmap ( parent , NULL ) ;
if ( ! regmap ) {
dev_err ( dev , " failed to get regmap: %d \n " , ret ) ;
return ret ;
}
regmap_read ( regmap , APCS_AUX_OFFSET , & val ) ;
regmap_update_bits ( regmap , APCS_AUX_OFFSET , APCS_AUX_DIV_MASK ,
FIELD_PREP ( APCS_AUX_DIV_MASK , APCS_AUX_DIV_2 ) ) ;
/*
* This clock is used during CPU cluster setup while setting up CPU PLLs .
* Add hardware mandated delay to make sure that the sys_apcs_aux clock
* is stable ( after setting the divider ) before continuing
* bootstrapping to keep CPUs from ending up in a weird state .
*/
udelay ( 5 ) ;
/*
* As this clocks is a parent of the CPU cluster clocks and is actually
* used as a parent during CPU clocks setup , we want for it to register
* as early as possible , without letting fw_devlink to delay probing of
* either of the drivers .
*
* The sys_apcs_aux is a child ( divider ) of gpll0 , but we register it
* as a fixed rate clock instead to ease bootstrapping procedure . By
* doing this we make sure that CPU cluster clocks are able to be setup
* early during the boot process ( as it is recommended by Qualcomm ) .
*/
hw = devm_clk_hw_register_fixed_rate ( dev , " sys_apcs_aux " , NULL , 0 , 300000000 ) ;
if ( IS_ERR ( hw ) )
return PTR_ERR ( hw ) ;
return devm_of_clk_add_hw_provider ( dev , of_clk_hw_simple_get , hw ) ;
}
static struct platform_driver qcom_apcs_msm8996_clk_driver = {
. probe = qcom_apcs_msm8996_clk_probe ,
. driver = {
. name = " qcom-apcs-msm8996-clk " ,
} ,
} ;
/* Register early enough to fix the clock to be used for other cores */
static int __init qcom_apcs_msm8996_clk_init ( void )
{
return platform_driver_register ( & qcom_apcs_msm8996_clk_driver ) ;
}
postcore_initcall ( qcom_apcs_msm8996_clk_init ) ;
static void __exit qcom_apcs_msm8996_clk_exit ( void )
{
platform_driver_unregister ( & qcom_apcs_msm8996_clk_driver ) ;
}
module_exit ( qcom_apcs_msm8996_clk_exit ) ;
MODULE_AUTHOR ( " Dmitry Baryshkov <dmitry.baryshkov@linaro.org> " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " Qualcomm MSM8996 APCS clock driver " ) ;