Commit cc323782 authored by Lars-Peter Clausen's avatar Lars-Peter Clausen Committed by Stephen Boyd

clk: vc5: Check IO access results

The devices of the versaclk clock generator family use an I2C control bus.
IO access on an I2C bus can fail for various reasons.

The driver currently ignores the return value of most IO operations. This
results in silent failure. To avoid this check the return value and in case
of an error abort the operation and propagate the error code to the caller.
Signed-off-by: default avatarLars-Peter Clausen <lars@metafoo.de>
Link: https://lore.kernel.org/r/20220719094637.844946-1-lars@metafoo.deReviewed-by: default avatarLuca Ceresoli <luca@lucaceresoli.net>
Signed-off-by: default avatarStephen Boyd <sboyd@kernel.org>
parent 568035b0
...@@ -230,8 +230,12 @@ static unsigned char vc5_mux_get_parent(struct clk_hw *hw) ...@@ -230,8 +230,12 @@ static unsigned char vc5_mux_get_parent(struct clk_hw *hw)
container_of(hw, struct vc5_driver_data, clk_mux); container_of(hw, struct vc5_driver_data, clk_mux);
const u8 mask = VC5_PRIM_SRC_SHDN_EN_XTAL | VC5_PRIM_SRC_SHDN_EN_CLKIN; const u8 mask = VC5_PRIM_SRC_SHDN_EN_XTAL | VC5_PRIM_SRC_SHDN_EN_CLKIN;
unsigned int src; unsigned int src;
int ret;
ret = regmap_read(vc5->regmap, VC5_PRIM_SRC_SHDN, &src);
if (ret)
return 0;
regmap_read(vc5->regmap, VC5_PRIM_SRC_SHDN, &src);
src &= mask; src &= mask;
if (src == VC5_PRIM_SRC_SHDN_EN_XTAL) if (src == VC5_PRIM_SRC_SHDN_EN_XTAL)
...@@ -286,8 +290,12 @@ static unsigned long vc5_dbl_recalc_rate(struct clk_hw *hw, ...@@ -286,8 +290,12 @@ static unsigned long vc5_dbl_recalc_rate(struct clk_hw *hw,
struct vc5_driver_data *vc5 = struct vc5_driver_data *vc5 =
container_of(hw, struct vc5_driver_data, clk_mul); container_of(hw, struct vc5_driver_data, clk_mul);
unsigned int premul; unsigned int premul;
int ret;
ret = regmap_read(vc5->regmap, VC5_PRIM_SRC_SHDN, &premul);
if (ret)
return 0;
regmap_read(vc5->regmap, VC5_PRIM_SRC_SHDN, &premul);
if (premul & VC5_PRIM_SRC_SHDN_EN_DOUBLE_XTAL_FREQ) if (premul & VC5_PRIM_SRC_SHDN_EN_DOUBLE_XTAL_FREQ)
parent_rate *= 2; parent_rate *= 2;
...@@ -315,11 +323,9 @@ static int vc5_dbl_set_rate(struct clk_hw *hw, unsigned long rate, ...@@ -315,11 +323,9 @@ static int vc5_dbl_set_rate(struct clk_hw *hw, unsigned long rate,
else else
mask = 0; mask = 0;
regmap_update_bits(vc5->regmap, VC5_PRIM_SRC_SHDN, return regmap_update_bits(vc5->regmap, VC5_PRIM_SRC_SHDN,
VC5_PRIM_SRC_SHDN_EN_DOUBLE_XTAL_FREQ, VC5_PRIM_SRC_SHDN_EN_DOUBLE_XTAL_FREQ,
mask); mask);
return 0;
} }
static const struct clk_ops vc5_dbl_ops = { static const struct clk_ops vc5_dbl_ops = {
...@@ -334,14 +340,19 @@ static unsigned long vc5_pfd_recalc_rate(struct clk_hw *hw, ...@@ -334,14 +340,19 @@ static unsigned long vc5_pfd_recalc_rate(struct clk_hw *hw,
struct vc5_driver_data *vc5 = struct vc5_driver_data *vc5 =
container_of(hw, struct vc5_driver_data, clk_pfd); container_of(hw, struct vc5_driver_data, clk_pfd);
unsigned int prediv, div; unsigned int prediv, div;
int ret;
regmap_read(vc5->regmap, VC5_VCO_CTRL_AND_PREDIV, &prediv); ret = regmap_read(vc5->regmap, VC5_VCO_CTRL_AND_PREDIV, &prediv);
if (ret)
return 0;
/* The bypass_prediv is set, PLL fed from Ref_in directly. */ /* The bypass_prediv is set, PLL fed from Ref_in directly. */
if (prediv & VC5_VCO_CTRL_AND_PREDIV_BYPASS_PREDIV) if (prediv & VC5_VCO_CTRL_AND_PREDIV_BYPASS_PREDIV)
return parent_rate; return parent_rate;
regmap_read(vc5->regmap, VC5_REF_DIVIDER, &div); ret = regmap_read(vc5->regmap, VC5_REF_DIVIDER, &div);
if (ret)
return 0;
/* The Sel_prediv2 is set, PLL fed from prediv2 (Ref_in / 2) */ /* The Sel_prediv2 is set, PLL fed from prediv2 (Ref_in / 2) */
if (div & VC5_REF_DIVIDER_SEL_PREDIV2) if (div & VC5_REF_DIVIDER_SEL_PREDIV2)
...@@ -376,15 +387,18 @@ static int vc5_pfd_set_rate(struct clk_hw *hw, unsigned long rate, ...@@ -376,15 +387,18 @@ static int vc5_pfd_set_rate(struct clk_hw *hw, unsigned long rate,
struct vc5_driver_data *vc5 = struct vc5_driver_data *vc5 =
container_of(hw, struct vc5_driver_data, clk_pfd); container_of(hw, struct vc5_driver_data, clk_pfd);
unsigned long idiv; unsigned long idiv;
int ret;
u8 div; u8 div;
/* CLKIN within range of PLL input, feed directly to PLL. */ /* CLKIN within range of PLL input, feed directly to PLL. */
if (parent_rate <= 50000000) { if (parent_rate <= 50000000) {
regmap_update_bits(vc5->regmap, VC5_VCO_CTRL_AND_PREDIV, ret = regmap_update_bits(vc5->regmap, VC5_VCO_CTRL_AND_PREDIV,
VC5_VCO_CTRL_AND_PREDIV_BYPASS_PREDIV, VC5_VCO_CTRL_AND_PREDIV_BYPASS_PREDIV,
VC5_VCO_CTRL_AND_PREDIV_BYPASS_PREDIV); VC5_VCO_CTRL_AND_PREDIV_BYPASS_PREDIV);
regmap_update_bits(vc5->regmap, VC5_REF_DIVIDER, 0xff, 0x00); if (ret)
return 0; return ret;
return regmap_update_bits(vc5->regmap, VC5_REF_DIVIDER, 0xff, 0x00);
} }
idiv = DIV_ROUND_UP(parent_rate, rate); idiv = DIV_ROUND_UP(parent_rate, rate);
...@@ -395,11 +409,12 @@ static int vc5_pfd_set_rate(struct clk_hw *hw, unsigned long rate, ...@@ -395,11 +409,12 @@ static int vc5_pfd_set_rate(struct clk_hw *hw, unsigned long rate,
else else
div = VC5_REF_DIVIDER_REF_DIV(idiv); div = VC5_REF_DIVIDER_REF_DIV(idiv);
regmap_update_bits(vc5->regmap, VC5_REF_DIVIDER, 0xff, div); ret = regmap_update_bits(vc5->regmap, VC5_REF_DIVIDER, 0xff, div);
regmap_update_bits(vc5->regmap, VC5_VCO_CTRL_AND_PREDIV, if (ret)
VC5_VCO_CTRL_AND_PREDIV_BYPASS_PREDIV, 0); return ret;
return 0; return regmap_update_bits(vc5->regmap, VC5_VCO_CTRL_AND_PREDIV,
VC5_VCO_CTRL_AND_PREDIV_BYPASS_PREDIV, 0);
} }
static const struct clk_ops vc5_pfd_ops = { static const struct clk_ops vc5_pfd_ops = {
...@@ -551,9 +566,12 @@ static int vc5_fod_set_rate(struct clk_hw *hw, unsigned long rate, ...@@ -551,9 +566,12 @@ static int vc5_fod_set_rate(struct clk_hw *hw, unsigned long rate,
hwdata->div_int >> 4, hwdata->div_int << 4, hwdata->div_int >> 4, hwdata->div_int << 4,
0 0
}; };
int ret;
regmap_bulk_write(vc5->regmap, VC5_OUT_DIV_FRAC(hwdata->num, 0), ret = regmap_bulk_write(vc5->regmap, VC5_OUT_DIV_FRAC(hwdata->num, 0),
data, 14); data, 14);
if (ret)
return ret;
/* /*
* Toggle magic bit in undocumented register for unknown reason. * Toggle magic bit in undocumented register for unknown reason.
...@@ -561,12 +579,14 @@ static int vc5_fod_set_rate(struct clk_hw *hw, unsigned long rate, ...@@ -561,12 +579,14 @@ static int vc5_fod_set_rate(struct clk_hw *hw, unsigned long rate,
* datasheet somewhat implies this is needed, but the register * datasheet somewhat implies this is needed, but the register
* and the bit is not documented. * and the bit is not documented.
*/ */
regmap_update_bits(vc5->regmap, VC5_GLOBAL_REGISTER, ret = regmap_update_bits(vc5->regmap, VC5_GLOBAL_REGISTER,
VC5_GLOBAL_REGISTER_GLOBAL_RESET, 0); VC5_GLOBAL_REGISTER_GLOBAL_RESET, 0);
regmap_update_bits(vc5->regmap, VC5_GLOBAL_REGISTER, if (ret)
VC5_GLOBAL_REGISTER_GLOBAL_RESET, return ret;
VC5_GLOBAL_REGISTER_GLOBAL_RESET);
return 0; return regmap_update_bits(vc5->regmap, VC5_GLOBAL_REGISTER,
VC5_GLOBAL_REGISTER_GLOBAL_RESET,
VC5_GLOBAL_REGISTER_GLOBAL_RESET);
} }
static const struct clk_ops vc5_fod_ops = { static const struct clk_ops vc5_fod_ops = {
...@@ -606,7 +626,10 @@ static int vc5_clk_out_prepare(struct clk_hw *hw) ...@@ -606,7 +626,10 @@ static int vc5_clk_out_prepare(struct clk_hw *hw)
* If the input mux is disabled, enable it first and * If the input mux is disabled, enable it first and
* select source from matching FOD. * select source from matching FOD.
*/ */
regmap_read(vc5->regmap, VC5_OUT_DIV_CONTROL(hwdata->num), &src); ret = regmap_read(vc5->regmap, VC5_OUT_DIV_CONTROL(hwdata->num), &src);
if (ret)
return ret;
if ((src & mask) == 0) { if ((src & mask) == 0) {
src = VC5_OUT_DIV_CONTROL_RESET | VC5_OUT_DIV_CONTROL_EN_FOD; src = VC5_OUT_DIV_CONTROL_RESET | VC5_OUT_DIV_CONTROL_EN_FOD;
ret = regmap_update_bits(vc5->regmap, ret = regmap_update_bits(vc5->regmap,
...@@ -617,18 +640,24 @@ static int vc5_clk_out_prepare(struct clk_hw *hw) ...@@ -617,18 +640,24 @@ static int vc5_clk_out_prepare(struct clk_hw *hw)
} }
/* Enable the clock buffer */ /* Enable the clock buffer */
regmap_update_bits(vc5->regmap, VC5_CLK_OUTPUT_CFG(hwdata->num, 1), ret = regmap_update_bits(vc5->regmap,
VC5_CLK_OUTPUT_CFG1_EN_CLKBUF, VC5_CLK_OUTPUT_CFG(hwdata->num, 1),
VC5_CLK_OUTPUT_CFG1_EN_CLKBUF); VC5_CLK_OUTPUT_CFG1_EN_CLKBUF,
VC5_CLK_OUTPUT_CFG1_EN_CLKBUF);
if (ret)
return ret;
if (hwdata->clk_output_cfg0_mask) { if (hwdata->clk_output_cfg0_mask) {
dev_dbg(&vc5->client->dev, "Update output %d mask 0x%0X val 0x%0X\n", dev_dbg(&vc5->client->dev, "Update output %d mask 0x%0X val 0x%0X\n",
hwdata->num, hwdata->clk_output_cfg0_mask, hwdata->num, hwdata->clk_output_cfg0_mask,
hwdata->clk_output_cfg0); hwdata->clk_output_cfg0);
regmap_update_bits(vc5->regmap, ret = regmap_update_bits(vc5->regmap,
VC5_CLK_OUTPUT_CFG(hwdata->num, 0), VC5_CLK_OUTPUT_CFG(hwdata->num, 0),
hwdata->clk_output_cfg0_mask, hwdata->clk_output_cfg0_mask,
hwdata->clk_output_cfg0); hwdata->clk_output_cfg0);
if (ret)
return ret;
} }
return 0; return 0;
...@@ -656,8 +685,12 @@ static unsigned char vc5_clk_out_get_parent(struct clk_hw *hw) ...@@ -656,8 +685,12 @@ static unsigned char vc5_clk_out_get_parent(struct clk_hw *hw)
const u8 extclk = VC5_OUT_DIV_CONTROL_SELB_NORM | const u8 extclk = VC5_OUT_DIV_CONTROL_SELB_NORM |
VC5_OUT_DIV_CONTROL_SEL_EXT; VC5_OUT_DIV_CONTROL_SEL_EXT;
unsigned int src; unsigned int src;
int ret;
ret = regmap_read(vc5->regmap, VC5_OUT_DIV_CONTROL(hwdata->num), &src);
if (ret)
return 0;
regmap_read(vc5->regmap, VC5_OUT_DIV_CONTROL(hwdata->num), &src);
src &= mask; src &= mask;
if (src == 0) /* Input mux set to DISABLED */ if (src == 0) /* Input mux set to DISABLED */
...@@ -819,22 +852,27 @@ static int vc5_update_cap_load(struct device_node *node, struct vc5_driver_data ...@@ -819,22 +852,27 @@ static int vc5_update_cap_load(struct device_node *node, struct vc5_driver_data
{ {
u32 value; u32 value;
int mapped_value; int mapped_value;
int ret;
if (!of_property_read_u32(node, "idt,xtal-load-femtofarads", &value)) { if (of_property_read_u32(node, "idt,xtal-load-femtofarads", &value))
mapped_value = vc5_map_cap_value(value); return 0;
if (mapped_value < 0)
return mapped_value;
/*
* The mapped_value is really the high 6 bits of
* VC5_XTAL_X1_LOAD_CAP and VC5_XTAL_X2_LOAD_CAP, so
* shift the value 2 places.
*/
regmap_update_bits(vc5->regmap, VC5_XTAL_X1_LOAD_CAP, ~0x03, mapped_value << 2);
regmap_update_bits(vc5->regmap, VC5_XTAL_X2_LOAD_CAP, ~0x03, mapped_value << 2);
}
return 0; mapped_value = vc5_map_cap_value(value);
if (mapped_value < 0)
return mapped_value;
/*
* The mapped_value is really the high 6 bits of
* VC5_XTAL_X1_LOAD_CAP and VC5_XTAL_X2_LOAD_CAP, so
* shift the value 2 places.
*/
ret = regmap_update_bits(vc5->regmap, VC5_XTAL_X1_LOAD_CAP, ~0x03,
mapped_value << 2);
if (ret)
return ret;
return regmap_update_bits(vc5->regmap, VC5_XTAL_X2_LOAD_CAP, ~0x03,
mapped_value << 2);
} }
static int vc5_update_slew(struct device_node *np_output, static int vc5_update_slew(struct device_node *np_output,
...@@ -956,7 +994,10 @@ static int vc5_probe(struct i2c_client *client) ...@@ -956,7 +994,10 @@ static int vc5_probe(struct i2c_client *client)
"could not read idt,output-enable-active\n"); "could not read idt,output-enable-active\n");
} }
regmap_update_bits(vc5->regmap, VC5_PRIM_SRC_SHDN, src_mask, src_val); ret = regmap_update_bits(vc5->regmap, VC5_PRIM_SRC_SHDN, src_mask,
src_val);
if (ret)
return ret;
/* Register clock input mux */ /* Register clock input mux */
memset(&init, 0, sizeof(init)); memset(&init, 0, sizeof(init));
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment