1a4422ff2SBryan O'Donoghue // SPDX-License-Identifier: GPL-2.0
2a4422ff2SBryan O'Donoghue /*
3a4422ff2SBryan O'Donoghue  * Copyright (c) 2023, Linaro Ltd. All rights reserved.
4a4422ff2SBryan O'Donoghue  */
5a4422ff2SBryan O'Donoghue 
6a4422ff2SBryan O'Donoghue #include <linux/err.h>
7a4422ff2SBryan O'Donoghue #include <linux/interrupt.h>
8a4422ff2SBryan O'Donoghue #include <linux/kernel.h>
9a4422ff2SBryan O'Donoghue #include <linux/mod_devicetable.h>
10a4422ff2SBryan O'Donoghue #include <linux/module.h>
11a4422ff2SBryan O'Donoghue #include <linux/platform_device.h>
12a4422ff2SBryan O'Donoghue #include <linux/regmap.h>
13a4422ff2SBryan O'Donoghue #include <linux/regulator/consumer.h>
14a4422ff2SBryan O'Donoghue #include <linux/slab.h>
15a4422ff2SBryan O'Donoghue #include <linux/usb/pd.h>
16a4422ff2SBryan O'Donoghue #include <linux/usb/tcpm.h>
17a4422ff2SBryan O'Donoghue #include "qcom_pmic_typec_pdphy.h"
18a4422ff2SBryan O'Donoghue 
19a4422ff2SBryan O'Donoghue struct pmic_typec_pdphy_irq_data {
20a4422ff2SBryan O'Donoghue 	int				virq;
21a4422ff2SBryan O'Donoghue 	int				irq;
22a4422ff2SBryan O'Donoghue 	struct pmic_typec_pdphy		*pmic_typec_pdphy;
23a4422ff2SBryan O'Donoghue };
24a4422ff2SBryan O'Donoghue 
25a4422ff2SBryan O'Donoghue struct pmic_typec_pdphy {
26a4422ff2SBryan O'Donoghue 	struct device			*dev;
27a4422ff2SBryan O'Donoghue 	struct tcpm_port		*tcpm_port;
28a4422ff2SBryan O'Donoghue 	struct regmap			*regmap;
29a4422ff2SBryan O'Donoghue 	u32				base;
30a4422ff2SBryan O'Donoghue 
31a4422ff2SBryan O'Donoghue 	unsigned int			nr_irqs;
32a4422ff2SBryan O'Donoghue 	struct pmic_typec_pdphy_irq_data	*irq_data;
33a4422ff2SBryan O'Donoghue 
34a4422ff2SBryan O'Donoghue 	struct work_struct		reset_work;
35a4422ff2SBryan O'Donoghue 	struct work_struct		receive_work;
36a4422ff2SBryan O'Donoghue 	struct regulator		*vdd_pdphy;
37a4422ff2SBryan O'Donoghue 	spinlock_t			lock;		/* Register atomicity */
38a4422ff2SBryan O'Donoghue };
39a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_reset_on(struct pmic_typec_pdphy * pmic_typec_pdphy)40a4422ff2SBryan O'Donoghue static void qcom_pmic_typec_pdphy_reset_on(struct pmic_typec_pdphy *pmic_typec_pdphy)
41a4422ff2SBryan O'Donoghue {
42a4422ff2SBryan O'Donoghue 	struct device *dev = pmic_typec_pdphy->dev;
43a4422ff2SBryan O'Donoghue 	int ret;
44a4422ff2SBryan O'Donoghue 
45a4422ff2SBryan O'Donoghue 	/* Terminate TX */
46a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
47a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_TX_CONTROL_REG, 0);
48a4422ff2SBryan O'Donoghue 	if (ret)
49a4422ff2SBryan O'Donoghue 		goto err;
50a4422ff2SBryan O'Donoghue 
51a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
52a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_FRAME_FILTER_REG, 0);
53a4422ff2SBryan O'Donoghue 	if (ret)
54a4422ff2SBryan O'Donoghue 		goto err;
55a4422ff2SBryan O'Donoghue 
56a4422ff2SBryan O'Donoghue 	return;
57a4422ff2SBryan O'Donoghue err:
58a4422ff2SBryan O'Donoghue 	dev_err(dev, "pd_reset_on error\n");
59a4422ff2SBryan O'Donoghue }
60a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_reset_off(struct pmic_typec_pdphy * pmic_typec_pdphy)61a4422ff2SBryan O'Donoghue static void qcom_pmic_typec_pdphy_reset_off(struct pmic_typec_pdphy *pmic_typec_pdphy)
62a4422ff2SBryan O'Donoghue {
63a4422ff2SBryan O'Donoghue 	struct device *dev = pmic_typec_pdphy->dev;
64a4422ff2SBryan O'Donoghue 	int ret;
65a4422ff2SBryan O'Donoghue 
66a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
67a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_FRAME_FILTER_REG,
68a4422ff2SBryan O'Donoghue 			   FRAME_FILTER_EN_SOP | FRAME_FILTER_EN_HARD_RESET);
69a4422ff2SBryan O'Donoghue 	if (ret)
70a4422ff2SBryan O'Donoghue 		dev_err(dev, "pd_reset_off error\n");
71a4422ff2SBryan O'Donoghue }
72a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_sig_reset_work(struct work_struct * work)73a4422ff2SBryan O'Donoghue static void qcom_pmic_typec_pdphy_sig_reset_work(struct work_struct *work)
74a4422ff2SBryan O'Donoghue {
75a4422ff2SBryan O'Donoghue 	struct pmic_typec_pdphy *pmic_typec_pdphy = container_of(work, struct pmic_typec_pdphy,
76a4422ff2SBryan O'Donoghue 						     reset_work);
77a4422ff2SBryan O'Donoghue 	unsigned long flags;
78a4422ff2SBryan O'Donoghue 
79a4422ff2SBryan O'Donoghue 	spin_lock_irqsave(&pmic_typec_pdphy->lock, flags);
80a4422ff2SBryan O'Donoghue 
81a4422ff2SBryan O'Donoghue 	qcom_pmic_typec_pdphy_reset_on(pmic_typec_pdphy);
82a4422ff2SBryan O'Donoghue 	qcom_pmic_typec_pdphy_reset_off(pmic_typec_pdphy);
83a4422ff2SBryan O'Donoghue 
84a4422ff2SBryan O'Donoghue 	spin_unlock_irqrestore(&pmic_typec_pdphy->lock, flags);
85a4422ff2SBryan O'Donoghue 
86a4422ff2SBryan O'Donoghue 	tcpm_pd_hard_reset(pmic_typec_pdphy->tcpm_port);
87a4422ff2SBryan O'Donoghue }
88a4422ff2SBryan O'Donoghue 
89a4422ff2SBryan O'Donoghue static int
qcom_pmic_typec_pdphy_clear_tx_control_reg(struct pmic_typec_pdphy * pmic_typec_pdphy)90a4422ff2SBryan O'Donoghue qcom_pmic_typec_pdphy_clear_tx_control_reg(struct pmic_typec_pdphy *pmic_typec_pdphy)
91a4422ff2SBryan O'Donoghue {
92a4422ff2SBryan O'Donoghue 	struct device *dev = pmic_typec_pdphy->dev;
93a4422ff2SBryan O'Donoghue 	unsigned int val;
94a4422ff2SBryan O'Donoghue 	int ret;
95a4422ff2SBryan O'Donoghue 
96a4422ff2SBryan O'Donoghue 	/* Clear TX control register */
97a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
98a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_TX_CONTROL_REG, 0);
99a4422ff2SBryan O'Donoghue 	if (ret)
100a4422ff2SBryan O'Donoghue 		goto done;
101a4422ff2SBryan O'Donoghue 
102a4422ff2SBryan O'Donoghue 	/* Perform readback to ensure sufficient delay for command to latch */
103a4422ff2SBryan O'Donoghue 	ret = regmap_read(pmic_typec_pdphy->regmap,
104a4422ff2SBryan O'Donoghue 			  pmic_typec_pdphy->base + USB_PDPHY_TX_CONTROL_REG, &val);
105a4422ff2SBryan O'Donoghue 
106a4422ff2SBryan O'Donoghue done:
107a4422ff2SBryan O'Donoghue 	if (ret)
108a4422ff2SBryan O'Donoghue 		dev_err(dev, "pd_clear_tx_control_reg: clear tx flag\n");
109a4422ff2SBryan O'Donoghue 
110a4422ff2SBryan O'Donoghue 	return ret;
111a4422ff2SBryan O'Donoghue }
112a4422ff2SBryan O'Donoghue 
113a4422ff2SBryan O'Donoghue static int
qcom_pmic_typec_pdphy_pd_transmit_signal(struct pmic_typec_pdphy * pmic_typec_pdphy,enum tcpm_transmit_type type,unsigned int negotiated_rev)114a4422ff2SBryan O'Donoghue qcom_pmic_typec_pdphy_pd_transmit_signal(struct pmic_typec_pdphy *pmic_typec_pdphy,
115a4422ff2SBryan O'Donoghue 					 enum tcpm_transmit_type type,
116a4422ff2SBryan O'Donoghue 					 unsigned int negotiated_rev)
117a4422ff2SBryan O'Donoghue {
118a4422ff2SBryan O'Donoghue 	struct device *dev = pmic_typec_pdphy->dev;
119a4422ff2SBryan O'Donoghue 	unsigned int val;
120a4422ff2SBryan O'Donoghue 	unsigned long flags;
121a4422ff2SBryan O'Donoghue 	int ret;
122a4422ff2SBryan O'Donoghue 
123a4422ff2SBryan O'Donoghue 	spin_lock_irqsave(&pmic_typec_pdphy->lock, flags);
124a4422ff2SBryan O'Donoghue 
125a4422ff2SBryan O'Donoghue 	/* Clear TX control register */
126a4422ff2SBryan O'Donoghue 	ret = qcom_pmic_typec_pdphy_clear_tx_control_reg(pmic_typec_pdphy);
127a4422ff2SBryan O'Donoghue 	if (ret)
128a4422ff2SBryan O'Donoghue 		goto done;
129a4422ff2SBryan O'Donoghue 
130a4422ff2SBryan O'Donoghue 	val = TX_CONTROL_SEND_SIGNAL;
131a4422ff2SBryan O'Donoghue 	if (negotiated_rev == PD_REV30)
132a4422ff2SBryan O'Donoghue 		val |= TX_CONTROL_RETRY_COUNT(2);
133a4422ff2SBryan O'Donoghue 	else
134a4422ff2SBryan O'Donoghue 		val |= TX_CONTROL_RETRY_COUNT(3);
135a4422ff2SBryan O'Donoghue 
136a4422ff2SBryan O'Donoghue 	if (type == TCPC_TX_CABLE_RESET || type == TCPC_TX_HARD_RESET)
137a4422ff2SBryan O'Donoghue 		val |= TX_CONTROL_FRAME_TYPE(1);
138a4422ff2SBryan O'Donoghue 
139a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
140a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_TX_CONTROL_REG, val);
141a4422ff2SBryan O'Donoghue 
142a4422ff2SBryan O'Donoghue done:
143a4422ff2SBryan O'Donoghue 	spin_unlock_irqrestore(&pmic_typec_pdphy->lock, flags);
144a4422ff2SBryan O'Donoghue 
145a4422ff2SBryan O'Donoghue 	dev_vdbg(dev, "pd_transmit_signal: type %d negotiate_rev %d send %d\n",
146a4422ff2SBryan O'Donoghue 		 type, negotiated_rev, ret);
147a4422ff2SBryan O'Donoghue 
148a4422ff2SBryan O'Donoghue 	return ret;
149a4422ff2SBryan O'Donoghue }
150a4422ff2SBryan O'Donoghue 
151a4422ff2SBryan O'Donoghue static int
qcom_pmic_typec_pdphy_pd_transmit_payload(struct pmic_typec_pdphy * pmic_typec_pdphy,enum tcpm_transmit_type type,const struct pd_message * msg,unsigned int negotiated_rev)152a4422ff2SBryan O'Donoghue qcom_pmic_typec_pdphy_pd_transmit_payload(struct pmic_typec_pdphy *pmic_typec_pdphy,
153a4422ff2SBryan O'Donoghue 					  enum tcpm_transmit_type type,
154a4422ff2SBryan O'Donoghue 					  const struct pd_message *msg,
155a4422ff2SBryan O'Donoghue 					  unsigned int negotiated_rev)
156a4422ff2SBryan O'Donoghue {
157a4422ff2SBryan O'Donoghue 	struct device *dev = pmic_typec_pdphy->dev;
158a4422ff2SBryan O'Donoghue 	unsigned int val, hdr_len, txbuf_len, txsize_len;
159a4422ff2SBryan O'Donoghue 	unsigned long flags;
160a4422ff2SBryan O'Donoghue 	int ret;
161a4422ff2SBryan O'Donoghue 
162a4422ff2SBryan O'Donoghue 	spin_lock_irqsave(&pmic_typec_pdphy->lock, flags);
163a4422ff2SBryan O'Donoghue 
164a4422ff2SBryan O'Donoghue 	ret = regmap_read(pmic_typec_pdphy->regmap,
165a4422ff2SBryan O'Donoghue 			  pmic_typec_pdphy->base + USB_PDPHY_RX_ACKNOWLEDGE_REG,
166a4422ff2SBryan O'Donoghue 			  &val);
167a4422ff2SBryan O'Donoghue 	if (ret)
168a4422ff2SBryan O'Donoghue 		goto done;
169a4422ff2SBryan O'Donoghue 
170a4422ff2SBryan O'Donoghue 	if (val) {
171a4422ff2SBryan O'Donoghue 		dev_err(dev, "pd_transmit_payload: RX message pending\n");
172a4422ff2SBryan O'Donoghue 		ret = -EBUSY;
173a4422ff2SBryan O'Donoghue 		goto done;
174a4422ff2SBryan O'Donoghue 	}
175a4422ff2SBryan O'Donoghue 
176a4422ff2SBryan O'Donoghue 	/* Clear TX control register */
177a4422ff2SBryan O'Donoghue 	ret = qcom_pmic_typec_pdphy_clear_tx_control_reg(pmic_typec_pdphy);
178a4422ff2SBryan O'Donoghue 	if (ret)
179a4422ff2SBryan O'Donoghue 		goto done;
180a4422ff2SBryan O'Donoghue 
181a4422ff2SBryan O'Donoghue 	hdr_len = sizeof(msg->header);
182a4422ff2SBryan O'Donoghue 	txbuf_len = pd_header_cnt_le(msg->header) * 4;
183a4422ff2SBryan O'Donoghue 	txsize_len = hdr_len + txbuf_len - 1;
184a4422ff2SBryan O'Donoghue 
185a4422ff2SBryan O'Donoghue 	/* Write message header sizeof(u16) to USB_PDPHY_TX_BUFFER_HDR_REG */
186a4422ff2SBryan O'Donoghue 	ret = regmap_bulk_write(pmic_typec_pdphy->regmap,
187a4422ff2SBryan O'Donoghue 				pmic_typec_pdphy->base + USB_PDPHY_TX_BUFFER_HDR_REG,
188a4422ff2SBryan O'Donoghue 				&msg->header, hdr_len);
189a4422ff2SBryan O'Donoghue 	if (ret)
190a4422ff2SBryan O'Donoghue 		goto done;
191a4422ff2SBryan O'Donoghue 
192a4422ff2SBryan O'Donoghue 	/* Write payload to USB_PDPHY_TX_BUFFER_DATA_REG for txbuf_len */
193a4422ff2SBryan O'Donoghue 	if (txbuf_len) {
194a4422ff2SBryan O'Donoghue 		ret = regmap_bulk_write(pmic_typec_pdphy->regmap,
195a4422ff2SBryan O'Donoghue 					pmic_typec_pdphy->base + USB_PDPHY_TX_BUFFER_DATA_REG,
196a4422ff2SBryan O'Donoghue 					&msg->payload, txbuf_len);
197a4422ff2SBryan O'Donoghue 		if (ret)
198a4422ff2SBryan O'Donoghue 			goto done;
199a4422ff2SBryan O'Donoghue 	}
200a4422ff2SBryan O'Donoghue 
201a4422ff2SBryan O'Donoghue 	/* Write total length ((header + data) - 1) to USB_PDPHY_TX_SIZE_REG */
202a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
203a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_TX_SIZE_REG,
204a4422ff2SBryan O'Donoghue 			   txsize_len);
205a4422ff2SBryan O'Donoghue 	if (ret)
206a4422ff2SBryan O'Donoghue 		goto done;
207a4422ff2SBryan O'Donoghue 
208a4422ff2SBryan O'Donoghue 	/* Clear TX control register */
209a4422ff2SBryan O'Donoghue 	ret = qcom_pmic_typec_pdphy_clear_tx_control_reg(pmic_typec_pdphy);
210a4422ff2SBryan O'Donoghue 	if (ret)
211a4422ff2SBryan O'Donoghue 		goto done;
212a4422ff2SBryan O'Donoghue 
213a4422ff2SBryan O'Donoghue 	/* Initiate transmit with retry count as indicated by PD revision */
214a4422ff2SBryan O'Donoghue 	val = TX_CONTROL_FRAME_TYPE(type) | TX_CONTROL_SEND_MSG;
215a4422ff2SBryan O'Donoghue 	if (pd_header_rev(msg->header) == PD_REV30)
216a4422ff2SBryan O'Donoghue 		val |= TX_CONTROL_RETRY_COUNT(2);
217a4422ff2SBryan O'Donoghue 	else
218a4422ff2SBryan O'Donoghue 		val |= TX_CONTROL_RETRY_COUNT(3);
219a4422ff2SBryan O'Donoghue 
220a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
221a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_TX_CONTROL_REG, val);
222a4422ff2SBryan O'Donoghue 
223a4422ff2SBryan O'Donoghue done:
224a4422ff2SBryan O'Donoghue 	spin_unlock_irqrestore(&pmic_typec_pdphy->lock, flags);
225a4422ff2SBryan O'Donoghue 
226a4422ff2SBryan O'Donoghue 	if (ret) {
227a4422ff2SBryan O'Donoghue 		dev_err(dev, "pd_transmit_payload: hdr %*ph data %*ph ret %d\n",
228a4422ff2SBryan O'Donoghue 			hdr_len, &msg->header, txbuf_len, &msg->payload, ret);
229a4422ff2SBryan O'Donoghue 	}
230a4422ff2SBryan O'Donoghue 
231a4422ff2SBryan O'Donoghue 	return ret;
232a4422ff2SBryan O'Donoghue }
233a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_pd_transmit(struct pmic_typec_pdphy * pmic_typec_pdphy,enum tcpm_transmit_type type,const struct pd_message * msg,unsigned int negotiated_rev)234a4422ff2SBryan O'Donoghue int qcom_pmic_typec_pdphy_pd_transmit(struct pmic_typec_pdphy *pmic_typec_pdphy,
235a4422ff2SBryan O'Donoghue 				      enum tcpm_transmit_type type,
236a4422ff2SBryan O'Donoghue 				      const struct pd_message *msg,
237a4422ff2SBryan O'Donoghue 				      unsigned int negotiated_rev)
238a4422ff2SBryan O'Donoghue {
239a4422ff2SBryan O'Donoghue 	struct device *dev = pmic_typec_pdphy->dev;
240a4422ff2SBryan O'Donoghue 	int ret;
241a4422ff2SBryan O'Donoghue 
242a4422ff2SBryan O'Donoghue 	if (msg) {
243a4422ff2SBryan O'Donoghue 		ret = qcom_pmic_typec_pdphy_pd_transmit_payload(pmic_typec_pdphy,
244a4422ff2SBryan O'Donoghue 								type, msg,
245a4422ff2SBryan O'Donoghue 								negotiated_rev);
246a4422ff2SBryan O'Donoghue 	} else {
247a4422ff2SBryan O'Donoghue 		ret = qcom_pmic_typec_pdphy_pd_transmit_signal(pmic_typec_pdphy,
248a4422ff2SBryan O'Donoghue 							       type,
249a4422ff2SBryan O'Donoghue 							       negotiated_rev);
250a4422ff2SBryan O'Donoghue 	}
251a4422ff2SBryan O'Donoghue 
252a4422ff2SBryan O'Donoghue 	if (ret)
253a4422ff2SBryan O'Donoghue 		dev_dbg(dev, "pd_transmit: type %x result %d\n", type, ret);
254a4422ff2SBryan O'Donoghue 
255a4422ff2SBryan O'Donoghue 	return ret;
256a4422ff2SBryan O'Donoghue }
257a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_pd_receive(struct pmic_typec_pdphy * pmic_typec_pdphy)258a4422ff2SBryan O'Donoghue static void qcom_pmic_typec_pdphy_pd_receive(struct pmic_typec_pdphy *pmic_typec_pdphy)
259a4422ff2SBryan O'Donoghue {
260a4422ff2SBryan O'Donoghue 	struct device *dev = pmic_typec_pdphy->dev;
261a4422ff2SBryan O'Donoghue 	struct pd_message msg;
262a4422ff2SBryan O'Donoghue 	unsigned int size, rx_status;
263a4422ff2SBryan O'Donoghue 	unsigned long flags;
264a4422ff2SBryan O'Donoghue 	int ret;
265a4422ff2SBryan O'Donoghue 
266a4422ff2SBryan O'Donoghue 	spin_lock_irqsave(&pmic_typec_pdphy->lock, flags);
267a4422ff2SBryan O'Donoghue 
268a4422ff2SBryan O'Donoghue 	ret = regmap_read(pmic_typec_pdphy->regmap,
269a4422ff2SBryan O'Donoghue 			  pmic_typec_pdphy->base + USB_PDPHY_RX_SIZE_REG, &size);
270a4422ff2SBryan O'Donoghue 	if (ret)
271a4422ff2SBryan O'Donoghue 		goto done;
272a4422ff2SBryan O'Donoghue 
273a4422ff2SBryan O'Donoghue 	/* Hardware requires +1 of the real read value to be passed */
274a4422ff2SBryan O'Donoghue 	if (size < 1 || size > sizeof(msg.payload) + 1) {
275a4422ff2SBryan O'Donoghue 		dev_dbg(dev, "pd_receive: invalid size %d\n", size);
276a4422ff2SBryan O'Donoghue 		goto done;
277a4422ff2SBryan O'Donoghue 	}
278a4422ff2SBryan O'Donoghue 
279a4422ff2SBryan O'Donoghue 	size += 1;
280a4422ff2SBryan O'Donoghue 	ret = regmap_read(pmic_typec_pdphy->regmap,
281a4422ff2SBryan O'Donoghue 			  pmic_typec_pdphy->base + USB_PDPHY_RX_STATUS_REG,
282a4422ff2SBryan O'Donoghue 			  &rx_status);
283a4422ff2SBryan O'Donoghue 
284a4422ff2SBryan O'Donoghue 	if (ret)
285a4422ff2SBryan O'Donoghue 		goto done;
286a4422ff2SBryan O'Donoghue 
287a4422ff2SBryan O'Donoghue 	ret = regmap_bulk_read(pmic_typec_pdphy->regmap,
288a4422ff2SBryan O'Donoghue 			       pmic_typec_pdphy->base + USB_PDPHY_RX_BUFFER_REG,
289a4422ff2SBryan O'Donoghue 			       (u8 *)&msg, size);
290a4422ff2SBryan O'Donoghue 	if (ret)
291a4422ff2SBryan O'Donoghue 		goto done;
292a4422ff2SBryan O'Donoghue 
293a4422ff2SBryan O'Donoghue 	/* Return ownership of RX buffer to hardware */
294a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
295a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_RX_ACKNOWLEDGE_REG, 0);
296a4422ff2SBryan O'Donoghue 
297a4422ff2SBryan O'Donoghue done:
298a4422ff2SBryan O'Donoghue 	spin_unlock_irqrestore(&pmic_typec_pdphy->lock, flags);
299a4422ff2SBryan O'Donoghue 
300a4422ff2SBryan O'Donoghue 	if (!ret) {
301a4422ff2SBryan O'Donoghue 		dev_vdbg(dev, "pd_receive: handing %d bytes to tcpm\n", size);
302a4422ff2SBryan O'Donoghue 		tcpm_pd_receive(pmic_typec_pdphy->tcpm_port, &msg);
303a4422ff2SBryan O'Donoghue 	}
304a4422ff2SBryan O'Donoghue }
305a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_isr(int irq,void * dev_id)306a4422ff2SBryan O'Donoghue static irqreturn_t qcom_pmic_typec_pdphy_isr(int irq, void *dev_id)
307a4422ff2SBryan O'Donoghue {
308a4422ff2SBryan O'Donoghue 	struct pmic_typec_pdphy_irq_data *irq_data = dev_id;
309a4422ff2SBryan O'Donoghue 	struct pmic_typec_pdphy *pmic_typec_pdphy = irq_data->pmic_typec_pdphy;
310a4422ff2SBryan O'Donoghue 	struct device *dev = pmic_typec_pdphy->dev;
311a4422ff2SBryan O'Donoghue 
312a4422ff2SBryan O'Donoghue 	switch (irq_data->virq) {
313a4422ff2SBryan O'Donoghue 	case PMIC_PDPHY_SIG_TX_IRQ:
314a4422ff2SBryan O'Donoghue 		dev_err(dev, "isr: tx_sig\n");
315a4422ff2SBryan O'Donoghue 		break;
316a4422ff2SBryan O'Donoghue 	case PMIC_PDPHY_SIG_RX_IRQ:
317a4422ff2SBryan O'Donoghue 		schedule_work(&pmic_typec_pdphy->reset_work);
318a4422ff2SBryan O'Donoghue 		break;
319a4422ff2SBryan O'Donoghue 	case PMIC_PDPHY_MSG_TX_IRQ:
320a4422ff2SBryan O'Donoghue 		tcpm_pd_transmit_complete(pmic_typec_pdphy->tcpm_port,
321a4422ff2SBryan O'Donoghue 					  TCPC_TX_SUCCESS);
322a4422ff2SBryan O'Donoghue 		break;
323a4422ff2SBryan O'Donoghue 	case PMIC_PDPHY_MSG_RX_IRQ:
324a4422ff2SBryan O'Donoghue 		qcom_pmic_typec_pdphy_pd_receive(pmic_typec_pdphy);
325a4422ff2SBryan O'Donoghue 		break;
326a4422ff2SBryan O'Donoghue 	case PMIC_PDPHY_MSG_TX_FAIL_IRQ:
327a4422ff2SBryan O'Donoghue 		tcpm_pd_transmit_complete(pmic_typec_pdphy->tcpm_port,
328a4422ff2SBryan O'Donoghue 					  TCPC_TX_FAILED);
329a4422ff2SBryan O'Donoghue 		break;
330a4422ff2SBryan O'Donoghue 	case PMIC_PDPHY_MSG_TX_DISCARD_IRQ:
331a4422ff2SBryan O'Donoghue 		tcpm_pd_transmit_complete(pmic_typec_pdphy->tcpm_port,
332a4422ff2SBryan O'Donoghue 					  TCPC_TX_DISCARDED);
333a4422ff2SBryan O'Donoghue 		break;
334a4422ff2SBryan O'Donoghue 	}
335a4422ff2SBryan O'Donoghue 
336a4422ff2SBryan O'Donoghue 	return IRQ_HANDLED;
337a4422ff2SBryan O'Donoghue }
338a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_set_pd_rx(struct pmic_typec_pdphy * pmic_typec_pdphy,bool on)339a4422ff2SBryan O'Donoghue int qcom_pmic_typec_pdphy_set_pd_rx(struct pmic_typec_pdphy *pmic_typec_pdphy, bool on)
340a4422ff2SBryan O'Donoghue {
341a4422ff2SBryan O'Donoghue 	unsigned long flags;
342a4422ff2SBryan O'Donoghue 	int ret;
343a4422ff2SBryan O'Donoghue 
344a4422ff2SBryan O'Donoghue 	spin_lock_irqsave(&pmic_typec_pdphy->lock, flags);
345a4422ff2SBryan O'Donoghue 
346a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
347a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_RX_ACKNOWLEDGE_REG, !on);
348a4422ff2SBryan O'Donoghue 
349a4422ff2SBryan O'Donoghue 	spin_unlock_irqrestore(&pmic_typec_pdphy->lock, flags);
350a4422ff2SBryan O'Donoghue 
351a4422ff2SBryan O'Donoghue 	dev_dbg(pmic_typec_pdphy->dev, "set_pd_rx: %s\n", on ? "on" : "off");
352a4422ff2SBryan O'Donoghue 
353a4422ff2SBryan O'Donoghue 	return ret;
354a4422ff2SBryan O'Donoghue }
355a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_set_roles(struct pmic_typec_pdphy * pmic_typec_pdphy,bool data_role_host,bool power_role_src)356a4422ff2SBryan O'Donoghue int qcom_pmic_typec_pdphy_set_roles(struct pmic_typec_pdphy *pmic_typec_pdphy,
357a4422ff2SBryan O'Donoghue 				    bool data_role_host, bool power_role_src)
358a4422ff2SBryan O'Donoghue {
359a4422ff2SBryan O'Donoghue 	struct device *dev = pmic_typec_pdphy->dev;
360a4422ff2SBryan O'Donoghue 	unsigned long flags;
361a4422ff2SBryan O'Donoghue 	int ret;
362a4422ff2SBryan O'Donoghue 
363a4422ff2SBryan O'Donoghue 	spin_lock_irqsave(&pmic_typec_pdphy->lock, flags);
364a4422ff2SBryan O'Donoghue 
365a4422ff2SBryan O'Donoghue 	ret = regmap_update_bits(pmic_typec_pdphy->regmap,
366a4422ff2SBryan O'Donoghue 				 pmic_typec_pdphy->base + USB_PDPHY_MSG_CONFIG_REG,
367a4422ff2SBryan O'Donoghue 				 MSG_CONFIG_PORT_DATA_ROLE |
368a4422ff2SBryan O'Donoghue 				 MSG_CONFIG_PORT_POWER_ROLE,
369a4422ff2SBryan O'Donoghue 				 data_role_host << 3 | power_role_src << 2);
370a4422ff2SBryan O'Donoghue 
371a4422ff2SBryan O'Donoghue 	spin_unlock_irqrestore(&pmic_typec_pdphy->lock, flags);
372a4422ff2SBryan O'Donoghue 
373a4422ff2SBryan O'Donoghue 	dev_dbg(dev, "pdphy_set_roles: data_role_host=%d power_role_src=%d\n",
374a4422ff2SBryan O'Donoghue 		data_role_host, power_role_src);
375a4422ff2SBryan O'Donoghue 
376a4422ff2SBryan O'Donoghue 	return ret;
377a4422ff2SBryan O'Donoghue }
378a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_enable(struct pmic_typec_pdphy * pmic_typec_pdphy)379a4422ff2SBryan O'Donoghue static int qcom_pmic_typec_pdphy_enable(struct pmic_typec_pdphy *pmic_typec_pdphy)
380a4422ff2SBryan O'Donoghue {
381a4422ff2SBryan O'Donoghue 	struct device *dev = pmic_typec_pdphy->dev;
382a4422ff2SBryan O'Donoghue 	int ret;
383a4422ff2SBryan O'Donoghue 
384a4422ff2SBryan O'Donoghue 	/* PD 2.0, DR=TYPEC_DEVICE, PR=TYPEC_SINK */
385a4422ff2SBryan O'Donoghue 	ret = regmap_update_bits(pmic_typec_pdphy->regmap,
386a4422ff2SBryan O'Donoghue 				 pmic_typec_pdphy->base + USB_PDPHY_MSG_CONFIG_REG,
387a4422ff2SBryan O'Donoghue 				 MSG_CONFIG_SPEC_REV_MASK, PD_REV20);
388a4422ff2SBryan O'Donoghue 	if (ret)
389a4422ff2SBryan O'Donoghue 		goto done;
390a4422ff2SBryan O'Donoghue 
391a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
392a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_EN_CONTROL_REG, 0);
393a4422ff2SBryan O'Donoghue 	if (ret)
394a4422ff2SBryan O'Donoghue 		goto done;
395a4422ff2SBryan O'Donoghue 
396a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
397a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_EN_CONTROL_REG,
398a4422ff2SBryan O'Donoghue 			   CONTROL_ENABLE);
399a4422ff2SBryan O'Donoghue 	if (ret)
400a4422ff2SBryan O'Donoghue 		goto done;
401a4422ff2SBryan O'Donoghue 
402a4422ff2SBryan O'Donoghue 	qcom_pmic_typec_pdphy_reset_off(pmic_typec_pdphy);
403a4422ff2SBryan O'Donoghue done:
404a4422ff2SBryan O'Donoghue 	if (ret) {
405a4422ff2SBryan O'Donoghue 		regulator_disable(pmic_typec_pdphy->vdd_pdphy);
406a4422ff2SBryan O'Donoghue 		dev_err(dev, "pdphy_enable fail %d\n", ret);
407a4422ff2SBryan O'Donoghue 	}
408a4422ff2SBryan O'Donoghue 
409a4422ff2SBryan O'Donoghue 	return ret;
410a4422ff2SBryan O'Donoghue }
411a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_disable(struct pmic_typec_pdphy * pmic_typec_pdphy)412a4422ff2SBryan O'Donoghue static int qcom_pmic_typec_pdphy_disable(struct pmic_typec_pdphy *pmic_typec_pdphy)
413a4422ff2SBryan O'Donoghue {
414a4422ff2SBryan O'Donoghue 	int ret;
415a4422ff2SBryan O'Donoghue 
416a4422ff2SBryan O'Donoghue 	qcom_pmic_typec_pdphy_reset_on(pmic_typec_pdphy);
417a4422ff2SBryan O'Donoghue 
418a4422ff2SBryan O'Donoghue 	ret = regmap_write(pmic_typec_pdphy->regmap,
419a4422ff2SBryan O'Donoghue 			   pmic_typec_pdphy->base + USB_PDPHY_EN_CONTROL_REG, 0);
420a4422ff2SBryan O'Donoghue 
421a4422ff2SBryan O'Donoghue 	return ret;
422a4422ff2SBryan O'Donoghue }
423a4422ff2SBryan O'Donoghue 
pmic_typec_pdphy_reset(struct pmic_typec_pdphy * pmic_typec_pdphy)424a4422ff2SBryan O'Donoghue static int pmic_typec_pdphy_reset(struct pmic_typec_pdphy *pmic_typec_pdphy)
425a4422ff2SBryan O'Donoghue {
426a4422ff2SBryan O'Donoghue 	int ret;
427a4422ff2SBryan O'Donoghue 
428a4422ff2SBryan O'Donoghue 	ret = qcom_pmic_typec_pdphy_disable(pmic_typec_pdphy);
429a4422ff2SBryan O'Donoghue 	if (ret)
430a4422ff2SBryan O'Donoghue 		goto done;
431a4422ff2SBryan O'Donoghue 
432a4422ff2SBryan O'Donoghue 	usleep_range(400, 500);
433a4422ff2SBryan O'Donoghue 	ret = qcom_pmic_typec_pdphy_enable(pmic_typec_pdphy);
434a4422ff2SBryan O'Donoghue done:
435a4422ff2SBryan O'Donoghue 	return ret;
436a4422ff2SBryan O'Donoghue }
437a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_start(struct pmic_typec_pdphy * pmic_typec_pdphy,struct tcpm_port * tcpm_port)438a4422ff2SBryan O'Donoghue int qcom_pmic_typec_pdphy_start(struct pmic_typec_pdphy *pmic_typec_pdphy,
439a4422ff2SBryan O'Donoghue 				struct tcpm_port *tcpm_port)
440a4422ff2SBryan O'Donoghue {
441a4422ff2SBryan O'Donoghue 	int i;
442a4422ff2SBryan O'Donoghue 	int ret;
443a4422ff2SBryan O'Donoghue 
444*76750f1dSHui Liu 	ret = regulator_enable(pmic_typec_pdphy->vdd_pdphy);
445*76750f1dSHui Liu 	if (ret)
446*76750f1dSHui Liu 		return ret;
447*76750f1dSHui Liu 
448a4422ff2SBryan O'Donoghue 	pmic_typec_pdphy->tcpm_port = tcpm_port;
449a4422ff2SBryan O'Donoghue 
450a4422ff2SBryan O'Donoghue 	ret = pmic_typec_pdphy_reset(pmic_typec_pdphy);
451a4422ff2SBryan O'Donoghue 	if (ret)
452a4422ff2SBryan O'Donoghue 		return ret;
453a4422ff2SBryan O'Donoghue 
454a4422ff2SBryan O'Donoghue 	for (i = 0; i < pmic_typec_pdphy->nr_irqs; i++)
455a4422ff2SBryan O'Donoghue 		enable_irq(pmic_typec_pdphy->irq_data[i].irq);
456a4422ff2SBryan O'Donoghue 
457a4422ff2SBryan O'Donoghue 	return 0;
458a4422ff2SBryan O'Donoghue }
459a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_stop(struct pmic_typec_pdphy * pmic_typec_pdphy)460a4422ff2SBryan O'Donoghue void qcom_pmic_typec_pdphy_stop(struct pmic_typec_pdphy *pmic_typec_pdphy)
461a4422ff2SBryan O'Donoghue {
462a4422ff2SBryan O'Donoghue 	int i;
463a4422ff2SBryan O'Donoghue 
464a4422ff2SBryan O'Donoghue 	for (i = 0; i < pmic_typec_pdphy->nr_irqs; i++)
465a4422ff2SBryan O'Donoghue 		disable_irq(pmic_typec_pdphy->irq_data[i].irq);
466a4422ff2SBryan O'Donoghue 
467a4422ff2SBryan O'Donoghue 	qcom_pmic_typec_pdphy_reset_on(pmic_typec_pdphy);
468*76750f1dSHui Liu 
469*76750f1dSHui Liu 	regulator_disable(pmic_typec_pdphy->vdd_pdphy);
470a4422ff2SBryan O'Donoghue }
471a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_alloc(struct device * dev)472a4422ff2SBryan O'Donoghue struct pmic_typec_pdphy *qcom_pmic_typec_pdphy_alloc(struct device *dev)
473a4422ff2SBryan O'Donoghue {
474a4422ff2SBryan O'Donoghue 	return devm_kzalloc(dev, sizeof(struct pmic_typec_pdphy), GFP_KERNEL);
475a4422ff2SBryan O'Donoghue }
476a4422ff2SBryan O'Donoghue 
qcom_pmic_typec_pdphy_probe(struct platform_device * pdev,struct pmic_typec_pdphy * pmic_typec_pdphy,struct pmic_typec_pdphy_resources * res,struct regmap * regmap,u32 base)477a4422ff2SBryan O'Donoghue int qcom_pmic_typec_pdphy_probe(struct platform_device *pdev,
478a4422ff2SBryan O'Donoghue 				struct pmic_typec_pdphy *pmic_typec_pdphy,
479a4422ff2SBryan O'Donoghue 				struct pmic_typec_pdphy_resources *res,
480a4422ff2SBryan O'Donoghue 				struct regmap *regmap,
481a4422ff2SBryan O'Donoghue 				u32 base)
482a4422ff2SBryan O'Donoghue {
483a4422ff2SBryan O'Donoghue 	struct device *dev = &pdev->dev;
484a4422ff2SBryan O'Donoghue 	struct pmic_typec_pdphy_irq_data *irq_data;
485a4422ff2SBryan O'Donoghue 	int i, ret, irq;
486a4422ff2SBryan O'Donoghue 
487a4422ff2SBryan O'Donoghue 	if (!res->nr_irqs || res->nr_irqs > PMIC_PDPHY_MAX_IRQS)
488a4422ff2SBryan O'Donoghue 		return -EINVAL;
489a4422ff2SBryan O'Donoghue 
490a4422ff2SBryan O'Donoghue 	irq_data = devm_kzalloc(dev, sizeof(*irq_data) * res->nr_irqs,
491a4422ff2SBryan O'Donoghue 				GFP_KERNEL);
492a4422ff2SBryan O'Donoghue 	if (!irq_data)
493a4422ff2SBryan O'Donoghue 		return -ENOMEM;
494a4422ff2SBryan O'Donoghue 
495a4422ff2SBryan O'Donoghue 	pmic_typec_pdphy->vdd_pdphy = devm_regulator_get(dev, "vdd-pdphy");
496a4422ff2SBryan O'Donoghue 	if (IS_ERR(pmic_typec_pdphy->vdd_pdphy))
497a4422ff2SBryan O'Donoghue 		return PTR_ERR(pmic_typec_pdphy->vdd_pdphy);
498a4422ff2SBryan O'Donoghue 
499a4422ff2SBryan O'Donoghue 	pmic_typec_pdphy->dev = dev;
500a4422ff2SBryan O'Donoghue 	pmic_typec_pdphy->base = base;
501a4422ff2SBryan O'Donoghue 	pmic_typec_pdphy->regmap = regmap;
502a4422ff2SBryan O'Donoghue 	pmic_typec_pdphy->nr_irqs = res->nr_irqs;
503a4422ff2SBryan O'Donoghue 	pmic_typec_pdphy->irq_data = irq_data;
504a4422ff2SBryan O'Donoghue 	spin_lock_init(&pmic_typec_pdphy->lock);
505a4422ff2SBryan O'Donoghue 	INIT_WORK(&pmic_typec_pdphy->reset_work, qcom_pmic_typec_pdphy_sig_reset_work);
506a4422ff2SBryan O'Donoghue 
507a4422ff2SBryan O'Donoghue 	for (i = 0; i < res->nr_irqs; i++, irq_data++) {
508a4422ff2SBryan O'Donoghue 		irq = platform_get_irq_byname(pdev, res->irq_params[i].irq_name);
509a4422ff2SBryan O'Donoghue 		if (irq < 0)
510a4422ff2SBryan O'Donoghue 			return irq;
511a4422ff2SBryan O'Donoghue 
512a4422ff2SBryan O'Donoghue 		irq_data->pmic_typec_pdphy = pmic_typec_pdphy;
513a4422ff2SBryan O'Donoghue 		irq_data->irq = irq;
514a4422ff2SBryan O'Donoghue 		irq_data->virq = res->irq_params[i].virq;
515a4422ff2SBryan O'Donoghue 
516a4422ff2SBryan O'Donoghue 		ret = devm_request_threaded_irq(dev, irq, NULL,
517a4422ff2SBryan O'Donoghue 						qcom_pmic_typec_pdphy_isr,
518a4422ff2SBryan O'Donoghue 						IRQF_ONESHOT | IRQF_NO_AUTOEN,
519a4422ff2SBryan O'Donoghue 						res->irq_params[i].irq_name,
520a4422ff2SBryan O'Donoghue 						irq_data);
521a4422ff2SBryan O'Donoghue 		if (ret)
522a4422ff2SBryan O'Donoghue 			return ret;
523a4422ff2SBryan O'Donoghue 	}
524a4422ff2SBryan O'Donoghue 
525a4422ff2SBryan O'Donoghue 	return 0;
526a4422ff2SBryan O'Donoghue }
527