phy-exynos-mipi-video.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Samsung S5P/Exynos SoC series MIPI CSIS/DSIM DPHY driver
  4. *
  5. * Copyright (C) 2013,2016 Samsung Electronics Co., Ltd.
  6. * Author: Sylwester Nawrocki <s.nawrocki@samsung.com>
  7. */
  8. #include <linux/err.h>
  9. #include <linux/io.h>
  10. #include <linux/kernel.h>
  11. #include <linux/module.h>
  12. #include <linux/of.h>
  13. #include <linux/phy/phy.h>
  14. #include <linux/platform_device.h>
  15. #include <linux/regmap.h>
  16. #include <linux/spinlock.h>
  17. #include <linux/soc/samsung/exynos-regs-pmu.h>
  18. #include <linux/mfd/syscon.h>
  19. enum exynos_mipi_phy_id {
  20. EXYNOS_MIPI_PHY_ID_NONE = -1,
  21. EXYNOS_MIPI_PHY_ID_CSIS0,
  22. EXYNOS_MIPI_PHY_ID_DSIM0,
  23. EXYNOS_MIPI_PHY_ID_CSIS1,
  24. EXYNOS_MIPI_PHY_ID_DSIM1,
  25. EXYNOS_MIPI_PHY_ID_CSIS2,
  26. EXYNOS_MIPI_PHYS_NUM
  27. };
  28. enum exynos_mipi_phy_regmap_id {
  29. EXYNOS_MIPI_REGMAP_PMU,
  30. EXYNOS_MIPI_REGMAP_DISP,
  31. EXYNOS_MIPI_REGMAP_CAM0,
  32. EXYNOS_MIPI_REGMAP_CAM1,
  33. EXYNOS_MIPI_REGMAPS_NUM
  34. };
  35. struct mipi_phy_device_desc {
  36. int num_phys;
  37. int num_regmaps;
  38. const char *regmap_names[EXYNOS_MIPI_REGMAPS_NUM];
  39. struct exynos_mipi_phy_desc {
  40. enum exynos_mipi_phy_id coupled_phy_id;
  41. u32 enable_val;
  42. unsigned int enable_reg;
  43. enum exynos_mipi_phy_regmap_id enable_map;
  44. u32 resetn_val;
  45. unsigned int resetn_reg;
  46. enum exynos_mipi_phy_regmap_id resetn_map;
  47. } phys[EXYNOS_MIPI_PHYS_NUM];
  48. };
  49. static const struct mipi_phy_device_desc s5pv210_mipi_phy = {
  50. .num_regmaps = 1,
  51. .regmap_names = {"syscon"},
  52. .num_phys = 4,
  53. .phys = {
  54. {
  55. /* EXYNOS_MIPI_PHY_ID_CSIS0 */
  56. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_DSIM0,
  57. .enable_val = EXYNOS4_PHY_ENABLE,
  58. .enable_reg = EXYNOS4_MIPI_PHY_CONTROL(0),
  59. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  60. .resetn_val = EXYNOS4_MIPI_PHY_SRESETN,
  61. .resetn_reg = EXYNOS4_MIPI_PHY_CONTROL(0),
  62. .resetn_map = EXYNOS_MIPI_REGMAP_PMU,
  63. }, {
  64. /* EXYNOS_MIPI_PHY_ID_DSIM0 */
  65. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_CSIS0,
  66. .enable_val = EXYNOS4_PHY_ENABLE,
  67. .enable_reg = EXYNOS4_MIPI_PHY_CONTROL(0),
  68. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  69. .resetn_val = EXYNOS4_MIPI_PHY_MRESETN,
  70. .resetn_reg = EXYNOS4_MIPI_PHY_CONTROL(0),
  71. .resetn_map = EXYNOS_MIPI_REGMAP_PMU,
  72. }, {
  73. /* EXYNOS_MIPI_PHY_ID_CSIS1 */
  74. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_DSIM1,
  75. .enable_val = EXYNOS4_PHY_ENABLE,
  76. .enable_reg = EXYNOS4_MIPI_PHY_CONTROL(1),
  77. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  78. .resetn_val = EXYNOS4_MIPI_PHY_SRESETN,
  79. .resetn_reg = EXYNOS4_MIPI_PHY_CONTROL(1),
  80. .resetn_map = EXYNOS_MIPI_REGMAP_PMU,
  81. }, {
  82. /* EXYNOS_MIPI_PHY_ID_DSIM1 */
  83. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_CSIS1,
  84. .enable_val = EXYNOS4_PHY_ENABLE,
  85. .enable_reg = EXYNOS4_MIPI_PHY_CONTROL(1),
  86. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  87. .resetn_val = EXYNOS4_MIPI_PHY_MRESETN,
  88. .resetn_reg = EXYNOS4_MIPI_PHY_CONTROL(1),
  89. .resetn_map = EXYNOS_MIPI_REGMAP_PMU,
  90. },
  91. },
  92. };
  93. static const struct mipi_phy_device_desc exynos5420_mipi_phy = {
  94. .num_regmaps = 1,
  95. .regmap_names = {"syscon"},
  96. .num_phys = 5,
  97. .phys = {
  98. {
  99. /* EXYNOS_MIPI_PHY_ID_CSIS0 */
  100. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_DSIM0,
  101. .enable_val = EXYNOS4_PHY_ENABLE,
  102. .enable_reg = EXYNOS5420_MIPI_PHY_CONTROL(0),
  103. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  104. .resetn_val = EXYNOS4_MIPI_PHY_SRESETN,
  105. .resetn_reg = EXYNOS5420_MIPI_PHY_CONTROL(0),
  106. .resetn_map = EXYNOS_MIPI_REGMAP_PMU,
  107. }, {
  108. /* EXYNOS_MIPI_PHY_ID_DSIM0 */
  109. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_CSIS0,
  110. .enable_val = EXYNOS4_PHY_ENABLE,
  111. .enable_reg = EXYNOS5420_MIPI_PHY_CONTROL(0),
  112. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  113. .resetn_val = EXYNOS4_MIPI_PHY_MRESETN,
  114. .resetn_reg = EXYNOS5420_MIPI_PHY_CONTROL(0),
  115. .resetn_map = EXYNOS_MIPI_REGMAP_PMU,
  116. }, {
  117. /* EXYNOS_MIPI_PHY_ID_CSIS1 */
  118. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_DSIM1,
  119. .enable_val = EXYNOS4_PHY_ENABLE,
  120. .enable_reg = EXYNOS5420_MIPI_PHY_CONTROL(1),
  121. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  122. .resetn_val = EXYNOS4_MIPI_PHY_SRESETN,
  123. .resetn_reg = EXYNOS5420_MIPI_PHY_CONTROL(1),
  124. .resetn_map = EXYNOS_MIPI_REGMAP_PMU,
  125. }, {
  126. /* EXYNOS_MIPI_PHY_ID_DSIM1 */
  127. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_CSIS1,
  128. .enable_val = EXYNOS4_PHY_ENABLE,
  129. .enable_reg = EXYNOS5420_MIPI_PHY_CONTROL(1),
  130. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  131. .resetn_val = EXYNOS4_MIPI_PHY_MRESETN,
  132. .resetn_reg = EXYNOS5420_MIPI_PHY_CONTROL(1),
  133. .resetn_map = EXYNOS_MIPI_REGMAP_PMU,
  134. }, {
  135. /* EXYNOS_MIPI_PHY_ID_CSIS2 */
  136. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_NONE,
  137. .enable_val = EXYNOS4_PHY_ENABLE,
  138. .enable_reg = EXYNOS5420_MIPI_PHY_CONTROL(2),
  139. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  140. .resetn_val = EXYNOS4_MIPI_PHY_SRESETN,
  141. .resetn_reg = EXYNOS5420_MIPI_PHY_CONTROL(2),
  142. .resetn_map = EXYNOS_MIPI_REGMAP_PMU,
  143. },
  144. },
  145. };
  146. #define EXYNOS5433_SYSREG_DISP_MIPI_PHY 0x100C
  147. #define EXYNOS5433_SYSREG_CAM0_MIPI_DPHY_CON 0x1014
  148. #define EXYNOS5433_SYSREG_CAM1_MIPI_DPHY_CON 0x1020
  149. static const struct mipi_phy_device_desc exynos5433_mipi_phy = {
  150. .num_regmaps = 4,
  151. .regmap_names = {
  152. "samsung,pmu-syscon",
  153. "samsung,disp-sysreg",
  154. "samsung,cam0-sysreg",
  155. "samsung,cam1-sysreg"
  156. },
  157. .num_phys = 5,
  158. .phys = {
  159. {
  160. /* EXYNOS_MIPI_PHY_ID_CSIS0 */
  161. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_DSIM0,
  162. .enable_val = EXYNOS4_PHY_ENABLE,
  163. .enable_reg = EXYNOS4_MIPI_PHY_CONTROL(0),
  164. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  165. .resetn_val = BIT(0),
  166. .resetn_reg = EXYNOS5433_SYSREG_CAM0_MIPI_DPHY_CON,
  167. .resetn_map = EXYNOS_MIPI_REGMAP_CAM0,
  168. }, {
  169. /* EXYNOS_MIPI_PHY_ID_DSIM0 */
  170. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_CSIS0,
  171. .enable_val = EXYNOS4_PHY_ENABLE,
  172. .enable_reg = EXYNOS4_MIPI_PHY_CONTROL(0),
  173. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  174. .resetn_val = BIT(0),
  175. .resetn_reg = EXYNOS5433_SYSREG_DISP_MIPI_PHY,
  176. .resetn_map = EXYNOS_MIPI_REGMAP_DISP,
  177. }, {
  178. /* EXYNOS_MIPI_PHY_ID_CSIS1 */
  179. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_NONE,
  180. .enable_val = EXYNOS4_PHY_ENABLE,
  181. .enable_reg = EXYNOS4_MIPI_PHY_CONTROL(1),
  182. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  183. .resetn_val = BIT(1),
  184. .resetn_reg = EXYNOS5433_SYSREG_CAM0_MIPI_DPHY_CON,
  185. .resetn_map = EXYNOS_MIPI_REGMAP_CAM0,
  186. }, {
  187. /* EXYNOS_MIPI_PHY_ID_DSIM1 */
  188. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_NONE,
  189. .enable_val = EXYNOS4_PHY_ENABLE,
  190. .enable_reg = EXYNOS4_MIPI_PHY_CONTROL(1),
  191. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  192. .resetn_val = BIT(1),
  193. .resetn_reg = EXYNOS5433_SYSREG_DISP_MIPI_PHY,
  194. .resetn_map = EXYNOS_MIPI_REGMAP_DISP,
  195. }, {
  196. /* EXYNOS_MIPI_PHY_ID_CSIS2 */
  197. .coupled_phy_id = EXYNOS_MIPI_PHY_ID_NONE,
  198. .enable_val = EXYNOS4_PHY_ENABLE,
  199. .enable_reg = EXYNOS4_MIPI_PHY_CONTROL(2),
  200. .enable_map = EXYNOS_MIPI_REGMAP_PMU,
  201. .resetn_val = BIT(0),
  202. .resetn_reg = EXYNOS5433_SYSREG_CAM1_MIPI_DPHY_CON,
  203. .resetn_map = EXYNOS_MIPI_REGMAP_CAM1,
  204. },
  205. },
  206. };
  207. struct exynos_mipi_video_phy {
  208. struct regmap *regmaps[EXYNOS_MIPI_REGMAPS_NUM];
  209. int num_phys;
  210. struct video_phy_desc {
  211. struct phy *phy;
  212. unsigned int index;
  213. const struct exynos_mipi_phy_desc *data;
  214. } phys[EXYNOS_MIPI_PHYS_NUM];
  215. spinlock_t slock;
  216. };
  217. static int __set_phy_state(const struct exynos_mipi_phy_desc *data,
  218. struct exynos_mipi_video_phy *state, unsigned int on)
  219. {
  220. struct regmap *enable_map = state->regmaps[data->enable_map];
  221. struct regmap *resetn_map = state->regmaps[data->resetn_map];
  222. spin_lock(&state->slock);
  223. /* disable in PMU sysreg */
  224. if (!on && data->coupled_phy_id >= 0 &&
  225. state->phys[data->coupled_phy_id].phy->power_count == 0)
  226. regmap_update_bits(enable_map, data->enable_reg,
  227. data->enable_val, 0);
  228. /* PHY reset */
  229. if (on)
  230. regmap_update_bits(resetn_map, data->resetn_reg,
  231. data->resetn_val, data->resetn_val);
  232. else
  233. regmap_update_bits(resetn_map, data->resetn_reg,
  234. data->resetn_val, 0);
  235. /* enable in PMU sysreg */
  236. if (on)
  237. regmap_update_bits(enable_map, data->enable_reg,
  238. data->enable_val, data->enable_val);
  239. spin_unlock(&state->slock);
  240. return 0;
  241. }
  242. #define to_mipi_video_phy(desc) \
  243. container_of((desc), struct exynos_mipi_video_phy, phys[(desc)->index])
  244. static int exynos_mipi_video_phy_power_on(struct phy *phy)
  245. {
  246. struct video_phy_desc *phy_desc = phy_get_drvdata(phy);
  247. struct exynos_mipi_video_phy *state = to_mipi_video_phy(phy_desc);
  248. return __set_phy_state(phy_desc->data, state, 1);
  249. }
  250. static int exynos_mipi_video_phy_power_off(struct phy *phy)
  251. {
  252. struct video_phy_desc *phy_desc = phy_get_drvdata(phy);
  253. struct exynos_mipi_video_phy *state = to_mipi_video_phy(phy_desc);
  254. return __set_phy_state(phy_desc->data, state, 0);
  255. }
  256. static struct phy *exynos_mipi_video_phy_xlate(struct device *dev,
  257. const struct of_phandle_args *args)
  258. {
  259. struct exynos_mipi_video_phy *state = dev_get_drvdata(dev);
  260. if (WARN_ON(args->args[0] >= state->num_phys))
  261. return ERR_PTR(-ENODEV);
  262. return state->phys[args->args[0]].phy;
  263. }
  264. static const struct phy_ops exynos_mipi_video_phy_ops = {
  265. .power_on = exynos_mipi_video_phy_power_on,
  266. .power_off = exynos_mipi_video_phy_power_off,
  267. .owner = THIS_MODULE,
  268. };
  269. static int exynos_mipi_video_phy_probe(struct platform_device *pdev)
  270. {
  271. const struct mipi_phy_device_desc *phy_dev;
  272. struct exynos_mipi_video_phy *state;
  273. struct device *dev = &pdev->dev;
  274. struct device_node *np = dev->of_node;
  275. struct phy_provider *phy_provider;
  276. unsigned int i = 0;
  277. phy_dev = of_device_get_match_data(dev);
  278. if (!phy_dev)
  279. return -ENODEV;
  280. state = devm_kzalloc(dev, sizeof(*state), GFP_KERNEL);
  281. if (!state)
  282. return -ENOMEM;
  283. state->regmaps[i] = syscon_node_to_regmap(dev->parent->of_node);
  284. if (!IS_ERR(state->regmaps[i]))
  285. i++;
  286. for (; i < phy_dev->num_regmaps; i++) {
  287. state->regmaps[i] = syscon_regmap_lookup_by_phandle(np,
  288. phy_dev->regmap_names[i]);
  289. if (IS_ERR(state->regmaps[i]))
  290. return PTR_ERR(state->regmaps[i]);
  291. }
  292. state->num_phys = phy_dev->num_phys;
  293. spin_lock_init(&state->slock);
  294. dev_set_drvdata(dev, state);
  295. for (i = 0; i < state->num_phys; i++) {
  296. struct phy *phy = devm_phy_create(dev, NULL,
  297. &exynos_mipi_video_phy_ops);
  298. if (IS_ERR(phy)) {
  299. dev_err(dev, "failed to create PHY %d\n", i);
  300. return PTR_ERR(phy);
  301. }
  302. state->phys[i].phy = phy;
  303. state->phys[i].index = i;
  304. state->phys[i].data = &phy_dev->phys[i];
  305. phy_set_drvdata(phy, &state->phys[i]);
  306. }
  307. phy_provider = devm_of_phy_provider_register(dev,
  308. exynos_mipi_video_phy_xlate);
  309. return PTR_ERR_OR_ZERO(phy_provider);
  310. }
  311. static const struct of_device_id exynos_mipi_video_phy_of_match[] = {
  312. {
  313. .compatible = "samsung,s5pv210-mipi-video-phy",
  314. .data = &s5pv210_mipi_phy,
  315. }, {
  316. .compatible = "samsung,exynos5420-mipi-video-phy",
  317. .data = &exynos5420_mipi_phy,
  318. }, {
  319. .compatible = "samsung,exynos5433-mipi-video-phy",
  320. .data = &exynos5433_mipi_phy,
  321. },
  322. { /* sentinel */ },
  323. };
  324. MODULE_DEVICE_TABLE(of, exynos_mipi_video_phy_of_match);
  325. static struct platform_driver exynos_mipi_video_phy_driver = {
  326. .probe = exynos_mipi_video_phy_probe,
  327. .driver = {
  328. .of_match_table = exynos_mipi_video_phy_of_match,
  329. .name = "exynos-mipi-video-phy",
  330. .suppress_bind_attrs = true,
  331. }
  332. };
  333. module_platform_driver(exynos_mipi_video_phy_driver);
  334. MODULE_DESCRIPTION("Samsung S5P/Exynos SoC MIPI CSI-2/DSI PHY driver");
  335. MODULE_AUTHOR("Sylwester Nawrocki <s.nawrocki@samsung.com>");
  336. MODULE_LICENSE("GPL v2");