xref: /openbmc/u-boot/drivers/pinctrl/meson/pinctrl-meson-gx-pmx.c (revision 5548c7a165bbd1ebb0d4599ef1d04751c37b3f79)
17c9dcfedSJerome Brunet // SPDX-License-Identifier: GPL-2.0+
27c9dcfedSJerome Brunet /*
37c9dcfedSJerome Brunet  * (C) Copyright 2016 - Beniamino Galvani <b.galvani@gmail.com>
47c9dcfedSJerome Brunet  */
57c9dcfedSJerome Brunet 
67c9dcfedSJerome Brunet #include <asm/gpio.h>
77c9dcfedSJerome Brunet #include <common.h>
87c9dcfedSJerome Brunet #include <dm.h>
97c9dcfedSJerome Brunet #include <dm/pinctrl.h>
107c9dcfedSJerome Brunet #include <linux/io.h>
117c9dcfedSJerome Brunet #include "pinctrl-meson-gx.h"
127c9dcfedSJerome Brunet 
meson_gx_pinmux_disable_other_groups(struct meson_pinctrl * priv,unsigned int pin,int sel_group)137c9dcfedSJerome Brunet static void meson_gx_pinmux_disable_other_groups(struct meson_pinctrl *priv,
147c9dcfedSJerome Brunet 						 unsigned int pin,
157c9dcfedSJerome Brunet 						 int sel_group)
167c9dcfedSJerome Brunet {
177c9dcfedSJerome Brunet 	struct meson_pmx_group *group;
187c9dcfedSJerome Brunet 	struct meson_gx_pmx_data *pmx_data;
197c9dcfedSJerome Brunet 	void __iomem *addr;
207c9dcfedSJerome Brunet 	int i, j;
217c9dcfedSJerome Brunet 
227c9dcfedSJerome Brunet 	for (i = 0; i < priv->data->num_groups; i++) {
237c9dcfedSJerome Brunet 		group = &priv->data->groups[i];
247c9dcfedSJerome Brunet 		pmx_data = (struct meson_gx_pmx_data *)group->data;
257c9dcfedSJerome Brunet 		if (pmx_data->is_gpio || i == sel_group)
267c9dcfedSJerome Brunet 			continue;
277c9dcfedSJerome Brunet 
287c9dcfedSJerome Brunet 		for (j = 0; j < group->num_pins; j++) {
297c9dcfedSJerome Brunet 			if (group->pins[j] == pin) {
307c9dcfedSJerome Brunet 				/* We have found a group using the pin */
317c9dcfedSJerome Brunet 				debug("pinmux: disabling %s\n", group->name);
327c9dcfedSJerome Brunet 				addr = priv->reg_mux + pmx_data->reg * 4;
337c9dcfedSJerome Brunet 				writel(readl(addr) & ~BIT(pmx_data->bit), addr);
347c9dcfedSJerome Brunet 			}
357c9dcfedSJerome Brunet 		}
367c9dcfedSJerome Brunet 	}
377c9dcfedSJerome Brunet }
387c9dcfedSJerome Brunet 
meson_gx_pinmux_group_set(struct udevice * dev,unsigned int group_selector,unsigned int func_selector)397c9dcfedSJerome Brunet static int meson_gx_pinmux_group_set(struct udevice *dev,
407c9dcfedSJerome Brunet 				     unsigned int group_selector,
417c9dcfedSJerome Brunet 				     unsigned int func_selector)
427c9dcfedSJerome Brunet {
437c9dcfedSJerome Brunet 	struct meson_pinctrl *priv = dev_get_priv(dev);
447c9dcfedSJerome Brunet 	const struct meson_pmx_group *group;
457c9dcfedSJerome Brunet 	const struct meson_pmx_func *func;
467c9dcfedSJerome Brunet 	struct meson_gx_pmx_data *pmx_data;
477c9dcfedSJerome Brunet 	void __iomem *addr;
487c9dcfedSJerome Brunet 	int i;
497c9dcfedSJerome Brunet 
507c9dcfedSJerome Brunet 	group = &priv->data->groups[group_selector];
517c9dcfedSJerome Brunet 	pmx_data = (struct meson_gx_pmx_data *)group->data;
527c9dcfedSJerome Brunet 	func = &priv->data->funcs[func_selector];
537c9dcfedSJerome Brunet 
547c9dcfedSJerome Brunet 	debug("pinmux: set group %s func %s\n", group->name, func->name);
557c9dcfedSJerome Brunet 
567c9dcfedSJerome Brunet 	/*
577c9dcfedSJerome Brunet 	 * Disable groups using the same pins.
587c9dcfedSJerome Brunet 	 * The selected group is not disabled to avoid glitches.
597c9dcfedSJerome Brunet 	 */
607c9dcfedSJerome Brunet 	for (i = 0; i < group->num_pins; i++) {
617c9dcfedSJerome Brunet 		meson_gx_pinmux_disable_other_groups(priv,
627c9dcfedSJerome Brunet 						     group->pins[i],
637c9dcfedSJerome Brunet 						     group_selector);
647c9dcfedSJerome Brunet 	}
657c9dcfedSJerome Brunet 
667c9dcfedSJerome Brunet 	/* Function 0 (GPIO) doesn't need any additional setting */
677c9dcfedSJerome Brunet 	if (func_selector) {
687c9dcfedSJerome Brunet 		addr = priv->reg_mux + pmx_data->reg * 4;
697c9dcfedSJerome Brunet 		writel(readl(addr) | BIT(pmx_data->bit), addr);
707c9dcfedSJerome Brunet 	}
717c9dcfedSJerome Brunet 
727c9dcfedSJerome Brunet 	return 0;
737c9dcfedSJerome Brunet }
747c9dcfedSJerome Brunet 
75*c4c726c2SJerome Brunet const struct pinconf_param meson_gx_pinconf_params[] = {
76*c4c726c2SJerome Brunet 	{ "bias-disable", PIN_CONFIG_BIAS_DISABLE, 0 },
77*c4c726c2SJerome Brunet 	{ "bias-pull-up", PIN_CONFIG_BIAS_PULL_UP, 1 },
78*c4c726c2SJerome Brunet 	{ "bias-pull-down", PIN_CONFIG_BIAS_PULL_DOWN, 1 },
79*c4c726c2SJerome Brunet };
80*c4c726c2SJerome Brunet 
817c9dcfedSJerome Brunet const struct pinctrl_ops meson_gx_pinctrl_ops = {
827c9dcfedSJerome Brunet 	.get_groups_count = meson_pinctrl_get_groups_count,
837c9dcfedSJerome Brunet 	.get_group_name = meson_pinctrl_get_group_name,
847c9dcfedSJerome Brunet 	.get_functions_count = meson_pinmux_get_functions_count,
857c9dcfedSJerome Brunet 	.get_function_name = meson_pinmux_get_function_name,
867c9dcfedSJerome Brunet 	.pinmux_group_set = meson_gx_pinmux_group_set,
877c9dcfedSJerome Brunet 	.set_state = pinctrl_generic_set_state,
88*c4c726c2SJerome Brunet 	.pinconf_params = meson_gx_pinconf_params,
89*c4c726c2SJerome Brunet 	.pinconf_num_params = ARRAY_SIZE(meson_gx_pinconf_params),
90*c4c726c2SJerome Brunet 	.pinconf_set = meson_pinconf_set,
91*c4c726c2SJerome Brunet 	.pinconf_group_set = meson_pinconf_group_set,
927c9dcfedSJerome Brunet };
937c9dcfedSJerome Brunet 
947c9dcfedSJerome Brunet static const struct dm_gpio_ops meson_gx_gpio_ops = {
957c9dcfedSJerome Brunet 	.set_value = meson_gpio_set,
967c9dcfedSJerome Brunet 	.get_value = meson_gpio_get,
977c9dcfedSJerome Brunet 	.get_function = meson_gpio_get_direction,
987c9dcfedSJerome Brunet 	.direction_input = meson_gpio_direction_input,
997c9dcfedSJerome Brunet 	.direction_output = meson_gpio_direction_output,
1007c9dcfedSJerome Brunet };
1017c9dcfedSJerome Brunet 
1027c9dcfedSJerome Brunet const struct driver meson_gx_gpio_driver = {
1037c9dcfedSJerome Brunet 	.name	= "meson-gx-gpio",
1047c9dcfedSJerome Brunet 	.id	= UCLASS_GPIO,
1057c9dcfedSJerome Brunet 	.probe	= meson_gpio_probe,
1067c9dcfedSJerome Brunet 	.ops	= &meson_gx_gpio_ops,
1077c9dcfedSJerome Brunet };
108