Lines Matching +full:vco +full:- +full:hz
1 // SPDX-License-Identifier: GPL-2.0
8 #include <linux/clk-provider.h>
45 /* VCO range is 10.8 .. 12.1 GHz, max depends on speed grade */
74 * struct clk_si544_muldiv - Multiplier/divider settings
79 * If ls_div_bits is non-zero, hs_div must be even
80 * @delta_m: Frequency shift for small -950..+950 ppm changes, 24 bit
93 return regmap_update_bits(data->regmap, SI544_REG_OE_STATE, in si544_enable_output()
117 err = regmap_read(data->regmap, SI544_REG_OE_STATE, &val); in si544_is_prepared()
131 err = regmap_bulk_read(data->regmap, SI544_REG_HS_DIV, reg, 2); in si544_get_muldiv()
135 settings->ls_div_bits = (reg[1] >> 4) & 0x07; in si544_get_muldiv()
136 settings->hs_div = (reg[1] & 0x07) << 8 | reg[0]; in si544_get_muldiv()
138 err = regmap_bulk_read(data->regmap, SI544_REG_FBDIV0, reg, 6); in si544_get_muldiv()
142 settings->fb_div_int = reg[4] | (reg[5] & 0x07) << 8; in si544_get_muldiv()
143 settings->fb_div_frac = reg[0] | reg[1] << 8 | reg[2] << 16 | in si544_get_muldiv()
146 err = regmap_bulk_read(data->regmap, SI544_REG_ADPLL_DELTA_M0, reg, 3); in si544_get_muldiv()
150 /* Interpret as 24-bit signed number */ in si544_get_muldiv()
151 settings->delta_m = reg[0] << 8 | reg[1] << 16 | reg[2] << 24; in si544_get_muldiv()
152 settings->delta_m >>= 8; in si544_get_muldiv()
165 return regmap_bulk_write(data->regmap, SI544_REG_ADPLL_DELTA_M0, in si544_set_delta_m()
175 reg[0] = settings->hs_div; in si544_set_muldiv()
176 reg[1] = settings->hs_div >> 8 | settings->ls_div_bits << 4; in si544_set_muldiv()
178 err = regmap_bulk_write(data->regmap, SI544_REG_HS_DIV, reg, 2); in si544_set_muldiv()
182 reg[0] = settings->fb_div_frac; in si544_set_muldiv()
183 reg[1] = settings->fb_div_frac >> 8; in si544_set_muldiv()
184 reg[2] = settings->fb_div_frac >> 16; in si544_set_muldiv()
185 reg[3] = settings->fb_div_frac >> 24; in si544_set_muldiv()
186 reg[4] = settings->fb_div_int; in si544_set_muldiv()
187 reg[5] = settings->fb_div_int >> 8; in si544_set_muldiv()
193 return regmap_bulk_write(data->regmap, SI544_REG_FBDIV0, reg, 6); in si544_set_muldiv()
204 switch (data->speed_grade) { in is_valid_frequency()
223 u64 vco; in si544_calc_muldiv() local
230 settings->ls_div_bits = 0; in si544_calc_muldiv()
233 settings->ls_div_bits = 0; in si544_calc_muldiv()
243 settings->ls_div_bits = res; in si544_calc_muldiv()
248 vco = FVCO_MIN + ls_freq - 1; in si544_calc_muldiv()
249 do_div(vco, ls_freq); in si544_calc_muldiv()
250 settings->hs_div = vco; in si544_calc_muldiv()
253 if ((settings->hs_div & 1) && in si544_calc_muldiv()
254 (settings->hs_div > HS_DIV_MAX_ODD || settings->ls_div_bits)) in si544_calc_muldiv()
255 ++settings->hs_div; in si544_calc_muldiv()
257 /* Calculate VCO frequency (in 10..12GHz range) */ in si544_calc_muldiv()
258 vco = (u64)ls_freq * settings->hs_div; in si544_calc_muldiv()
261 tmp = do_div(vco, FXO); in si544_calc_muldiv()
262 settings->fb_div_int = vco; in si544_calc_muldiv()
265 vco = (u64)tmp << 32; in si544_calc_muldiv()
266 vco += FXO / 2; /* Round to nearest multiple */ in si544_calc_muldiv()
267 do_div(vco, FXO); in si544_calc_muldiv()
268 settings->fb_div_frac = vco; in si544_calc_muldiv()
271 settings->delta_m = 0; in si544_calc_muldiv()
280 u32 d = settings->hs_div * BIT(settings->ls_div_bits); in si544_calc_center_rate()
281 u64 vco; in si544_calc_center_rate() local
283 /* Calculate VCO from the fractional part */ in si544_calc_center_rate()
284 vco = (u64)settings->fb_div_frac * FXO; in si544_calc_center_rate()
285 vco += (FXO / 2); in si544_calc_center_rate()
286 vco >>= 32; in si544_calc_center_rate()
288 /* Add the integer part of the VCO frequency */ in si544_calc_center_rate()
289 vco += (u64)settings->fb_div_int * FXO; in si544_calc_center_rate()
292 do_div(vco, d); in si544_calc_center_rate()
294 return vco; in si544_calc_center_rate()
300 s64 delta = (s64)rate * (DELTA_M_FRAC_NUM * settings->delta_m); in si544_calc_rate()
303 * The clock adjustment is much smaller than 1 Hz, round to the in si544_calc_rate()
307 if (settings->delta_m < 0) in si544_calc_rate()
308 delta -= ((s64)DELTA_M_MAX * DELTA_M_FRAC_DEN) / 2; in si544_calc_rate()
336 return -EINVAL; in si544_round_rate()
338 /* The accuracy is less than 1 Hz, so any rate is possible */ in si544_round_rate()
372 return -EINVAL; in si544_set_rate()
381 delta = rate - center; in si544_set_rate()
392 err = regmap_read(data->regmap, SI544_REG_OE_STATE, &old_oe_state); in si544_set_rate()
399 err = regmap_write(data->regmap, SI544_REG_FCAL_OVR, 0); in si544_set_rate()
412 err = regmap_write(data->regmap, SI544_REG_CONTROL, in si544_set_rate()
469 data = devm_kzalloc(&client->dev, sizeof(*data), GFP_KERNEL); in si544_probe()
471 return -ENOMEM; in si544_probe()
476 data->hw.init = &init; in si544_probe()
477 data->i2c_client = client; in si544_probe()
478 data->speed_grade = id->driver_data; in si544_probe()
480 if (of_property_read_string(client->dev.of_node, "clock-output-names", in si544_probe()
482 init.name = client->dev.of_node->name; in si544_probe()
484 data->regmap = devm_regmap_init_i2c(client, &si544_regmap_config); in si544_probe()
485 if (IS_ERR(data->regmap)) in si544_probe()
486 return PTR_ERR(data->regmap); in si544_probe()
491 err = regmap_write(data->regmap, SI544_REG_PAGE_SELECT, 0); in si544_probe()
495 err = devm_clk_hw_register(&client->dev, &data->hw); in si544_probe()
497 dev_err(&client->dev, "clock registration failed\n"); in si544_probe()
500 err = devm_of_clk_add_hw_provider(&client->dev, of_clk_hw_simple_get, in si544_probe()
501 &data->hw); in si544_probe()
503 dev_err(&client->dev, "unable to add clk provider\n"); in si544_probe()