mirror of https://gitee.com/openkylin/linux.git
131 lines
3.6 KiB
C
131 lines
3.6 KiB
C
|
/*
|
||
|
* OMAP2 and OMAP3 clockdomain control
|
||
|
*
|
||
|
* Copyright (C) 2008-2010 Texas Instruments, Inc.
|
||
|
* Copyright (C) 2008-2010 Nokia Corporation
|
||
|
*
|
||
|
* Derived from mach-omap2/clockdomain.c written by Paul Walmsley
|
||
|
* Rajendra Nayak <rnayak@ti.com>
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify
|
||
|
* it under the terms of the GNU General Public License version 2 as
|
||
|
* published by the Free Software Foundation.
|
||
|
*/
|
||
|
|
||
|
#include <linux/types.h>
|
||
|
#include <plat/prcm.h>
|
||
|
#include "prm.h"
|
||
|
#include "prm2xxx_3xxx.h"
|
||
|
#include "cm.h"
|
||
|
#include "cm2xxx_3xxx.h"
|
||
|
#include "cm-regbits-24xx.h"
|
||
|
#include "cm-regbits-34xx.h"
|
||
|
#include "clockdomain.h"
|
||
|
|
||
|
static int omap2_clkdm_add_wkdep(struct clockdomain *clkdm1,
|
||
|
struct clockdomain *clkdm2)
|
||
|
{
|
||
|
omap2_prm_set_mod_reg_bits((1 << clkdm2->dep_bit),
|
||
|
clkdm1->pwrdm.ptr->prcm_offs, PM_WKDEP);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int omap2_clkdm_del_wkdep(struct clockdomain *clkdm1,
|
||
|
struct clockdomain *clkdm2)
|
||
|
{
|
||
|
omap2_prm_clear_mod_reg_bits((1 << clkdm2->dep_bit),
|
||
|
clkdm1->pwrdm.ptr->prcm_offs, PM_WKDEP);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int omap2_clkdm_read_wkdep(struct clockdomain *clkdm1,
|
||
|
struct clockdomain *clkdm2)
|
||
|
{
|
||
|
return omap2_prm_read_mod_bits_shift(clkdm1->pwrdm.ptr->prcm_offs,
|
||
|
PM_WKDEP, (1 << clkdm2->dep_bit));
|
||
|
}
|
||
|
|
||
|
static int omap2_clkdm_clear_all_wkdeps(struct clockdomain *clkdm)
|
||
|
{
|
||
|
struct clkdm_dep *cd;
|
||
|
u32 mask = 0;
|
||
|
|
||
|
for (cd = clkdm->wkdep_srcs; cd && cd->clkdm_name; cd++) {
|
||
|
if (!omap_chip_is(cd->omap_chip))
|
||
|
continue;
|
||
|
if (!cd->clkdm)
|
||
|
continue; /* only happens if data is erroneous */
|
||
|
|
||
|
/* PRM accesses are slow, so minimize them */
|
||
|
mask |= 1 << cd->clkdm->dep_bit;
|
||
|
atomic_set(&cd->wkdep_usecount, 0);
|
||
|
}
|
||
|
|
||
|
omap2_prm_clear_mod_reg_bits(mask, clkdm->pwrdm.ptr->prcm_offs,
|
||
|
PM_WKDEP);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int omap3_clkdm_add_sleepdep(struct clockdomain *clkdm1,
|
||
|
struct clockdomain *clkdm2)
|
||
|
{
|
||
|
omap2_cm_set_mod_reg_bits((1 << clkdm2->dep_bit),
|
||
|
clkdm1->pwrdm.ptr->prcm_offs,
|
||
|
OMAP3430_CM_SLEEPDEP);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int omap3_clkdm_del_sleepdep(struct clockdomain *clkdm1,
|
||
|
struct clockdomain *clkdm2)
|
||
|
{
|
||
|
omap2_cm_clear_mod_reg_bits((1 << clkdm2->dep_bit),
|
||
|
clkdm1->pwrdm.ptr->prcm_offs,
|
||
|
OMAP3430_CM_SLEEPDEP);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int omap3_clkdm_read_sleepdep(struct clockdomain *clkdm1,
|
||
|
struct clockdomain *clkdm2)
|
||
|
{
|
||
|
return omap2_prm_read_mod_bits_shift(clkdm1->pwrdm.ptr->prcm_offs,
|
||
|
OMAP3430_CM_SLEEPDEP, (1 << clkdm2->dep_bit));
|
||
|
}
|
||
|
|
||
|
static int omap3_clkdm_clear_all_sleepdeps(struct clockdomain *clkdm)
|
||
|
{
|
||
|
struct clkdm_dep *cd;
|
||
|
u32 mask = 0;
|
||
|
|
||
|
for (cd = clkdm->sleepdep_srcs; cd && cd->clkdm_name; cd++) {
|
||
|
if (!omap_chip_is(cd->omap_chip))
|
||
|
continue;
|
||
|
if (!cd->clkdm)
|
||
|
continue; /* only happens if data is erroneous */
|
||
|
|
||
|
/* PRM accesses are slow, so minimize them */
|
||
|
mask |= 1 << cd->clkdm->dep_bit;
|
||
|
atomic_set(&cd->sleepdep_usecount, 0);
|
||
|
}
|
||
|
omap2_prm_clear_mod_reg_bits(mask, clkdm->pwrdm.ptr->prcm_offs,
|
||
|
OMAP3430_CM_SLEEPDEP);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
struct clkdm_ops omap2_clkdm_operations = {
|
||
|
.clkdm_add_wkdep = omap2_clkdm_add_wkdep,
|
||
|
.clkdm_del_wkdep = omap2_clkdm_del_wkdep,
|
||
|
.clkdm_read_wkdep = omap2_clkdm_read_wkdep,
|
||
|
.clkdm_clear_all_wkdeps = omap2_clkdm_clear_all_wkdeps,
|
||
|
};
|
||
|
|
||
|
struct clkdm_ops omap3_clkdm_operations = {
|
||
|
.clkdm_add_wkdep = omap2_clkdm_add_wkdep,
|
||
|
.clkdm_del_wkdep = omap2_clkdm_del_wkdep,
|
||
|
.clkdm_read_wkdep = omap2_clkdm_read_wkdep,
|
||
|
.clkdm_clear_all_wkdeps = omap2_clkdm_clear_all_wkdeps,
|
||
|
.clkdm_add_sleepdep = omap3_clkdm_add_sleepdep,
|
||
|
.clkdm_del_sleepdep = omap3_clkdm_del_sleepdep,
|
||
|
.clkdm_read_sleepdep = omap3_clkdm_read_sleepdep,
|
||
|
.clkdm_clear_all_sleepdeps = omap3_clkdm_clear_all_sleepdeps,
|
||
|
};
|