1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3  * Copyright 2023 CR GROUP France
4  * Christophe Leroy <christophe.leroy@csgroup.eu>
5  */
6 
7 #include <common.h>
8 #include <dm.h>
9 #include <mapmem.h>
10 #include <asm/gpio.h>
11 #include <asm/immap_83xx.h>
12 #include <asm/io.h>
13 #include <dm/of_access.h>
14 
15 #define QE_DIR_NONE	0
16 #define QE_DIR_OUT	1
17 #define QE_DIR_IN	2
18 #define QE_DIR_IN_OUT	3
19 
20 struct qe_gpio_data {
21 	/* The bank's register base in memory */
22 	struct gpio_n __iomem *base;
23 	/* The address of the registers; used to identify the bank */
24 	phys_addr_t addr;
25 };
26 
gpio_mask(uint gpio)27 static inline u32 gpio_mask(uint gpio)
28 {
29 	return 1U << (31 - (gpio));
30 }
31 
gpio_mask2(uint gpio)32 static inline u32 gpio_mask2(uint gpio)
33 {
34 	return 1U << (30 - ((gpio & 15) << 1));
35 }
36 
qe_gpio_direction_input(struct udevice * dev,uint gpio)37 static int qe_gpio_direction_input(struct udevice *dev, uint gpio)
38 {
39 	struct qe_gpio_data *data = dev_get_priv(dev);
40 	struct gpio_n __iomem *base = data->base;
41 	u32 mask2 = gpio_mask2(gpio);
42 
43 	if (gpio < 16)
44 		clrsetbits_be32(&base->dir1, mask2 * QE_DIR_OUT, mask2 * QE_DIR_IN);
45 	else
46 		clrsetbits_be32(&base->dir2, mask2 * QE_DIR_OUT, mask2 * QE_DIR_IN);
47 
48 	return 0;
49 }
50 
qe_gpio_set_value(struct udevice * dev,uint gpio,int value)51 static int qe_gpio_set_value(struct udevice *dev, uint gpio, int value)
52 {
53 	struct qe_gpio_data *data = dev_get_priv(dev);
54 	struct gpio_n __iomem *base = data->base;
55 	u32 mask = gpio_mask(gpio);
56 	u32 mask2 = gpio_mask2(gpio);
57 
58 	if (gpio < 16)
59 		clrsetbits_be32(&base->dir1, mask2 * QE_DIR_IN, mask2 * QE_DIR_OUT);
60 	else
61 		clrsetbits_be32(&base->dir2, mask2 * QE_DIR_IN, mask2 * QE_DIR_OUT);
62 
63 	if (value)
64 		setbits_be32(&base->pdat, mask);
65 	else
66 		clrbits_be32(&base->pdat, mask);
67 
68 	return 0;
69 }
70 
qe_gpio_get_value(struct udevice * dev,uint gpio)71 static int qe_gpio_get_value(struct udevice *dev, uint gpio)
72 {
73 	struct qe_gpio_data *data = dev_get_priv(dev);
74 	struct gpio_n __iomem *base = data->base;
75 	u32 mask = gpio_mask(gpio);
76 
77 	return !!(in_be32(&base->pdat) & mask);
78 }
79 
qe_gpio_get_function(struct udevice * dev,uint gpio)80 static int qe_gpio_get_function(struct udevice *dev, uint gpio)
81 {
82 	struct qe_gpio_data *data = dev_get_priv(dev);
83 	struct gpio_n __iomem *base = data->base;
84 	u32 mask2 = gpio_mask2(gpio);
85 	int dir;
86 
87 	if (gpio < 16)
88 		dir = in_be32(&base->dir1);
89 	else
90 		dir = in_be32(&base->dir2);
91 
92 	if ((dir & (mask2 * QE_DIR_IN_OUT)) == (mask2 & QE_DIR_IN))
93 		return GPIOF_INPUT;
94 	else if ((dir & (mask2 * QE_DIR_IN_OUT)) == (mask2 & QE_DIR_OUT))
95 		return GPIOF_OUTPUT;
96 	else
97 		return GPIOF_UNKNOWN;
98 }
99 
qe_gpio_of_to_plat(struct udevice * dev)100 static int qe_gpio_of_to_plat(struct udevice *dev)
101 {
102 	struct qe_gpio_plat *plat = dev_get_plat(dev);
103 
104 	plat->addr = dev_read_addr_size_index(dev, 0, (fdt_size_t *)&plat->size);
105 
106 	return 0;
107 }
108 
qe_gpio_plat_to_priv(struct udevice * dev)109 static int qe_gpio_plat_to_priv(struct udevice *dev)
110 {
111 	struct qe_gpio_data *priv = dev_get_priv(dev);
112 	struct qe_gpio_plat *plat = dev_get_plat(dev);
113 	unsigned long size = plat->size;
114 
115 	if (size == 0)
116 		size = sizeof(struct gpio_n);
117 
118 	priv->addr = plat->addr;
119 	priv->base = (void __iomem *)plat->addr;
120 
121 	if (!priv->base)
122 		return -ENOMEM;
123 
124 	return 0;
125 }
126 
qe_gpio_probe(struct udevice * dev)127 static int qe_gpio_probe(struct udevice *dev)
128 {
129 	struct gpio_dev_priv *uc_priv = dev_get_uclass_priv(dev);
130 	struct qe_gpio_data *data = dev_get_priv(dev);
131 	char name[32], *str;
132 
133 	qe_gpio_plat_to_priv(dev);
134 
135 	snprintf(name, sizeof(name), "QE@%.8llx",
136 		 (unsigned long long)data->addr);
137 	str = strdup(name);
138 
139 	if (!str)
140 		return -ENOMEM;
141 
142 	uc_priv->bank_name = str;
143 	uc_priv->gpio_count = 32;
144 
145 	return 0;
146 }
147 
148 static const struct dm_gpio_ops gpio_qe_ops = {
149 	.direction_input	= qe_gpio_direction_input,
150 	.direction_output	= qe_gpio_set_value,
151 	.get_value		= qe_gpio_get_value,
152 	.set_value		= qe_gpio_set_value,
153 	.get_function		= qe_gpio_get_function,
154 };
155 
156 static const struct udevice_id qe_gpio_ids[] = {
157 	{ .compatible = "fsl,mpc8323-qe-pario-bank"},
158 	{ /* sentinel */ }
159 };
160 
161 U_BOOT_DRIVER(gpio_qe) = {
162 	.name	= "gpio_qe",
163 	.id	= UCLASS_GPIO,
164 	.ops	= &gpio_qe_ops,
165 	.of_to_plat = qe_gpio_of_to_plat,
166 	.plat_auto	= sizeof(struct qe_gpio_plat),
167 	.of_match = qe_gpio_ids,
168 	.probe	= qe_gpio_probe,
169 	.priv_auto	= sizeof(struct qe_gpio_data),
170 };
171