clk: bcm281xx: define CCU clock data statically
Rather than "manually" setting up each CCU's clock entries at run time, define a flexible array of generic Kona clock structures within the CCU structure itself. Each of these entries contains generic kona clock information (like its CCU pointer and clock framework initialization data). Each also has a pointer to a structure contianing clock type-dependent initialization data (like register definitions). Since we'll iterate over these arrays we need to be sure they have slots for all potential clock index values. (E.g. for the root CCU we must have at least BCM281XX_ROOT_CCU_CLOCK_COUNT slots.) To ensure this we always define an extra entry and fill it using the special initializer LAST_KONA_CLK. Just about everything we need to know about a clock can be defined statically. As a result, kona_clk_setup() can be changed to take just a kona_clk structure as its argument, and peri_clk_setup() can be simplified. With the information pre-defined we are also able to handle most clock setup genericially. We can do away with the CCU-specific callback functions that previously were needed to set up the entries in CCU's clock array. Move the definition of the ccu_data structure down in "clk-kona.h" to avoid a forward dependency. Signed-off-by: Alex Elder <elder@linaro.org> Signed-off-by: Mike Turquette <mturquette@linaro.org>
This commit is contained in:
parent
b12151ca5c
commit
03548ec06a
|
@ -41,6 +41,11 @@ static struct peri_clk_data frac_1m_data = {
|
||||||
|
|
||||||
static struct ccu_data root_ccu_data = {
|
static struct ccu_data root_ccu_data = {
|
||||||
BCM281XX_CCU_COMMON(root, ROOT),
|
BCM281XX_CCU_COMMON(root, ROOT),
|
||||||
|
.kona_clks = {
|
||||||
|
[BCM281XX_ROOT_CCU_FRAC_1M] =
|
||||||
|
KONA_CLK(root, frac_1m, peri),
|
||||||
|
[BCM281XX_ROOT_CCU_CLOCK_COUNT] = LAST_KONA_CLK,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
/* AON CCU */
|
/* AON CCU */
|
||||||
|
@ -74,6 +79,15 @@ static struct peri_clk_data pmu_bsc_var_data = {
|
||||||
|
|
||||||
static struct ccu_data aon_ccu_data = {
|
static struct ccu_data aon_ccu_data = {
|
||||||
BCM281XX_CCU_COMMON(aon, AON),
|
BCM281XX_CCU_COMMON(aon, AON),
|
||||||
|
.kona_clks = {
|
||||||
|
[BCM281XX_AON_CCU_HUB_TIMER] =
|
||||||
|
KONA_CLK(aon, hub_timer, peri),
|
||||||
|
[BCM281XX_AON_CCU_PMU_BSC] =
|
||||||
|
KONA_CLK(aon, pmu_bsc, peri),
|
||||||
|
[BCM281XX_AON_CCU_PMU_BSC_VAR] =
|
||||||
|
KONA_CLK(aon, pmu_bsc_var, peri),
|
||||||
|
[BCM281XX_AON_CCU_CLOCK_COUNT] = LAST_KONA_CLK,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Hub CCU */
|
/* Hub CCU */
|
||||||
|
@ -88,6 +102,11 @@ static struct peri_clk_data tmon_1m_data = {
|
||||||
|
|
||||||
static struct ccu_data hub_ccu_data = {
|
static struct ccu_data hub_ccu_data = {
|
||||||
BCM281XX_CCU_COMMON(hub, HUB),
|
BCM281XX_CCU_COMMON(hub, HUB),
|
||||||
|
.kona_clks = {
|
||||||
|
[BCM281XX_HUB_CCU_TMON_1M] =
|
||||||
|
KONA_CLK(hub, tmon_1m, peri),
|
||||||
|
[BCM281XX_HUB_CCU_CLOCK_COUNT] = LAST_KONA_CLK,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Master CCU */
|
/* Master CCU */
|
||||||
|
@ -175,6 +194,23 @@ static struct peri_clk_data hsic2_12m_data = {
|
||||||
|
|
||||||
static struct ccu_data master_ccu_data = {
|
static struct ccu_data master_ccu_data = {
|
||||||
BCM281XX_CCU_COMMON(master, MASTER),
|
BCM281XX_CCU_COMMON(master, MASTER),
|
||||||
|
.kona_clks = {
|
||||||
|
[BCM281XX_MASTER_CCU_SDIO1] =
|
||||||
|
KONA_CLK(master, sdio1, peri),
|
||||||
|
[BCM281XX_MASTER_CCU_SDIO2] =
|
||||||
|
KONA_CLK(master, sdio2, peri),
|
||||||
|
[BCM281XX_MASTER_CCU_SDIO3] =
|
||||||
|
KONA_CLK(master, sdio3, peri),
|
||||||
|
[BCM281XX_MASTER_CCU_SDIO4] =
|
||||||
|
KONA_CLK(master, sdio4, peri),
|
||||||
|
[BCM281XX_MASTER_CCU_USB_IC] =
|
||||||
|
KONA_CLK(master, usb_ic, peri),
|
||||||
|
[BCM281XX_MASTER_CCU_HSIC2_48M] =
|
||||||
|
KONA_CLK(master, hsic2_48m, peri),
|
||||||
|
[BCM281XX_MASTER_CCU_HSIC2_12M] =
|
||||||
|
KONA_CLK(master, hsic2_12m, peri),
|
||||||
|
[BCM281XX_MASTER_CCU_CLOCK_COUNT] = LAST_KONA_CLK,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Slave CCU */
|
/* Slave CCU */
|
||||||
|
@ -287,96 +323,56 @@ static struct peri_clk_data pwm_data = {
|
||||||
|
|
||||||
static struct ccu_data slave_ccu_data = {
|
static struct ccu_data slave_ccu_data = {
|
||||||
BCM281XX_CCU_COMMON(slave, SLAVE),
|
BCM281XX_CCU_COMMON(slave, SLAVE),
|
||||||
|
.kona_clks = {
|
||||||
|
[BCM281XX_SLAVE_CCU_UARTB] =
|
||||||
|
KONA_CLK(slave, uartb, peri),
|
||||||
|
[BCM281XX_SLAVE_CCU_UARTB2] =
|
||||||
|
KONA_CLK(slave, uartb2, peri),
|
||||||
|
[BCM281XX_SLAVE_CCU_UARTB3] =
|
||||||
|
KONA_CLK(slave, uartb3, peri),
|
||||||
|
[BCM281XX_SLAVE_CCU_UARTB4] =
|
||||||
|
KONA_CLK(slave, uartb4, peri),
|
||||||
|
[BCM281XX_SLAVE_CCU_SSP0] =
|
||||||
|
KONA_CLK(slave, ssp0, peri),
|
||||||
|
[BCM281XX_SLAVE_CCU_SSP2] =
|
||||||
|
KONA_CLK(slave, ssp2, peri),
|
||||||
|
[BCM281XX_SLAVE_CCU_BSC1] =
|
||||||
|
KONA_CLK(slave, bsc1, peri),
|
||||||
|
[BCM281XX_SLAVE_CCU_BSC2] =
|
||||||
|
KONA_CLK(slave, bsc2, peri),
|
||||||
|
[BCM281XX_SLAVE_CCU_BSC3] =
|
||||||
|
KONA_CLK(slave, bsc3, peri),
|
||||||
|
[BCM281XX_SLAVE_CCU_PWM] =
|
||||||
|
KONA_CLK(slave, pwm, peri),
|
||||||
|
[BCM281XX_SLAVE_CCU_CLOCK_COUNT] = LAST_KONA_CLK,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
|
||||||
* CCU setup routines
|
|
||||||
*
|
|
||||||
* These are called from kona_dt_ccu_setup() to initialize the array
|
|
||||||
* of clocks provided by the CCU. Once allocated, the entries in
|
|
||||||
* the array are initialized by calling kona_clk_setup() with the
|
|
||||||
* initialization data for each clock. They return 0 if successful
|
|
||||||
* or an error code otherwise.
|
|
||||||
*/
|
|
||||||
static int __init bcm281xx_root_ccu_clks_setup(struct ccu_data *ccu)
|
|
||||||
{
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_ROOT_CCU_FRAC_1M, frac_1m);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int __init bcm281xx_aon_ccu_clks_setup(struct ccu_data *ccu)
|
|
||||||
{
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_AON_CCU_HUB_TIMER, hub_timer);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_AON_CCU_PMU_BSC, pmu_bsc);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_AON_CCU_PMU_BSC_VAR, pmu_bsc_var);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int __init bcm281xx_hub_ccu_clks_setup(struct ccu_data *ccu)
|
|
||||||
{
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_HUB_CCU_TMON_1M, tmon_1m);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int __init bcm281xx_master_ccu_clks_setup(struct ccu_data *ccu)
|
|
||||||
{
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_MASTER_CCU_SDIO1, sdio1);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_MASTER_CCU_SDIO2, sdio2);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_MASTER_CCU_SDIO3, sdio3);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_MASTER_CCU_SDIO4, sdio4);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_MASTER_CCU_USB_IC, usb_ic);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_MASTER_CCU_HSIC2_48M, hsic2_48m);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_MASTER_CCU_HSIC2_12M, hsic2_12m);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int __init bcm281xx_slave_ccu_clks_setup(struct ccu_data *ccu)
|
|
||||||
{
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_SLAVE_CCU_UARTB, uartb);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_SLAVE_CCU_UARTB2, uartb2);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_SLAVE_CCU_UARTB3, uartb3);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_SLAVE_CCU_UARTB4, uartb4);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_SLAVE_CCU_SSP0, ssp0);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_SLAVE_CCU_SSP2, ssp2);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_SLAVE_CCU_BSC1, bsc1);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_SLAVE_CCU_BSC2, bsc2);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_SLAVE_CCU_BSC3, bsc3);
|
|
||||||
PERI_CLK_SETUP(ccu, BCM281XX_SLAVE_CCU_PWM, pwm);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Device tree match table callback functions */
|
/* Device tree match table callback functions */
|
||||||
|
|
||||||
static void __init kona_dt_root_ccu_setup(struct device_node *node)
|
static void __init kona_dt_root_ccu_setup(struct device_node *node)
|
||||||
{
|
{
|
||||||
kona_dt_ccu_setup(&root_ccu_data, node, bcm281xx_root_ccu_clks_setup);
|
kona_dt_ccu_setup(&root_ccu_data, node);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __init kona_dt_aon_ccu_setup(struct device_node *node)
|
static void __init kona_dt_aon_ccu_setup(struct device_node *node)
|
||||||
{
|
{
|
||||||
kona_dt_ccu_setup(&aon_ccu_data, node, bcm281xx_aon_ccu_clks_setup);
|
kona_dt_ccu_setup(&aon_ccu_data, node);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __init kona_dt_hub_ccu_setup(struct device_node *node)
|
static void __init kona_dt_hub_ccu_setup(struct device_node *node)
|
||||||
{
|
{
|
||||||
kona_dt_ccu_setup(&hub_ccu_data, node, bcm281xx_hub_ccu_clks_setup);
|
kona_dt_ccu_setup(&hub_ccu_data, node);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __init kona_dt_master_ccu_setup(struct device_node *node)
|
static void __init kona_dt_master_ccu_setup(struct device_node *node)
|
||||||
{
|
{
|
||||||
kona_dt_ccu_setup(&master_ccu_data, node,
|
kona_dt_ccu_setup(&master_ccu_data, node);
|
||||||
bcm281xx_master_ccu_clks_setup);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __init kona_dt_slave_ccu_setup(struct device_node *node)
|
static void __init kona_dt_slave_ccu_setup(struct device_node *node)
|
||||||
{
|
{
|
||||||
kona_dt_ccu_setup(&slave_ccu_data, node,
|
kona_dt_ccu_setup(&slave_ccu_data, node);
|
||||||
bcm281xx_slave_ccu_clks_setup);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
CLK_OF_DECLARE(bcm281xx_root_ccu, BCM281XX_DT_ROOT_CCU_COMPAT,
|
CLK_OF_DECLARE(bcm281xx_root_ccu, BCM281XX_DT_ROOT_CCU_COMPAT,
|
||||||
|
|
|
@ -567,7 +567,6 @@ static void peri_clk_teardown(struct peri_clk_data *data,
|
||||||
struct clk_init_data *init_data)
|
struct clk_init_data *init_data)
|
||||||
{
|
{
|
||||||
clk_sel_teardown(&data->sel, init_data);
|
clk_sel_teardown(&data->sel, init_data);
|
||||||
init_data->ops = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -576,10 +575,9 @@ static void peri_clk_teardown(struct peri_clk_data *data,
|
||||||
* that can be assigned if the clock has one or more parent clocks
|
* that can be assigned if the clock has one or more parent clocks
|
||||||
* associated with it.
|
* associated with it.
|
||||||
*/
|
*/
|
||||||
static int peri_clk_setup(struct ccu_data *ccu, struct peri_clk_data *data,
|
static int
|
||||||
struct clk_init_data *init_data)
|
peri_clk_setup(struct peri_clk_data *data, struct clk_init_data *init_data)
|
||||||
{
|
{
|
||||||
init_data->ops = &kona_peri_clk_ops;
|
|
||||||
init_data->flags = CLK_IGNORE_UNUSED;
|
init_data->flags = CLK_IGNORE_UNUSED;
|
||||||
|
|
||||||
return clk_sel_setup(data->clocks, &data->sel, init_data);
|
return clk_sel_setup(data->clocks, &data->sel, init_data);
|
||||||
|
@ -617,39 +615,26 @@ static void kona_clk_teardown(struct clk *clk)
|
||||||
bcm_clk_teardown(bcm_clk);
|
bcm_clk_teardown(bcm_clk);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct clk *kona_clk_setup(struct ccu_data *ccu, const char *name,
|
struct clk *kona_clk_setup(struct kona_clk *bcm_clk)
|
||||||
enum bcm_clk_type type, void *data)
|
|
||||||
{
|
{
|
||||||
struct kona_clk *bcm_clk;
|
struct clk_init_data *init_data = &bcm_clk->init_data;
|
||||||
struct clk_init_data *init_data;
|
|
||||||
struct clk *clk = NULL;
|
struct clk *clk = NULL;
|
||||||
|
|
||||||
bcm_clk = kzalloc(sizeof(*bcm_clk), GFP_KERNEL);
|
switch (bcm_clk->type) {
|
||||||
if (!bcm_clk) {
|
|
||||||
pr_err("%s: failed to allocate bcm_clk for %s\n", __func__,
|
|
||||||
name);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
bcm_clk->ccu = ccu;
|
|
||||||
bcm_clk->init_data.name = name;
|
|
||||||
|
|
||||||
init_data = &bcm_clk->init_data;
|
|
||||||
init_data->name = name;
|
|
||||||
switch (type) {
|
|
||||||
case bcm_clk_peri:
|
case bcm_clk_peri:
|
||||||
if (peri_clk_setup(ccu, data, init_data))
|
if (peri_clk_setup(bcm_clk->u.data, init_data))
|
||||||
goto out_free;
|
return NULL;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
data = NULL;
|
pr_err("%s: clock type %d invalid for %s\n", __func__,
|
||||||
break;
|
(int)bcm_clk->type, init_data->name);
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
bcm_clk->type = type;
|
|
||||||
bcm_clk->u.data = data;
|
|
||||||
|
|
||||||
/* Make sure everything makes sense before we set it up */
|
/* Make sure everything makes sense before we set it up */
|
||||||
if (!kona_clk_valid(bcm_clk)) {
|
if (!kona_clk_valid(bcm_clk)) {
|
||||||
pr_err("%s: clock data invalid for %s\n", __func__, name);
|
pr_err("%s: clock data invalid for %s\n", __func__,
|
||||||
|
init_data->name);
|
||||||
goto out_teardown;
|
goto out_teardown;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -657,7 +642,7 @@ struct clk *kona_clk_setup(struct ccu_data *ccu, const char *name,
|
||||||
clk = clk_register(NULL, &bcm_clk->hw);
|
clk = clk_register(NULL, &bcm_clk->hw);
|
||||||
if (IS_ERR(clk)) {
|
if (IS_ERR(clk)) {
|
||||||
pr_err("%s: error registering clock %s (%ld)\n", __func__,
|
pr_err("%s: error registering clock %s (%ld)\n", __func__,
|
||||||
name, PTR_ERR(clk));
|
init_data->name, PTR_ERR(clk));
|
||||||
goto out_teardown;
|
goto out_teardown;
|
||||||
}
|
}
|
||||||
BUG_ON(!clk);
|
BUG_ON(!clk);
|
||||||
|
@ -665,8 +650,6 @@ struct clk *kona_clk_setup(struct ccu_data *ccu, const char *name,
|
||||||
return clk;
|
return clk;
|
||||||
out_teardown:
|
out_teardown:
|
||||||
bcm_clk_teardown(bcm_clk);
|
bcm_clk_teardown(bcm_clk);
|
||||||
out_free:
|
|
||||||
kfree(bcm_clk);
|
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
@ -701,11 +684,11 @@ static void kona_ccu_teardown(struct ccu_data *ccu)
|
||||||
* initialize the array of clocks provided by the CCU.
|
* initialize the array of clocks provided by the CCU.
|
||||||
*/
|
*/
|
||||||
void __init kona_dt_ccu_setup(struct ccu_data *ccu,
|
void __init kona_dt_ccu_setup(struct ccu_data *ccu,
|
||||||
struct device_node *node,
|
struct device_node *node)
|
||||||
int (*ccu_clks_setup)(struct ccu_data *))
|
|
||||||
{
|
{
|
||||||
struct resource res = { 0 };
|
struct resource res = { 0 };
|
||||||
resource_size_t range;
|
resource_size_t range;
|
||||||
|
unsigned int i;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
if (ccu->clk_data.clk_num) {
|
if (ccu->clk_data.clk_num) {
|
||||||
|
@ -744,9 +727,16 @@ void __init kona_dt_ccu_setup(struct ccu_data *ccu,
|
||||||
ccu->node = of_node_get(node);
|
ccu->node = of_node_get(node);
|
||||||
list_add_tail(&ccu->links, &ccu_list);
|
list_add_tail(&ccu->links, &ccu_list);
|
||||||
|
|
||||||
/* Set up clocks array (in ccu->clk_data) */
|
/*
|
||||||
if (ccu_clks_setup(ccu))
|
* Set up each defined kona clock and save the result in
|
||||||
goto out_err;
|
* the clock framework clock array (in ccu->data). Then
|
||||||
|
* register as a provider for these clocks.
|
||||||
|
*/
|
||||||
|
for (i = 0; i < ccu->clk_data.clk_num; i++) {
|
||||||
|
if (!ccu->kona_clks[i].ccu)
|
||||||
|
continue;
|
||||||
|
ccu->clk_data.clks[i] = kona_clk_setup(&ccu->kona_clks[i]);
|
||||||
|
}
|
||||||
|
|
||||||
ret = of_clk_add_provider(node, of_clk_src_onecell_get, &ccu->clk_data);
|
ret = of_clk_add_provider(node, of_clk_src_onecell_get, &ccu->clk_data);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
|
|
@ -70,35 +70,6 @@ enum bcm_clk_type {
|
||||||
bcm_clk_peri
|
bcm_clk_peri
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
|
||||||
* Each CCU defines a mapped area of memory containing registers
|
|
||||||
* used to manage clocks implemented by the CCU. Access to memory
|
|
||||||
* within the CCU's space is serialized by a spinlock. Before any
|
|
||||||
* (other) address can be written, a special access "password" value
|
|
||||||
* must be written to its WR_ACCESS register (located at the base
|
|
||||||
* address of the range). We keep track of the name of each CCU as
|
|
||||||
* it is set up, and maintain them in a list.
|
|
||||||
*/
|
|
||||||
struct ccu_data {
|
|
||||||
void __iomem *base; /* base of mapped address space */
|
|
||||||
spinlock_t lock; /* serialization lock */
|
|
||||||
bool write_enabled; /* write access is currently enabled */
|
|
||||||
struct list_head links; /* for ccu_list */
|
|
||||||
struct device_node *node;
|
|
||||||
struct clk_onecell_data clk_data;
|
|
||||||
const char *name;
|
|
||||||
u32 range; /* byte range of address space */
|
|
||||||
};
|
|
||||||
|
|
||||||
/* Initialization for common fields in a Kona ccu_data structure */
|
|
||||||
#define KONA_CCU_COMMON(_prefix, _name, _ucase_name) \
|
|
||||||
.name = #_name "_ccu", \
|
|
||||||
.lock = __SPIN_LOCK_UNLOCKED(_name ## _ccu_data.lock), \
|
|
||||||
.links = LIST_HEAD_INIT(_name ## _ccu_data.links), \
|
|
||||||
.clk_data = { \
|
|
||||||
.clk_num = _prefix ## _ ## _ucase_name ## _CCU_CLOCK_COUNT, \
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Gating control and status is managed by a 32-bit gate register.
|
* Gating control and status is managed by a 32-bit gate register.
|
||||||
*
|
*
|
||||||
|
@ -393,18 +364,53 @@ struct kona_clk {
|
||||||
#define to_kona_clk(_hw) \
|
#define to_kona_clk(_hw) \
|
||||||
container_of(_hw, struct kona_clk, hw)
|
container_of(_hw, struct kona_clk, hw)
|
||||||
|
|
||||||
|
/* Initialization macro for an entry in a CCU's kona_clks[] array. */
|
||||||
|
#define KONA_CLK(_ccu_name, _clk_name, _type) \
|
||||||
|
{ \
|
||||||
|
.init_data = { \
|
||||||
|
.name = #_clk_name, \
|
||||||
|
.ops = &kona_ ## _type ## _clk_ops, \
|
||||||
|
}, \
|
||||||
|
.ccu = &_ccu_name ## _ccu_data, \
|
||||||
|
.type = bcm_clk_ ## _type, \
|
||||||
|
.u.data = &_clk_name ## _data, \
|
||||||
|
}
|
||||||
|
#define LAST_KONA_CLK { .type = bcm_clk_none }
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Each CCU defines a mapped area of memory containing registers
|
||||||
|
* used to manage clocks implemented by the CCU. Access to memory
|
||||||
|
* within the CCU's space is serialized by a spinlock. Before any
|
||||||
|
* (other) address can be written, a special access "password" value
|
||||||
|
* must be written to its WR_ACCESS register (located at the base
|
||||||
|
* address of the range). We keep track of the name of each CCU as
|
||||||
|
* it is set up, and maintain them in a list.
|
||||||
|
*/
|
||||||
|
struct ccu_data {
|
||||||
|
void __iomem *base; /* base of mapped address space */
|
||||||
|
spinlock_t lock; /* serialization lock */
|
||||||
|
bool write_enabled; /* write access is currently enabled */
|
||||||
|
struct list_head links; /* for ccu_list */
|
||||||
|
struct device_node *node;
|
||||||
|
struct clk_onecell_data clk_data;
|
||||||
|
const char *name;
|
||||||
|
u32 range; /* byte range of address space */
|
||||||
|
struct kona_clk kona_clks[]; /* must be last */
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Initialization for common fields in a Kona ccu_data structure */
|
||||||
|
#define KONA_CCU_COMMON(_prefix, _name, _ccuname) \
|
||||||
|
.name = #_name "_ccu", \
|
||||||
|
.lock = __SPIN_LOCK_UNLOCKED(_name ## _ccu_data.lock), \
|
||||||
|
.links = LIST_HEAD_INIT(_name ## _ccu_data.links), \
|
||||||
|
.clk_data = { \
|
||||||
|
.clk_num = _prefix ## _ ## _ccuname ## _CCU_CLOCK_COUNT, \
|
||||||
|
}
|
||||||
|
|
||||||
/* Exported globals */
|
/* Exported globals */
|
||||||
|
|
||||||
extern struct clk_ops kona_peri_clk_ops;
|
extern struct clk_ops kona_peri_clk_ops;
|
||||||
|
|
||||||
/* Help functions */
|
|
||||||
|
|
||||||
#define KONA_CLK_SETUP(_ccu, _type, _name) \
|
|
||||||
kona_clk_setup((_ccu), #_name, bcm_clk_## _type, &_name ## _data)
|
|
||||||
|
|
||||||
#define PERI_CLK_SETUP(_ccu, _id, _name) \
|
|
||||||
(_ccu)->clk_data.clks[_id] = KONA_CLK_SETUP((_ccu), peri, _name)
|
|
||||||
|
|
||||||
/* Externally visible functions */
|
/* Externally visible functions */
|
||||||
|
|
||||||
extern u64 do_div_round_closest(u64 dividend, unsigned long divisor);
|
extern u64 do_div_round_closest(u64 dividend, unsigned long divisor);
|
||||||
|
@ -412,11 +418,9 @@ extern u64 scaled_div_max(struct bcm_clk_div *div);
|
||||||
extern u64 scaled_div_build(struct bcm_clk_div *div, u32 div_value,
|
extern u64 scaled_div_build(struct bcm_clk_div *div, u32 div_value,
|
||||||
u32 billionths);
|
u32 billionths);
|
||||||
|
|
||||||
extern struct clk *kona_clk_setup(struct ccu_data *ccu, const char *name,
|
extern struct clk *kona_clk_setup(struct kona_clk *bcm_clk);
|
||||||
enum bcm_clk_type type, void *data);
|
|
||||||
extern void __init kona_dt_ccu_setup(struct ccu_data *ccu,
|
extern void __init kona_dt_ccu_setup(struct ccu_data *ccu,
|
||||||
struct device_node *node,
|
struct device_node *node);
|
||||||
int (*ccu_clks_setup)(struct ccu_data *));
|
|
||||||
extern bool __init kona_ccu_init(struct ccu_data *ccu);
|
extern bool __init kona_ccu_init(struct ccu_data *ccu);
|
||||||
|
|
||||||
#endif /* _CLK_KONA_H */
|
#endif /* _CLK_KONA_H */
|
||||||
|
|
Loading…
Reference in New Issue