1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3 * Copyright (C) 2020-21 Intel Corporation.
4 */
5
6 #include <linux/delay.h>
7
8 #include "iosm_ipc_chnl_cfg.h"
9 #include "iosm_ipc_imem.h"
10 #include "iosm_ipc_imem_ops.h"
11 #include "iosm_ipc_port.h"
12 #include "iosm_ipc_task_queue.h"
13
14 /* Open a packet data online channel between the network layer and CP. */
ipc_imem_sys_wwan_open(struct iosm_imem * ipc_imem,int if_id)15 int ipc_imem_sys_wwan_open(struct iosm_imem *ipc_imem, int if_id)
16 {
17 dev_dbg(ipc_imem->dev, "%s if id: %d",
18 ipc_imem_phase_get_string(ipc_imem->phase), if_id);
19
20 /* The network interface is only supported in the runtime phase. */
21 if (ipc_imem_phase_update(ipc_imem) != IPC_P_RUN) {
22 dev_err(ipc_imem->dev, "net:%d : refused phase %s", if_id,
23 ipc_imem_phase_get_string(ipc_imem->phase));
24 return -EIO;
25 }
26
27 return ipc_mux_open_session(ipc_imem->mux, if_id);
28 }
29
30 /* Release a net link to CP. */
ipc_imem_sys_wwan_close(struct iosm_imem * ipc_imem,int if_id,int channel_id)31 void ipc_imem_sys_wwan_close(struct iosm_imem *ipc_imem, int if_id,
32 int channel_id)
33 {
34 if (ipc_imem->mux && if_id >= IP_MUX_SESSION_START &&
35 if_id <= IP_MUX_SESSION_END)
36 ipc_mux_close_session(ipc_imem->mux, if_id);
37 }
38
39 /* Tasklet call to do uplink transfer. */
ipc_imem_tq_cdev_write(struct iosm_imem * ipc_imem,int arg,void * msg,size_t size)40 static int ipc_imem_tq_cdev_write(struct iosm_imem *ipc_imem, int arg,
41 void *msg, size_t size)
42 {
43 ipc_imem->ev_cdev_write_pending = false;
44 ipc_imem_ul_send(ipc_imem);
45
46 return 0;
47 }
48
49 /* Through tasklet to do sio write. */
ipc_imem_call_cdev_write(struct iosm_imem * ipc_imem)50 static int ipc_imem_call_cdev_write(struct iosm_imem *ipc_imem)
51 {
52 if (ipc_imem->ev_cdev_write_pending)
53 return -1;
54
55 ipc_imem->ev_cdev_write_pending = true;
56
57 return ipc_task_queue_send_task(ipc_imem, ipc_imem_tq_cdev_write, 0,
58 NULL, 0, false);
59 }
60
61 /* Function for transfer UL data */
ipc_imem_sys_wwan_transmit(struct iosm_imem * ipc_imem,int if_id,int channel_id,struct sk_buff * skb)62 int ipc_imem_sys_wwan_transmit(struct iosm_imem *ipc_imem,
63 int if_id, int channel_id, struct sk_buff *skb)
64 {
65 int ret = -EINVAL;
66
67 if (!ipc_imem || channel_id < 0)
68 goto out;
69
70 /* Is CP Running? */
71 if (ipc_imem->phase != IPC_P_RUN) {
72 dev_dbg(ipc_imem->dev, "phase %s transmit",
73 ipc_imem_phase_get_string(ipc_imem->phase));
74 ret = -EIO;
75 goto out;
76 }
77
78 /* Route the UL packet through IP MUX Layer */
79 ret = ipc_mux_ul_trigger_encode(ipc_imem->mux, if_id, skb);
80 out:
81 return ret;
82 }
83
84 /* Initialize wwan channel */
ipc_imem_wwan_channel_init(struct iosm_imem * ipc_imem,enum ipc_mux_protocol mux_type)85 void ipc_imem_wwan_channel_init(struct iosm_imem *ipc_imem,
86 enum ipc_mux_protocol mux_type)
87 {
88 struct ipc_chnl_cfg chnl_cfg = { 0 };
89
90 ipc_imem->cp_version = ipc_mmio_get_cp_version(ipc_imem->mmio);
91
92 /* If modem version is invalid (0xffffffff), do not initialize WWAN. */
93 if (ipc_imem->cp_version == -1) {
94 dev_err(ipc_imem->dev, "invalid CP version");
95 return;
96 }
97
98 ipc_chnl_cfg_get(&chnl_cfg, ipc_imem->nr_of_channels);
99 ipc_imem_channel_init(ipc_imem, IPC_CTYPE_WWAN, chnl_cfg,
100 IRQ_MOD_OFF);
101
102 /* WWAN registration. */
103 ipc_imem->wwan = ipc_wwan_init(ipc_imem, ipc_imem->dev);
104 if (!ipc_imem->wwan)
105 dev_err(ipc_imem->dev,
106 "failed to register the ipc_wwan interfaces");
107 }
108
109 /* Map SKB to DMA for transfer */
ipc_imem_map_skb_to_dma(struct iosm_imem * ipc_imem,struct sk_buff * skb)110 static int ipc_imem_map_skb_to_dma(struct iosm_imem *ipc_imem,
111 struct sk_buff *skb)
112 {
113 struct iosm_pcie *ipc_pcie = ipc_imem->pcie;
114 char *buf = skb->data;
115 int len = skb->len;
116 dma_addr_t mapping;
117 int ret;
118
119 ret = ipc_pcie_addr_map(ipc_pcie, buf, len, &mapping, DMA_TO_DEVICE);
120
121 if (ret)
122 goto err;
123
124 BUILD_BUG_ON(sizeof(*IPC_CB(skb)) > sizeof(skb->cb));
125
126 IPC_CB(skb)->mapping = mapping;
127 IPC_CB(skb)->direction = DMA_TO_DEVICE;
128 IPC_CB(skb)->len = len;
129 IPC_CB(skb)->op_type = (u8)UL_DEFAULT;
130
131 err:
132 return ret;
133 }
134
135 /* return true if channel is ready for use */
ipc_imem_is_channel_active(struct iosm_imem * ipc_imem,struct ipc_mem_channel * channel)136 static bool ipc_imem_is_channel_active(struct iosm_imem *ipc_imem,
137 struct ipc_mem_channel *channel)
138 {
139 enum ipc_phase phase;
140
141 /* Update the current operation phase. */
142 phase = ipc_imem->phase;
143
144 /* Select the operation depending on the execution stage. */
145 switch (phase) {
146 case IPC_P_RUN:
147 case IPC_P_PSI:
148 case IPC_P_EBL:
149 break;
150
151 case IPC_P_ROM:
152 /* Prepare the PSI image for the CP ROM driver and
153 * suspend the flash app.
154 */
155 if (channel->state != IMEM_CHANNEL_RESERVED) {
156 dev_err(ipc_imem->dev,
157 "ch[%d]:invalid channel state %d,expected %d",
158 channel->channel_id, channel->state,
159 IMEM_CHANNEL_RESERVED);
160 goto channel_unavailable;
161 }
162 goto channel_available;
163
164 default:
165 /* Ignore uplink actions in all other phases. */
166 dev_err(ipc_imem->dev, "ch[%d]: confused phase %d",
167 channel->channel_id, phase);
168 goto channel_unavailable;
169 }
170 /* Check the full availability of the channel. */
171 if (channel->state != IMEM_CHANNEL_ACTIVE) {
172 dev_err(ipc_imem->dev, "ch[%d]: confused channel state %d",
173 channel->channel_id, channel->state);
174 goto channel_unavailable;
175 }
176
177 channel_available:
178 return true;
179
180 channel_unavailable:
181 return false;
182 }
183
184 /* Release a sio link to CP. */
ipc_imem_sys_cdev_close(struct iosm_cdev * ipc_cdev)185 void ipc_imem_sys_cdev_close(struct iosm_cdev *ipc_cdev)
186 {
187 struct iosm_imem *ipc_imem = ipc_cdev->ipc_imem;
188 struct ipc_mem_channel *channel = ipc_cdev->channel;
189 enum ipc_phase curr_phase;
190 int status = 0;
191 u32 tail = 0;
192
193 curr_phase = ipc_imem->phase;
194
195 /* If current phase is IPC_P_OFF or SIO ID is -ve then
196 * channel is already freed. Nothing to do.
197 */
198 if (curr_phase == IPC_P_OFF) {
199 dev_err(ipc_imem->dev,
200 "nothing to do. Current Phase: %s",
201 ipc_imem_phase_get_string(curr_phase));
202 return;
203 }
204
205 if (channel->state == IMEM_CHANNEL_FREE) {
206 dev_err(ipc_imem->dev, "ch[%d]: invalid channel state %d",
207 channel->channel_id, channel->state);
208 return;
209 }
210
211 /* If there are any pending TDs then wait for Timeout/Completion before
212 * closing pipe.
213 */
214 if (channel->ul_pipe.old_tail != channel->ul_pipe.old_head) {
215 ipc_imem->app_notify_ul_pend = 1;
216
217 /* Suspend the user app and wait a certain time for processing
218 * UL Data.
219 */
220 status = wait_for_completion_interruptible_timeout
221 (&ipc_imem->ul_pend_sem,
222 msecs_to_jiffies(IPC_PEND_DATA_TIMEOUT));
223 if (status == 0) {
224 dev_dbg(ipc_imem->dev,
225 "Pend data Timeout UL-Pipe:%d Head:%d Tail:%d",
226 channel->ul_pipe.pipe_nr,
227 channel->ul_pipe.old_head,
228 channel->ul_pipe.old_tail);
229 }
230
231 ipc_imem->app_notify_ul_pend = 0;
232 }
233
234 /* If there are any pending TDs then wait for Timeout/Completion before
235 * closing pipe.
236 */
237 ipc_protocol_get_head_tail_index(ipc_imem->ipc_protocol,
238 &channel->dl_pipe, NULL, &tail);
239
240 if (tail != channel->dl_pipe.old_tail) {
241 ipc_imem->app_notify_dl_pend = 1;
242
243 /* Suspend the user app and wait a certain time for processing
244 * DL Data.
245 */
246 status = wait_for_completion_interruptible_timeout
247 (&ipc_imem->dl_pend_sem,
248 msecs_to_jiffies(IPC_PEND_DATA_TIMEOUT));
249 if (status == 0) {
250 dev_dbg(ipc_imem->dev,
251 "Pend data Timeout DL-Pipe:%d Head:%d Tail:%d",
252 channel->dl_pipe.pipe_nr,
253 channel->dl_pipe.old_head,
254 channel->dl_pipe.old_tail);
255 }
256
257 ipc_imem->app_notify_dl_pend = 0;
258 }
259
260 /* Due to wait for completion in messages, there is a small window
261 * between closing the pipe and updating the channel is closed. In this
262 * small window there could be HP update from Host Driver. Hence update
263 * the channel state as CLOSING to aviod unnecessary interrupt
264 * towards CP.
265 */
266 channel->state = IMEM_CHANNEL_CLOSING;
267
268 ipc_imem_pipe_close(ipc_imem, &channel->ul_pipe);
269 ipc_imem_pipe_close(ipc_imem, &channel->dl_pipe);
270
271 ipc_imem_channel_free(channel);
272 }
273
274 /* Open a PORT link to CP and return the channel */
ipc_imem_sys_port_open(struct iosm_imem * ipc_imem,int chl_id,int hp_id)275 struct ipc_mem_channel *ipc_imem_sys_port_open(struct iosm_imem *ipc_imem,
276 int chl_id, int hp_id)
277 {
278 struct ipc_mem_channel *channel;
279 int ch_id;
280
281 /* The PORT interface is only supported in the runtime phase. */
282 if (ipc_imem_phase_update(ipc_imem) != IPC_P_RUN) {
283 dev_err(ipc_imem->dev, "PORT open refused, phase %s",
284 ipc_imem_phase_get_string(ipc_imem->phase));
285 return NULL;
286 }
287
288 ch_id = ipc_imem_channel_alloc(ipc_imem, chl_id, IPC_CTYPE_CTRL);
289
290 if (ch_id < 0) {
291 dev_err(ipc_imem->dev, "reservation of an PORT chnl id failed");
292 return NULL;
293 }
294
295 channel = ipc_imem_channel_open(ipc_imem, ch_id, hp_id);
296
297 if (!channel) {
298 dev_err(ipc_imem->dev, "PORT channel id open failed");
299 return NULL;
300 }
301
302 return channel;
303 }
304
305 /* transfer skb to modem */
ipc_imem_sys_cdev_write(struct iosm_cdev * ipc_cdev,struct sk_buff * skb)306 int ipc_imem_sys_cdev_write(struct iosm_cdev *ipc_cdev, struct sk_buff *skb)
307 {
308 struct ipc_mem_channel *channel = ipc_cdev->channel;
309 struct iosm_imem *ipc_imem = ipc_cdev->ipc_imem;
310 int ret = -EIO;
311
312 if (!ipc_imem_is_channel_active(ipc_imem, channel) ||
313 ipc_imem->phase == IPC_P_OFF_REQ)
314 goto out;
315
316 ret = ipc_imem_map_skb_to_dma(ipc_imem, skb);
317
318 if (ret)
319 goto out;
320
321 /* Add skb to the uplink skbuf accumulator. */
322 skb_queue_tail(&channel->ul_list, skb);
323
324 ret = ipc_imem_call_cdev_write(ipc_imem);
325
326 if (ret) {
327 skb_dequeue_tail(&channel->ul_list);
328 dev_err(ipc_cdev->dev, "channel id[%d] write failed\n",
329 ipc_cdev->channel->channel_id);
330 }
331 out:
332 return ret;
333 }
334