xref: /rk3399_rockchip-uboot/drivers/power/pmic/as3722_gpio.c (revision 90aa625c9a9e1fb7a2f001fd8e50099bacaf92b8)
1c2012cb4SSimon Glass /*
2c2012cb4SSimon Glass  * Copyright (C) 2014 NVIDIA Corporation
3c2012cb4SSimon Glass  *
4c2012cb4SSimon Glass  * SPDX-License-Identifier: GPL-2.0+
5c2012cb4SSimon Glass  */
6c2012cb4SSimon Glass 
7c2012cb4SSimon Glass #include <common.h>
8c2012cb4SSimon Glass #include <dm.h>
9c2012cb4SSimon Glass #include <asm/gpio.h>
10c2012cb4SSimon Glass #include <power/as3722.h>
11c2012cb4SSimon Glass #include <power/pmic.h>
12c2012cb4SSimon Glass 
13c2012cb4SSimon Glass #define NUM_GPIOS	8
14c2012cb4SSimon Glass 
as3722_gpio_configure(struct udevice * pmic,unsigned int gpio,unsigned long flags)15c2012cb4SSimon Glass int as3722_gpio_configure(struct udevice *pmic, unsigned int gpio,
16c2012cb4SSimon Glass 			  unsigned long flags)
17c2012cb4SSimon Glass {
18c2012cb4SSimon Glass 	u8 value = 0;
19c2012cb4SSimon Glass 	int err;
20c2012cb4SSimon Glass 
21c2012cb4SSimon Glass 	if (flags & AS3722_GPIO_OUTPUT_VDDH)
22c2012cb4SSimon Glass 		value |= AS3722_GPIO_CONTROL_MODE_OUTPUT_VDDH;
23c2012cb4SSimon Glass 
24c2012cb4SSimon Glass 	if (flags & AS3722_GPIO_INVERT)
25c2012cb4SSimon Glass 		value |= AS3722_GPIO_CONTROL_INVERT;
26c2012cb4SSimon Glass 
27c2012cb4SSimon Glass 	err = pmic_reg_write(pmic, AS3722_GPIO_CONTROL(gpio), value);
28c2012cb4SSimon Glass 	if (err) {
29*90aa625cSMasahiro Yamada 		pr_err("failed to configure GPIO#%u: %d", gpio, err);
30c2012cb4SSimon Glass 		return err;
31c2012cb4SSimon Glass 	}
32c2012cb4SSimon Glass 
33c2012cb4SSimon Glass 	return 0;
34c2012cb4SSimon Glass }
35c2012cb4SSimon Glass 
as3722_gpio_set_value(struct udevice * dev,unsigned int gpio,int level)36c2012cb4SSimon Glass static int as3722_gpio_set_value(struct udevice *dev, unsigned int gpio,
37c2012cb4SSimon Glass 				 int level)
38c2012cb4SSimon Glass {
39c2012cb4SSimon Glass 	struct udevice *pmic = dev_get_parent(dev);
40c2012cb4SSimon Glass 	const char *l;
41c2012cb4SSimon Glass 	u8 value;
42c2012cb4SSimon Glass 	int err;
43c2012cb4SSimon Glass 
44c2012cb4SSimon Glass 	if (gpio >= NUM_GPIOS)
45c2012cb4SSimon Glass 		return -EINVAL;
46c2012cb4SSimon Glass 
47c2012cb4SSimon Glass 	err = pmic_reg_read(pmic, AS3722_GPIO_SIGNAL_OUT);
48c2012cb4SSimon Glass 	if (err < 0) {
49*90aa625cSMasahiro Yamada 		pr_err("failed to read GPIO signal out register: %d", err);
50c2012cb4SSimon Glass 		return err;
51c2012cb4SSimon Glass 	}
52c2012cb4SSimon Glass 	value = err;
53c2012cb4SSimon Glass 
54c2012cb4SSimon Glass 	if (level == 0) {
55c2012cb4SSimon Glass 		value &= ~(1 << gpio);
56c2012cb4SSimon Glass 		l = "low";
57c2012cb4SSimon Glass 	} else {
58c2012cb4SSimon Glass 		value |= 1 << gpio;
59c2012cb4SSimon Glass 		l = "high";
60c2012cb4SSimon Glass 	}
61c2012cb4SSimon Glass 
62c2012cb4SSimon Glass 	err = pmic_reg_write(pmic, AS3722_GPIO_SIGNAL_OUT, value);
63c2012cb4SSimon Glass 	if (err) {
64*90aa625cSMasahiro Yamada 		pr_err("failed to set GPIO#%u %s: %d", gpio, l, err);
65c2012cb4SSimon Glass 		return err;
66c2012cb4SSimon Glass 	}
67c2012cb4SSimon Glass 
68c2012cb4SSimon Glass 	return 0;
69c2012cb4SSimon Glass }
70c2012cb4SSimon Glass 
as3722_gpio_direction_output(struct udevice * dev,unsigned int gpio,int value)71c2012cb4SSimon Glass int as3722_gpio_direction_output(struct udevice *dev, unsigned int gpio,
72c2012cb4SSimon Glass 				 int value)
73c2012cb4SSimon Glass {
74c2012cb4SSimon Glass 	struct udevice *pmic = dev_get_parent(dev);
75c2012cb4SSimon Glass 	int err;
76c2012cb4SSimon Glass 
77c2012cb4SSimon Glass 	if (gpio > 7)
78c2012cb4SSimon Glass 		return -EINVAL;
79c2012cb4SSimon Glass 
80c2012cb4SSimon Glass 	if (value == 0)
81c2012cb4SSimon Glass 		value = AS3722_GPIO_CONTROL_MODE_OUTPUT_VDDL;
82c2012cb4SSimon Glass 	else
83c2012cb4SSimon Glass 		value = AS3722_GPIO_CONTROL_MODE_OUTPUT_VDDH;
84c2012cb4SSimon Glass 
85c2012cb4SSimon Glass 	err = pmic_reg_write(pmic, AS3722_GPIO_CONTROL(gpio), value);
86c2012cb4SSimon Glass 	if (err) {
87*90aa625cSMasahiro Yamada 		pr_err("failed to configure GPIO#%u as output: %d", gpio, err);
88c2012cb4SSimon Glass 		return err;
89c2012cb4SSimon Glass 	}
90c2012cb4SSimon Glass 
91c2012cb4SSimon Glass 	err = as3722_gpio_set_value(pmic, gpio, value);
92c2012cb4SSimon Glass 	if (err < 0) {
93*90aa625cSMasahiro Yamada 		pr_err("failed to set GPIO#%u high: %d", gpio, err);
94c2012cb4SSimon Glass 		return err;
95c2012cb4SSimon Glass 	}
96c2012cb4SSimon Glass 
97c2012cb4SSimon Glass 	return 0;
98c2012cb4SSimon Glass }
99c2012cb4SSimon Glass 
as3722_gpio_probe(struct udevice * dev)100c2012cb4SSimon Glass static int as3722_gpio_probe(struct udevice *dev)
101c2012cb4SSimon Glass {
102c2012cb4SSimon Glass 	struct gpio_dev_priv *uc_priv = dev_get_uclass_priv(dev);
103c2012cb4SSimon Glass 
104c2012cb4SSimon Glass 	uc_priv->gpio_count = NUM_GPIOS;
105c2012cb4SSimon Glass 	uc_priv->bank_name = "as3722_";
106c2012cb4SSimon Glass 
107c2012cb4SSimon Glass 	return 0;
108c2012cb4SSimon Glass }
109c2012cb4SSimon Glass 
110c2012cb4SSimon Glass static const struct dm_gpio_ops gpio_as3722_ops = {
111c2012cb4SSimon Glass 	.direction_output	= as3722_gpio_direction_output,
112c2012cb4SSimon Glass 	.set_value		= as3722_gpio_set_value,
113c2012cb4SSimon Glass };
114c2012cb4SSimon Glass 
115c2012cb4SSimon Glass U_BOOT_DRIVER(gpio_as3722) = {
116c2012cb4SSimon Glass 	.name	= "gpio_as3722",
117c2012cb4SSimon Glass 	.id	= UCLASS_GPIO,
118c2012cb4SSimon Glass 	.ops	= &gpio_as3722_ops,
119c2012cb4SSimon Glass 	.probe	= as3722_gpio_probe,
120c2012cb4SSimon Glass };
121