On 09/28, Eric Anholt wrote:
> +
> +static const char *bcm2835_clock_per_parents[] = {
> +static const char *bcm2835_clock_vpu_parents[] = {
> +static const char *bcm2835_clock_osc_parents[] = {

Can these parent arrays be const char * const ?

> +     "gnd",
> +     "xosc",
> +     "testdebug0",
> +     "testdebug1"
> +};
> +
> +/*
> + * Used for a 1Mhz clock for the system clocksource, and also used by
> + * the watchdog timer and the camera pulse generator.
> + */
> +static struct bcm2835_clock_data bcm2835_clock_timer_data = {
> +static struct bcm2835_clock_data bcm2835_clock_otp_data = {
> +static struct bcm2835_clock_data bcm2835_clock_vpu_data = {
> +static struct bcm2835_clock_data bcm2835_clock_v3d_data = {
> +static struct bcm2835_clock_data bcm2835_clock_isp_data = {
> +static struct bcm2835_clock_data bcm2835_clock_h264_data = {
> +static struct bcm2835_clock_data bcm2835_clock_vec_data = {
> +static struct bcm2835_clock_data bcm2835_clock_uart_data = {
> +static struct bcm2835_clock_data bcm2835_clock_hsm_data = {
> +static struct bcm2835_clock_data bcm2835_clock_sdram_data = {
> +static struct bcm2835_clock_data bcm2835_clock_tsens_data = {
> +static struct bcm2835_clock_data bcm2835_clock_emmc_data = {

Can all these data structures be const?

> +static int bcm2835_pll_is_on(struct clk_hw *hw)
> +{
> +     struct bcm2835_pll *pll = container_of(hw, struct bcm2835_pll, hw);
> +     struct bcm2835_cprman *cprman = pll->cprman;
> +     const struct bcm2835_pll_data *data = pll->data;
> +
> +     return (cprman_read(cprman, data->a2w_ctrl_reg) &
> +             A2W_PLL_CTRL_PRST_DISABLE);

Useless parenthesis.

> +}
> +
> +static void bcm2835_pll_choose_ndiv_and_fdiv(unsigned long rate,
> +                                          unsigned long parent_rate,
> +                                          u32 *ndiv, u32 *fdiv)
> +{
> +     u64 div;
> +
> +     div = ((u64)rate << A2W_PLL_FRAC_BITS);
> +     do_div(div, parent_rate);
> +
> +     *ndiv = div >> A2W_PLL_FRAC_BITS;
> +     *fdiv = div & ((1 << A2W_PLL_FRAC_BITS) - 1);
> +}
[..]
> +static unsigned long bcm2835_pll_get_rate(struct clk_hw *hw,
> +                                       unsigned long parent_rate)
> +{
> +     struct bcm2835_pll *pll = container_of(hw, struct bcm2835_pll, hw);
> +     struct bcm2835_cprman *cprman = pll->cprman;
> +     const struct bcm2835_pll_data *data = pll->data;
> +     u32 a2wctrl = cprman_read(cprman, data->a2w_ctrl_reg);
> +     u32 ndiv, pdiv, fdiv;
> +
> +     if (parent_rate == 0)
> +             return 0;
> +
> +     fdiv = cprman_read(cprman, data->frac_reg) & A2W_PLL_FRAC_MASK;
> +     ndiv = (a2wctrl & A2W_PLL_CTRL_NDIV_MASK) >> A2W_PLL_CTRL_NDIV_SHIFT;
> +     pdiv = (a2wctrl & A2W_PLL_CTRL_PDIV_MASK) >> A2W_PLL_CTRL_PDIV_SHIFT;
> +
> +     if (cprman_read(cprman, data->ana_reg_base + 4) &
> +         data->ana->fb_prediv_mask) {
> +             ndiv *= 2;
> +     }

How about a local variable so that we can put the if on one line
and drop the braces?

> +
> +     return bcm2835_pll_rate_from_divisors(parent_rate, ndiv, fdiv, pdiv);
> +}
> +
[..]
> +static int bcm2835_pll_on(struct clk_hw *hw)
> +{
> +     struct bcm2835_pll *pll = container_of(hw, struct bcm2835_pll, hw);
> +     struct bcm2835_cprman *cprman = pll->cprman;
> +     const struct bcm2835_pll_data *data = pll->data;
> +
> +     /* Take the PLL out of reset. */
> +     cprman_write(cprman, data->cm_ctrl_reg,
> +                  cprman_read(cprman, data->cm_ctrl_reg) & ~CM_PLL_ANARST);
> +
> +     /* Wait for the PLL to lock. */
> +     while (!(cprman_read(cprman, CM_LOCK) & data->lock_mask))
> +             cpu_relax();

Is there any reasonable timeout that we can put here? Hopefully
this isn't an infinite loop.

> +
> +     return 0;
> +}
> +
> +static int bcm2835_pll_set_rate(struct clk_hw *hw,
> +                             unsigned long rate, unsigned long parent_rate)
> +{
[..]
> +
> +     /* Unmask the reference clock from the oscillator. */
> +     cprman_write(cprman, A2W_XOSC_CTRL,
> +                  cprman_read(cprman, A2W_XOSC_CTRL) |
> +                  data->reference_enable_mask);
> +
> +     if (do_ana_setup_first) {
> +             cprman_write(cprman, data->ana_reg_base + 12, ana3);
> +             cprman_write(cprman, data->ana_reg_base + 8, ana2);

ana2 never changes, so why do we need to write it again?

> +             cprman_write(cprman, data->ana_reg_base + 4, ana1);
> +             cprman_write(cprman, data->ana_reg_base + 0, ana0);

Maybe this should be a function that takes a u32 array of size 4.

> +     }
> +
> +     /* Set the PLL multiplier from the oscillator. */
> +     cprman_write(cprman, data->frac_reg, fdiv);
> +     cprman_write(cprman, data->a2w_ctrl_reg,
> +                  (cprman_read(cprman, data->a2w_ctrl_reg) &
> +                   ~(A2W_PLL_CTRL_NDIV_MASK |
> +                     A2W_PLL_CTRL_PDIV_MASK)) |
> +                  (ndiv << A2W_PLL_CTRL_NDIV_SHIFT) |
> +                  (pdiv << A2W_PLL_CTRL_PDIV_SHIFT));

This is a 6 line write. Can we get some local variables and do
the bit setting in different statements?

> +
> +     if (!do_ana_setup_first) {
> +             cprman_write(cprman, data->ana_reg_base + 12, ana3);
> +             cprman_write(cprman, data->ana_reg_base + 8, ana2);
> +             cprman_write(cprman, data->ana_reg_base + 4, ana1);
> +             cprman_write(cprman, data->ana_reg_base + 0, ana0);

Function would help because we do this twice.

> +     }
> +
> +     bcm2835_pll_get_rate(&pll->hw, parent_rate);
> +
> +     return 0;
> +}
> +
[...]
> +static unsigned long bcm2835_pll_divider_get_rate(struct clk_hw *hw,
> +                                               unsigned long parent_rate)
> +{
> +     struct bcm2835_pll_divider *divider =
> +             container_of(hw, struct bcm2835_pll_divider, div.hw);
> +     struct bcm2835_cprman *cprman = divider->cprman;
> +     const struct bcm2835_pll_divider_data *data = divider->data;
> +     u32 div = cprman_read(cprman, data->a2w_reg);
> +
> +     div &= ((1 << A2W_PLL_DIV_BITS) - 1);

One too many parenthesis here.

> +     if (div == 0)
> +             div = 256;
> +
> +     return parent_rate / div;
> +}
> +
[..]
> +
> +static int bcm2835_clock_is_on(struct clk_hw *hw)
> +{
> +     struct bcm2835_clock *clock =
> +             container_of(hw, struct bcm2835_clock, hw);
> +     struct bcm2835_cprman *cprman = clock->cprman;
> +     const struct bcm2835_clock_data *data = clock->data;
> +
> +     /*
> +      * The VPU clock is always on, regardless of what we might set
> +      * the enable bit to.
> +      */
> +     if (data->is_nonstop)

Maybe the variable should be called is_vpu_clock then? Or the
comment is going to go out of date soon.

> +             return true;
> +
> +     return (cprman_read(cprman, data->ctl_reg) & CM_ENABLE) != 0;
> +}
> +
> +static u32 bcm2835_clock_choose_div(struct clk_hw *hw,
> +                                 unsigned long rate,
> +                                 unsigned long parent_rate)
> +{
> +     struct bcm2835_clock *clock =
> +             container_of(hw, struct bcm2835_clock, hw);
> +     const struct bcm2835_clock_data *data = clock->data;
> +     u32 unused_frac_mask = (1 << (CM_DIV_FRAC_BITS - data->frac_bits)) - 1;

We have GENMASK for this sort of stuff.

> +     u64 temp = (u64)parent_rate << CM_DIV_FRAC_BITS;
> +     u32 div;
> +
> +     do_div(temp, rate);
> +     div = temp;
> +
> +     /* Round and mask off the unused bits */
> +     if (unused_frac_mask != 0) {
> +             div += unused_frac_mask >> 1;
> +             div &= ~unused_frac_mask;
> +     }
> +
> +     /* Clamp to the limits. */
> +     div = max(div, unused_frac_mask + 1);
> +     div = min(div, (((1 << (data->int_bits + CM_DIV_FRAC_BITS)) - 1)) &
> +               ~unused_frac_mask);
> +
> +     return div;
> +}
> +
> +static long bcm2835_clock_rate_from_divisor(struct bcm2835_clock *clock,
> +                                         unsigned long parent_rate,
> +                                         u32 div)
> +{
> +     const struct bcm2835_clock_data *data = clock->data;
> +     u64 temp;
> +
> +     /*
> +      * The divisor is a 12.12 fixed point field, but only some of
> +      * the bits are populated in any given clock.
> +      */
> +     div >>= (CM_DIV_FRAC_BITS - data->frac_bits);

Useless parenthesis here.

> +     div &= (1 << (data->int_bits + data->frac_bits)) - 1;
> +
> +     if (div == 0)
> +             return 0;
> +
> +     temp = (u64)parent_rate << data->frac_bits;
> +
> +     do_div(temp, div);
> +
> +     return temp;
> +}
> +
> +static long bcm2835_clock_round_rate(struct clk_hw *hw,
> +                                  unsigned long rate,
> +                                  unsigned long *parent_rate)
> +{
> +     struct bcm2835_clock *clock =
> +             container_of(hw, struct bcm2835_clock, hw);

Would be nice to have a macro to get this onto one line

        struct bcm2835_clock *clock = to_bcm2385_clock(hw);

> +     u32 div = bcm2835_clock_choose_div(hw, rate, *parent_rate);
> +
> +     return bcm2835_clock_rate_from_divisor(clock, *parent_rate, div);
> +}
> +
> +static unsigned long bcm2835_clock_get_rate(struct clk_hw *hw,
> +                                         unsigned long parent_rate)
> +{
> +     struct bcm2835_clock *clock =
> +             container_of(hw, struct bcm2835_clock, hw);
> +     struct bcm2835_cprman *cprman = clock->cprman;
> +     const struct bcm2835_clock_data *data = clock->data;
> +     u32 div = cprman_read(cprman, data->div_reg);
> +
> +     return bcm2835_clock_rate_from_divisor(clock, parent_rate, div);
> +}
> +
> +static void bcm2835_clock_wait_busy(struct bcm2835_clock *clock)
> +{
> +     struct bcm2835_cprman *cprman = clock->cprman;
> +     const struct bcm2835_clock_data *data = clock->data;
> +
> +     while (cprman_read(cprman, data->ctl_reg) & CM_BUSY)
> +             cpu_relax();
> +}
> +
> +static void bcm2835_clock_off(struct clk_hw *hw)
> +{
> +     struct bcm2835_clock *clock =
> +             container_of(hw, struct bcm2835_clock, hw);
> +     struct bcm2835_cprman *cprman = clock->cprman;
> +     const struct bcm2835_clock_data *data = clock->data;
> +
> +     if (data->is_nonstop)

Or we should have different clk_ops for clocks that are "nonstop" so
that we don't do any sorts of checks here.

> +             return;
> +
> +     spin_lock(&cprman->regs_lock);
> +     cprman_write(cprman, data->ctl_reg,
> +                  cprman_read(cprman, data->ctl_reg) & ~CM_ENABLE);
> +     spin_unlock(&cprman->regs_lock);
> +
> +     /* BUSY will remain high until the divider completes its cycle. */
> +     bcm2835_clock_wait_busy(clock);
> +}
> +
[..]
> +static struct clk *
> +bcm2835_register_pll_divider(struct bcm2835_cprman *cprman,
> +                          const struct bcm2835_pll_divider_data *data)
> +{
[..]
> +     clk = clk_register(cprman->dev, &divider->div.hw);

What if clk_register() fails?

> +
> +     /*
> +      * PLLH's channels have a fixed divide by 10 afterwards, which
> +      * is what our consumers are actually using.
> +      */
> +     if (data->fixed_divider != 1) {
> +             return clk_register_fixed_factor(cprman->dev, data->name,
> +                                              divider_name,
> +                                              CLK_SET_RATE_PARENT,
> +                                              1,
> +                                              data->fixed_divider);
> +     } else {
> +             return clk;
> +     }

Just return clk instead of the else return clk compound
statement.

> +}
> +
> +static struct clk *bcm2835_register_clock(struct bcm2835_cprman *cprman,
> +                                       const struct bcm2835_clock_data *data)
> +{
> +     struct bcm2835_clock *clock;
> +     struct clk_init_data init;
> +     const char *parent;
> +
> +     /*
> +      * Most of the clock generators have a mux field, so we
> +      * instantiate a generic mux as our parent to handle it.
> +      */
> +     if (data->num_mux_parents) {
> +             int i;
> +
> +             parent = kasprintf(GFP_KERNEL, "mux_%s", data->name);
> +             if (!parent)
> +                     return NULL;
> +
> +             /*
> +              * Replace our "xosc" references with the actual
> +              * oscillator's name.
> +              */
> +             for (i = 0; i < data->num_mux_parents; i++) {
> +                     if (strcmp(data->parents[i], "xosc") == 0)
> +                             data->parents[i] = cprman->osc_name;
> +             }

Braces aren't needed here.


-- 
Qualcomm Innovation Center, Inc. is a member of Code Aurora Forum,
a Linux Foundation Collaborative Project
--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to