Commit 0ae67123 authored by Marijn Suijten's avatar Marijn Suijten Committed by Stephen Boyd

clk: qcom: rcg2: Rectify clk_gfx3d rate rounding without mux division

In case the mux is not divided parent_req was mistakenly not assigned to
leading __clk_determine_rate to determine the best frequency setting for
a requested rate of 0, resulting in the msm8996 platform not booting.
Rectify this by refactoring the logic to unconditionally assign to
parent_req.rate with the clock rate the caller is expecting.

Fixes: 7cbb78a9 ("clk: qcom: rcg2: Stop hardcoding gfx3d pingpong parent numbers")
Reported-by: default avatarKonrad Dybcio <konrad.dybcio@somainline.org>
Tested-by: default avatarKonrad Dybcio <konrad.dybcio@somainline.org>
Reviewed-By: default avatarAngeloGioacchino Del Regno <angelogioacchino.delregno@somainline.org>
Signed-off-by: default avatarMarijn Suijten <marijn.suijten@somainline.org>
Link: https://lore.kernel.org/r/20210302234106.3418665-1-marijn.suijten@somainline.orgSigned-off-by: default avatarStephen Boyd <sboyd@kernel.org>
parent c9b86db2
...@@ -730,7 +730,8 @@ static int clk_gfx3d_determine_rate(struct clk_hw *hw, ...@@ -730,7 +730,8 @@ static int clk_gfx3d_determine_rate(struct clk_hw *hw,
struct clk_rate_request parent_req = { }; struct clk_rate_request parent_req = { };
struct clk_rcg2_gfx3d *cgfx = to_clk_rcg2_gfx3d(hw); struct clk_rcg2_gfx3d *cgfx = to_clk_rcg2_gfx3d(hw);
struct clk_hw *xo, *p0, *p1, *p2; struct clk_hw *xo, *p0, *p1, *p2;
unsigned long request, p0_rate; unsigned long p0_rate;
u8 mux_div = cgfx->div;
int ret; int ret;
p0 = cgfx->hws[0]; p0 = cgfx->hws[0];
...@@ -750,14 +751,15 @@ static int clk_gfx3d_determine_rate(struct clk_hw *hw, ...@@ -750,14 +751,15 @@ static int clk_gfx3d_determine_rate(struct clk_hw *hw,
return 0; return 0;
} }
request = req->rate; if (mux_div == 0)
if (cgfx->div > 1) mux_div = 1;
parent_req.rate = request = request * cgfx->div;
parent_req.rate = req->rate * mux_div;
/* This has to be a fixed rate PLL */ /* This has to be a fixed rate PLL */
p0_rate = clk_hw_get_rate(p0); p0_rate = clk_hw_get_rate(p0);
if (request == p0_rate) { if (parent_req.rate == p0_rate) {
req->rate = req->best_parent_rate = p0_rate; req->rate = req->best_parent_rate = p0_rate;
req->best_parent_hw = p0; req->best_parent_hw = p0;
return 0; return 0;
...@@ -765,7 +767,7 @@ static int clk_gfx3d_determine_rate(struct clk_hw *hw, ...@@ -765,7 +767,7 @@ static int clk_gfx3d_determine_rate(struct clk_hw *hw,
if (req->best_parent_hw == p0) { if (req->best_parent_hw == p0) {
/* Are we going back to a previously used rate? */ /* Are we going back to a previously used rate? */
if (clk_hw_get_rate(p2) == request) if (clk_hw_get_rate(p2) == parent_req.rate)
req->best_parent_hw = p2; req->best_parent_hw = p2;
else else
req->best_parent_hw = p1; req->best_parent_hw = p1;
...@@ -780,8 +782,7 @@ static int clk_gfx3d_determine_rate(struct clk_hw *hw, ...@@ -780,8 +782,7 @@ static int clk_gfx3d_determine_rate(struct clk_hw *hw,
return ret; return ret;
req->rate = req->best_parent_rate = parent_req.rate; req->rate = req->best_parent_rate = parent_req.rate;
if (cgfx->div > 1) req->rate /= mux_div;
req->rate /= cgfx->div;
return 0; return 0;
} }
......
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