18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * First generation of pinmux driver for Amlogic Meson SoCs 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright (C) 2014 Beniamino Galvani <b.galvani@gmail.com> 68c2ecf20Sopenharmony_ci * Copyright (C) 2017 Jerome Brunet <jbrunet@baylibre.com> 78c2ecf20Sopenharmony_ci */ 88c2ecf20Sopenharmony_ci 98c2ecf20Sopenharmony_ci/* For this first generation of pinctrl driver every pinmux group can be 108c2ecf20Sopenharmony_ci * enabled by a specific bit in the first register range. When all groups for 118c2ecf20Sopenharmony_ci * a given pin are disabled the pin acts as a GPIO. 128c2ecf20Sopenharmony_ci */ 138c2ecf20Sopenharmony_ci#include <linux/device.h> 148c2ecf20Sopenharmony_ci#include <linux/regmap.h> 158c2ecf20Sopenharmony_ci#include <linux/pinctrl/pinctrl.h> 168c2ecf20Sopenharmony_ci#include <linux/pinctrl/pinmux.h> 178c2ecf20Sopenharmony_ci 188c2ecf20Sopenharmony_ci#include "pinctrl-meson.h" 198c2ecf20Sopenharmony_ci#include "pinctrl-meson8-pmx.h" 208c2ecf20Sopenharmony_ci 218c2ecf20Sopenharmony_ci/** 228c2ecf20Sopenharmony_ci * meson8_pmx_disable_other_groups() - disable other groups using a given pin 238c2ecf20Sopenharmony_ci * 248c2ecf20Sopenharmony_ci * @pc: meson pin controller device 258c2ecf20Sopenharmony_ci * @pin: number of the pin 268c2ecf20Sopenharmony_ci * @sel_group: index of the selected group, or -1 if none 278c2ecf20Sopenharmony_ci * 288c2ecf20Sopenharmony_ci * The function disables all pinmux groups using a pin except the 298c2ecf20Sopenharmony_ci * selected one. If @sel_group is -1 all groups are disabled, leaving 308c2ecf20Sopenharmony_ci * the pin in GPIO mode. 318c2ecf20Sopenharmony_ci */ 328c2ecf20Sopenharmony_cistatic void meson8_pmx_disable_other_groups(struct meson_pinctrl *pc, 338c2ecf20Sopenharmony_ci unsigned int pin, int sel_group) 348c2ecf20Sopenharmony_ci{ 358c2ecf20Sopenharmony_ci struct meson_pmx_group *group; 368c2ecf20Sopenharmony_ci struct meson8_pmx_data *pmx_data; 378c2ecf20Sopenharmony_ci int i, j; 388c2ecf20Sopenharmony_ci 398c2ecf20Sopenharmony_ci for (i = 0; i < pc->data->num_groups; i++) { 408c2ecf20Sopenharmony_ci group = &pc->data->groups[i]; 418c2ecf20Sopenharmony_ci pmx_data = (struct meson8_pmx_data *)group->data; 428c2ecf20Sopenharmony_ci if (pmx_data->is_gpio || i == sel_group) 438c2ecf20Sopenharmony_ci continue; 448c2ecf20Sopenharmony_ci 458c2ecf20Sopenharmony_ci for (j = 0; j < group->num_pins; j++) { 468c2ecf20Sopenharmony_ci if (group->pins[j] == pin) { 478c2ecf20Sopenharmony_ci /* We have found a group using the pin */ 488c2ecf20Sopenharmony_ci regmap_update_bits(pc->reg_mux, 498c2ecf20Sopenharmony_ci pmx_data->reg * 4, 508c2ecf20Sopenharmony_ci BIT(pmx_data->bit), 0); 518c2ecf20Sopenharmony_ci } 528c2ecf20Sopenharmony_ci } 538c2ecf20Sopenharmony_ci } 548c2ecf20Sopenharmony_ci} 558c2ecf20Sopenharmony_ci 568c2ecf20Sopenharmony_cistatic int meson8_pmx_set_mux(struct pinctrl_dev *pcdev, unsigned func_num, 578c2ecf20Sopenharmony_ci unsigned group_num) 588c2ecf20Sopenharmony_ci{ 598c2ecf20Sopenharmony_ci struct meson_pinctrl *pc = pinctrl_dev_get_drvdata(pcdev); 608c2ecf20Sopenharmony_ci struct meson_pmx_func *func = &pc->data->funcs[func_num]; 618c2ecf20Sopenharmony_ci struct meson_pmx_group *group = &pc->data->groups[group_num]; 628c2ecf20Sopenharmony_ci struct meson8_pmx_data *pmx_data = 638c2ecf20Sopenharmony_ci (struct meson8_pmx_data *)group->data; 648c2ecf20Sopenharmony_ci int i, ret = 0; 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_ci dev_dbg(pc->dev, "enable function %s, group %s\n", func->name, 678c2ecf20Sopenharmony_ci group->name); 688c2ecf20Sopenharmony_ci 698c2ecf20Sopenharmony_ci /* 708c2ecf20Sopenharmony_ci * Disable groups using the same pin. 718c2ecf20Sopenharmony_ci * The selected group is not disabled to avoid glitches. 728c2ecf20Sopenharmony_ci */ 738c2ecf20Sopenharmony_ci for (i = 0; i < group->num_pins; i++) 748c2ecf20Sopenharmony_ci meson8_pmx_disable_other_groups(pc, group->pins[i], group_num); 758c2ecf20Sopenharmony_ci 768c2ecf20Sopenharmony_ci /* Function 0 (GPIO) doesn't need any additional setting */ 778c2ecf20Sopenharmony_ci if (func_num) 788c2ecf20Sopenharmony_ci ret = regmap_update_bits(pc->reg_mux, pmx_data->reg * 4, 798c2ecf20Sopenharmony_ci BIT(pmx_data->bit), 808c2ecf20Sopenharmony_ci BIT(pmx_data->bit)); 818c2ecf20Sopenharmony_ci 828c2ecf20Sopenharmony_ci return ret; 838c2ecf20Sopenharmony_ci} 848c2ecf20Sopenharmony_ci 858c2ecf20Sopenharmony_cistatic int meson8_pmx_request_gpio(struct pinctrl_dev *pcdev, 868c2ecf20Sopenharmony_ci struct pinctrl_gpio_range *range, 878c2ecf20Sopenharmony_ci unsigned offset) 888c2ecf20Sopenharmony_ci{ 898c2ecf20Sopenharmony_ci struct meson_pinctrl *pc = pinctrl_dev_get_drvdata(pcdev); 908c2ecf20Sopenharmony_ci 918c2ecf20Sopenharmony_ci meson8_pmx_disable_other_groups(pc, offset, -1); 928c2ecf20Sopenharmony_ci 938c2ecf20Sopenharmony_ci return 0; 948c2ecf20Sopenharmony_ci} 958c2ecf20Sopenharmony_ci 968c2ecf20Sopenharmony_ciconst struct pinmux_ops meson8_pmx_ops = { 978c2ecf20Sopenharmony_ci .set_mux = meson8_pmx_set_mux, 988c2ecf20Sopenharmony_ci .get_functions_count = meson_pmx_get_funcs_count, 998c2ecf20Sopenharmony_ci .get_function_name = meson_pmx_get_func_name, 1008c2ecf20Sopenharmony_ci .get_function_groups = meson_pmx_get_groups, 1018c2ecf20Sopenharmony_ci .gpio_request_enable = meson8_pmx_request_gpio, 1028c2ecf20Sopenharmony_ci}; 103