pinctrl-meson-axg-pmx.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. // SPDX-License-Identifier: (GPL-2.0+ OR MIT)
  2. /*
  3. * Second generation of pinmux driver for Amlogic Meson-AXG SoC.
  4. *
  5. * Copyright (c) 2017 Baylibre SAS.
  6. * Author: Jerome Brunet <jbrunet@baylibre.com>
  7. *
  8. * Copyright (c) 2017 Amlogic, Inc. All rights reserved.
  9. * Author: Xingyu Chen <xingyu.chen@amlogic.com>
  10. */
  11. /*
  12. * This new generation of pinctrl IP is mainly adopted by the
  13. * Meson-AXG SoC and later series, which use 4-width continuous
  14. * register bit to select the function for each pin.
  15. *
  16. * The value 0 is always selecting the GPIO mode, while other
  17. * values (start from 1) for selecting the function mode.
  18. */
  19. #include <linux/device.h>
  20. #include <linux/regmap.h>
  21. #include <linux/pinctrl/pinctrl.h>
  22. #include <linux/pinctrl/pinmux.h>
  23. #include "pinctrl-meson.h"
  24. #include "pinctrl-meson-axg-pmx.h"
  25. static int meson_axg_pmx_get_bank(struct meson_pinctrl *pc,
  26. unsigned int pin,
  27. const struct meson_pmx_bank **bank)
  28. {
  29. int i;
  30. const struct meson_axg_pmx_data *pmx = pc->data->pmx_data;
  31. for (i = 0; i < pmx->num_pmx_banks; i++)
  32. if (pin >= pmx->pmx_banks[i].first &&
  33. pin <= pmx->pmx_banks[i].last) {
  34. *bank = &pmx->pmx_banks[i];
  35. return 0;
  36. }
  37. return -EINVAL;
  38. }
  39. static int meson_pmx_calc_reg_and_offset(const struct meson_pmx_bank *bank,
  40. unsigned int pin, unsigned int *reg,
  41. unsigned int *offset)
  42. {
  43. int shift;
  44. shift = pin - bank->first;
  45. *reg = bank->reg + (bank->offset + (shift << 2)) / 32;
  46. *offset = (bank->offset + (shift << 2)) % 32;
  47. return 0;
  48. }
  49. static int meson_axg_pmx_update_function(struct meson_pinctrl *pc,
  50. unsigned int pin, unsigned int func)
  51. {
  52. const struct meson_pmx_bank *bank;
  53. int ret;
  54. int reg;
  55. int offset;
  56. ret = meson_axg_pmx_get_bank(pc, pin, &bank);
  57. if (ret)
  58. return ret;
  59. meson_pmx_calc_reg_and_offset(bank, pin, &reg, &offset);
  60. ret = regmap_update_bits(pc->reg_mux, reg << 2,
  61. 0xf << offset, (func & 0xf) << offset);
  62. return ret;
  63. }
  64. static int meson_axg_pmx_set_mux(struct pinctrl_dev *pcdev,
  65. unsigned int func_num, unsigned int group_num)
  66. {
  67. int i;
  68. int ret;
  69. struct meson_pinctrl *pc = pinctrl_dev_get_drvdata(pcdev);
  70. const struct meson_pmx_func *func = &pc->data->funcs[func_num];
  71. const struct meson_pmx_group *group = &pc->data->groups[group_num];
  72. struct meson_pmx_axg_data *pmx_data =
  73. (struct meson_pmx_axg_data *)group->data;
  74. dev_dbg(pc->dev, "enable function %s, group %s\n", func->name,
  75. group->name);
  76. for (i = 0; i < group->num_pins; i++) {
  77. ret = meson_axg_pmx_update_function(pc, group->pins[i],
  78. pmx_data->func);
  79. if (ret)
  80. return ret;
  81. }
  82. return 0;
  83. }
  84. static int meson_axg_pmx_request_gpio(struct pinctrl_dev *pcdev,
  85. struct pinctrl_gpio_range *range, unsigned int offset)
  86. {
  87. struct meson_pinctrl *pc = pinctrl_dev_get_drvdata(pcdev);
  88. return meson_axg_pmx_update_function(pc, offset, 0);
  89. }
  90. const struct pinmux_ops meson_axg_pmx_ops = {
  91. .set_mux = meson_axg_pmx_set_mux,
  92. .get_functions_count = meson_pmx_get_funcs_count,
  93. .get_function_name = meson_pmx_get_func_name,
  94. .get_function_groups = meson_pmx_get_groups,
  95. .gpio_request_enable = meson_axg_pmx_request_gpio,
  96. };
  97. EXPORT_SYMBOL_GPL(meson_axg_pmx_ops);
  98. MODULE_DESCRIPTION("Amlogic Meson AXG second generation pinmux driver");
  99. MODULE_LICENSE("Dual BSD/GPL");