| // SPDX-License-Identifier: GPL-2.0-or-later |
| /* |
| * Realtek DHC pin controller driver |
| * |
| * Copyright (c) 2023 Realtek Semiconductor Corp. |
| */ |
| |
| #include <linux/bitops.h> |
| #include <linux/io.h> |
| #include <linux/module.h> |
| #include <linux/of.h> |
| #include <linux/of_address.h> |
| #include <linux/pinctrl/machine.h> |
| #include <linux/pinctrl/pinconf.h> |
| #include <linux/pinctrl/pinconf-generic.h> |
| #include <linux/pinctrl/pinctrl.h> |
| #include <linux/pinctrl/pinmux.h> |
| #include <linux/platform_device.h> |
| #include <linux/seq_file.h> |
| #include <linux/regmap.h> |
| #include <linux/slab.h> |
| #include "../core.h" |
| #include "../pinctrl-utils.h" |
| #include "pinctrl-rtd.h" |
| |
| struct rtd_pinctrl { |
| struct device *dev; |
| struct pinctrl_dev *pcdev; |
| void __iomem *base; |
| struct pinctrl_desc desc; |
| const struct rtd_pinctrl_desc *info; |
| struct regmap *regmap_pinctrl; |
| }; |
| |
| /* custom pinconf parameters */ |
| #define RTD_DRIVE_STRENGH_P (PIN_CONFIG_END + 1) |
| #define RTD_DRIVE_STRENGH_N (PIN_CONFIG_END + 2) |
| #define RTD_DUTY_CYCLE (PIN_CONFIG_END + 3) |
| |
| static const struct pinconf_generic_params rtd_custom_bindings[] = { |
| {"realtek,drive-strength-p", RTD_DRIVE_STRENGH_P, 0}, |
| {"realtek,drive-strength-n", RTD_DRIVE_STRENGH_N, 0}, |
| {"realtek,duty-cycle", RTD_DUTY_CYCLE, 0}, |
| }; |
| |
| static int rtd_pinctrl_get_groups_count(struct pinctrl_dev *pcdev) |
| { |
| struct rtd_pinctrl *data = pinctrl_dev_get_drvdata(pcdev); |
| |
| return data->info->num_groups; |
| } |
| |
| static const char *rtd_pinctrl_get_group_name(struct pinctrl_dev *pcdev, |
| unsigned int selector) |
| { |
| struct rtd_pinctrl *data = pinctrl_dev_get_drvdata(pcdev); |
| |
| return data->info->groups[selector].name; |
| } |
| |
| static int rtd_pinctrl_get_group_pins(struct pinctrl_dev *pcdev, |
| unsigned int selector, |
| const unsigned int **pins, |
| unsigned int *num_pins) |
| { |
| struct rtd_pinctrl *data = pinctrl_dev_get_drvdata(pcdev); |
| |
| *pins = data->info->groups[selector].pins; |
| *num_pins = data->info->groups[selector].num_pins; |
| |
| return 0; |
| } |
| |
| static void rtd_pinctrl_dbg_show(struct pinctrl_dev *pcdev, |
| struct seq_file *s, |
| unsigned int offset) |
| { |
| struct rtd_pinctrl *data = pinctrl_dev_get_drvdata(pcdev); |
| const struct rtd_pin_desc *mux = &data->info->muxes[offset]; |
| const struct rtd_pin_mux_desc *func; |
| u32 val; |
| u32 mask; |
| u32 pin_val; |
| int is_map; |
| |
| if (!mux->name) { |
| seq_puts(s, "[not defined]"); |
| return; |
| } |
| val = readl_relaxed(data->base + mux->mux_offset); |
| mask = mux->mux_mask; |
| pin_val = val & mask; |
| |
| is_map = 0; |
| func = &mux->functions[0]; |
| seq_puts(s, "function: "); |
| while (func->name) { |
| if (func->mux_value == pin_val) { |
| is_map = 1; |
| seq_printf(s, "[%s] ", func->name); |
| } else { |
| seq_printf(s, "%s ", func->name); |
| } |
| func++; |
| } |
| if (!is_map) |
| seq_puts(s, "[not defined]"); |
| } |
| |
| static const struct pinctrl_ops rtd_pinctrl_ops = { |
| .dt_node_to_map = pinconf_generic_dt_node_to_map_all, |
| .dt_free_map = pinctrl_utils_free_map, |
| .get_groups_count = rtd_pinctrl_get_groups_count, |
| .get_group_name = rtd_pinctrl_get_group_name, |
| .get_group_pins = rtd_pinctrl_get_group_pins, |
| .pin_dbg_show = rtd_pinctrl_dbg_show, |
| }; |
| |
| static int rtd_pinctrl_get_functions_count(struct pinctrl_dev *pcdev) |
| { |
| struct rtd_pinctrl *data = pinctrl_dev_get_drvdata(pcdev); |
| |
| return data->info->num_functions; |
| } |
| |
| static const char *rtd_pinctrl_get_function_name(struct pinctrl_dev *pcdev, |
| unsigned int selector) |
| { |
| struct rtd_pinctrl *data = pinctrl_dev_get_drvdata(pcdev); |
| |
| return data->info->functions[selector].name; |
| } |
| |
| static int rtd_pinctrl_get_function_groups(struct pinctrl_dev *pcdev, |
| unsigned int selector, |
| const char * const **groups, |
| unsigned int * const num_groups) |
| { |
| struct rtd_pinctrl *data = pinctrl_dev_get_drvdata(pcdev); |
| |
| *groups = data->info->functions[selector].groups; |
| *num_groups = data->info->functions[selector].num_groups; |
| |
| return 0; |
| } |
| |
| static const struct rtd_pin_desc *rtd_pinctrl_find_mux(struct rtd_pinctrl *data, unsigned int pin) |
| { |
| if (data->info->muxes[pin].name) |
| return &data->info->muxes[pin]; |
| |
| return NULL; |
| } |
| |
| static int rtd_pinctrl_set_one_mux(struct pinctrl_dev *pcdev, |
| unsigned int pin, const char *func_name) |
| { |
| struct rtd_pinctrl *data = pinctrl_dev_get_drvdata(pcdev); |
| const struct rtd_pin_desc *mux; |
| int ret = 0; |
| int i; |
| |
| mux = rtd_pinctrl_find_mux(data, pin); |
| if (!mux) |
| return 0; |
| |
| if (!mux->functions) { |
| if (!mux->name) |
| dev_err(pcdev->dev, "NULL pin has no functions\n"); |
| else |
| dev_err(pcdev->dev, "No functions available for pin %s\n", mux->name); |
| return -ENOTSUPP; |
| } |
| |
| for (i = 0; mux->functions[i].name; i++) { |
| if (strcmp(mux->functions[i].name, func_name) != 0) |
| continue; |
| ret = regmap_update_bits(data->regmap_pinctrl, mux->mux_offset, mux->mux_mask, |
| mux->functions[i].mux_value); |
| return ret; |
| } |
| |
| if (!mux->name) { |
| dev_err(pcdev->dev, "NULL pin provided for function %s\n", func_name); |
| return -EINVAL; |
| } |
| |
| dev_err(pcdev->dev, "No function %s available for pin %s\n", func_name, mux->name); |
| |
| return -EINVAL; |
| } |
| |
| static int rtd_pinctrl_set_mux(struct pinctrl_dev *pcdev, |
| unsigned int function, unsigned int group) |
| { |
| struct rtd_pinctrl *data = pinctrl_dev_get_drvdata(pcdev); |
| const unsigned int *pins; |
| unsigned int num_pins; |
| const char *func_name; |
| const char *group_name; |
| int i, ret; |
| |
| func_name = data->info->functions[function].name; |
| group_name = data->info->groups[group].name; |
| |
| ret = rtd_pinctrl_get_group_pins(pcdev, group, &pins, &num_pins); |
| if (ret) { |
| dev_err(pcdev->dev, "Getting pins for group %s failed\n", group_name); |
| return ret; |
| } |
| |
| for (i = 0; i < num_pins; i++) { |
| ret = rtd_pinctrl_set_one_mux(pcdev, pins[i], func_name); |
| if (ret) |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| static int rtd_pinctrl_gpio_request_enable(struct pinctrl_dev *pcdev, |
| struct pinctrl_gpio_range *range, |
| unsigned int offset) |
| { |
| return rtd_pinctrl_set_one_mux(pcdev, offset, "gpio"); |
| } |
| |
| static const struct pinmux_ops rtd_pinmux_ops = { |
| .get_functions_count = rtd_pinctrl_get_functions_count, |
| .get_function_name = rtd_pinctrl_get_function_name, |
| .get_function_groups = rtd_pinctrl_get_function_groups, |
| .set_mux = rtd_pinctrl_set_mux, |
| .gpio_request_enable = rtd_pinctrl_gpio_request_enable, |
| }; |
| |
| static const struct pinctrl_pin_desc |
| *rtd_pinctrl_get_pin_by_number(struct rtd_pinctrl *data, int number) |
| { |
| int i; |
| |
| for (i = 0; i < data->info->num_pins; i++) { |
| if (data->info->pins[i].number == number) |
| return &data->info->pins[i]; |
| } |
| |
| return NULL; |
| } |
| |
| static const struct rtd_pin_config_desc |
| *rtd_pinctrl_find_config(struct rtd_pinctrl *data, unsigned int pin) |
| { |
| if (data->info->configs[pin].name) |
| return &data->info->configs[pin]; |
| |
| return NULL; |
| } |
| |
| static const struct rtd_pin_sconfig_desc *rtd_pinctrl_find_sconfig(struct rtd_pinctrl *data, |
| unsigned int pin) |
| { |
| int i; |
| const struct pinctrl_pin_desc *pin_desc; |
| const char *pin_name; |
| |
| pin_desc = rtd_pinctrl_get_pin_by_number(data, pin); |
| if (!pin_desc) |
| return NULL; |
| |
| pin_name = pin_desc->name; |
| |
| for (i = 0; i < data->info->num_sconfigs; i++) { |
| if (strcmp(data->info->sconfigs[i].name, pin_name) == 0) |
| return &data->info->sconfigs[i]; |
| } |
| |
| return NULL; |
| } |
| |
| static int rtd_pconf_parse_conf(struct rtd_pinctrl *data, |
| unsigned int pinnr, |
| enum pin_config_param param, |
| enum pin_config_param arg) |
| { |
| const struct rtd_pin_config_desc *config_desc; |
| const struct rtd_pin_sconfig_desc *sconfig_desc; |
| u8 set_val = 0; |
| u16 strength; |
| u32 val; |
| u32 mask; |
| u32 pulsel_off, pulen_off, smt_off, curr_off, pow_off, reg_off, p_off, n_off; |
| const char *name = data->info->pins[pinnr].name; |
| int ret = 0; |
| |
| config_desc = rtd_pinctrl_find_config(data, pinnr); |
| if (!config_desc) { |
| dev_err(data->dev, "Not support pin config for pin: %s\n", name); |
| return -ENOTSUPP; |
| } |
| switch ((u32)param) { |
| case PIN_CONFIG_INPUT_SCHMITT: |
| case PIN_CONFIG_INPUT_SCHMITT_ENABLE: |
| if (config_desc->smt_offset == NA) { |
| dev_err(data->dev, "Not support input schmitt for pin: %s\n", name); |
| return -ENOTSUPP; |
| } |
| smt_off = config_desc->base_bit + config_desc->smt_offset; |
| reg_off = config_desc->reg_offset; |
| set_val = arg; |
| |
| mask = BIT(smt_off); |
| val = set_val ? BIT(smt_off) : 0; |
| break; |
| |
| case PIN_CONFIG_DRIVE_PUSH_PULL: |
| if (config_desc->pud_en_offset == NA) { |
| dev_err(data->dev, "Not support push pull for pin: %s\n", name); |
| return -ENOTSUPP; |
| } |
| pulen_off = config_desc->base_bit + config_desc->pud_en_offset; |
| reg_off = config_desc->reg_offset; |
| |
| mask = BIT(pulen_off); |
| val = 0; |
| break; |
| |
| case PIN_CONFIG_BIAS_DISABLE: |
| if (config_desc->pud_en_offset == NA) { |
| dev_err(data->dev, "Not support bias disable for pin: %s\n", name); |
| return -ENOTSUPP; |
| } |
| pulen_off = config_desc->base_bit + config_desc->pud_en_offset; |
| reg_off = config_desc->reg_offset; |
| |
| mask = BIT(pulen_off); |
| val = 0; |
| break; |
| |
| case PIN_CONFIG_BIAS_PULL_UP: |
| if (config_desc->pud_en_offset == NA) { |
| dev_err(data->dev, "Not support bias pull up for pin:%s\n", name); |
| return -ENOTSUPP; |
| } |
| pulen_off = config_desc->base_bit + config_desc->pud_en_offset; |
| pulsel_off = config_desc->base_bit + config_desc->pud_sel_offset; |
| reg_off = config_desc->reg_offset; |
| |
| mask = BIT(pulen_off) | BIT(pulsel_off); |
| val = mask; |
| break; |
| |
| case PIN_CONFIG_BIAS_PULL_DOWN: |
| if (config_desc->pud_en_offset == NA) { |
| dev_err(data->dev, "Not support bias pull down for pin: %s\n", name); |
| return -ENOTSUPP; |
| } |
| pulen_off = config_desc->base_bit + config_desc->pud_en_offset; |
| pulsel_off = config_desc->base_bit + config_desc->pud_sel_offset; |
| reg_off = config_desc->reg_offset; |
| |
| mask = BIT(pulen_off) | BIT(pulsel_off); |
| val = BIT(pulen_off); |
| break; |
| |
| case PIN_CONFIG_DRIVE_STRENGTH: |
| curr_off = config_desc->base_bit + config_desc->curr_offset; |
| reg_off = config_desc->reg_offset; |
| strength = arg; |
| val = 0; |
| switch (config_desc->curr_type) { |
| case PADDRI_4_8: |
| if (strength == 4) |
| val = 0; |
| else if (strength == 8) |
| val = BIT(curr_off); |
| else |
| return -EINVAL; |
| break; |
| case PADDRI_2_4: |
| if (strength == 2) |
| val = 0; |
| else if (strength == 4) |
| val = BIT(curr_off); |
| else |
| return -EINVAL; |
| break; |
| case NA: |
| dev_err(data->dev, "Not support drive strength for pin: %s\n", name); |
| return -ENOTSUPP; |
| default: |
| return -EINVAL; |
| } |
| mask = BIT(curr_off); |
| break; |
| |
| case PIN_CONFIG_POWER_SOURCE: |
| if (config_desc->power_offset == NA) { |
| dev_err(data->dev, "Not support power source for pin: %s\n", name); |
| return -ENOTSUPP; |
| } |
| reg_off = config_desc->reg_offset; |
| pow_off = config_desc->base_bit + config_desc->power_offset; |
| if (pow_off >= 32) { |
| reg_off += 0x4; |
| pow_off -= 32; |
| } |
| set_val = arg; |
| mask = BIT(pow_off); |
| val = set_val ? mask : 0; |
| break; |
| |
| case RTD_DRIVE_STRENGH_P: |
| sconfig_desc = rtd_pinctrl_find_sconfig(data, pinnr); |
| if (!sconfig_desc) { |
| dev_err(data->dev, "Not support P driving for pin: %s\n", name); |
| return -ENOTSUPP; |
| } |
| set_val = arg; |
| reg_off = sconfig_desc->reg_offset; |
| p_off = sconfig_desc->pdrive_offset; |
| if (p_off >= 32) { |
| reg_off += 0x4; |
| p_off -= 32; |
| } |
| mask = GENMASK(p_off + sconfig_desc->pdrive_maskbits - 1, p_off); |
| val = set_val << p_off; |
| break; |
| |
| case RTD_DRIVE_STRENGH_N: |
| sconfig_desc = rtd_pinctrl_find_sconfig(data, pinnr); |
| if (!sconfig_desc) { |
| dev_err(data->dev, "Not support N driving for pin: %s\n", name); |
| return -ENOTSUPP; |
| } |
| set_val = arg; |
| reg_off = sconfig_desc->reg_offset; |
| n_off = sconfig_desc->ndrive_offset; |
| if (n_off >= 32) { |
| reg_off += 0x4; |
| n_off -= 32; |
| } |
| mask = GENMASK(n_off + sconfig_desc->ndrive_maskbits - 1, n_off); |
| val = set_val << n_off; |
| break; |
| |
| case RTD_DUTY_CYCLE: |
| sconfig_desc = rtd_pinctrl_find_sconfig(data, pinnr); |
| if (!sconfig_desc || sconfig_desc->dcycle_offset == NA) { |
| dev_err(data->dev, "Not support duty cycle for pin: %s\n", name); |
| return -ENOTSUPP; |
| } |
| set_val = arg; |
| reg_off = config_desc->reg_offset; |
| mask = GENMASK(sconfig_desc->dcycle_offset + |
| sconfig_desc->dcycle_maskbits - 1, sconfig_desc->dcycle_offset); |
| val = set_val << sconfig_desc->dcycle_offset; |
| break; |
| |
| default: |
| dev_err(data->dev, "unsupported pinconf: %d\n", (u32)param); |
| return -EINVAL; |
| } |
| |
| ret = regmap_update_bits(data->regmap_pinctrl, reg_off, mask, val); |
| if (ret) |
| dev_err(data->dev, "could not update pinconf(%d) for pin(%s)\n", (u32)param, name); |
| |
| return ret; |
| } |
| |
| static int rtd_pin_config_get(struct pinctrl_dev *pcdev, unsigned int pinnr, |
| unsigned long *config) |
| { |
| unsigned int param = pinconf_to_config_param(*config); |
| unsigned int arg = 0; |
| |
| switch (param) { |
| default: |
| return -ENOTSUPP; |
| } |
| |
| *config = pinconf_to_config_packed(param, arg); |
| return 0; |
| } |
| |
| static int rtd_pin_config_set(struct pinctrl_dev *pcdev, unsigned int pinnr, |
| unsigned long *configs, unsigned int num_configs) |
| { |
| struct rtd_pinctrl *data = pinctrl_dev_get_drvdata(pcdev); |
| int i; |
| int ret = 0; |
| |
| for (i = 0; i < num_configs; i++) { |
| ret = rtd_pconf_parse_conf(data, pinnr, |
| pinconf_to_config_param(configs[i]), |
| pinconf_to_config_argument(configs[i])); |
| if (ret < 0) |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| static int rtd_pin_config_group_set(struct pinctrl_dev *pcdev, unsigned int group, |
| unsigned long *configs, unsigned int num_configs) |
| { |
| struct rtd_pinctrl *data = pinctrl_dev_get_drvdata(pcdev); |
| const unsigned int *pins; |
| unsigned int num_pins; |
| const char *group_name; |
| int i, ret; |
| |
| group_name = data->info->groups[group].name; |
| |
| ret = rtd_pinctrl_get_group_pins(pcdev, group, &pins, &num_pins); |
| if (ret) { |
| dev_err(pcdev->dev, "Getting pins for group %s failed\n", group_name); |
| return ret; |
| } |
| |
| for (i = 0; i < num_pins; i++) { |
| ret = rtd_pin_config_set(pcdev, pins[i], configs, num_configs); |
| if (ret) |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| static const struct pinconf_ops rtd_pinconf_ops = { |
| .is_generic = true, |
| .pin_config_get = rtd_pin_config_get, |
| .pin_config_set = rtd_pin_config_set, |
| .pin_config_group_set = rtd_pin_config_group_set, |
| }; |
| |
| static struct regmap_config rtd_pinctrl_regmap_config = { |
| .reg_bits = 32, |
| .val_bits = 32, |
| .reg_stride = 4, |
| .use_relaxed_mmio = true, |
| }; |
| |
| int rtd_pinctrl_probe(struct platform_device *pdev, const struct rtd_pinctrl_desc *desc) |
| { |
| struct rtd_pinctrl *data; |
| int ret; |
| |
| data = devm_kzalloc(&pdev->dev, sizeof(*data), GFP_KERNEL); |
| if (!data) |
| return -ENOMEM; |
| |
| data->base = of_iomap(pdev->dev.of_node, 0); |
| if (!data->base) |
| return -ENOMEM; |
| |
| data->dev = &pdev->dev; |
| data->info = desc; |
| data->desc.name = dev_name(&pdev->dev); |
| data->desc.pins = data->info->pins; |
| data->desc.npins = data->info->num_pins; |
| data->desc.pctlops = &rtd_pinctrl_ops; |
| data->desc.pmxops = &rtd_pinmux_ops; |
| data->desc.confops = &rtd_pinconf_ops; |
| data->desc.custom_params = rtd_custom_bindings; |
| data->desc.num_custom_params = ARRAY_SIZE(rtd_custom_bindings); |
| data->desc.owner = THIS_MODULE; |
| data->regmap_pinctrl = devm_regmap_init_mmio(data->dev, data->base, |
| &rtd_pinctrl_regmap_config); |
| |
| if (IS_ERR(data->regmap_pinctrl)) { |
| dev_err(data->dev, "failed to init regmap: %ld\n", |
| PTR_ERR(data->regmap_pinctrl)); |
| ret = PTR_ERR(data->regmap_pinctrl); |
| goto unmap; |
| } |
| |
| data->pcdev = pinctrl_register(&data->desc, &pdev->dev, data); |
| if (IS_ERR(data->pcdev)) { |
| ret = PTR_ERR(data->pcdev); |
| goto unmap; |
| } |
| |
| platform_set_drvdata(pdev, data); |
| |
| dev_dbg(&pdev->dev, "probed\n"); |
| |
| return 0; |
| |
| unmap: |
| iounmap(data->base); |
| return ret; |
| } |
| EXPORT_SYMBOL(rtd_pinctrl_probe); |
| |
| MODULE_DESCRIPTION("Realtek DHC SoC pinctrl driver"); |
| MODULE_LICENSE("GPL v2"); |