1 /*
2  * Copyright (c) 2019, Henrik Brix Andersen <henrik@brixandersen.dk>
3  *
4  * Based on the i2c_mcux_lpi2c.c driver, which is:
5  * Copyright (c) 2016 Freescale Semiconductor, Inc.
6  * Copyright (c) 2019, NXP
7  *
8  * SPDX-License-Identifier: Apache-2.0
9  */
10 
11 #define DT_DRV_COMPAT openisa_rv32m1_lpi2c
12 
13 #include <zephyr/drivers/i2c.h>
14 #include <zephyr/drivers/clock_control.h>
15 #include <zephyr/irq.h>
16 #include <fsl_lpi2c.h>
17 #include <zephyr/logging/log.h>
18 #include <soc.h>
19 #include <zephyr/drivers/pinctrl.h>
20 
21 LOG_MODULE_REGISTER(rv32m1_lpi2c);
22 
23 #include "i2c-priv.h"
24 
25 struct rv32m1_lpi2c_config {
26 	LPI2C_Type *base;
27 	const struct device *clock_dev;
28 	clock_control_subsys_t clock_subsys;
29 	clock_ip_name_t clock_ip_name;
30 	uint32_t clock_ip_src;
31 	uint32_t bitrate;
32 	void (*irq_config_func)(const struct device *dev);
33 	const struct pinctrl_dev_config *pincfg;
34 };
35 
36 struct rv32m1_lpi2c_data {
37 	lpi2c_master_handle_t handle;
38 	struct k_sem transfer_sync;
39 	struct k_sem completion_sync;
40 	status_t completion_status;
41 };
42 
rv32m1_lpi2c_configure(const struct device * dev,uint32_t dev_config)43 static int rv32m1_lpi2c_configure(const struct device *dev,
44 				  uint32_t dev_config)
45 {
46 	const struct rv32m1_lpi2c_config *config = dev->config;
47 	uint32_t baudrate;
48 	uint32_t clk_freq;
49 	int err;
50 
51 	if (!(I2C_MODE_CONTROLLER & dev_config)) {
52 		/* Slave mode not supported - yet */
53 		LOG_ERR("Slave mode not supported");
54 		return -ENOTSUP;
55 	}
56 
57 	if (I2C_ADDR_10_BITS & dev_config) {
58 		/* FSL LPI2C driver only supports 7-bit addressing */
59 		LOG_ERR("10 bit addressing not supported");
60 		return -ENOTSUP;
61 	}
62 
63 	switch (I2C_SPEED_GET(dev_config)) {
64 	case I2C_SPEED_STANDARD:
65 		baudrate = KHZ(100);
66 		break;
67 	case I2C_SPEED_FAST:
68 		baudrate = KHZ(400);
69 		break;
70 	case I2C_SPEED_FAST_PLUS:
71 		baudrate = MHZ(1);
72 		break;
73 	/* TODO: only if SCL pin implements current source pull-up */
74 	/* case I2C_SPEED_HIGH: */
75 	/*      baudrate = KHZ(3400); */
76 	/*      break; */
77 	/* TODO: ultra-fast requires pin_config setting */
78 	/* case I2C_SPEED_ULTRA: */
79 	/*      baudrate = MHZ(5); */
80 	/*      break; */
81 	default:
82 		LOG_ERR("Unsupported speed");
83 		return -ENOTSUP;
84 	}
85 
86 	err = clock_control_get_rate(config->clock_dev, config->clock_subsys, &clk_freq);
87 	if (err) {
88 		LOG_ERR("Could not get clock frequency (err %d)", err);
89 		return -EINVAL;
90 	}
91 
92 	LPI2C_MasterSetBaudRate(config->base, clk_freq, baudrate);
93 
94 	return 0;
95 }
96 
rv32m1_lpi2c_master_transfer_callback(LPI2C_Type * base,lpi2c_master_handle_t * handle,status_t completionStatus,void * userData)97 static void rv32m1_lpi2c_master_transfer_callback(LPI2C_Type *base,
98 						  lpi2c_master_handle_t *handle,
99 						  status_t completionStatus,
100 						  void *userData)
101 {
102 	struct rv32m1_lpi2c_data *data = userData;
103 
104 	ARG_UNUSED(base);
105 	ARG_UNUSED(handle);
106 
107 	data->completion_status = completionStatus;
108 	k_sem_give(&data->completion_sync);
109 }
110 
rv32m1_lpi2c_convert_flags(int msg_flags)111 static uint32_t rv32m1_lpi2c_convert_flags(int msg_flags)
112 {
113 	uint32_t flags = 0U;
114 
115 	if (!(msg_flags & I2C_MSG_STOP)) {
116 		flags |= kLPI2C_TransferNoStopFlag;
117 	}
118 
119 	if (msg_flags & I2C_MSG_RESTART) {
120 		flags |= kLPI2C_TransferRepeatedStartFlag;
121 	}
122 
123 	return flags;
124 }
125 
rv32m1_lpi2c_transfer(const struct device * dev,struct i2c_msg * msgs,uint8_t num_msgs,uint16_t addr)126 static int rv32m1_lpi2c_transfer(const struct device *dev,
127 				 struct i2c_msg *msgs,
128 				 uint8_t num_msgs, uint16_t addr)
129 {
130 	const struct rv32m1_lpi2c_config *config = dev->config;
131 	struct rv32m1_lpi2c_data *data = dev->data;
132 	lpi2c_master_transfer_t transfer;
133 	status_t status;
134 	int ret = 0;
135 
136 	k_sem_take(&data->transfer_sync, K_FOREVER);
137 
138 	/* Iterate over all the messages */
139 	for (int i = 0; i < num_msgs; i++) {
140 		if (I2C_MSG_ADDR_10_BITS & msgs->flags) {
141 			ret = -ENOTSUP;
142 			goto out;
143 		}
144 
145 		/* Initialize the transfer descriptor */
146 		transfer.flags = rv32m1_lpi2c_convert_flags(msgs->flags);
147 
148 		/* Prevent the controller to send a start condition between
149 		 * messages, except if explicitly requested.
150 		 */
151 		if (i != 0 && !(msgs->flags & I2C_MSG_RESTART)) {
152 			transfer.flags |= kLPI2C_TransferNoStartFlag;
153 		}
154 
155 		transfer.slaveAddress = addr;
156 		transfer.direction = (msgs->flags & I2C_MSG_READ)
157 			? kLPI2C_Read : kLPI2C_Write;
158 		transfer.subaddress = 0;
159 		transfer.subaddressSize = 0;
160 		transfer.data = msgs->buf;
161 		transfer.dataSize = msgs->len;
162 
163 		/* Start the transfer */
164 		status = LPI2C_MasterTransferNonBlocking(config->base,
165 							 &data->handle,
166 							 &transfer);
167 
168 		/* Return an error if the transfer didn't start successfully
169 		 * e.g., if the bus was busy
170 		 */
171 		if (status != kStatus_Success) {
172 			LOG_DBG("Could not start transfer (status %d)", status);
173 			ret = -EIO;
174 			goto out;
175 		}
176 
177 		/* Wait for the transfer to complete */
178 		k_sem_take(&data->completion_sync, K_FOREVER);
179 
180 		/* Return an error if the transfer didn't complete
181 		 * successfully. e.g., nak, timeout, lost arbitration
182 		 */
183 		if (data->completion_status != kStatus_Success) {
184 			LOG_DBG("Transfer failed (status %d)",
185 				data->completion_status);
186 			LPI2C_MasterTransferAbort(config->base, &data->handle);
187 			ret = -EIO;
188 			goto out;
189 		}
190 
191 		/* Move to the next message */
192 		msgs++;
193 	}
194 
195 out:
196 	k_sem_give(&data->transfer_sync);
197 	return ret;
198 }
199 
rv32m1_lpi2c_isr(const struct device * dev)200 static void rv32m1_lpi2c_isr(const struct device *dev)
201 {
202 	const struct rv32m1_lpi2c_config *config = dev->config;
203 	struct rv32m1_lpi2c_data *data = dev->data;
204 
205 	LPI2C_MasterTransferHandleIRQ(config->base, &data->handle);
206 }
207 
rv32m1_lpi2c_init(const struct device * dev)208 static int rv32m1_lpi2c_init(const struct device *dev)
209 {
210 	const struct rv32m1_lpi2c_config *config = dev->config;
211 	struct rv32m1_lpi2c_data *data = dev->data;
212 	lpi2c_master_config_t master_config;
213 	uint32_t clk_freq, dev_cfg;
214 	int err;
215 
216 	CLOCK_SetIpSrc(config->clock_ip_name, config->clock_ip_src);
217 
218 	if (!device_is_ready(config->clock_dev)) {
219 		LOG_ERR("clock control device not ready");
220 		return -ENODEV;
221 	}
222 
223 	err = clock_control_on(config->clock_dev, config->clock_subsys);
224 	if (err) {
225 		LOG_ERR("Could not turn on clock (err %d)", err);
226 		return -EINVAL;
227 	}
228 
229 	err = clock_control_get_rate(config->clock_dev, config->clock_subsys, &clk_freq);
230 	if (err) {
231 		LOG_ERR("Could not get clock frequency (err %d)", err);
232 		return -EINVAL;
233 	}
234 
235 	LPI2C_MasterGetDefaultConfig(&master_config);
236 	LPI2C_MasterInit(config->base, &master_config, clk_freq);
237 	LPI2C_MasterTransferCreateHandle(config->base, &data->handle,
238 					 rv32m1_lpi2c_master_transfer_callback,
239 					 data);
240 
241 	dev_cfg = i2c_map_dt_bitrate(config->bitrate);
242 	err = rv32m1_lpi2c_configure(dev, dev_cfg | I2C_MODE_CONTROLLER);
243 	if (err) {
244 		LOG_ERR("Could not configure controller (err %d)", err);
245 		return err;
246 	}
247 
248 	err = pinctrl_apply_state(config->pincfg, PINCTRL_STATE_DEFAULT);
249 	if (err != 0) {
250 		return err;
251 	}
252 
253 	config->irq_config_func(dev);
254 
255 	return 0;
256 }
257 
258 static DEVICE_API(i2c, rv32m1_lpi2c_driver_api) = {
259 	.configure = rv32m1_lpi2c_configure,
260 	.transfer = rv32m1_lpi2c_transfer,
261 #ifdef CONFIG_I2C_RTIO
262 	.iodev_submit = i2c_iodev_submit_fallback,
263 #endif
264 };
265 
266 #define RV32M1_LPI2C_DEVICE(id)                                                \
267 	PINCTRL_DT_INST_DEFINE(id);					       \
268 	static void rv32m1_lpi2c_irq_config_func_##id(const struct device *dev);     \
269 	static const struct rv32m1_lpi2c_config rv32m1_lpi2c_##id##_config = { \
270 		.base =                                                        \
271 		(LPI2C_Type *)DT_INST_REG_ADDR(id),                            \
272 		.clock_dev = DEVICE_DT_GET(DT_INST_CLOCKS_CTLR(id)),	       \
273 		.clock_subsys =                                                \
274 			(clock_control_subsys_t) DT_INST_CLOCKS_CELL(id, name),\
275 		.clock_ip_name = INST_DT_CLOCK_IP_NAME(id),                    \
276 		.clock_ip_src  = kCLOCK_IpSrcFircAsync,                        \
277 		.bitrate = DT_INST_PROP(id, clock_frequency),                  \
278 		.irq_config_func = rv32m1_lpi2c_irq_config_func_##id,          \
279 		.pincfg = PINCTRL_DT_INST_DEV_CONFIG_GET(id),		       \
280 	};                                                                     \
281 	static struct rv32m1_lpi2c_data rv32m1_lpi2c_##id##_data = {           \
282 		.transfer_sync = Z_SEM_INITIALIZER(                            \
283 			rv32m1_lpi2c_##id##_data.transfer_sync, 1, 1),         \
284 		.completion_sync = Z_SEM_INITIALIZER(                          \
285 			rv32m1_lpi2c_##id##_data.completion_sync, 0, 1),       \
286 	};                                                                     \
287 	I2C_DEVICE_DT_INST_DEFINE(id,                                          \
288 			    rv32m1_lpi2c_init,                                 \
289 			    NULL,                                              \
290 			    &rv32m1_lpi2c_##id##_data,                         \
291 			    &rv32m1_lpi2c_##id##_config,                       \
292 			    POST_KERNEL, CONFIG_I2C_INIT_PRIORITY,             \
293 			    &rv32m1_lpi2c_driver_api);	                       \
294 	static void rv32m1_lpi2c_irq_config_func_##id(const struct device *dev)      \
295 	{                                                                      \
296 		IRQ_CONNECT(DT_INST_IRQN(id),                                  \
297 			    0,						       \
298 			    rv32m1_lpi2c_isr, DEVICE_DT_INST_GET(id),	       \
299 			    0);                                                \
300 		irq_enable(DT_INST_IRQN(id));                                  \
301 	}                                                                      \
302 
303 DT_INST_FOREACH_STATUS_OKAY(RV32M1_LPI2C_DEVICE)
304