]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
clk: zynqmp: fix check for fractional clock
authorMichael Tretter <m.tretter@pengutronix.de>
Tue, 19 Mar 2019 10:01:46 +0000 (11:01 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 31 May 2019 13:45:07 +0000 (06:45 -0700)
[ Upstream commit c06e64407e031e71c67f45f07981510ca4c880a1 ]

The firmware sets BIT(13) in clkflag to mark a divider as fractional
divider. The clock driver copies the clkflag straight to the flags of
the common clock framework. In the common clk framework flags, BIT(13)
is defined as CLK_DUTY_CYCLE_PARENT.

Add a new field to the zynqmp_clk_divider to specify if a divider is a
fractional devider. Set this field based on the clkflag when registering
a divider.

At the same time, unset BIT(13) from clkflag when copying the flags to
the common clk framework flags.

Signed-off-by: Michael Tretter <m.tretter@pengutronix.de>
Signed-off-by: Stephen Boyd <sboyd@kernel.org>
Signed-off-by: Sasha Levin <sashal@kernel.org>
drivers/clk/zynqmp/divider.c

index a371c66e72ef682d04655e2eaa0e288475380727..bd9b5fbc443b3b14a5736d356dfff888f3c8daef 100644 (file)
  * struct zynqmp_clk_divider - adjustable divider clock
  * @hw:                handle between common and hardware-specific interfaces
  * @flags:     Hardware specific flags
+ * @is_frac:   The divider is a fractional divider
  * @clk_id:    Id of clock
  * @div_type:  divisor type (TYPE_DIV1 or TYPE_DIV2)
  */
 struct zynqmp_clk_divider {
        struct clk_hw hw;
        u8 flags;
+       bool is_frac;
        u32 clk_id;
        u32 div_type;
 };
@@ -116,8 +118,7 @@ static long zynqmp_clk_divider_round_rate(struct clk_hw *hw,
 
        bestdiv = zynqmp_divider_get_val(*prate, rate);
 
-       if ((clk_hw_get_flags(hw) & CLK_SET_RATE_PARENT) &&
-           (divider->flags & CLK_FRAC))
+       if ((clk_hw_get_flags(hw) & CLK_SET_RATE_PARENT) && divider->is_frac)
                bestdiv = rate % *prate ? 1 : bestdiv;
        *prate = rate * bestdiv;
 
@@ -195,11 +196,13 @@ struct clk_hw *zynqmp_clk_register_divider(const char *name,
 
        init.name = name;
        init.ops = &zynqmp_clk_divider_ops;
-       init.flags = nodes->flag;
+       /* CLK_FRAC is not defined in the common clk framework */
+       init.flags = nodes->flag & ~CLK_FRAC;
        init.parent_names = parents;
        init.num_parents = 1;
 
        /* struct clk_divider assignments */
+       div->is_frac = !!(nodes->flag & CLK_FRAC);
        div->flags = nodes->type_flag;
        div->hw.init = &init;
        div->clk_id = clk_id;