clk-versaclock3.c 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Driver for Renesas Versaclock 3
  4. *
  5. * Copyright (C) 2023 Renesas Electronics Corp.
  6. */
  7. #include <linux/clk-provider.h>
  8. #include <linux/i2c.h>
  9. #include <linux/limits.h>
  10. #include <linux/module.h>
  11. #include <linux/regmap.h>
  12. #define NUM_CONFIG_REGISTERS 37
  13. #define VC3_GENERAL_CTR 0x0
  14. #define VC3_GENERAL_CTR_DIV1_SRC_SEL BIT(3)
  15. #define VC3_GENERAL_CTR_PLL3_REFIN_SEL BIT(2)
  16. #define VC3_PLL3_M_DIVIDER 0x3
  17. #define VC3_PLL3_M_DIV1 BIT(7)
  18. #define VC3_PLL3_M_DIV2 BIT(6)
  19. #define VC3_PLL3_M_DIV(n) ((n) & GENMASK(5, 0))
  20. #define VC3_PLL3_N_DIVIDER 0x4
  21. #define VC3_PLL3_LOOP_FILTER_N_DIV_MSB 0x5
  22. #define VC3_PLL3_CHARGE_PUMP_CTRL 0x6
  23. #define VC3_PLL3_CHARGE_PUMP_CTRL_OUTDIV3_SRC_SEL BIT(7)
  24. #define VC3_PLL1_CTRL_OUTDIV5 0x7
  25. #define VC3_PLL1_CTRL_OUTDIV5_PLL1_MDIV_DOUBLER BIT(7)
  26. #define VC3_PLL1_M_DIVIDER 0x8
  27. #define VC3_PLL1_M_DIV1 BIT(7)
  28. #define VC3_PLL1_M_DIV2 BIT(6)
  29. #define VC3_PLL1_M_DIV(n) ((n) & GENMASK(5, 0))
  30. #define VC3_PLL1_VCO_N_DIVIDER 0x9
  31. #define VC3_PLL1_LOOP_FILTER_N_DIV_MSB 0xa
  32. #define VC3_OUT_DIV1_DIV2_CTRL 0xf
  33. #define VC3_PLL2_FB_INT_DIV_MSB 0x10
  34. #define VC3_PLL2_FB_INT_DIV_LSB 0x11
  35. #define VC3_PLL2_FB_FRC_DIV_MSB 0x12
  36. #define VC3_PLL2_FB_FRC_DIV_LSB 0x13
  37. #define VC3_PLL2_M_DIVIDER 0x1a
  38. #define VC3_PLL2_MDIV_DOUBLER BIT(7)
  39. #define VC3_PLL2_M_DIV1 BIT(6)
  40. #define VC3_PLL2_M_DIV2 BIT(5)
  41. #define VC3_PLL2_M_DIV(n) ((n) & GENMASK(4, 0))
  42. #define VC3_OUT_DIV3_DIV4_CTRL 0x1b
  43. #define VC3_PLL_OP_CTRL 0x1c
  44. #define VC3_PLL_OP_CTRL_PLL2_REFIN_SEL 6
  45. #define VC3_OUTPUT_CTR 0x1d
  46. #define VC3_OUTPUT_CTR_DIV4_SRC_SEL BIT(3)
  47. #define VC3_SE2_CTRL_REG0 0x1f
  48. #define VC3_SE2_CTRL_REG0_SE2_CLK_SEL BIT(6)
  49. #define VC3_SE3_DIFF1_CTRL_REG 0x21
  50. #define VC3_SE3_DIFF1_CTRL_REG_SE3_CLK_SEL BIT(6)
  51. #define VC3_DIFF1_CTRL_REG 0x22
  52. #define VC3_DIFF1_CTRL_REG_DIFF1_CLK_SEL BIT(7)
  53. #define VC3_DIFF2_CTRL_REG 0x23
  54. #define VC3_DIFF2_CTRL_REG_DIFF2_CLK_SEL BIT(7)
  55. #define VC3_SE1_DIV4_CTRL 0x24
  56. #define VC3_SE1_DIV4_CTRL_SE1_CLK_SEL BIT(3)
  57. #define VC3_PLL1_VCO_MIN 300000000UL
  58. #define VC3_PLL1_VCO_MAX 600000000UL
  59. #define VC3_PLL2_VCO_MIN 400000000UL
  60. #define VC3_PLL2_VCO_MAX 1200000000UL
  61. #define VC3_PLL3_VCO_MIN 300000000UL
  62. #define VC3_PLL3_VCO_MAX 800000000UL
  63. #define VC3_2_POW_16 (U16_MAX + 1)
  64. #define VC3_DIV_MASK(width) ((1 << (width)) - 1)
  65. enum vc3_pfd_mux {
  66. VC3_PFD2_MUX,
  67. VC3_PFD3_MUX,
  68. };
  69. enum vc3_pfd {
  70. VC3_PFD1,
  71. VC3_PFD2,
  72. VC3_PFD3,
  73. };
  74. enum vc3_pll {
  75. VC3_PLL1,
  76. VC3_PLL2,
  77. VC3_PLL3,
  78. };
  79. enum vc3_div_mux {
  80. VC3_DIV1_MUX,
  81. VC3_DIV3_MUX,
  82. VC3_DIV4_MUX,
  83. };
  84. enum vc3_div {
  85. VC3_DIV1,
  86. VC3_DIV2,
  87. VC3_DIV3,
  88. VC3_DIV4,
  89. VC3_DIV5,
  90. };
  91. enum vc3_clk {
  92. VC3_REF,
  93. VC3_SE1,
  94. VC3_SE2,
  95. VC3_SE3,
  96. VC3_DIFF1,
  97. VC3_DIFF2,
  98. };
  99. enum vc3_clk_mux {
  100. VC3_SE1_MUX = VC3_SE1 - 1,
  101. VC3_SE2_MUX = VC3_SE2 - 1,
  102. VC3_SE3_MUX = VC3_SE3 - 1,
  103. VC3_DIFF1_MUX = VC3_DIFF1 - 1,
  104. VC3_DIFF2_MUX = VC3_DIFF2 - 1,
  105. };
  106. struct vc3_clk_data {
  107. u8 offs;
  108. u8 bitmsk;
  109. };
  110. struct vc3_pfd_data {
  111. u8 num;
  112. u8 offs;
  113. u8 mdiv1_bitmsk;
  114. u8 mdiv2_bitmsk;
  115. };
  116. struct vc3_pll_data {
  117. unsigned long vco_min;
  118. unsigned long vco_max;
  119. u8 num;
  120. u8 int_div_msb_offs;
  121. u8 int_div_lsb_offs;
  122. };
  123. struct vc3_div_data {
  124. const struct clk_div_table *table;
  125. u8 offs;
  126. u8 shift;
  127. u8 width;
  128. u8 flags;
  129. };
  130. struct vc3_hw_data {
  131. struct clk_hw hw;
  132. struct regmap *regmap;
  133. const void *data;
  134. u32 div_int;
  135. u32 div_frc;
  136. };
  137. static const struct clk_div_table div1_divs[] = {
  138. { .val = 0, .div = 1, }, { .val = 1, .div = 4, },
  139. { .val = 2, .div = 5, }, { .val = 3, .div = 6, },
  140. { .val = 4, .div = 2, }, { .val = 5, .div = 8, },
  141. { .val = 6, .div = 10, }, { .val = 7, .div = 12, },
  142. { .val = 8, .div = 4, }, { .val = 9, .div = 16, },
  143. { .val = 10, .div = 20, }, { .val = 11, .div = 24, },
  144. { .val = 12, .div = 8, }, { .val = 13, .div = 32, },
  145. { .val = 14, .div = 40, }, { .val = 15, .div = 48, },
  146. {}
  147. };
  148. static const struct clk_div_table div245_divs[] = {
  149. { .val = 0, .div = 1, }, { .val = 1, .div = 3, },
  150. { .val = 2, .div = 5, }, { .val = 3, .div = 10, },
  151. { .val = 4, .div = 2, }, { .val = 5, .div = 6, },
  152. { .val = 6, .div = 10, }, { .val = 7, .div = 20, },
  153. { .val = 8, .div = 4, }, { .val = 9, .div = 12, },
  154. { .val = 10, .div = 20, }, { .val = 11, .div = 40, },
  155. { .val = 12, .div = 5, }, { .val = 13, .div = 15, },
  156. { .val = 14, .div = 25, }, { .val = 15, .div = 50, },
  157. {}
  158. };
  159. static const struct clk_div_table div3_divs[] = {
  160. { .val = 0, .div = 1, }, { .val = 1, .div = 3, },
  161. { .val = 2, .div = 5, }, { .val = 3, .div = 10, },
  162. { .val = 4, .div = 2, }, { .val = 5, .div = 6, },
  163. { .val = 6, .div = 10, }, { .val = 7, .div = 20, },
  164. { .val = 8, .div = 4, }, { .val = 9, .div = 12, },
  165. { .val = 10, .div = 20, }, { .val = 11, .div = 40, },
  166. { .val = 12, .div = 8, }, { .val = 13, .div = 24, },
  167. { .val = 14, .div = 40, }, { .val = 15, .div = 80, },
  168. {}
  169. };
  170. static struct clk_hw *clk_out[6];
  171. static u8 vc3_pfd_mux_get_parent(struct clk_hw *hw)
  172. {
  173. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  174. const struct vc3_clk_data *pfd_mux = vc3->data;
  175. u32 src;
  176. regmap_read(vc3->regmap, pfd_mux->offs, &src);
  177. return !!(src & pfd_mux->bitmsk);
  178. }
  179. static int vc3_pfd_mux_set_parent(struct clk_hw *hw, u8 index)
  180. {
  181. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  182. const struct vc3_clk_data *pfd_mux = vc3->data;
  183. return regmap_update_bits(vc3->regmap, pfd_mux->offs, pfd_mux->bitmsk,
  184. index ? pfd_mux->bitmsk : 0);
  185. }
  186. static const struct clk_ops vc3_pfd_mux_ops = {
  187. .determine_rate = clk_hw_determine_rate_no_reparent,
  188. .set_parent = vc3_pfd_mux_set_parent,
  189. .get_parent = vc3_pfd_mux_get_parent,
  190. };
  191. static unsigned long vc3_pfd_recalc_rate(struct clk_hw *hw,
  192. unsigned long parent_rate)
  193. {
  194. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  195. const struct vc3_pfd_data *pfd = vc3->data;
  196. unsigned int prediv, premul;
  197. unsigned long rate;
  198. u8 mdiv;
  199. regmap_read(vc3->regmap, pfd->offs, &prediv);
  200. if (pfd->num == VC3_PFD1) {
  201. /* The bypass_prediv is set, PLL fed from Ref_in directly. */
  202. if (prediv & pfd->mdiv1_bitmsk) {
  203. /* check doubler is set or not */
  204. regmap_read(vc3->regmap, VC3_PLL1_CTRL_OUTDIV5, &premul);
  205. if (premul & VC3_PLL1_CTRL_OUTDIV5_PLL1_MDIV_DOUBLER)
  206. parent_rate *= 2;
  207. return parent_rate;
  208. }
  209. mdiv = VC3_PLL1_M_DIV(prediv);
  210. } else if (pfd->num == VC3_PFD2) {
  211. /* The bypass_prediv is set, PLL fed from Ref_in directly. */
  212. if (prediv & pfd->mdiv1_bitmsk) {
  213. regmap_read(vc3->regmap, VC3_PLL2_M_DIVIDER, &premul);
  214. /* check doubler is set or not */
  215. if (premul & VC3_PLL2_MDIV_DOUBLER)
  216. parent_rate *= 2;
  217. return parent_rate;
  218. }
  219. mdiv = VC3_PLL2_M_DIV(prediv);
  220. } else {
  221. /* The bypass_prediv is set, PLL fed from Ref_in directly. */
  222. if (prediv & pfd->mdiv1_bitmsk)
  223. return parent_rate;
  224. mdiv = VC3_PLL3_M_DIV(prediv);
  225. }
  226. if (prediv & pfd->mdiv2_bitmsk)
  227. rate = parent_rate / 2;
  228. else
  229. rate = parent_rate / mdiv;
  230. return rate;
  231. }
  232. static long vc3_pfd_round_rate(struct clk_hw *hw, unsigned long rate,
  233. unsigned long *parent_rate)
  234. {
  235. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  236. const struct vc3_pfd_data *pfd = vc3->data;
  237. unsigned long idiv;
  238. /* PLL cannot operate with input clock above 50 MHz. */
  239. if (rate > 50000000)
  240. return -EINVAL;
  241. /* CLKIN within range of PLL input, feed directly to PLL. */
  242. if (*parent_rate <= 50000000)
  243. return *parent_rate;
  244. idiv = DIV_ROUND_UP(*parent_rate, rate);
  245. if (pfd->num == VC3_PFD1 || pfd->num == VC3_PFD3) {
  246. if (idiv > 63)
  247. return -EINVAL;
  248. } else {
  249. if (idiv > 31)
  250. return -EINVAL;
  251. }
  252. return *parent_rate / idiv;
  253. }
  254. static int vc3_pfd_set_rate(struct clk_hw *hw, unsigned long rate,
  255. unsigned long parent_rate)
  256. {
  257. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  258. const struct vc3_pfd_data *pfd = vc3->data;
  259. unsigned long idiv;
  260. u8 div;
  261. /* CLKIN within range of PLL input, feed directly to PLL. */
  262. if (parent_rate <= 50000000) {
  263. regmap_update_bits(vc3->regmap, pfd->offs, pfd->mdiv1_bitmsk,
  264. pfd->mdiv1_bitmsk);
  265. regmap_update_bits(vc3->regmap, pfd->offs, pfd->mdiv2_bitmsk, 0);
  266. return 0;
  267. }
  268. idiv = DIV_ROUND_UP(parent_rate, rate);
  269. /* We have dedicated div-2 predivider. */
  270. if (idiv == 2) {
  271. regmap_update_bits(vc3->regmap, pfd->offs, pfd->mdiv2_bitmsk,
  272. pfd->mdiv2_bitmsk);
  273. regmap_update_bits(vc3->regmap, pfd->offs, pfd->mdiv1_bitmsk, 0);
  274. } else {
  275. if (pfd->num == VC3_PFD1)
  276. div = VC3_PLL1_M_DIV(idiv);
  277. else if (pfd->num == VC3_PFD2)
  278. div = VC3_PLL2_M_DIV(idiv);
  279. else
  280. div = VC3_PLL3_M_DIV(idiv);
  281. regmap_write(vc3->regmap, pfd->offs, div);
  282. }
  283. return 0;
  284. }
  285. static const struct clk_ops vc3_pfd_ops = {
  286. .recalc_rate = vc3_pfd_recalc_rate,
  287. .round_rate = vc3_pfd_round_rate,
  288. .set_rate = vc3_pfd_set_rate,
  289. };
  290. static unsigned long vc3_pll_recalc_rate(struct clk_hw *hw,
  291. unsigned long parent_rate)
  292. {
  293. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  294. const struct vc3_pll_data *pll = vc3->data;
  295. u32 div_int, div_frc, val;
  296. unsigned long rate;
  297. regmap_read(vc3->regmap, pll->int_div_msb_offs, &val);
  298. div_int = (val & GENMASK(2, 0)) << 8;
  299. regmap_read(vc3->regmap, pll->int_div_lsb_offs, &val);
  300. div_int |= val;
  301. if (pll->num == VC3_PLL2) {
  302. regmap_read(vc3->regmap, VC3_PLL2_FB_FRC_DIV_MSB, &val);
  303. div_frc = val << 8;
  304. regmap_read(vc3->regmap, VC3_PLL2_FB_FRC_DIV_LSB, &val);
  305. div_frc |= val;
  306. rate = (parent_rate *
  307. (div_int * VC3_2_POW_16 + div_frc) / VC3_2_POW_16);
  308. } else {
  309. rate = parent_rate * div_int;
  310. }
  311. return rate;
  312. }
  313. static long vc3_pll_round_rate(struct clk_hw *hw, unsigned long rate,
  314. unsigned long *parent_rate)
  315. {
  316. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  317. const struct vc3_pll_data *pll = vc3->data;
  318. u64 div_frc;
  319. if (rate < pll->vco_min)
  320. rate = pll->vco_min;
  321. if (rate > pll->vco_max)
  322. rate = pll->vco_max;
  323. vc3->div_int = rate / *parent_rate;
  324. if (pll->num == VC3_PLL2) {
  325. if (vc3->div_int > 0x7ff)
  326. rate = *parent_rate * 0x7ff;
  327. /* Determine best fractional part, which is 16 bit wide */
  328. div_frc = rate % *parent_rate;
  329. div_frc *= BIT(16) - 1;
  330. vc3->div_frc = min_t(u64, div64_ul(div_frc, *parent_rate), U16_MAX);
  331. rate = (*parent_rate *
  332. (vc3->div_int * VC3_2_POW_16 + vc3->div_frc) / VC3_2_POW_16);
  333. } else {
  334. rate = *parent_rate * vc3->div_int;
  335. }
  336. return rate;
  337. }
  338. static int vc3_pll_set_rate(struct clk_hw *hw, unsigned long rate,
  339. unsigned long parent_rate)
  340. {
  341. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  342. const struct vc3_pll_data *pll = vc3->data;
  343. u32 val;
  344. regmap_read(vc3->regmap, pll->int_div_msb_offs, &val);
  345. val = (val & 0xf8) | ((vc3->div_int >> 8) & 0x7);
  346. regmap_write(vc3->regmap, pll->int_div_msb_offs, val);
  347. regmap_write(vc3->regmap, pll->int_div_lsb_offs, vc3->div_int & 0xff);
  348. if (pll->num == VC3_PLL2) {
  349. regmap_write(vc3->regmap, VC3_PLL2_FB_FRC_DIV_MSB,
  350. vc3->div_frc >> 8);
  351. regmap_write(vc3->regmap, VC3_PLL2_FB_FRC_DIV_LSB,
  352. vc3->div_frc & 0xff);
  353. }
  354. return 0;
  355. }
  356. static const struct clk_ops vc3_pll_ops = {
  357. .recalc_rate = vc3_pll_recalc_rate,
  358. .round_rate = vc3_pll_round_rate,
  359. .set_rate = vc3_pll_set_rate,
  360. };
  361. static u8 vc3_div_mux_get_parent(struct clk_hw *hw)
  362. {
  363. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  364. const struct vc3_clk_data *div_mux = vc3->data;
  365. u32 src;
  366. regmap_read(vc3->regmap, div_mux->offs, &src);
  367. return !!(src & div_mux->bitmsk);
  368. }
  369. static int vc3_div_mux_set_parent(struct clk_hw *hw, u8 index)
  370. {
  371. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  372. const struct vc3_clk_data *div_mux = vc3->data;
  373. return regmap_update_bits(vc3->regmap, div_mux->offs, div_mux->bitmsk,
  374. index ? div_mux->bitmsk : 0);
  375. }
  376. static const struct clk_ops vc3_div_mux_ops = {
  377. .determine_rate = clk_hw_determine_rate_no_reparent,
  378. .set_parent = vc3_div_mux_set_parent,
  379. .get_parent = vc3_div_mux_get_parent,
  380. };
  381. static unsigned int vc3_get_div(const struct clk_div_table *table,
  382. unsigned int val, unsigned long flag)
  383. {
  384. const struct clk_div_table *clkt;
  385. for (clkt = table; clkt->div; clkt++)
  386. if (clkt->val == val)
  387. return clkt->div;
  388. return 1;
  389. }
  390. static unsigned long vc3_div_recalc_rate(struct clk_hw *hw,
  391. unsigned long parent_rate)
  392. {
  393. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  394. const struct vc3_div_data *div_data = vc3->data;
  395. unsigned int val;
  396. regmap_read(vc3->regmap, div_data->offs, &val);
  397. val >>= div_data->shift;
  398. val &= VC3_DIV_MASK(div_data->width);
  399. return divider_recalc_rate(hw, parent_rate, val, div_data->table,
  400. div_data->flags, div_data->width);
  401. }
  402. static long vc3_div_round_rate(struct clk_hw *hw, unsigned long rate,
  403. unsigned long *parent_rate)
  404. {
  405. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  406. const struct vc3_div_data *div_data = vc3->data;
  407. unsigned int bestdiv;
  408. /* if read only, just return current value */
  409. if (div_data->flags & CLK_DIVIDER_READ_ONLY) {
  410. regmap_read(vc3->regmap, div_data->offs, &bestdiv);
  411. bestdiv >>= div_data->shift;
  412. bestdiv &= VC3_DIV_MASK(div_data->width);
  413. bestdiv = vc3_get_div(div_data->table, bestdiv, div_data->flags);
  414. return DIV_ROUND_UP(*parent_rate, bestdiv);
  415. }
  416. return divider_round_rate(hw, rate, parent_rate, div_data->table,
  417. div_data->width, div_data->flags);
  418. }
  419. static int vc3_div_set_rate(struct clk_hw *hw, unsigned long rate,
  420. unsigned long parent_rate)
  421. {
  422. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  423. const struct vc3_div_data *div_data = vc3->data;
  424. unsigned int value;
  425. value = divider_get_val(rate, parent_rate, div_data->table,
  426. div_data->width, div_data->flags);
  427. return regmap_update_bits(vc3->regmap, div_data->offs,
  428. VC3_DIV_MASK(div_data->width) << div_data->shift,
  429. value << div_data->shift);
  430. }
  431. static const struct clk_ops vc3_div_ops = {
  432. .recalc_rate = vc3_div_recalc_rate,
  433. .round_rate = vc3_div_round_rate,
  434. .set_rate = vc3_div_set_rate,
  435. };
  436. static int vc3_clk_mux_determine_rate(struct clk_hw *hw,
  437. struct clk_rate_request *req)
  438. {
  439. int frc;
  440. if (clk_mux_determine_rate_flags(hw, req, CLK_SET_RATE_PARENT)) {
  441. /* The below check is equivalent to (best_parent_rate/rate) */
  442. if (req->best_parent_rate >= req->rate) {
  443. frc = DIV_ROUND_CLOSEST_ULL(req->best_parent_rate,
  444. req->rate);
  445. req->rate *= frc;
  446. return clk_mux_determine_rate_flags(hw, req,
  447. CLK_SET_RATE_PARENT);
  448. }
  449. }
  450. return 0;
  451. }
  452. static u8 vc3_clk_mux_get_parent(struct clk_hw *hw)
  453. {
  454. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  455. const struct vc3_clk_data *clk_mux = vc3->data;
  456. u32 val;
  457. regmap_read(vc3->regmap, clk_mux->offs, &val);
  458. return !!(val & clk_mux->bitmsk);
  459. }
  460. static int vc3_clk_mux_set_parent(struct clk_hw *hw, u8 index)
  461. {
  462. struct vc3_hw_data *vc3 = container_of(hw, struct vc3_hw_data, hw);
  463. const struct vc3_clk_data *clk_mux = vc3->data;
  464. return regmap_update_bits(vc3->regmap, clk_mux->offs, clk_mux->bitmsk,
  465. index ? clk_mux->bitmsk : 0);
  466. }
  467. static const struct clk_ops vc3_clk_mux_ops = {
  468. .determine_rate = vc3_clk_mux_determine_rate,
  469. .set_parent = vc3_clk_mux_set_parent,
  470. .get_parent = vc3_clk_mux_get_parent,
  471. };
  472. static const struct regmap_config vc3_regmap_config = {
  473. .reg_bits = 8,
  474. .val_bits = 8,
  475. .cache_type = REGCACHE_MAPLE,
  476. .max_register = 0x24,
  477. };
  478. static struct vc3_hw_data clk_div[5];
  479. static const struct clk_parent_data pfd_mux_parent_data[] = {
  480. { .index = 0, },
  481. { .hw = &clk_div[VC3_DIV2].hw }
  482. };
  483. static struct vc3_hw_data clk_pfd_mux[] = {
  484. [VC3_PFD2_MUX] = {
  485. .data = &(struct vc3_clk_data) {
  486. .offs = VC3_PLL_OP_CTRL,
  487. .bitmsk = BIT(VC3_PLL_OP_CTRL_PLL2_REFIN_SEL)
  488. },
  489. .hw.init = &(struct clk_init_data) {
  490. .name = "pfd2_mux",
  491. .ops = &vc3_pfd_mux_ops,
  492. .parent_data = pfd_mux_parent_data,
  493. .num_parents = 2,
  494. .flags = CLK_SET_RATE_PARENT | CLK_SET_RATE_NO_REPARENT
  495. }
  496. },
  497. [VC3_PFD3_MUX] = {
  498. .data = &(struct vc3_clk_data) {
  499. .offs = VC3_GENERAL_CTR,
  500. .bitmsk = BIT(VC3_GENERAL_CTR_PLL3_REFIN_SEL)
  501. },
  502. .hw.init = &(struct clk_init_data) {
  503. .name = "pfd3_mux",
  504. .ops = &vc3_pfd_mux_ops,
  505. .parent_data = pfd_mux_parent_data,
  506. .num_parents = 2,
  507. .flags = CLK_SET_RATE_PARENT | CLK_SET_RATE_NO_REPARENT
  508. }
  509. }
  510. };
  511. static struct vc3_hw_data clk_pfd[] = {
  512. [VC3_PFD1] = {
  513. .data = &(struct vc3_pfd_data) {
  514. .num = VC3_PFD1,
  515. .offs = VC3_PLL1_M_DIVIDER,
  516. .mdiv1_bitmsk = VC3_PLL1_M_DIV1,
  517. .mdiv2_bitmsk = VC3_PLL1_M_DIV2
  518. },
  519. .hw.init = &(struct clk_init_data) {
  520. .name = "pfd1",
  521. .ops = &vc3_pfd_ops,
  522. .parent_data = &(const struct clk_parent_data) {
  523. .index = 0
  524. },
  525. .num_parents = 1,
  526. .flags = CLK_SET_RATE_PARENT
  527. }
  528. },
  529. [VC3_PFD2] = {
  530. .data = &(struct vc3_pfd_data) {
  531. .num = VC3_PFD2,
  532. .offs = VC3_PLL2_M_DIVIDER,
  533. .mdiv1_bitmsk = VC3_PLL2_M_DIV1,
  534. .mdiv2_bitmsk = VC3_PLL2_M_DIV2
  535. },
  536. .hw.init = &(struct clk_init_data) {
  537. .name = "pfd2",
  538. .ops = &vc3_pfd_ops,
  539. .parent_hws = (const struct clk_hw *[]) {
  540. &clk_pfd_mux[VC3_PFD2_MUX].hw
  541. },
  542. .num_parents = 1,
  543. .flags = CLK_SET_RATE_PARENT
  544. }
  545. },
  546. [VC3_PFD3] = {
  547. .data = &(struct vc3_pfd_data) {
  548. .num = VC3_PFD3,
  549. .offs = VC3_PLL3_M_DIVIDER,
  550. .mdiv1_bitmsk = VC3_PLL3_M_DIV1,
  551. .mdiv2_bitmsk = VC3_PLL3_M_DIV2
  552. },
  553. .hw.init = &(struct clk_init_data) {
  554. .name = "pfd3",
  555. .ops = &vc3_pfd_ops,
  556. .parent_hws = (const struct clk_hw *[]) {
  557. &clk_pfd_mux[VC3_PFD3_MUX].hw
  558. },
  559. .num_parents = 1,
  560. .flags = CLK_SET_RATE_PARENT
  561. }
  562. }
  563. };
  564. static struct vc3_hw_data clk_pll[] = {
  565. [VC3_PLL1] = {
  566. .data = &(struct vc3_pll_data) {
  567. .num = VC3_PLL1,
  568. .int_div_msb_offs = VC3_PLL1_LOOP_FILTER_N_DIV_MSB,
  569. .int_div_lsb_offs = VC3_PLL1_VCO_N_DIVIDER,
  570. .vco_min = VC3_PLL1_VCO_MIN,
  571. .vco_max = VC3_PLL1_VCO_MAX
  572. },
  573. .hw.init = &(struct clk_init_data) {
  574. .name = "pll1",
  575. .ops = &vc3_pll_ops,
  576. .parent_hws = (const struct clk_hw *[]) {
  577. &clk_pfd[VC3_PFD1].hw
  578. },
  579. .num_parents = 1,
  580. .flags = CLK_SET_RATE_PARENT
  581. }
  582. },
  583. [VC3_PLL2] = {
  584. .data = &(struct vc3_pll_data) {
  585. .num = VC3_PLL2,
  586. .int_div_msb_offs = VC3_PLL2_FB_INT_DIV_MSB,
  587. .int_div_lsb_offs = VC3_PLL2_FB_INT_DIV_LSB,
  588. .vco_min = VC3_PLL2_VCO_MIN,
  589. .vco_max = VC3_PLL2_VCO_MAX
  590. },
  591. .hw.init = &(struct clk_init_data) {
  592. .name = "pll2",
  593. .ops = &vc3_pll_ops,
  594. .parent_hws = (const struct clk_hw *[]) {
  595. &clk_pfd[VC3_PFD2].hw
  596. },
  597. .num_parents = 1,
  598. .flags = CLK_SET_RATE_PARENT
  599. }
  600. },
  601. [VC3_PLL3] = {
  602. .data = &(struct vc3_pll_data) {
  603. .num = VC3_PLL3,
  604. .int_div_msb_offs = VC3_PLL3_LOOP_FILTER_N_DIV_MSB,
  605. .int_div_lsb_offs = VC3_PLL3_N_DIVIDER,
  606. .vco_min = VC3_PLL3_VCO_MIN,
  607. .vco_max = VC3_PLL3_VCO_MAX
  608. },
  609. .hw.init = &(struct clk_init_data) {
  610. .name = "pll3",
  611. .ops = &vc3_pll_ops,
  612. .parent_hws = (const struct clk_hw *[]) {
  613. &clk_pfd[VC3_PFD3].hw
  614. },
  615. .num_parents = 1,
  616. .flags = CLK_SET_RATE_PARENT
  617. }
  618. }
  619. };
  620. static const struct clk_parent_data div_mux_parent_data[][2] = {
  621. [VC3_DIV1_MUX] = {
  622. { .hw = &clk_pll[VC3_PLL1].hw },
  623. { .index = 0 }
  624. },
  625. [VC3_DIV3_MUX] = {
  626. { .hw = &clk_pll[VC3_PLL2].hw },
  627. { .hw = &clk_pll[VC3_PLL3].hw }
  628. },
  629. [VC3_DIV4_MUX] = {
  630. { .hw = &clk_pll[VC3_PLL2].hw },
  631. { .index = 0 }
  632. }
  633. };
  634. static struct vc3_hw_data clk_div_mux[] = {
  635. [VC3_DIV1_MUX] = {
  636. .data = &(struct vc3_clk_data) {
  637. .offs = VC3_GENERAL_CTR,
  638. .bitmsk = VC3_GENERAL_CTR_DIV1_SRC_SEL
  639. },
  640. .hw.init = &(struct clk_init_data) {
  641. .name = "div1_mux",
  642. .ops = &vc3_div_mux_ops,
  643. .parent_data = div_mux_parent_data[VC3_DIV1_MUX],
  644. .num_parents = 2,
  645. .flags = CLK_SET_RATE_PARENT | CLK_SET_RATE_NO_REPARENT
  646. }
  647. },
  648. [VC3_DIV3_MUX] = {
  649. .data = &(struct vc3_clk_data) {
  650. .offs = VC3_PLL3_CHARGE_PUMP_CTRL,
  651. .bitmsk = VC3_PLL3_CHARGE_PUMP_CTRL_OUTDIV3_SRC_SEL
  652. },
  653. .hw.init = &(struct clk_init_data) {
  654. .name = "div3_mux",
  655. .ops = &vc3_div_mux_ops,
  656. .parent_data = div_mux_parent_data[VC3_DIV3_MUX],
  657. .num_parents = 2,
  658. .flags = CLK_SET_RATE_PARENT | CLK_SET_RATE_NO_REPARENT
  659. }
  660. },
  661. [VC3_DIV4_MUX] = {
  662. .data = &(struct vc3_clk_data) {
  663. .offs = VC3_OUTPUT_CTR,
  664. .bitmsk = VC3_OUTPUT_CTR_DIV4_SRC_SEL
  665. },
  666. .hw.init = &(struct clk_init_data) {
  667. .name = "div4_mux",
  668. .ops = &vc3_div_mux_ops,
  669. .parent_data = div_mux_parent_data[VC3_DIV4_MUX],
  670. .num_parents = 2,
  671. .flags = CLK_SET_RATE_PARENT | CLK_SET_RATE_NO_REPARENT
  672. }
  673. }
  674. };
  675. static struct vc3_hw_data clk_div[] = {
  676. [VC3_DIV1] = {
  677. .data = &(struct vc3_div_data) {
  678. .offs = VC3_OUT_DIV1_DIV2_CTRL,
  679. .table = div1_divs,
  680. .shift = 4,
  681. .width = 4,
  682. .flags = CLK_DIVIDER_READ_ONLY
  683. },
  684. .hw.init = &(struct clk_init_data) {
  685. .name = "div1",
  686. .ops = &vc3_div_ops,
  687. .parent_hws = (const struct clk_hw *[]) {
  688. &clk_div_mux[VC3_DIV1_MUX].hw
  689. },
  690. .num_parents = 1,
  691. .flags = CLK_SET_RATE_PARENT
  692. }
  693. },
  694. [VC3_DIV2] = {
  695. .data = &(struct vc3_div_data) {
  696. .offs = VC3_OUT_DIV1_DIV2_CTRL,
  697. .table = div245_divs,
  698. .shift = 0,
  699. .width = 4,
  700. .flags = CLK_DIVIDER_READ_ONLY
  701. },
  702. .hw.init = &(struct clk_init_data) {
  703. .name = "div2",
  704. .ops = &vc3_div_ops,
  705. .parent_hws = (const struct clk_hw *[]) {
  706. &clk_pll[VC3_PLL1].hw
  707. },
  708. .num_parents = 1,
  709. .flags = CLK_SET_RATE_PARENT
  710. }
  711. },
  712. [VC3_DIV3] = {
  713. .data = &(struct vc3_div_data) {
  714. .offs = VC3_OUT_DIV3_DIV4_CTRL,
  715. .table = div3_divs,
  716. .shift = 4,
  717. .width = 4,
  718. .flags = CLK_DIVIDER_READ_ONLY
  719. },
  720. .hw.init = &(struct clk_init_data) {
  721. .name = "div3",
  722. .ops = &vc3_div_ops,
  723. .parent_hws = (const struct clk_hw *[]) {
  724. &clk_div_mux[VC3_DIV3_MUX].hw
  725. },
  726. .num_parents = 1,
  727. .flags = CLK_SET_RATE_PARENT
  728. }
  729. },
  730. [VC3_DIV4] = {
  731. .data = &(struct vc3_div_data) {
  732. .offs = VC3_OUT_DIV3_DIV4_CTRL,
  733. .table = div245_divs,
  734. .shift = 0,
  735. .width = 4,
  736. .flags = CLK_DIVIDER_READ_ONLY
  737. },
  738. .hw.init = &(struct clk_init_data) {
  739. .name = "div4",
  740. .ops = &vc3_div_ops,
  741. .parent_hws = (const struct clk_hw *[]) {
  742. &clk_div_mux[VC3_DIV4_MUX].hw
  743. },
  744. .num_parents = 1,
  745. .flags = CLK_SET_RATE_PARENT
  746. }
  747. },
  748. [VC3_DIV5] = {
  749. .data = &(struct vc3_div_data) {
  750. .offs = VC3_PLL1_CTRL_OUTDIV5,
  751. .table = div245_divs,
  752. .shift = 0,
  753. .width = 4,
  754. .flags = CLK_DIVIDER_READ_ONLY
  755. },
  756. .hw.init = &(struct clk_init_data) {
  757. .name = "div5",
  758. .ops = &vc3_div_ops,
  759. .parent_hws = (const struct clk_hw *[]) {
  760. &clk_pll[VC3_PLL3].hw
  761. },
  762. .num_parents = 1,
  763. .flags = CLK_SET_RATE_PARENT
  764. }
  765. }
  766. };
  767. static struct vc3_hw_data clk_mux[] = {
  768. [VC3_SE1_MUX] = {
  769. .data = &(struct vc3_clk_data) {
  770. .offs = VC3_SE1_DIV4_CTRL,
  771. .bitmsk = VC3_SE1_DIV4_CTRL_SE1_CLK_SEL
  772. },
  773. .hw.init = &(struct clk_init_data) {
  774. .name = "se1_mux",
  775. .ops = &vc3_clk_mux_ops,
  776. .parent_hws = (const struct clk_hw *[]) {
  777. &clk_div[VC3_DIV5].hw,
  778. &clk_div[VC3_DIV4].hw
  779. },
  780. .num_parents = 2,
  781. .flags = CLK_SET_RATE_PARENT
  782. }
  783. },
  784. [VC3_SE2_MUX] = {
  785. .data = &(struct vc3_clk_data) {
  786. .offs = VC3_SE2_CTRL_REG0,
  787. .bitmsk = VC3_SE2_CTRL_REG0_SE2_CLK_SEL
  788. },
  789. .hw.init = &(struct clk_init_data) {
  790. .name = "se2_mux",
  791. .ops = &vc3_clk_mux_ops,
  792. .parent_hws = (const struct clk_hw *[]) {
  793. &clk_div[VC3_DIV5].hw,
  794. &clk_div[VC3_DIV4].hw
  795. },
  796. .num_parents = 2,
  797. .flags = CLK_SET_RATE_PARENT
  798. }
  799. },
  800. [VC3_SE3_MUX] = {
  801. .data = &(struct vc3_clk_data) {
  802. .offs = VC3_SE3_DIFF1_CTRL_REG,
  803. .bitmsk = VC3_SE3_DIFF1_CTRL_REG_SE3_CLK_SEL
  804. },
  805. .hw.init = &(struct clk_init_data) {
  806. .name = "se3_mux",
  807. .ops = &vc3_clk_mux_ops,
  808. .parent_hws = (const struct clk_hw *[]) {
  809. &clk_div[VC3_DIV2].hw,
  810. &clk_div[VC3_DIV4].hw
  811. },
  812. .num_parents = 2,
  813. .flags = CLK_SET_RATE_PARENT
  814. }
  815. },
  816. [VC3_DIFF1_MUX] = {
  817. .data = &(struct vc3_clk_data) {
  818. .offs = VC3_DIFF1_CTRL_REG,
  819. .bitmsk = VC3_DIFF1_CTRL_REG_DIFF1_CLK_SEL
  820. },
  821. .hw.init = &(struct clk_init_data) {
  822. .name = "diff1_mux",
  823. .ops = &vc3_clk_mux_ops,
  824. .parent_hws = (const struct clk_hw *[]) {
  825. &clk_div[VC3_DIV1].hw,
  826. &clk_div[VC3_DIV3].hw
  827. },
  828. .num_parents = 2,
  829. .flags = CLK_SET_RATE_PARENT
  830. }
  831. },
  832. [VC3_DIFF2_MUX] = {
  833. .data = &(struct vc3_clk_data) {
  834. .offs = VC3_DIFF2_CTRL_REG,
  835. .bitmsk = VC3_DIFF2_CTRL_REG_DIFF2_CLK_SEL
  836. },
  837. .hw.init = &(struct clk_init_data) {
  838. .name = "diff2_mux",
  839. .ops = &vc3_clk_mux_ops,
  840. .parent_hws = (const struct clk_hw *[]) {
  841. &clk_div[VC3_DIV1].hw,
  842. &clk_div[VC3_DIV3].hw
  843. },
  844. .num_parents = 2,
  845. .flags = CLK_SET_RATE_PARENT
  846. }
  847. }
  848. };
  849. static struct clk_hw *vc3_of_clk_get(struct of_phandle_args *clkspec,
  850. void *data)
  851. {
  852. unsigned int idx = clkspec->args[0];
  853. struct clk_hw **clkout_hw = data;
  854. if (idx >= ARRAY_SIZE(clk_out)) {
  855. pr_err("invalid clk index %u for provider %pOF\n", idx, clkspec->np);
  856. return ERR_PTR(-EINVAL);
  857. }
  858. return clkout_hw[idx];
  859. }
  860. static int vc3_probe(struct i2c_client *client)
  861. {
  862. struct device *dev = &client->dev;
  863. u8 settings[NUM_CONFIG_REGISTERS];
  864. struct regmap *regmap;
  865. const char *name;
  866. int ret, i;
  867. regmap = devm_regmap_init_i2c(client, &vc3_regmap_config);
  868. if (IS_ERR(regmap))
  869. return dev_err_probe(dev, PTR_ERR(regmap),
  870. "failed to allocate register map\n");
  871. ret = of_property_read_u8_array(dev->of_node, "renesas,settings",
  872. settings, ARRAY_SIZE(settings));
  873. if (!ret) {
  874. /*
  875. * A raw settings array was specified in the DT. Write the
  876. * settings to the device immediately.
  877. */
  878. for (i = 0; i < NUM_CONFIG_REGISTERS; i++) {
  879. ret = regmap_write(regmap, i, settings[i]);
  880. if (ret) {
  881. dev_err(dev, "error writing to chip (%i)\n", ret);
  882. return ret;
  883. }
  884. }
  885. } else if (ret == -EOVERFLOW) {
  886. dev_err(&client->dev, "EOVERFLOW reg settings. ARRAY_SIZE: %zu\n",
  887. ARRAY_SIZE(settings));
  888. return ret;
  889. }
  890. /* Register pfd muxes */
  891. for (i = 0; i < ARRAY_SIZE(clk_pfd_mux); i++) {
  892. clk_pfd_mux[i].regmap = regmap;
  893. ret = devm_clk_hw_register(dev, &clk_pfd_mux[i].hw);
  894. if (ret)
  895. return dev_err_probe(dev, ret, "%s failed\n",
  896. clk_pfd_mux[i].hw.init->name);
  897. }
  898. /* Register pfd's */
  899. for (i = 0; i < ARRAY_SIZE(clk_pfd); i++) {
  900. clk_pfd[i].regmap = regmap;
  901. ret = devm_clk_hw_register(dev, &clk_pfd[i].hw);
  902. if (ret)
  903. return dev_err_probe(dev, ret, "%s failed\n",
  904. clk_pfd[i].hw.init->name);
  905. }
  906. /* Register pll's */
  907. for (i = 0; i < ARRAY_SIZE(clk_pll); i++) {
  908. clk_pll[i].regmap = regmap;
  909. ret = devm_clk_hw_register(dev, &clk_pll[i].hw);
  910. if (ret)
  911. return dev_err_probe(dev, ret, "%s failed\n",
  912. clk_pll[i].hw.init->name);
  913. }
  914. /* Register divider muxes */
  915. for (i = 0; i < ARRAY_SIZE(clk_div_mux); i++) {
  916. clk_div_mux[i].regmap = regmap;
  917. ret = devm_clk_hw_register(dev, &clk_div_mux[i].hw);
  918. if (ret)
  919. return dev_err_probe(dev, ret, "%s failed\n",
  920. clk_div_mux[i].hw.init->name);
  921. }
  922. /* Register dividers */
  923. for (i = 0; i < ARRAY_SIZE(clk_div); i++) {
  924. clk_div[i].regmap = regmap;
  925. ret = devm_clk_hw_register(dev, &clk_div[i].hw);
  926. if (ret)
  927. return dev_err_probe(dev, ret, "%s failed\n",
  928. clk_div[i].hw.init->name);
  929. }
  930. /* Register clk muxes */
  931. for (i = 0; i < ARRAY_SIZE(clk_mux); i++) {
  932. clk_mux[i].regmap = regmap;
  933. ret = devm_clk_hw_register(dev, &clk_mux[i].hw);
  934. if (ret)
  935. return dev_err_probe(dev, ret, "%s failed\n",
  936. clk_mux[i].hw.init->name);
  937. }
  938. /* Register clk outputs */
  939. for (i = 0; i < ARRAY_SIZE(clk_out); i++) {
  940. switch (i) {
  941. case VC3_DIFF2:
  942. name = "diff2";
  943. break;
  944. case VC3_DIFF1:
  945. name = "diff1";
  946. break;
  947. case VC3_SE3:
  948. name = "se3";
  949. break;
  950. case VC3_SE2:
  951. name = "se2";
  952. break;
  953. case VC3_SE1:
  954. name = "se1";
  955. break;
  956. case VC3_REF:
  957. name = "ref";
  958. break;
  959. default:
  960. return dev_err_probe(dev, -EINVAL, "invalid clk output %d\n", i);
  961. }
  962. if (i == VC3_REF)
  963. clk_out[i] = devm_clk_hw_register_fixed_factor_index(dev,
  964. name, 0, CLK_SET_RATE_PARENT, 1, 1);
  965. else
  966. clk_out[i] = devm_clk_hw_register_fixed_factor_parent_hw(dev,
  967. name, &clk_mux[i - 1].hw, CLK_SET_RATE_PARENT, 1, 1);
  968. if (IS_ERR(clk_out[i]))
  969. return PTR_ERR(clk_out[i]);
  970. }
  971. ret = devm_of_clk_add_hw_provider(dev, vc3_of_clk_get, clk_out);
  972. if (ret)
  973. return dev_err_probe(dev, ret, "unable to add clk provider\n");
  974. return ret;
  975. }
  976. static const struct of_device_id dev_ids[] = {
  977. { .compatible = "renesas,5p35023" },
  978. { /* Sentinel */ }
  979. };
  980. MODULE_DEVICE_TABLE(of, dev_ids);
  981. static struct i2c_driver vc3_driver = {
  982. .driver = {
  983. .name = "vc3",
  984. .of_match_table = of_match_ptr(dev_ids),
  985. },
  986. .probe = vc3_probe,
  987. };
  988. module_i2c_driver(vc3_driver);
  989. MODULE_AUTHOR("Biju Das <biju.das.jz@bp.renesas.com>");
  990. MODULE_DESCRIPTION("Renesas VersaClock 3 driver");
  991. MODULE_LICENSE("GPL");