1// SPDX-License-Identifier: GPL-2.0-only
2/*
3 * Copyright (c) 2021, MediaTek Inc.
4 * Copyright (c) 2021-2022, Intel Corporation.
5 *
6 * Authors:
7 * Haijun Liu <haijun.liu@mediatek.com>
8 * Ricardo Martinez <ricardo.martinez@linux.intel.com>
9 * Moises Veleta <moises.veleta@intel.com>
10 *
11 * Contributors:
12 * Amir Hanania <amir.hanania@intel.com>
13 * Chiranjeevi Rapolu <chiranjeevi.rapolu@intel.com>
14 * Eliot Lee <eliot.lee@intel.com>
15 * Sreehari Kancharla <sreehari.kancharla@intel.com>
16 */
17
18#include <linux/bitfield.h>
19#include <linux/device.h>
20#include <linux/err.h>
21#include <linux/kthread.h>
22#include <linux/netdevice.h>
23#include <linux/skbuff.h>
24#include <linux/spinlock.h>
25
26#include "t7xx_port.h"
27#include "t7xx_port_proxy.h"
28#include "t7xx_state_monitor.h"
29
30#define PORT_MSG_VERSION GENMASK(31, 16)
31#define PORT_MSG_PRT_CNT GENMASK(15, 0)
32
33struct port_msg {
34 __le32 head_pattern;
35 __le32 info;
36 __le32 tail_pattern;
37 __le32 data[];
38};
39
40static int port_ctl_send_msg_to_md(struct t7xx_port *port, unsigned int msg, unsigned int ex_msg)
41{
42 struct sk_buff *skb;
43 int ret;
44
45 skb = t7xx_ctrl_alloc_skb(payload: 0);
46 if (!skb)
47 return -ENOMEM;
48
49 ret = t7xx_port_send_ctl_skb(port, skb, msg, ex_msg);
50 if (ret)
51 dev_kfree_skb_any(skb);
52
53 return ret;
54}
55
56static int fsm_ee_message_handler(struct t7xx_port *port, struct t7xx_fsm_ctl *ctl,
57 struct sk_buff *skb)
58{
59 struct ctrl_msg_header *ctrl_msg_h = (struct ctrl_msg_header *)skb->data;
60 struct device *dev = &ctl->md->t7xx_dev->pdev->dev;
61 enum md_state md_state;
62 int ret = -EINVAL;
63
64 md_state = t7xx_fsm_get_md_state(ctl);
65 if (md_state != MD_STATE_EXCEPTION) {
66 dev_err(dev, "Receive invalid MD_EX %x when MD state is %d\n",
67 ctrl_msg_h->ex_msg, md_state);
68 return -EINVAL;
69 }
70
71 switch (le32_to_cpu(ctrl_msg_h->ctrl_msg_id)) {
72 case CTL_ID_MD_EX:
73 if (le32_to_cpu(ctrl_msg_h->ex_msg) != MD_EX_CHK_ID) {
74 dev_err(dev, "Receive invalid MD_EX %x\n", ctrl_msg_h->ex_msg);
75 break;
76 }
77
78 ret = port_ctl_send_msg_to_md(port, CTL_ID_MD_EX, MD_EX_CHK_ID);
79 if (ret) {
80 dev_err(dev, "Failed to send exception message to modem\n");
81 break;
82 }
83
84 ret = t7xx_fsm_append_event(ctl, event_id: FSM_EVENT_MD_EX, NULL, length: 0);
85 if (ret)
86 dev_err(dev, "Failed to append Modem Exception event");
87
88 break;
89
90 case CTL_ID_MD_EX_ACK:
91 if (le32_to_cpu(ctrl_msg_h->ex_msg) != MD_EX_CHK_ACK_ID) {
92 dev_err(dev, "Receive invalid MD_EX_ACK %x\n", ctrl_msg_h->ex_msg);
93 break;
94 }
95
96 ret = t7xx_fsm_append_event(ctl, event_id: FSM_EVENT_MD_EX_REC_OK, NULL, length: 0);
97 if (ret)
98 dev_err(dev, "Failed to append Modem Exception Received event");
99
100 break;
101
102 case CTL_ID_MD_EX_PASS:
103 ret = t7xx_fsm_append_event(ctl, event_id: FSM_EVENT_MD_EX_PASS, NULL, length: 0);
104 if (ret)
105 dev_err(dev, "Failed to append Modem Exception Passed event");
106
107 break;
108
109 case CTL_ID_DRV_VER_ERROR:
110 dev_err(dev, "AP/MD driver version mismatch\n");
111 }
112
113 return ret;
114}
115
116/**
117 * t7xx_port_enum_msg_handler() - Parse the port enumeration message to create/remove nodes.
118 * @md: Modem context.
119 * @msg: Message.
120 *
121 * Used to control create/remove device node.
122 *
123 * Return:
124 * * 0 - Success.
125 * * -EFAULT - Message check failure.
126 */
127int t7xx_port_enum_msg_handler(struct t7xx_modem *md, void *msg)
128{
129 struct device *dev = &md->t7xx_dev->pdev->dev;
130 unsigned int version, port_count, i;
131 struct port_msg *port_msg = msg;
132
133 version = FIELD_GET(PORT_MSG_VERSION, le32_to_cpu(port_msg->info));
134 if (version != PORT_ENUM_VER ||
135 le32_to_cpu(port_msg->head_pattern) != PORT_ENUM_HEAD_PATTERN ||
136 le32_to_cpu(port_msg->tail_pattern) != PORT_ENUM_TAIL_PATTERN) {
137 dev_err(dev, "Invalid port control message %x:%x:%x\n",
138 version, le32_to_cpu(port_msg->head_pattern),
139 le32_to_cpu(port_msg->tail_pattern));
140 return -EFAULT;
141 }
142
143 port_count = FIELD_GET(PORT_MSG_PRT_CNT, le32_to_cpu(port_msg->info));
144 for (i = 0; i < port_count; i++) {
145 u32 port_info = le32_to_cpu(port_msg->data[i]);
146 unsigned int ch_id;
147 bool en_flag;
148
149 ch_id = FIELD_GET(PORT_INFO_CH_ID, port_info);
150 en_flag = port_info & PORT_INFO_ENFLG;
151 if (t7xx_port_proxy_chl_enable_disable(port_prox: md->port_prox, ch_id, en_flag))
152 dev_dbg(dev, "Port:%x not found\n", ch_id);
153 }
154
155 return 0;
156}
157
158static int control_msg_handler(struct t7xx_port *port, struct sk_buff *skb)
159{
160 const struct t7xx_port_conf *port_conf = port->port_conf;
161 struct t7xx_fsm_ctl *ctl = port->t7xx_dev->md->fsm_ctl;
162 struct ctrl_msg_header *ctrl_msg_h;
163 int ret = 0;
164
165 ctrl_msg_h = (struct ctrl_msg_header *)skb->data;
166 switch (le32_to_cpu(ctrl_msg_h->ctrl_msg_id)) {
167 case CTL_ID_HS2_MSG:
168 skb_pull(skb, len: sizeof(*ctrl_msg_h));
169
170 if (port_conf->rx_ch == PORT_CH_CONTROL_RX ||
171 port_conf->rx_ch == PORT_CH_AP_CONTROL_RX) {
172 int event = port_conf->rx_ch == PORT_CH_CONTROL_RX ?
173 FSM_EVENT_MD_HS2 : FSM_EVENT_AP_HS2;
174
175 ret = t7xx_fsm_append_event(ctl, event_id: event, data: skb->data,
176 le32_to_cpu(ctrl_msg_h->data_length));
177 if (ret)
178 dev_err(port->dev, "Failed to append Handshake 2 event");
179 }
180
181 dev_kfree_skb_any(skb);
182 break;
183
184 case CTL_ID_MD_EX:
185 case CTL_ID_MD_EX_ACK:
186 case CTL_ID_MD_EX_PASS:
187 case CTL_ID_DRV_VER_ERROR:
188 ret = fsm_ee_message_handler(port, ctl, skb);
189 dev_kfree_skb_any(skb);
190 break;
191
192 case CTL_ID_PORT_ENUM:
193 skb_pull(skb, len: sizeof(*ctrl_msg_h));
194 ret = t7xx_port_enum_msg_handler(md: ctl->md, msg: (struct port_msg *)skb->data);
195 if (!ret)
196 ret = port_ctl_send_msg_to_md(port, CTL_ID_PORT_ENUM, ex_msg: 0);
197 else
198 ret = port_ctl_send_msg_to_md(port, CTL_ID_PORT_ENUM,
199 PORT_ENUM_VER_MISMATCH);
200
201 break;
202
203 default:
204 ret = -EINVAL;
205 dev_err(port->dev, "Unknown control message ID to FSM %x\n",
206 le32_to_cpu(ctrl_msg_h->ctrl_msg_id));
207 break;
208 }
209
210 if (ret)
211 dev_err(port->dev, "%s control message handle error: %d\n", port_conf->name, ret);
212
213 return ret;
214}
215
216static int port_ctl_rx_thread(void *arg)
217{
218 while (!kthread_should_stop()) {
219 struct t7xx_port *port = arg;
220 struct sk_buff *skb;
221 unsigned long flags;
222
223 spin_lock_irqsave(&port->rx_wq.lock, flags);
224 if (skb_queue_empty(list: &port->rx_skb_list) &&
225 wait_event_interruptible_locked_irq(port->rx_wq,
226 !skb_queue_empty(&port->rx_skb_list) ||
227 kthread_should_stop())) {
228 spin_unlock_irqrestore(lock: &port->rx_wq.lock, flags);
229 continue;
230 }
231 if (kthread_should_stop()) {
232 spin_unlock_irqrestore(lock: &port->rx_wq.lock, flags);
233 break;
234 }
235 skb = __skb_dequeue(list: &port->rx_skb_list);
236 spin_unlock_irqrestore(lock: &port->rx_wq.lock, flags);
237
238 control_msg_handler(port, skb);
239 }
240
241 return 0;
242}
243
244static int port_ctl_init(struct t7xx_port *port)
245{
246 const struct t7xx_port_conf *port_conf = port->port_conf;
247
248 port->thread = kthread_run(port_ctl_rx_thread, port, "%s", port_conf->name);
249 if (IS_ERR(ptr: port->thread)) {
250 dev_err(port->dev, "Failed to start port control thread\n");
251 return PTR_ERR(ptr: port->thread);
252 }
253
254 port->rx_length_th = CTRL_QUEUE_MAXLEN;
255 return 0;
256}
257
258static void port_ctl_uninit(struct t7xx_port *port)
259{
260 unsigned long flags;
261 struct sk_buff *skb;
262
263 if (port->thread)
264 kthread_stop(k: port->thread);
265
266 spin_lock_irqsave(&port->rx_wq.lock, flags);
267 port->rx_length_th = 0;
268 while ((skb = __skb_dequeue(list: &port->rx_skb_list)) != NULL)
269 dev_kfree_skb_any(skb);
270 spin_unlock_irqrestore(lock: &port->rx_wq.lock, flags);
271}
272
273struct port_ops ctl_port_ops = {
274 .init = port_ctl_init,
275 .recv_skb = t7xx_port_enqueue_skb,
276 .uninit = port_ctl_uninit,
277};
278

source code of linux/drivers/net/wwan/t7xx/t7xx_port_ctrl_msg.c