pinctrl-spear.c 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354
  1. /*
  2. * Driver for the ST Microelectronics SPEAr pinmux
  3. *
  4. * Copyright (C) 2012 ST Microelectronics
  5. * Viresh Kumar <viresh.linux@gmail.com>
  6. *
  7. * Inspired from:
  8. * - U300 Pinctl drivers
  9. * - Tegra Pinctl drivers
  10. *
  11. * This file is licensed under the terms of the GNU General Public
  12. * License version 2. This program is licensed "as is" without any
  13. * warranty of any kind, whether express or implied.
  14. */
  15. #include <linux/err.h>
  16. #include <linux/io.h>
  17. #include <linux/module.h>
  18. #include <linux/of.h>
  19. #include <linux/of_address.h>
  20. #include <linux/pinctrl/machine.h>
  21. #include <linux/pinctrl/pinctrl.h>
  22. #include <linux/pinctrl/pinmux.h>
  23. #include <linux/platform_device.h>
  24. #include <linux/slab.h>
  25. #include "pinctrl-spear.h"
  26. #define DRIVER_NAME "spear-pinmux"
  27. static inline u32 pmx_readl(struct spear_pmx *pmx, u32 reg)
  28. {
  29. return readl_relaxed(pmx->vbase + reg);
  30. }
  31. static inline void pmx_writel(struct spear_pmx *pmx, u32 val, u32 reg)
  32. {
  33. writel_relaxed(val, pmx->vbase + reg);
  34. }
  35. static int set_mode(struct spear_pmx *pmx, int mode)
  36. {
  37. struct spear_pmx_mode *pmx_mode = NULL;
  38. int i;
  39. u32 val;
  40. if (!pmx->machdata->pmx_modes || !pmx->machdata->npmx_modes)
  41. return -EINVAL;
  42. for (i = 0; i < pmx->machdata->npmx_modes; i++) {
  43. if (pmx->machdata->pmx_modes[i]->mode == (1 << mode)) {
  44. pmx_mode = pmx->machdata->pmx_modes[i];
  45. break;
  46. }
  47. }
  48. if (!pmx_mode)
  49. return -EINVAL;
  50. val = pmx_readl(pmx, pmx_mode->reg);
  51. val &= ~pmx_mode->mask;
  52. val |= pmx_mode->val;
  53. pmx_writel(pmx, val, pmx_mode->reg);
  54. pmx->machdata->mode = pmx_mode->mode;
  55. dev_info(pmx->dev, "Configured Mode: %s with id: %x\n\n",
  56. pmx_mode->name ? pmx_mode->name : "no_name",
  57. pmx_mode->reg);
  58. return 0;
  59. }
  60. void __devinit pmx_init_addr(struct spear_pinctrl_machdata *machdata, u16 reg)
  61. {
  62. struct spear_pingroup *pgroup;
  63. struct spear_modemux *modemux;
  64. int i, j, group;
  65. for (group = 0; group < machdata->ngroups; group++) {
  66. pgroup = machdata->groups[group];
  67. for (i = 0; i < pgroup->nmodemuxs; i++) {
  68. modemux = &pgroup->modemuxs[i];
  69. for (j = 0; j < modemux->nmuxregs; j++)
  70. if (modemux->muxregs[j].reg == 0xFFFF)
  71. modemux->muxregs[j].reg = reg;
  72. }
  73. }
  74. }
  75. static int spear_pinctrl_get_groups_cnt(struct pinctrl_dev *pctldev)
  76. {
  77. struct spear_pmx *pmx = pinctrl_dev_get_drvdata(pctldev);
  78. return pmx->machdata->ngroups;
  79. }
  80. static const char *spear_pinctrl_get_group_name(struct pinctrl_dev *pctldev,
  81. unsigned group)
  82. {
  83. struct spear_pmx *pmx = pinctrl_dev_get_drvdata(pctldev);
  84. return pmx->machdata->groups[group]->name;
  85. }
  86. static int spear_pinctrl_get_group_pins(struct pinctrl_dev *pctldev,
  87. unsigned group, const unsigned **pins, unsigned *num_pins)
  88. {
  89. struct spear_pmx *pmx = pinctrl_dev_get_drvdata(pctldev);
  90. *pins = pmx->machdata->groups[group]->pins;
  91. *num_pins = pmx->machdata->groups[group]->npins;
  92. return 0;
  93. }
  94. static void spear_pinctrl_pin_dbg_show(struct pinctrl_dev *pctldev,
  95. struct seq_file *s, unsigned offset)
  96. {
  97. seq_printf(s, " " DRIVER_NAME);
  98. }
  99. int spear_pinctrl_dt_node_to_map(struct pinctrl_dev *pctldev,
  100. struct device_node *np_config,
  101. struct pinctrl_map **map, unsigned *num_maps)
  102. {
  103. struct spear_pmx *pmx = pinctrl_dev_get_drvdata(pctldev);
  104. struct device_node *np;
  105. struct property *prop;
  106. const char *function, *group;
  107. int ret, index = 0, count = 0;
  108. /* calculate number of maps required */
  109. for_each_child_of_node(np_config, np) {
  110. ret = of_property_read_string(np, "st,function", &function);
  111. if (ret < 0)
  112. return ret;
  113. ret = of_property_count_strings(np, "st,pins");
  114. if (ret < 0)
  115. return ret;
  116. count += ret;
  117. }
  118. if (!count) {
  119. dev_err(pmx->dev, "No child nodes passed via DT\n");
  120. return -ENODEV;
  121. }
  122. *map = kzalloc(sizeof(**map) * count, GFP_KERNEL);
  123. if (!*map)
  124. return -ENOMEM;
  125. for_each_child_of_node(np_config, np) {
  126. of_property_read_string(np, "st,function", &function);
  127. of_property_for_each_string(np, "st,pins", prop, group) {
  128. (*map)[index].type = PIN_MAP_TYPE_MUX_GROUP;
  129. (*map)[index].data.mux.group = group;
  130. (*map)[index].data.mux.function = function;
  131. index++;
  132. }
  133. }
  134. *num_maps = count;
  135. return 0;
  136. }
  137. void spear_pinctrl_dt_free_map(struct pinctrl_dev *pctldev,
  138. struct pinctrl_map *map, unsigned num_maps)
  139. {
  140. kfree(map);
  141. }
  142. static struct pinctrl_ops spear_pinctrl_ops = {
  143. .get_groups_count = spear_pinctrl_get_groups_cnt,
  144. .get_group_name = spear_pinctrl_get_group_name,
  145. .get_group_pins = spear_pinctrl_get_group_pins,
  146. .pin_dbg_show = spear_pinctrl_pin_dbg_show,
  147. .dt_node_to_map = spear_pinctrl_dt_node_to_map,
  148. .dt_free_map = spear_pinctrl_dt_free_map,
  149. };
  150. static int spear_pinctrl_get_funcs_count(struct pinctrl_dev *pctldev)
  151. {
  152. struct spear_pmx *pmx = pinctrl_dev_get_drvdata(pctldev);
  153. return pmx->machdata->nfunctions;
  154. }
  155. static const char *spear_pinctrl_get_func_name(struct pinctrl_dev *pctldev,
  156. unsigned function)
  157. {
  158. struct spear_pmx *pmx = pinctrl_dev_get_drvdata(pctldev);
  159. return pmx->machdata->functions[function]->name;
  160. }
  161. static int spear_pinctrl_get_func_groups(struct pinctrl_dev *pctldev,
  162. unsigned function, const char *const **groups,
  163. unsigned * const ngroups)
  164. {
  165. struct spear_pmx *pmx = pinctrl_dev_get_drvdata(pctldev);
  166. *groups = pmx->machdata->functions[function]->groups;
  167. *ngroups = pmx->machdata->functions[function]->ngroups;
  168. return 0;
  169. }
  170. static int spear_pinctrl_endisable(struct pinctrl_dev *pctldev,
  171. unsigned function, unsigned group, bool enable)
  172. {
  173. struct spear_pmx *pmx = pinctrl_dev_get_drvdata(pctldev);
  174. const struct spear_pingroup *pgroup;
  175. const struct spear_modemux *modemux;
  176. struct spear_muxreg *muxreg;
  177. u32 val, temp;
  178. int i, j;
  179. bool found = false;
  180. pgroup = pmx->machdata->groups[group];
  181. for (i = 0; i < pgroup->nmodemuxs; i++) {
  182. modemux = &pgroup->modemuxs[i];
  183. /* SoC have any modes */
  184. if (pmx->machdata->modes_supported) {
  185. if (!(pmx->machdata->mode & modemux->modes))
  186. continue;
  187. }
  188. found = true;
  189. for (j = 0; j < modemux->nmuxregs; j++) {
  190. muxreg = &modemux->muxregs[j];
  191. val = pmx_readl(pmx, muxreg->reg);
  192. val &= ~muxreg->mask;
  193. if (enable)
  194. temp = muxreg->val;
  195. else
  196. temp = ~muxreg->val;
  197. val |= temp;
  198. pmx_writel(pmx, val, muxreg->reg);
  199. }
  200. }
  201. if (!found) {
  202. dev_err(pmx->dev, "pinmux group: %s not supported\n",
  203. pgroup->name);
  204. return -ENODEV;
  205. }
  206. return 0;
  207. }
  208. static int spear_pinctrl_enable(struct pinctrl_dev *pctldev, unsigned function,
  209. unsigned group)
  210. {
  211. return spear_pinctrl_endisable(pctldev, function, group, true);
  212. }
  213. static void spear_pinctrl_disable(struct pinctrl_dev *pctldev,
  214. unsigned function, unsigned group)
  215. {
  216. spear_pinctrl_endisable(pctldev, function, group, false);
  217. }
  218. static struct pinmux_ops spear_pinmux_ops = {
  219. .get_functions_count = spear_pinctrl_get_funcs_count,
  220. .get_function_name = spear_pinctrl_get_func_name,
  221. .get_function_groups = spear_pinctrl_get_func_groups,
  222. .enable = spear_pinctrl_enable,
  223. .disable = spear_pinctrl_disable,
  224. };
  225. static struct pinctrl_desc spear_pinctrl_desc = {
  226. .name = DRIVER_NAME,
  227. .pctlops = &spear_pinctrl_ops,
  228. .pmxops = &spear_pinmux_ops,
  229. .owner = THIS_MODULE,
  230. };
  231. int __devinit spear_pinctrl_probe(struct platform_device *pdev,
  232. struct spear_pinctrl_machdata *machdata)
  233. {
  234. struct device_node *np = pdev->dev.of_node;
  235. struct resource *res;
  236. struct spear_pmx *pmx;
  237. if (!machdata)
  238. return -ENODEV;
  239. res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  240. if (!res)
  241. return -EINVAL;
  242. pmx = devm_kzalloc(&pdev->dev, sizeof(*pmx), GFP_KERNEL);
  243. if (!pmx) {
  244. dev_err(&pdev->dev, "Can't alloc spear_pmx\n");
  245. return -ENOMEM;
  246. }
  247. pmx->vbase = devm_ioremap(&pdev->dev, res->start, resource_size(res));
  248. if (!pmx->vbase) {
  249. dev_err(&pdev->dev, "Couldn't ioremap at index 0\n");
  250. return -ENODEV;
  251. }
  252. pmx->dev = &pdev->dev;
  253. pmx->machdata = machdata;
  254. /* configure mode, if supported by SoC */
  255. if (machdata->modes_supported) {
  256. int mode = 0;
  257. if (of_property_read_u32(np, "st,pinmux-mode", &mode)) {
  258. dev_err(&pdev->dev, "OF: pinmux mode not passed\n");
  259. return -EINVAL;
  260. }
  261. if (set_mode(pmx, mode)) {
  262. dev_err(&pdev->dev, "OF: Couldn't configure mode: %x\n",
  263. mode);
  264. return -EINVAL;
  265. }
  266. }
  267. platform_set_drvdata(pdev, pmx);
  268. spear_pinctrl_desc.pins = machdata->pins;
  269. spear_pinctrl_desc.npins = machdata->npins;
  270. pmx->pctl = pinctrl_register(&spear_pinctrl_desc, &pdev->dev, pmx);
  271. if (!pmx->pctl) {
  272. dev_err(&pdev->dev, "Couldn't register pinctrl driver\n");
  273. return -ENODEV;
  274. }
  275. return 0;
  276. }
  277. int __devexit spear_pinctrl_remove(struct platform_device *pdev)
  278. {
  279. struct spear_pmx *pmx = platform_get_drvdata(pdev);
  280. pinctrl_unregister(pmx->pctl);
  281. return 0;
  282. }