blob: 573b9a397f2b978c58e308d2bd1a09f68c2c3fce [file] [log] [blame]
Karthikeyan Ramasubramanian5c568fb2013-01-10 17:09:13 -07001/* Copyright (c) 2011-2013, The Linux Foundation. All rights reserved.
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12
13#define DEBUG
14
15#include <linux/slab.h>
16#include <linux/module.h>
17#include <linux/kernel.h>
18#include <linux/string.h>
19#include <linux/errno.h>
20#include <linux/init.h>
21#include <linux/types.h>
22#include <linux/delay.h>
23#include <linux/err.h>
24#include <linux/sched.h>
25#include <linux/poll.h>
26#include <linux/wakelock.h>
27#include <linux/platform_device.h>
28#include <linux/uaccess.h>
29#include <linux/debugfs.h>
30
31#include <asm/uaccess.h>
32#include <asm/byteorder.h>
33
34#include <mach/smem_log.h>
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -060035#include <mach/subsystem_notif.h>
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -060036#include <mach/msm_ipc_router.h>
Karthikeyan Ramasubramanian2bfe8ec2013-03-22 10:47:20 -060037#include <mach/msm_ipc_logging.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070038
39#include "ipc_router.h"
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -060040#include "modem_notifier.h"
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -060041#include "msm_ipc_router_security.h"
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070042
43enum {
44 SMEM_LOG = 1U << 0,
45 RTR_DBG = 1U << 1,
46 R2R_MSG = 1U << 2,
47 R2R_RAW = 1U << 3,
48 NTFY_MSG = 1U << 4,
49 R2R_RAW_HDR = 1U << 5,
50};
51
52static int msm_ipc_router_debug_mask;
53module_param_named(debug_mask, msm_ipc_router_debug_mask,
54 int, S_IRUGO | S_IWUSR | S_IWGRP);
55
Karthikeyan Ramasubramanian2bfe8ec2013-03-22 10:47:20 -060056static void *ipc_rtr_log_ctxt;
57#define IPC_RTR_LOG_PAGES 5
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070058#define DIAG(x...) pr_info("[RR] ERROR " x)
59
60#if defined(DEBUG)
61#define D(x...) do { \
Karthikeyan Ramasubramanian2bfe8ec2013-03-22 10:47:20 -060062if (ipc_rtr_log_ctxt) \
63 ipc_log_string(ipc_rtr_log_ctxt, x); \
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070064if (msm_ipc_router_debug_mask & RTR_DBG) \
65 pr_info(x); \
66} while (0)
67
68#define RR(x...) do { \
Karthikeyan Ramasubramanian2bfe8ec2013-03-22 10:47:20 -060069if (ipc_rtr_log_ctxt) \
70 ipc_log_string(ipc_rtr_log_ctxt, x); \
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070071if (msm_ipc_router_debug_mask & R2R_MSG) \
72 pr_info("[RR] "x); \
73} while (0)
74
75#define RAW(x...) do { \
76if (msm_ipc_router_debug_mask & R2R_RAW) \
77 pr_info("[RAW] "x); \
78} while (0)
79
80#define NTFY(x...) do { \
81if (msm_ipc_router_debug_mask & NTFY_MSG) \
82 pr_info("[NOTIFY] "x); \
83} while (0)
84
85#define RAW_HDR(x...) do { \
86if (msm_ipc_router_debug_mask & R2R_RAW_HDR) \
87 pr_info("[HDR] "x); \
88} while (0)
89#else
90#define D(x...) do { } while (0)
91#define RR(x...) do { } while (0)
92#define RAW(x...) do { } while (0)
93#define RAW_HDR(x...) do { } while (0)
94#define NTFY(x...) do { } while (0)
95#endif
96
97#define IPC_ROUTER_LOG_EVENT_ERROR 0x10
98#define IPC_ROUTER_LOG_EVENT_TX 0x11
99#define IPC_ROUTER_LOG_EVENT_RX 0x12
100
101static LIST_HEAD(control_ports);
102static DEFINE_MUTEX(control_ports_lock);
103
104#define LP_HASH_SIZE 32
105static struct list_head local_ports[LP_HASH_SIZE];
106static DEFINE_MUTEX(local_ports_lock);
107
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600108/*
109 * Server info is organized as a hash table. The server's service ID is
110 * used to index into the hash table. The instance ID of most of the servers
111 * are 1 or 2. The service IDs are well distributed compared to the instance
112 * IDs and hence choosing service ID to index into this hash table optimizes
113 * the hash table operations like add, lookup, destroy.
114 */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700115#define SRV_HASH_SIZE 32
116static struct list_head server_list[SRV_HASH_SIZE];
117static DEFINE_MUTEX(server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700118
119struct msm_ipc_server {
120 struct list_head list;
121 struct msm_ipc_port_name name;
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600122 char pdev_name[32];
123 int next_pdev_id;
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -0600124 int synced_sec_rule;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700125 struct list_head server_port_list;
126};
127
128struct msm_ipc_server_port {
129 struct list_head list;
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600130 struct platform_device pdev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700131 struct msm_ipc_port_addr server_addr;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600132 struct msm_ipc_router_xprt_info *xprt_info;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700133};
134
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530135struct msm_ipc_resume_tx_port {
136 struct list_head list;
137 uint32_t port_id;
138 uint32_t node_id;
139};
140
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700141#define RP_HASH_SIZE 32
142struct msm_ipc_router_remote_port {
143 struct list_head list;
144 uint32_t node_id;
145 uint32_t port_id;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600146 uint32_t restart_state;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700147 uint32_t tx_quota_cnt;
148 struct mutex quota_lock;
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530149 struct list_head resume_tx_port_list;
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -0600150 void *sec_rule;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700151};
152
153struct msm_ipc_router_xprt_info {
154 struct list_head list;
155 struct msm_ipc_router_xprt *xprt;
156 uint32_t remote_node_id;
157 uint32_t initialized;
158 struct list_head pkt_list;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700159 struct wake_lock wakelock;
160 struct mutex rx_lock;
161 struct mutex tx_lock;
162 uint32_t need_len;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600163 uint32_t abort_data_read;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700164 struct work_struct read_data;
165 struct workqueue_struct *workqueue;
166};
167
168#define RT_HASH_SIZE 4
169struct msm_ipc_routing_table_entry {
170 struct list_head list;
171 uint32_t node_id;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600172 uint32_t neighbor_node_id;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700173 struct list_head remote_port_list[RP_HASH_SIZE];
174 struct msm_ipc_router_xprt_info *xprt_info;
175 struct mutex lock;
176 unsigned long num_tx_bytes;
177 unsigned long num_rx_bytes;
178};
179
180static struct list_head routing_table[RT_HASH_SIZE];
181static DEFINE_MUTEX(routing_table_lock);
182static int routing_table_inited;
183
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700184static void do_read_data(struct work_struct *work);
185
186#define RR_STATE_IDLE 0
187#define RR_STATE_HEADER 1
188#define RR_STATE_BODY 2
189#define RR_STATE_ERROR 3
190
191#define RESTART_NORMAL 0
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600192#define RESTART_PEND 1
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700193
194/* State for remote ep following restart */
195#define RESTART_QUOTA_ABORT 1
196
197static LIST_HEAD(xprt_info_list);
198static DEFINE_MUTEX(xprt_info_list_lock);
199
Karthikeyan Ramasubramanian4af9f7c2011-09-30 15:55:18 -0600200static DECLARE_COMPLETION(msm_ipc_local_router_up);
201#define IPC_ROUTER_INIT_TIMEOUT (10 * HZ)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700202
203static uint32_t next_port_id;
204static DEFINE_MUTEX(next_port_id_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600205static atomic_t pending_close_count = ATOMIC_INIT(0);
206static wait_queue_head_t subsystem_restart_wait;
207static struct workqueue_struct *msm_ipc_router_workqueue;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700208
209enum {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700210 DOWN,
211 UP,
212};
213
214static void init_routing_table(void)
215{
216 int i;
217 for (i = 0; i < RT_HASH_SIZE; i++)
218 INIT_LIST_HEAD(&routing_table[i]);
219}
220
221static struct msm_ipc_routing_table_entry *alloc_routing_table_entry(
222 uint32_t node_id)
223{
224 int i;
225 struct msm_ipc_routing_table_entry *rt_entry;
226
227 rt_entry = kmalloc(sizeof(struct msm_ipc_routing_table_entry),
228 GFP_KERNEL);
229 if (!rt_entry) {
230 pr_err("%s: rt_entry allocation failed for %d\n",
231 __func__, node_id);
232 return NULL;
233 }
234
235 for (i = 0; i < RP_HASH_SIZE; i++)
236 INIT_LIST_HEAD(&rt_entry->remote_port_list[i]);
237
238 mutex_init(&rt_entry->lock);
239 rt_entry->node_id = node_id;
240 rt_entry->xprt_info = NULL;
241 return rt_entry;
242}
243
244/*Please take routing_table_lock before calling this function*/
245static int add_routing_table_entry(
246 struct msm_ipc_routing_table_entry *rt_entry)
247{
248 uint32_t key;
249
250 if (!rt_entry)
251 return -EINVAL;
252
253 key = (rt_entry->node_id % RT_HASH_SIZE);
254 list_add_tail(&rt_entry->list, &routing_table[key]);
255 return 0;
256}
257
258/*Please take routing_table_lock before calling this function*/
259static struct msm_ipc_routing_table_entry *lookup_routing_table(
260 uint32_t node_id)
261{
262 uint32_t key = (node_id % RT_HASH_SIZE);
263 struct msm_ipc_routing_table_entry *rt_entry;
264
265 list_for_each_entry(rt_entry, &routing_table[key], list) {
266 if (rt_entry->node_id == node_id)
267 return rt_entry;
268 }
269 return NULL;
270}
271
272struct rr_packet *rr_read(struct msm_ipc_router_xprt_info *xprt_info)
273{
274 struct rr_packet *temp_pkt;
275
276 if (!xprt_info)
277 return NULL;
278
279 mutex_lock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600280 if (xprt_info->abort_data_read) {
281 mutex_unlock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -0600282 pr_err("%s detected SSR & exiting now\n",
283 xprt_info->xprt->name);
284 return NULL;
285 }
286
287 if (list_empty(&xprt_info->pkt_list)) {
288 mutex_unlock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600289 return NULL;
290 }
291
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700292 temp_pkt = list_first_entry(&xprt_info->pkt_list,
293 struct rr_packet, list);
294 list_del(&temp_pkt->list);
295 if (list_empty(&xprt_info->pkt_list))
296 wake_unlock(&xprt_info->wakelock);
297 mutex_unlock(&xprt_info->rx_lock);
298 return temp_pkt;
299}
300
301struct rr_packet *clone_pkt(struct rr_packet *pkt)
302{
303 struct rr_packet *cloned_pkt;
304 struct sk_buff *temp_skb, *cloned_skb;
305 struct sk_buff_head *pkt_fragment_q;
306
307 cloned_pkt = kzalloc(sizeof(struct rr_packet), GFP_KERNEL);
308 if (!cloned_pkt) {
309 pr_err("%s: failure\n", __func__);
310 return NULL;
311 }
312
313 pkt_fragment_q = kmalloc(sizeof(struct sk_buff_head), GFP_KERNEL);
314 if (!pkt_fragment_q) {
315 pr_err("%s: pkt_frag_q alloc failure\n", __func__);
316 kfree(cloned_pkt);
317 return NULL;
318 }
319 skb_queue_head_init(pkt_fragment_q);
320
321 skb_queue_walk(pkt->pkt_fragment_q, temp_skb) {
322 cloned_skb = skb_clone(temp_skb, GFP_KERNEL);
323 if (!cloned_skb)
324 goto fail_clone;
325 skb_queue_tail(pkt_fragment_q, cloned_skb);
326 }
327 cloned_pkt->pkt_fragment_q = pkt_fragment_q;
328 cloned_pkt->length = pkt->length;
329 return cloned_pkt;
330
331fail_clone:
332 while (!skb_queue_empty(pkt_fragment_q)) {
333 temp_skb = skb_dequeue(pkt_fragment_q);
334 kfree_skb(temp_skb);
335 }
336 kfree(pkt_fragment_q);
337 kfree(cloned_pkt);
338 return NULL;
339}
340
341struct rr_packet *create_pkt(struct sk_buff_head *data)
342{
343 struct rr_packet *pkt;
344 struct sk_buff *temp_skb;
345
346 pkt = kzalloc(sizeof(struct rr_packet), GFP_KERNEL);
347 if (!pkt) {
348 pr_err("%s: failure\n", __func__);
349 return NULL;
350 }
351
352 pkt->pkt_fragment_q = data;
353 skb_queue_walk(pkt->pkt_fragment_q, temp_skb)
354 pkt->length += temp_skb->len;
355 return pkt;
356}
357
358void release_pkt(struct rr_packet *pkt)
359{
360 struct sk_buff *temp_skb;
361
362 if (!pkt)
363 return;
364
365 if (!pkt->pkt_fragment_q) {
366 kfree(pkt);
367 return;
368 }
369
370 while (!skb_queue_empty(pkt->pkt_fragment_q)) {
371 temp_skb = skb_dequeue(pkt->pkt_fragment_q);
372 kfree_skb(temp_skb);
373 }
374 kfree(pkt->pkt_fragment_q);
375 kfree(pkt);
376 return;
377}
378
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600379static struct sk_buff_head *msm_ipc_router_buf_to_skb(void *buf,
380 unsigned int buf_len)
381{
382 struct sk_buff_head *skb_head;
383 struct sk_buff *skb;
384 int first = 1, offset = 0;
385 int skb_size, data_size;
386 void *data;
387
388 skb_head = kmalloc(sizeof(struct sk_buff_head), GFP_KERNEL);
389 if (!skb_head) {
390 pr_err("%s: Couldnot allocate skb_head\n", __func__);
391 return NULL;
392 }
393 skb_queue_head_init(skb_head);
394
395 data_size = buf_len;
396 while (offset != buf_len) {
397 skb_size = data_size;
398 if (first)
399 skb_size += IPC_ROUTER_HDR_SIZE;
400
401 skb = alloc_skb(skb_size, GFP_KERNEL);
402 if (!skb) {
403 if (skb_size <= (PAGE_SIZE/2)) {
404 pr_err("%s: cannot allocate skb\n", __func__);
405 goto buf_to_skb_error;
406 }
407 data_size = data_size / 2;
408 continue;
409 }
410
411 if (first) {
412 skb_reserve(skb, IPC_ROUTER_HDR_SIZE);
413 first = 0;
414 }
415
416 data = skb_put(skb, data_size);
417 memcpy(skb->data, buf + offset, data_size);
418 skb_queue_tail(skb_head, skb);
419 offset += data_size;
420 data_size = buf_len - offset;
421 }
422 return skb_head;
423
424buf_to_skb_error:
425 while (!skb_queue_empty(skb_head)) {
426 skb = skb_dequeue(skb_head);
427 kfree_skb(skb);
428 }
429 kfree(skb_head);
430 return NULL;
431}
432
433static void *msm_ipc_router_skb_to_buf(struct sk_buff_head *skb_head,
434 unsigned int len)
435{
436 struct sk_buff *temp;
437 int offset = 0, buf_len = 0, copy_len;
438 void *buf;
439
440 if (!skb_head) {
441 pr_err("%s: NULL skb_head\n", __func__);
442 return NULL;
443 }
444
445 temp = skb_peek(skb_head);
446 buf_len = len;
447 buf = kmalloc(buf_len, GFP_KERNEL);
448 if (!buf) {
449 pr_err("%s: cannot allocate buf\n", __func__);
450 return NULL;
451 }
452 skb_queue_walk(skb_head, temp) {
453 copy_len = buf_len < temp->len ? buf_len : temp->len;
454 memcpy(buf + offset, temp->data, copy_len);
455 offset += copy_len;
456 buf_len -= copy_len;
457 }
458 return buf;
459}
460
461static void msm_ipc_router_free_skb(struct sk_buff_head *skb_head)
462{
463 struct sk_buff *temp_skb;
464
465 if (!skb_head)
466 return;
467
468 while (!skb_queue_empty(skb_head)) {
469 temp_skb = skb_dequeue(skb_head);
470 kfree_skb(temp_skb);
471 }
472 kfree(skb_head);
473}
474
Karthikeyan Ramasubramanian6dbb00f2013-05-17 15:19:56 -0600475static int post_pkt_to_port(struct msm_ipc_port *port_ptr,
476 struct rr_packet *pkt, int clone)
477{
478 struct rr_packet *temp_pkt = pkt;
479
480 if (unlikely(!port_ptr || !pkt))
481 return -EINVAL;
482
483 if (clone) {
484 temp_pkt = clone_pkt(pkt);
485 if (!temp_pkt) {
486 pr_err("%s: Error cloning packet for port %08x:%08x\n",
487 __func__, port_ptr->this_port.node_id,
488 port_ptr->this_port.port_id);
489 return -ENOMEM;
490 }
491 }
492
493 mutex_lock(&port_ptr->port_rx_q_lock);
494 wake_lock(&port_ptr->port_rx_wake_lock);
495 list_add_tail(&temp_pkt->list, &port_ptr->port_rx_q);
496 wake_up(&port_ptr->port_rx_wait_q);
497 if (port_ptr->notify)
498 port_ptr->notify(MSM_IPC_ROUTER_READ_CB, port_ptr->priv);
499 mutex_unlock(&port_ptr->port_rx_q_lock);
500 return 0;
501}
502
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700503static int post_control_ports(struct rr_packet *pkt)
504{
505 struct msm_ipc_port *port_ptr;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700506
507 if (!pkt)
508 return -EINVAL;
509
510 mutex_lock(&control_ports_lock);
Karthikeyan Ramasubramanian6dbb00f2013-05-17 15:19:56 -0600511 list_for_each_entry(port_ptr, &control_ports, list)
512 post_pkt_to_port(port_ptr, pkt, 1);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700513 mutex_unlock(&control_ports_lock);
514 return 0;
515}
516
517static uint32_t allocate_port_id(void)
518{
519 uint32_t port_id = 0, prev_port_id, key;
520 struct msm_ipc_port *port_ptr;
521
522 mutex_lock(&next_port_id_lock);
523 prev_port_id = next_port_id;
524 mutex_lock(&local_ports_lock);
525 do {
526 next_port_id++;
527 if ((next_port_id & 0xFFFFFFFE) == 0xFFFFFFFE)
528 next_port_id = 1;
529
530 key = (next_port_id & (LP_HASH_SIZE - 1));
531 if (list_empty(&local_ports[key])) {
532 port_id = next_port_id;
533 break;
534 }
535 list_for_each_entry(port_ptr, &local_ports[key], list) {
536 if (port_ptr->this_port.port_id == next_port_id) {
537 port_id = next_port_id;
538 break;
539 }
540 }
541 if (!port_id) {
542 port_id = next_port_id;
543 break;
544 }
545 port_id = 0;
546 } while (next_port_id != prev_port_id);
547 mutex_unlock(&local_ports_lock);
548 mutex_unlock(&next_port_id_lock);
549
550 return port_id;
551}
552
553void msm_ipc_router_add_local_port(struct msm_ipc_port *port_ptr)
554{
555 uint32_t key;
556
557 if (!port_ptr)
558 return;
559
560 key = (port_ptr->this_port.port_id & (LP_HASH_SIZE - 1));
561 mutex_lock(&local_ports_lock);
562 list_add_tail(&port_ptr->list, &local_ports[key]);
563 mutex_unlock(&local_ports_lock);
564}
565
566struct msm_ipc_port *msm_ipc_router_create_raw_port(void *endpoint,
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600567 void (*notify)(unsigned event, void *priv),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700568 void *priv)
569{
570 struct msm_ipc_port *port_ptr;
571
572 port_ptr = kzalloc(sizeof(struct msm_ipc_port), GFP_KERNEL);
573 if (!port_ptr)
574 return NULL;
575
576 port_ptr->this_port.node_id = IPC_ROUTER_NID_LOCAL;
577 port_ptr->this_port.port_id = allocate_port_id();
578 if (!port_ptr->this_port.port_id) {
579 pr_err("%s: All port ids are in use\n", __func__);
580 kfree(port_ptr);
581 return NULL;
582 }
583
584 spin_lock_init(&port_ptr->port_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700585 INIT_LIST_HEAD(&port_ptr->port_rx_q);
586 mutex_init(&port_ptr->port_rx_q_lock);
587 init_waitqueue_head(&port_ptr->port_rx_wait_q);
Karthikeyan Ramasubramanianb4aeeb92012-03-13 12:31:43 -0600588 snprintf(port_ptr->rx_wakelock_name, MAX_WAKELOCK_NAME_SZ,
Karthikeyan Ramasubramanian6396bdd2013-02-14 13:53:20 -0700589 "ipc%08x_%s",
590 port_ptr->this_port.port_id,
591 current->comm);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700592 wake_lock_init(&port_ptr->port_rx_wake_lock,
Karthikeyan Ramasubramanianb4aeeb92012-03-13 12:31:43 -0600593 WAKE_LOCK_SUSPEND, port_ptr->rx_wakelock_name);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700594
595 port_ptr->endpoint = endpoint;
596 port_ptr->notify = notify;
597 port_ptr->priv = priv;
598
599 msm_ipc_router_add_local_port(port_ptr);
600 return port_ptr;
601}
602
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -0600603/*
604 * Should be called with local_ports_lock locked
605 */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700606static struct msm_ipc_port *msm_ipc_router_lookup_local_port(uint32_t port_id)
607{
608 int key = (port_id & (LP_HASH_SIZE - 1));
609 struct msm_ipc_port *port_ptr;
610
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700611 list_for_each_entry(port_ptr, &local_ports[key], list) {
612 if (port_ptr->this_port.port_id == port_id) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700613 return port_ptr;
614 }
615 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700616 return NULL;
617}
618
619static struct msm_ipc_router_remote_port *msm_ipc_router_lookup_remote_port(
620 uint32_t node_id,
621 uint32_t port_id)
622{
623 struct msm_ipc_router_remote_port *rport_ptr;
624 struct msm_ipc_routing_table_entry *rt_entry;
625 int key = (port_id & (RP_HASH_SIZE - 1));
626
627 mutex_lock(&routing_table_lock);
628 rt_entry = lookup_routing_table(node_id);
629 if (!rt_entry) {
630 mutex_unlock(&routing_table_lock);
631 pr_err("%s: Node is not up\n", __func__);
632 return NULL;
633 }
634
635 mutex_lock(&rt_entry->lock);
636 list_for_each_entry(rport_ptr,
637 &rt_entry->remote_port_list[key], list) {
638 if (rport_ptr->port_id == port_id) {
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600639 if (rport_ptr->restart_state != RESTART_NORMAL)
640 rport_ptr = NULL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700641 mutex_unlock(&rt_entry->lock);
642 mutex_unlock(&routing_table_lock);
643 return rport_ptr;
644 }
645 }
646 mutex_unlock(&rt_entry->lock);
647 mutex_unlock(&routing_table_lock);
648 return NULL;
649}
650
651static struct msm_ipc_router_remote_port *msm_ipc_router_create_remote_port(
652 uint32_t node_id,
653 uint32_t port_id)
654{
655 struct msm_ipc_router_remote_port *rport_ptr;
656 struct msm_ipc_routing_table_entry *rt_entry;
657 int key = (port_id & (RP_HASH_SIZE - 1));
658
659 mutex_lock(&routing_table_lock);
660 rt_entry = lookup_routing_table(node_id);
661 if (!rt_entry) {
662 mutex_unlock(&routing_table_lock);
663 pr_err("%s: Node is not up\n", __func__);
664 return NULL;
665 }
666
667 mutex_lock(&rt_entry->lock);
668 rport_ptr = kmalloc(sizeof(struct msm_ipc_router_remote_port),
669 GFP_KERNEL);
670 if (!rport_ptr) {
671 mutex_unlock(&rt_entry->lock);
672 mutex_unlock(&routing_table_lock);
673 pr_err("%s: Remote port alloc failed\n", __func__);
674 return NULL;
675 }
676 rport_ptr->port_id = port_id;
677 rport_ptr->node_id = node_id;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600678 rport_ptr->restart_state = RESTART_NORMAL;
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -0600679 rport_ptr->sec_rule = NULL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700680 rport_ptr->tx_quota_cnt = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700681 mutex_init(&rport_ptr->quota_lock);
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530682 INIT_LIST_HEAD(&rport_ptr->resume_tx_port_list);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700683 list_add_tail(&rport_ptr->list,
684 &rt_entry->remote_port_list[key]);
685 mutex_unlock(&rt_entry->lock);
686 mutex_unlock(&routing_table_lock);
687 return rport_ptr;
688}
689
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530690/**
691 * msm_ipc_router_free_resume_tx_port() - Free the resume_tx ports
692 * @rport_ptr: Pointer to the remote port.
693 *
694 * This function deletes all the resume_tx ports associated with a remote port
695 * and frees the memory allocated to each resume_tx port.
696 *
697 * Must be called with rport_ptr->quota_lock locked.
698 */
699static void msm_ipc_router_free_resume_tx_port(
700 struct msm_ipc_router_remote_port *rport_ptr)
701{
702 struct msm_ipc_resume_tx_port *rtx_port, *tmp_rtx_port;
703
704 list_for_each_entry_safe(rtx_port, tmp_rtx_port,
705 &rport_ptr->resume_tx_port_list, list) {
706 list_del(&rtx_port->list);
707 kfree(rtx_port);
708 }
709}
710
711/**
712 * msm_ipc_router_lookup_resume_tx_port() - Lookup resume_tx port list
713 * @rport_ptr: Remote port whose resume_tx port list needs to be looked.
714 * @port_id: Port ID which needs to be looked from the list.
715 *
716 * return 1 if the port_id is found in the list, else 0.
717 *
718 * This function is used to lookup the existence of a local port in
719 * remote port's resume_tx list. This function is used to ensure that
720 * the same port is not added to the remote_port's resume_tx list repeatedly.
721 *
722 * Must be called with rport_ptr->quota_lock locked.
723 */
724static int msm_ipc_router_lookup_resume_tx_port(
725 struct msm_ipc_router_remote_port *rport_ptr, uint32_t port_id)
726{
727 struct msm_ipc_resume_tx_port *rtx_port;
728
729 list_for_each_entry(rtx_port, &rport_ptr->resume_tx_port_list, list) {
730 if (port_id == rtx_port->port_id)
731 return 1;
732 }
733 return 0;
734}
735
736/**
737 * post_resume_tx() - Post the resume_tx event
738 * @rport_ptr: Pointer to the remote port
739 * @pkt : The data packet that is received on a resume_tx event
740 *
741 * This function informs about the reception of the resume_tx message from a
742 * remote port pointed by rport_ptr to all the local ports that are in the
743 * resume_tx_ports_list of this remote port. On posting the information, this
744 * function sequentially deletes each entry in the resume_tx_port_list of the
745 * remote port.
746 *
747 * Must be called with rport_ptr->quota_lock locked.
748 */
749static void post_resume_tx(struct msm_ipc_router_remote_port *rport_ptr,
750 struct rr_packet *pkt)
751{
752 struct msm_ipc_resume_tx_port *rtx_port, *tmp_rtx_port;
753 struct msm_ipc_port *local_port;
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530754
755 list_for_each_entry_safe(rtx_port, tmp_rtx_port,
756 &rport_ptr->resume_tx_port_list, list) {
757 mutex_lock(&local_ports_lock);
758 local_port =
759 msm_ipc_router_lookup_local_port(rtx_port->port_id);
Karthikeyan Ramasubramanian6dbb00f2013-05-17 15:19:56 -0600760 if (local_port)
761 post_pkt_to_port(local_port, pkt, 1);
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530762 mutex_unlock(&local_ports_lock);
763 list_del(&rtx_port->list);
764 kfree(rtx_port);
765 }
766}
767
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700768static void msm_ipc_router_destroy_remote_port(
769 struct msm_ipc_router_remote_port *rport_ptr)
770{
771 uint32_t node_id;
772 struct msm_ipc_routing_table_entry *rt_entry;
773
774 if (!rport_ptr)
775 return;
776
777 node_id = rport_ptr->node_id;
778 mutex_lock(&routing_table_lock);
779 rt_entry = lookup_routing_table(node_id);
780 if (!rt_entry) {
781 mutex_unlock(&routing_table_lock);
782 pr_err("%s: Node %d is not up\n", __func__, node_id);
783 return;
784 }
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530785 mutex_lock(&rport_ptr->quota_lock);
786 msm_ipc_router_free_resume_tx_port(rport_ptr);
787 mutex_unlock(&rport_ptr->quota_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700788 mutex_lock(&rt_entry->lock);
789 list_del(&rport_ptr->list);
790 kfree(rport_ptr);
791 mutex_unlock(&rt_entry->lock);
792 mutex_unlock(&routing_table_lock);
793 return;
794}
795
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600796/**
797 * msm_ipc_router_lookup_server() - Lookup server information
798 * @service: Service ID of the server info to be looked up.
799 * @instance: Instance ID of the server info to be looked up.
800 * @node_id: Node/Processor ID in which the server is hosted.
801 * @port_id: Port ID within the node in which the server is hosted.
802 *
803 * @return: If found Pointer to server structure, else NULL.
804 *
805 * Note1: Lock the server_list_lock before accessing this function.
806 * Note2: If the <node_id:port_id> are <0:0>, then the lookup is restricted
807 * to <service:instance>. Used only when a client wants to send a
808 * message to any QMI server.
809 */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700810static struct msm_ipc_server *msm_ipc_router_lookup_server(
811 uint32_t service,
812 uint32_t instance,
813 uint32_t node_id,
814 uint32_t port_id)
815{
816 struct msm_ipc_server *server;
817 struct msm_ipc_server_port *server_port;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600818 int key = (service & (SRV_HASH_SIZE - 1));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700819
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700820 list_for_each_entry(server, &server_list[key], list) {
821 if ((server->name.service != service) ||
822 (server->name.instance != instance))
823 continue;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600824 if ((node_id == 0) && (port_id == 0))
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700825 return server;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700826 list_for_each_entry(server_port, &server->server_port_list,
827 list) {
828 if ((server_port->server_addr.node_id == node_id) &&
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600829 (server_port->server_addr.port_id == port_id))
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700830 return server;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700831 }
832 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700833 return NULL;
834}
835
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600836static void dummy_release(struct device *dev)
837{
838}
839
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600840/**
841 * msm_ipc_router_create_server() - Add server info to hash table
842 * @service: Service ID of the server info to be created.
843 * @instance: Instance ID of the server info to be created.
844 * @node_id: Node/Processor ID in which the server is hosted.
845 * @port_id: Port ID within the node in which the server is hosted.
846 * @xprt_info: XPRT through which the node hosting the server is reached.
847 *
848 * @return: Pointer to server structure on success, else NULL.
849 *
850 * This function adds the server info to the hash table. If the same
851 * server(i.e. <service_id:instance_id>) is hosted in different nodes,
852 * they are maintained as list of "server_port" under "server" structure.
853 * Note: Lock the server_list_lock before accessing this function.
854 */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700855static struct msm_ipc_server *msm_ipc_router_create_server(
856 uint32_t service,
857 uint32_t instance,
858 uint32_t node_id,
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600859 uint32_t port_id,
860 struct msm_ipc_router_xprt_info *xprt_info)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700861{
862 struct msm_ipc_server *server = NULL;
863 struct msm_ipc_server_port *server_port;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600864 int key = (service & (SRV_HASH_SIZE - 1));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700865
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700866 list_for_each_entry(server, &server_list[key], list) {
867 if ((server->name.service == service) &&
868 (server->name.instance == instance))
869 goto create_srv_port;
870 }
871
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600872 server = kzalloc(sizeof(struct msm_ipc_server), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700873 if (!server) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700874 pr_err("%s: Server allocation failed\n", __func__);
875 return NULL;
876 }
877 server->name.service = service;
878 server->name.instance = instance;
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -0600879 server->synced_sec_rule = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700880 INIT_LIST_HEAD(&server->server_port_list);
881 list_add_tail(&server->list, &server_list[key]);
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600882 scnprintf(server->pdev_name, sizeof(server->pdev_name),
883 "QMI%08x:%08x", service, instance);
884 server->next_pdev_id = 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700885
886create_srv_port:
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600887 server_port = kzalloc(sizeof(struct msm_ipc_server_port), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700888 if (!server_port) {
889 if (list_empty(&server->server_port_list)) {
890 list_del(&server->list);
891 kfree(server);
892 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700893 pr_err("%s: Server Port allocation failed\n", __func__);
894 return NULL;
895 }
896 server_port->server_addr.node_id = node_id;
897 server_port->server_addr.port_id = port_id;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600898 server_port->xprt_info = xprt_info;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700899 list_add_tail(&server_port->list, &server->server_port_list);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700900
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600901 server_port->pdev.name = server->pdev_name;
902 server_port->pdev.id = server->next_pdev_id++;
903 server_port->pdev.dev.release = dummy_release;
904 platform_device_register(&server_port->pdev);
905
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700906 return server;
907}
908
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600909/**
910 * msm_ipc_router_destroy_server() - Remove server info from hash table
911 * @server: Server info to be removed.
912 * @node_id: Node/Processor ID in which the server is hosted.
913 * @port_id: Port ID within the node in which the server is hosted.
914 *
915 * This function removes the server_port identified using <node_id:port_id>
916 * from the server structure. If the server_port list under server structure
917 * is empty after removal, then remove the server structure from the server
918 * hash table.
919 * Note: Lock the server_list_lock before accessing this function.
920 */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700921static void msm_ipc_router_destroy_server(struct msm_ipc_server *server,
922 uint32_t node_id, uint32_t port_id)
923{
924 struct msm_ipc_server_port *server_port;
925
926 if (!server)
927 return;
928
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700929 list_for_each_entry(server_port, &server->server_port_list, list) {
930 if ((server_port->server_addr.node_id == node_id) &&
931 (server_port->server_addr.port_id == port_id))
932 break;
933 }
934 if (server_port) {
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600935 platform_device_unregister(&server_port->pdev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700936 list_del(&server_port->list);
937 kfree(server_port);
938 }
939 if (list_empty(&server->server_port_list)) {
940 list_del(&server->list);
941 kfree(server);
942 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700943 return;
944}
945
946static int msm_ipc_router_send_control_msg(
947 struct msm_ipc_router_xprt_info *xprt_info,
948 union rr_control_msg *msg)
949{
950 struct rr_packet *pkt;
951 struct sk_buff *ipc_rtr_pkt;
952 struct rr_header *hdr;
953 int pkt_size;
954 void *data;
955 struct sk_buff_head *pkt_fragment_q;
956 int ret;
957
958 if (!xprt_info || ((msg->cmd != IPC_ROUTER_CTRL_CMD_HELLO) &&
959 !xprt_info->initialized)) {
960 pr_err("%s: xprt_info not initialized\n", __func__);
961 return -EINVAL;
962 }
963
964 if (xprt_info->remote_node_id == IPC_ROUTER_NID_LOCAL)
965 return 0;
966
967 pkt = kzalloc(sizeof(struct rr_packet), GFP_KERNEL);
968 if (!pkt) {
969 pr_err("%s: pkt alloc failed\n", __func__);
970 return -ENOMEM;
971 }
972
973 pkt_fragment_q = kmalloc(sizeof(struct sk_buff_head), GFP_KERNEL);
974 if (!pkt_fragment_q) {
975 pr_err("%s: pkt_fragment_q alloc failed\n", __func__);
976 kfree(pkt);
977 return -ENOMEM;
978 }
979 skb_queue_head_init(pkt_fragment_q);
980
981 pkt_size = IPC_ROUTER_HDR_SIZE + sizeof(*msg);
982 ipc_rtr_pkt = alloc_skb(pkt_size, GFP_KERNEL);
983 if (!ipc_rtr_pkt) {
984 pr_err("%s: ipc_rtr_pkt alloc failed\n", __func__);
985 kfree(pkt_fragment_q);
986 kfree(pkt);
987 return -ENOMEM;
988 }
989
990 skb_reserve(ipc_rtr_pkt, IPC_ROUTER_HDR_SIZE);
991 data = skb_put(ipc_rtr_pkt, sizeof(*msg));
992 memcpy(data, msg, sizeof(*msg));
993 hdr = (struct rr_header *)skb_push(ipc_rtr_pkt, IPC_ROUTER_HDR_SIZE);
994 if (!hdr) {
995 pr_err("%s: skb_push failed\n", __func__);
996 kfree_skb(ipc_rtr_pkt);
997 kfree(pkt_fragment_q);
998 kfree(pkt);
999 return -ENOMEM;
1000 }
1001
1002 hdr->version = IPC_ROUTER_VERSION;
1003 hdr->type = msg->cmd;
1004 hdr->src_node_id = IPC_ROUTER_NID_LOCAL;
1005 hdr->src_port_id = IPC_ROUTER_ADDRESS;
1006 hdr->confirm_rx = 0;
1007 hdr->size = sizeof(*msg);
1008 hdr->dst_node_id = xprt_info->remote_node_id;
1009 hdr->dst_port_id = IPC_ROUTER_ADDRESS;
1010 skb_queue_tail(pkt_fragment_q, ipc_rtr_pkt);
1011 pkt->pkt_fragment_q = pkt_fragment_q;
1012 pkt->length = pkt_size;
1013
1014 mutex_lock(&xprt_info->tx_lock);
Karthikeyan Ramasubramanian8cec5922012-02-16 17:41:58 -07001015 ret = xprt_info->xprt->write(pkt, pkt_size, xprt_info->xprt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001016 mutex_unlock(&xprt_info->tx_lock);
1017
1018 release_pkt(pkt);
1019 return ret;
1020}
1021
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001022static int msm_ipc_router_send_server_list(uint32_t node_id,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001023 struct msm_ipc_router_xprt_info *xprt_info)
1024{
1025 union rr_control_msg ctl;
1026 struct msm_ipc_server *server;
1027 struct msm_ipc_server_port *server_port;
1028 int i;
1029
1030 if (!xprt_info || !xprt_info->initialized) {
1031 pr_err("%s: Xprt info not initialized\n", __func__);
1032 return -EINVAL;
1033 }
1034
1035 ctl.cmd = IPC_ROUTER_CTRL_CMD_NEW_SERVER;
1036
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001037 for (i = 0; i < SRV_HASH_SIZE; i++) {
1038 list_for_each_entry(server, &server_list[i], list) {
1039 ctl.srv.service = server->name.service;
1040 ctl.srv.instance = server->name.instance;
1041 list_for_each_entry(server_port,
1042 &server->server_port_list, list) {
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001043 if (server_port->server_addr.node_id !=
1044 node_id)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001045 continue;
1046
1047 ctl.srv.node_id =
1048 server_port->server_addr.node_id;
1049 ctl.srv.port_id =
1050 server_port->server_addr.port_id;
1051 msm_ipc_router_send_control_msg(xprt_info,
1052 &ctl);
1053 }
1054 }
1055 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001056
1057 return 0;
1058}
1059
1060#if defined(DEBUG)
1061static char *type_to_str(int i)
1062{
1063 switch (i) {
1064 case IPC_ROUTER_CTRL_CMD_DATA:
1065 return "data ";
1066 case IPC_ROUTER_CTRL_CMD_HELLO:
1067 return "hello ";
1068 case IPC_ROUTER_CTRL_CMD_BYE:
1069 return "bye ";
1070 case IPC_ROUTER_CTRL_CMD_NEW_SERVER:
1071 return "new_srvr";
1072 case IPC_ROUTER_CTRL_CMD_REMOVE_SERVER:
1073 return "rmv_srvr";
1074 case IPC_ROUTER_CTRL_CMD_REMOVE_CLIENT:
1075 return "rmv_clnt";
1076 case IPC_ROUTER_CTRL_CMD_RESUME_TX:
1077 return "resum_tx";
1078 case IPC_ROUTER_CTRL_CMD_EXIT:
1079 return "cmd_exit";
1080 default:
1081 return "invalid";
1082 }
1083}
1084#endif
1085
1086static int broadcast_ctl_msg_locally(union rr_control_msg *msg)
1087{
1088 struct rr_packet *pkt;
1089 struct sk_buff *ipc_rtr_pkt;
1090 struct rr_header *hdr;
1091 int pkt_size;
1092 void *data;
1093 struct sk_buff_head *pkt_fragment_q;
1094 int ret;
1095
1096 pkt = kzalloc(sizeof(struct rr_packet), GFP_KERNEL);
1097 if (!pkt) {
1098 pr_err("%s: pkt alloc failed\n", __func__);
1099 return -ENOMEM;
1100 }
1101
1102 pkt_fragment_q = kmalloc(sizeof(struct sk_buff_head), GFP_KERNEL);
1103 if (!pkt_fragment_q) {
1104 pr_err("%s: pkt_fragment_q alloc failed\n", __func__);
1105 kfree(pkt);
1106 return -ENOMEM;
1107 }
1108 skb_queue_head_init(pkt_fragment_q);
1109
1110 pkt_size = IPC_ROUTER_HDR_SIZE + sizeof(*msg);
1111 ipc_rtr_pkt = alloc_skb(pkt_size, GFP_KERNEL);
1112 if (!ipc_rtr_pkt) {
1113 pr_err("%s: ipc_rtr_pkt alloc failed\n", __func__);
1114 kfree(pkt_fragment_q);
1115 kfree(pkt);
1116 return -ENOMEM;
1117 }
1118
1119 skb_reserve(ipc_rtr_pkt, IPC_ROUTER_HDR_SIZE);
1120 data = skb_put(ipc_rtr_pkt, sizeof(*msg));
1121 memcpy(data, msg, sizeof(*msg));
1122 hdr = (struct rr_header *)skb_push(ipc_rtr_pkt, IPC_ROUTER_HDR_SIZE);
1123 if (!hdr) {
1124 pr_err("%s: skb_push failed\n", __func__);
1125 kfree_skb(ipc_rtr_pkt);
1126 kfree(pkt_fragment_q);
1127 kfree(pkt);
1128 return -ENOMEM;
1129 }
1130 hdr->version = IPC_ROUTER_VERSION;
1131 hdr->type = msg->cmd;
1132 hdr->src_node_id = IPC_ROUTER_NID_LOCAL;
1133 hdr->src_port_id = IPC_ROUTER_ADDRESS;
1134 hdr->confirm_rx = 0;
1135 hdr->size = sizeof(*msg);
1136 hdr->dst_node_id = IPC_ROUTER_NID_LOCAL;
1137 hdr->dst_port_id = IPC_ROUTER_ADDRESS;
1138 skb_queue_tail(pkt_fragment_q, ipc_rtr_pkt);
1139 pkt->pkt_fragment_q = pkt_fragment_q;
1140 pkt->length = pkt_size;
1141
1142 ret = post_control_ports(pkt);
1143 release_pkt(pkt);
1144 return ret;
1145}
1146
1147static int broadcast_ctl_msg(union rr_control_msg *ctl)
1148{
1149 struct msm_ipc_router_xprt_info *xprt_info;
1150
1151 mutex_lock(&xprt_info_list_lock);
1152 list_for_each_entry(xprt_info, &xprt_info_list, list) {
1153 msm_ipc_router_send_control_msg(xprt_info, ctl);
1154 }
1155 mutex_unlock(&xprt_info_list_lock);
1156
1157 return 0;
1158}
1159
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001160static int relay_ctl_msg(struct msm_ipc_router_xprt_info *xprt_info,
1161 union rr_control_msg *ctl)
1162{
1163 struct msm_ipc_router_xprt_info *fwd_xprt_info;
1164
1165 if (!xprt_info || !ctl)
1166 return -EINVAL;
1167
1168 mutex_lock(&xprt_info_list_lock);
1169 list_for_each_entry(fwd_xprt_info, &xprt_info_list, list) {
1170 if (xprt_info->xprt->link_id != fwd_xprt_info->xprt->link_id)
1171 msm_ipc_router_send_control_msg(fwd_xprt_info, ctl);
1172 }
1173 mutex_unlock(&xprt_info_list_lock);
1174
1175 return 0;
1176}
1177
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001178static int relay_msg(struct msm_ipc_router_xprt_info *xprt_info,
1179 struct rr_packet *pkt)
1180{
1181 struct msm_ipc_router_xprt_info *fwd_xprt_info;
1182
1183 if (!xprt_info || !pkt)
1184 return -EINVAL;
1185
1186 mutex_lock(&xprt_info_list_lock);
1187 list_for_each_entry(fwd_xprt_info, &xprt_info_list, list) {
1188 mutex_lock(&fwd_xprt_info->tx_lock);
1189 if (xprt_info->xprt->link_id != fwd_xprt_info->xprt->link_id)
Karthikeyan Ramasubramanian8cec5922012-02-16 17:41:58 -07001190 fwd_xprt_info->xprt->write(pkt, pkt->length,
1191 fwd_xprt_info->xprt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001192 mutex_unlock(&fwd_xprt_info->tx_lock);
1193 }
1194 mutex_unlock(&xprt_info_list_lock);
1195 return 0;
1196}
1197
1198static int forward_msg(struct msm_ipc_router_xprt_info *xprt_info,
1199 struct rr_packet *pkt)
1200{
1201 uint32_t dst_node_id;
1202 struct sk_buff *head_pkt;
1203 struct rr_header *hdr;
1204 struct msm_ipc_router_xprt_info *fwd_xprt_info;
1205 struct msm_ipc_routing_table_entry *rt_entry;
1206
1207 if (!xprt_info || !pkt)
1208 return -EINVAL;
1209
1210 head_pkt = skb_peek(pkt->pkt_fragment_q);
1211 if (!head_pkt)
1212 return -EINVAL;
1213
1214 hdr = (struct rr_header *)head_pkt->data;
1215 dst_node_id = hdr->dst_node_id;
1216 mutex_lock(&routing_table_lock);
1217 rt_entry = lookup_routing_table(dst_node_id);
1218 if (!(rt_entry) || !(rt_entry->xprt_info)) {
1219 mutex_unlock(&routing_table_lock);
1220 pr_err("%s: Routing table not initialized\n", __func__);
1221 return -ENODEV;
1222 }
1223
1224 mutex_lock(&rt_entry->lock);
1225 fwd_xprt_info = rt_entry->xprt_info;
1226 mutex_lock(&fwd_xprt_info->tx_lock);
1227 if (xprt_info->remote_node_id == fwd_xprt_info->remote_node_id) {
1228 mutex_unlock(&fwd_xprt_info->tx_lock);
1229 mutex_unlock(&rt_entry->lock);
1230 mutex_unlock(&routing_table_lock);
1231 pr_err("%s: Discarding Command to route back\n", __func__);
1232 return -EINVAL;
1233 }
1234
1235 if (xprt_info->xprt->link_id == fwd_xprt_info->xprt->link_id) {
1236 mutex_unlock(&fwd_xprt_info->tx_lock);
1237 mutex_unlock(&rt_entry->lock);
1238 mutex_unlock(&routing_table_lock);
1239 pr_err("%s: DST in the same cluster\n", __func__);
1240 return 0;
1241 }
Karthikeyan Ramasubramanian8cec5922012-02-16 17:41:58 -07001242 fwd_xprt_info->xprt->write(pkt, pkt->length, fwd_xprt_info->xprt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001243 mutex_unlock(&fwd_xprt_info->tx_lock);
1244 mutex_unlock(&rt_entry->lock);
1245 mutex_unlock(&routing_table_lock);
1246
1247 return 0;
1248}
1249
Karthikeyan Ramasubramanian96cced72013-05-02 17:25:54 -06001250static int msm_ipc_router_send_remove_client(struct comm_mode_info *mode_info,
1251 uint32_t node_id, uint32_t port_id)
1252{
1253 union rr_control_msg msg;
1254 struct msm_ipc_router_xprt_info *tmp_xprt_info;
1255 int mode;
1256 void *xprt_info;
1257 int rc = 0;
1258
1259 if (!mode_info) {
1260 pr_err("%s: NULL mode_info\n", __func__);
1261 return -EINVAL;
1262 }
1263 mode = mode_info->mode;
1264 xprt_info = mode_info->xprt_info;
1265
1266 msg.cmd = IPC_ROUTER_CTRL_CMD_REMOVE_CLIENT;
1267 msg.cli.node_id = node_id;
1268 msg.cli.port_id = port_id;
1269
1270 if ((mode == SINGLE_LINK_MODE) && xprt_info) {
1271 mutex_lock(&xprt_info_list_lock);
1272 list_for_each_entry(tmp_xprt_info, &xprt_info_list, list) {
1273 if (tmp_xprt_info != xprt_info)
1274 continue;
1275 msm_ipc_router_send_control_msg(tmp_xprt_info, &msg);
1276 break;
1277 }
1278 mutex_unlock(&xprt_info_list_lock);
1279 } else if ((mode == SINGLE_LINK_MODE) && !xprt_info) {
1280 broadcast_ctl_msg_locally(&msg);
1281 } else if (mode == MULTI_LINK_MODE) {
1282 broadcast_ctl_msg(&msg);
1283 broadcast_ctl_msg_locally(&msg);
1284 } else if (mode != NULL_MODE) {
1285 pr_err("%s: Invalid mode(%d) + xprt_inf(%p) for %08x:%08x\n",
1286 __func__, mode, xprt_info, node_id, port_id);
1287 rc = -EINVAL;
1288 }
1289 return rc;
1290}
1291
1292static void update_comm_mode_info(struct comm_mode_info *mode_info,
1293 struct msm_ipc_router_xprt_info *xprt_info)
1294{
1295 if (!mode_info) {
1296 pr_err("%s: NULL mode_info\n", __func__);
1297 return;
1298 }
1299
1300 if (mode_info->mode == NULL_MODE) {
1301 mode_info->xprt_info = xprt_info;
1302 mode_info->mode = SINGLE_LINK_MODE;
1303 } else if (mode_info->mode == SINGLE_LINK_MODE &&
1304 mode_info->xprt_info != xprt_info) {
1305 mode_info->mode = MULTI_LINK_MODE;
1306 }
1307
1308 return;
1309}
1310
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001311static void reset_remote_port_info(uint32_t node_id, uint32_t port_id)
1312{
1313 struct msm_ipc_router_remote_port *rport_ptr;
1314
1315 rport_ptr = msm_ipc_router_lookup_remote_port(node_id, port_id);
1316 if (!rport_ptr) {
1317 pr_err("%s: No such remote port %08x:%08x\n",
1318 __func__, node_id, port_id);
1319 return;
1320 }
1321 mutex_lock(&rport_ptr->quota_lock);
1322 rport_ptr->restart_state = RESTART_PEND;
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05301323 msm_ipc_router_free_resume_tx_port(rport_ptr);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001324 mutex_unlock(&rport_ptr->quota_lock);
1325 return;
1326}
1327
1328static void msm_ipc_cleanup_remote_server_info(
1329 struct msm_ipc_router_xprt_info *xprt_info)
1330{
1331 struct msm_ipc_server *svr, *tmp_svr;
1332 struct msm_ipc_server_port *svr_port, *tmp_svr_port;
1333 int i;
1334 union rr_control_msg ctl;
1335
1336 if (!xprt_info) {
1337 pr_err("%s: Invalid xprt_info\n", __func__);
1338 return;
1339 }
1340
1341 ctl.cmd = IPC_ROUTER_CTRL_CMD_REMOVE_SERVER;
1342 mutex_lock(&server_list_lock);
1343 for (i = 0; i < SRV_HASH_SIZE; i++) {
1344 list_for_each_entry_safe(svr, tmp_svr, &server_list[i], list) {
1345 ctl.srv.service = svr->name.service;
1346 ctl.srv.instance = svr->name.instance;
1347 list_for_each_entry_safe(svr_port, tmp_svr_port,
1348 &svr->server_port_list, list) {
1349 if (svr_port->xprt_info != xprt_info)
1350 continue;
1351 D("Remove server %08x:%08x - %08x:%08x",
1352 ctl.srv.service, ctl.srv.instance,
1353 svr_port->server_addr.node_id,
1354 svr_port->server_addr.port_id);
1355 reset_remote_port_info(
1356 svr_port->server_addr.node_id,
1357 svr_port->server_addr.port_id);
1358 ctl.srv.node_id = svr_port->server_addr.node_id;
1359 ctl.srv.port_id = svr_port->server_addr.port_id;
1360 relay_ctl_msg(xprt_info, &ctl);
1361 broadcast_ctl_msg_locally(&ctl);
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06001362 platform_device_unregister(&svr_port->pdev);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001363 list_del(&svr_port->list);
1364 kfree(svr_port);
1365 }
1366 if (list_empty(&svr->server_port_list)) {
1367 list_del(&svr->list);
1368 kfree(svr);
1369 }
1370 }
1371 }
1372 mutex_unlock(&server_list_lock);
1373}
1374
1375static void msm_ipc_cleanup_remote_client_info(
1376 struct msm_ipc_router_xprt_info *xprt_info)
1377{
1378 struct msm_ipc_routing_table_entry *rt_entry;
1379 struct msm_ipc_router_remote_port *rport_ptr;
1380 int i, j;
1381 union rr_control_msg ctl;
1382
1383 if (!xprt_info) {
1384 pr_err("%s: Invalid xprt_info\n", __func__);
1385 return;
1386 }
1387
1388 ctl.cmd = IPC_ROUTER_CTRL_CMD_REMOVE_CLIENT;
1389 mutex_lock(&routing_table_lock);
1390 for (i = 0; i < RT_HASH_SIZE; i++) {
1391 list_for_each_entry(rt_entry, &routing_table[i], list) {
1392 mutex_lock(&rt_entry->lock);
1393 if (rt_entry->xprt_info != xprt_info) {
1394 mutex_unlock(&rt_entry->lock);
1395 continue;
1396 }
1397 for (j = 0; j < RP_HASH_SIZE; j++) {
1398 list_for_each_entry(rport_ptr,
1399 &rt_entry->remote_port_list[j], list) {
1400 if (rport_ptr->restart_state ==
1401 RESTART_PEND)
1402 continue;
1403 mutex_lock(&rport_ptr->quota_lock);
1404 rport_ptr->restart_state = RESTART_PEND;
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05301405 msm_ipc_router_free_resume_tx_port(
1406 rport_ptr);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001407 mutex_unlock(&rport_ptr->quota_lock);
1408 ctl.cli.node_id = rport_ptr->node_id;
1409 ctl.cli.port_id = rport_ptr->port_id;
1410 broadcast_ctl_msg_locally(&ctl);
1411 }
1412 }
1413 mutex_unlock(&rt_entry->lock);
1414 }
1415 }
1416 mutex_unlock(&routing_table_lock);
1417}
1418
1419static void msm_ipc_cleanup_remote_port_info(uint32_t node_id)
1420{
1421 struct msm_ipc_routing_table_entry *rt_entry, *tmp_rt_entry;
1422 struct msm_ipc_router_remote_port *rport_ptr, *tmp_rport_ptr;
1423 int i, j;
1424
1425 mutex_lock(&routing_table_lock);
1426 for (i = 0; i < RT_HASH_SIZE; i++) {
1427 list_for_each_entry_safe(rt_entry, tmp_rt_entry,
1428 &routing_table[i], list) {
1429 mutex_lock(&rt_entry->lock);
1430 if (rt_entry->neighbor_node_id != node_id) {
1431 mutex_unlock(&rt_entry->lock);
1432 continue;
1433 }
1434 for (j = 0; j < RP_HASH_SIZE; j++) {
1435 list_for_each_entry_safe(rport_ptr,
1436 tmp_rport_ptr,
1437 &rt_entry->remote_port_list[j], list) {
1438 list_del(&rport_ptr->list);
1439 kfree(rport_ptr);
1440 }
1441 }
1442 mutex_unlock(&rt_entry->lock);
1443 }
1444 }
1445 mutex_unlock(&routing_table_lock);
1446}
1447
1448static void msm_ipc_cleanup_routing_table(
1449 struct msm_ipc_router_xprt_info *xprt_info)
1450{
1451 int i;
1452 struct msm_ipc_routing_table_entry *rt_entry;
1453
1454 if (!xprt_info) {
1455 pr_err("%s: Invalid xprt_info\n", __func__);
1456 return;
1457 }
1458
1459 mutex_lock(&routing_table_lock);
1460 for (i = 0; i < RT_HASH_SIZE; i++) {
1461 list_for_each_entry(rt_entry, &routing_table[i], list) {
1462 mutex_lock(&rt_entry->lock);
1463 if (rt_entry->xprt_info == xprt_info)
1464 rt_entry->xprt_info = NULL;
1465 mutex_unlock(&rt_entry->lock);
1466 }
1467 }
1468 mutex_unlock(&routing_table_lock);
1469}
1470
1471static void modem_reset_cleanup(struct msm_ipc_router_xprt_info *xprt_info)
1472{
1473
1474 if (!xprt_info) {
1475 pr_err("%s: Invalid xprt_info\n", __func__);
1476 return;
1477 }
1478
1479 msm_ipc_cleanup_remote_server_info(xprt_info);
1480 msm_ipc_cleanup_remote_client_info(xprt_info);
1481 msm_ipc_cleanup_routing_table(xprt_info);
1482}
1483
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -06001484/**
1485 * sync_sec_rule() - Synchrnoize the security rule into the server structure
1486 * @server: Server structure where the rule has to be synchronized.
1487 * @rule: Security tule to be synchronized.
1488 *
1489 * This function is used to update the server structure with the security
1490 * rule configured for the <service:instance> corresponding to that server.
1491 */
1492static void sync_sec_rule(struct msm_ipc_server *server, void *rule)
1493{
1494 struct msm_ipc_server_port *server_port;
1495 struct msm_ipc_router_remote_port *rport_ptr = NULL;
1496
1497 list_for_each_entry(server_port, &server->server_port_list, list) {
1498 rport_ptr = msm_ipc_router_lookup_remote_port(
1499 server_port->server_addr.node_id,
1500 server_port->server_addr.port_id);
1501 if (!rport_ptr)
1502 continue;
1503 rport_ptr->sec_rule = rule;
1504 }
1505 server->synced_sec_rule = 1;
1506}
1507
1508/**
1509 * msm_ipc_sync_sec_rule() - Sync the security rule to the service
1510 * @service: Service for which the rule has to be synchronized.
1511 * @instance: Instance for which the rule has to be synchronized.
1512 * @rule: Security rule to be synchronized.
1513 *
1514 * This function is used to syncrhonize the security rule with the server
1515 * hash table, if the user-space script configures the rule after the service
1516 * has come up. This function is used to synchronize the security rule to a
1517 * specific service and optionally a specific instance.
1518 */
1519void msm_ipc_sync_sec_rule(uint32_t service, uint32_t instance, void *rule)
1520{
1521 int key = (service & (SRV_HASH_SIZE - 1));
1522 struct msm_ipc_server *server;
1523
1524 mutex_lock(&server_list_lock);
1525 list_for_each_entry(server, &server_list[key], list) {
1526 if (server->name.service != service)
1527 continue;
1528
1529 if (server->name.instance != instance &&
1530 instance != ALL_INSTANCE)
1531 continue;
1532
1533 /*
1534 * If the rule applies to all instances and if the specific
1535 * instance of a service has a rule synchronized already,
1536 * do not apply the rule for that specific instance.
1537 */
1538 if (instance == ALL_INSTANCE && server->synced_sec_rule)
1539 continue;
1540
1541 sync_sec_rule(server, rule);
1542 }
1543 mutex_unlock(&server_list_lock);
1544}
1545
1546/**
1547 * msm_ipc_sync_default_sec_rule() - Default security rule to all services
1548 * @rule: Security rule to be synchronized.
1549 *
1550 * This function is used to syncrhonize the security rule with the server
1551 * hash table, if the user-space script configures the rule after the service
1552 * has come up. This function is used to synchronize the security rule that
1553 * applies to all services, if the concerned service do not have any rule
1554 * defined.
1555 */
1556void msm_ipc_sync_default_sec_rule(void *rule)
1557{
1558 int key;
1559 struct msm_ipc_server *server;
1560
1561 mutex_lock(&server_list_lock);
1562 for (key = 0; key < SRV_HASH_SIZE; key++) {
1563 list_for_each_entry(server, &server_list[key], list) {
1564 if (server->synced_sec_rule)
1565 continue;
1566
1567 sync_sec_rule(server, rule);
1568 }
1569 }
1570 mutex_unlock(&server_list_lock);
1571}
1572
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001573static int process_hello_msg(struct msm_ipc_router_xprt_info *xprt_info,
1574 struct rr_header *hdr)
1575{
1576 int i, rc = 0;
1577 union rr_control_msg ctl;
1578 struct msm_ipc_routing_table_entry *rt_entry;
1579
1580 if (!hdr)
1581 return -EINVAL;
1582
1583 RR("o HELLO NID %d\n", hdr->src_node_id);
1584
1585 xprt_info->remote_node_id = hdr->src_node_id;
1586 /*
1587 * Find the entry from Routing Table corresponding to Node ID.
1588 * Under SSR, an entry will be found. When the system boots up
1589 * for the 1st time, an entry will not be found and hence allocate
1590 * an entry. Update the entry with the Node ID that it corresponds
1591 * to and the XPRT through which it can be reached.
1592 */
1593 mutex_lock(&routing_table_lock);
1594 rt_entry = lookup_routing_table(hdr->src_node_id);
1595 if (!rt_entry) {
1596 rt_entry = alloc_routing_table_entry(hdr->src_node_id);
1597 if (!rt_entry) {
1598 mutex_unlock(&routing_table_lock);
1599 pr_err("%s: rt_entry allocation failed\n", __func__);
1600 return -ENOMEM;
1601 }
1602 add_routing_table_entry(rt_entry);
1603 }
1604 mutex_lock(&rt_entry->lock);
1605 rt_entry->neighbor_node_id = xprt_info->remote_node_id;
1606 rt_entry->xprt_info = xprt_info;
1607 mutex_unlock(&rt_entry->lock);
1608 mutex_unlock(&routing_table_lock);
1609
1610 /* Cleanup any remote ports, if the node is coming out of reset */
1611 msm_ipc_cleanup_remote_port_info(xprt_info->remote_node_id);
1612
1613 /* Send a reply HELLO message */
1614 memset(&ctl, 0, sizeof(ctl));
1615 ctl.cmd = IPC_ROUTER_CTRL_CMD_HELLO;
1616 rc = msm_ipc_router_send_control_msg(xprt_info, &ctl);
1617 if (rc < 0) {
1618 pr_err("%s: Error sending reply HELLO message\n", __func__);
1619 return rc;
1620 }
1621 xprt_info->initialized = 1;
1622
1623 /*
1624 * Send list of servers from the local node and from nodes
1625 * outside the mesh network in which this XPRT is part of.
1626 */
Karthikeyan Ramasubramanian05066a82012-10-31 15:12:08 -06001627 mutex_lock(&server_list_lock);
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001628 mutex_lock(&routing_table_lock);
1629 for (i = 0; i < RT_HASH_SIZE; i++) {
1630 list_for_each_entry(rt_entry, &routing_table[i], list) {
1631 if ((rt_entry->node_id != IPC_ROUTER_NID_LOCAL) &&
Karthikeyan Ramasubramanianc4c0aaa2013-01-30 14:17:57 -07001632 (!rt_entry->xprt_info ||
1633 (rt_entry->xprt_info->xprt->link_id ==
1634 xprt_info->xprt->link_id)))
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001635 continue;
1636 rc = msm_ipc_router_send_server_list(rt_entry->node_id,
1637 xprt_info);
1638 if (rc < 0) {
1639 mutex_unlock(&routing_table_lock);
Karthikeyan Ramasubramanian05066a82012-10-31 15:12:08 -06001640 mutex_unlock(&server_list_lock);
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001641 return rc;
1642 }
1643 }
1644 }
1645 mutex_unlock(&routing_table_lock);
Karthikeyan Ramasubramanian05066a82012-10-31 15:12:08 -06001646 mutex_unlock(&server_list_lock);
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001647 RR("HELLO message processed\n");
1648 return rc;
1649}
1650
Karthikeyan Ramasubramaniane2e15fa2013-05-17 18:18:20 -06001651static int process_resume_tx_msg(union rr_control_msg *msg,
1652 struct rr_packet *pkt)
1653{
1654 struct msm_ipc_router_remote_port *rport_ptr;
1655
1656 RR("o RESUME_TX id=%d:%08x\n", msg->cli.node_id, msg->cli.port_id);
1657
1658 rport_ptr = msm_ipc_router_lookup_remote_port(msg->cli.node_id,
1659 msg->cli.port_id);
1660 if (!rport_ptr) {
1661 pr_err("%s: Unable to resume client\n", __func__);
1662 return -ENODEV;
1663 }
1664 mutex_lock(&rport_ptr->quota_lock);
1665 rport_ptr->tx_quota_cnt = 0;
1666 post_resume_tx(rport_ptr, pkt);
1667 mutex_unlock(&rport_ptr->quota_lock);
1668 return 0;
1669}
1670
1671static int process_new_server_msg(struct msm_ipc_router_xprt_info *xprt_info,
1672 union rr_control_msg *msg, struct rr_packet *pkt)
1673{
1674 struct msm_ipc_routing_table_entry *rt_entry;
1675 struct msm_ipc_server *server;
1676 struct msm_ipc_router_remote_port *rport_ptr;
1677
1678 if (msg->srv.instance == 0) {
1679 pr_err("%s: Server %08x create rejected, version = 0\n",
1680 __func__, msg->srv.service);
1681 return -EINVAL;
1682 }
1683
1684 RR("o NEW_SERVER id=%d:%08x service=%08x:%08x\n", msg->srv.node_id,
1685 msg->srv.port_id, msg->srv.service, msg->srv.instance);
1686 /*
1687 * Find the entry from Routing Table corresponding to Node ID.
1688 * Under SSR, an entry will be found. When the subsystem hosting
1689 * service is not adjacent, an entry will not be found and hence
1690 * allocate an entry. Update the entry with the Node ID that it
1691 * corresponds to and the XPRT through which it can be reached.
1692 */
1693 mutex_lock(&routing_table_lock);
1694 rt_entry = lookup_routing_table(msg->srv.node_id);
1695 if (!rt_entry) {
1696 rt_entry = alloc_routing_table_entry(msg->srv.node_id);
1697 if (!rt_entry) {
1698 mutex_unlock(&routing_table_lock);
1699 pr_err("%s: rt_entry allocation failed\n", __func__);
1700 return -ENOMEM;
1701 }
1702 mutex_lock(&rt_entry->lock);
1703 rt_entry->neighbor_node_id = xprt_info->remote_node_id;
1704 rt_entry->xprt_info = xprt_info;
1705 mutex_unlock(&rt_entry->lock);
1706 add_routing_table_entry(rt_entry);
1707 }
1708 mutex_unlock(&routing_table_lock);
1709
1710 /*
1711 * If the service does not exist already in the database, create and
1712 * store the service info. Create a remote port structure in which
1713 * the service is hosted and cache the security rule for the service
1714 * in that remote port structure.
1715 */
1716 mutex_lock(&server_list_lock);
1717 server = msm_ipc_router_lookup_server(msg->srv.service,
1718 msg->srv.instance, msg->srv.node_id, msg->srv.port_id);
1719 if (!server) {
1720 server = msm_ipc_router_create_server(
1721 msg->srv.service, msg->srv.instance,
1722 msg->srv.node_id, msg->srv.port_id, xprt_info);
1723 if (!server) {
1724 mutex_unlock(&server_list_lock);
1725 pr_err("%s: Server Create failed\n", __func__);
1726 return -ENOMEM;
1727 }
1728
1729 if (!msm_ipc_router_lookup_remote_port(
1730 msg->srv.node_id, msg->srv.port_id)) {
1731 rport_ptr = msm_ipc_router_create_remote_port(
1732 msg->srv.node_id, msg->srv.port_id);
1733 if (!rport_ptr) {
1734 mutex_unlock(&server_list_lock);
1735 return -ENOMEM;
1736 }
1737 rport_ptr->sec_rule = msm_ipc_get_security_rule(
1738 msg->srv.service,
1739 msg->srv.instance);
1740 }
1741 }
1742 mutex_unlock(&server_list_lock);
1743
1744 /*
1745 * Relay the new server message to other subsystems that do not belong
1746 * to the cluster from which this message is received. Notify the
1747 * local clients waiting for this service.
1748 */
1749 relay_msg(xprt_info, pkt);
1750 post_control_ports(pkt);
1751 return 0;
1752}
1753
1754static int process_rmv_server_msg(struct msm_ipc_router_xprt_info *xprt_info,
1755 union rr_control_msg *msg, struct rr_packet *pkt)
1756{
1757 struct msm_ipc_server *server;
1758
1759 RR("o REMOVE_SERVER service=%08x:%d\n",
1760 msg->srv.service, msg->srv.instance);
1761 mutex_lock(&server_list_lock);
1762 server = msm_ipc_router_lookup_server(msg->srv.service,
1763 msg->srv.instance, msg->srv.node_id, msg->srv.port_id);
1764 if (server) {
1765 msm_ipc_router_destroy_server(server, msg->srv.node_id,
1766 msg->srv.port_id);
1767 /*
1768 * Relay the new server message to other subsystems that do not
1769 * belong to the cluster from which this message is received.
1770 * Notify the local clients communicating with the service.
1771 */
1772 relay_msg(xprt_info, pkt);
1773 post_control_ports(pkt);
1774 }
1775 mutex_unlock(&server_list_lock);
1776 return 0;
1777}
1778
1779static int process_rmv_client_msg(struct msm_ipc_router_xprt_info *xprt_info,
1780 union rr_control_msg *msg, struct rr_packet *pkt)
1781{
1782 struct msm_ipc_router_remote_port *rport_ptr;
1783
1784 RR("o REMOVE_CLIENT id=%d:%08x\n", msg->cli.node_id, msg->cli.port_id);
1785 rport_ptr = msm_ipc_router_lookup_remote_port(msg->cli.node_id,
1786 msg->cli.port_id);
1787 if (rport_ptr)
1788 msm_ipc_router_destroy_remote_port(rport_ptr);
1789
1790 relay_msg(xprt_info, pkt);
1791 post_control_ports(pkt);
1792 return 0;
1793}
1794
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001795static int process_control_msg(struct msm_ipc_router_xprt_info *xprt_info,
1796 struct rr_packet *pkt)
1797{
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001798 union rr_control_msg *msg;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001799 int rc = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001800 struct sk_buff *temp_ptr;
1801 struct rr_header *hdr;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001802
1803 if (pkt->length != (IPC_ROUTER_HDR_SIZE + sizeof(*msg))) {
1804 pr_err("%s: r2r msg size %d != %d\n", __func__, pkt->length,
1805 (IPC_ROUTER_HDR_SIZE + sizeof(*msg)));
1806 return -EINVAL;
1807 }
1808
1809 temp_ptr = skb_peek(pkt->pkt_fragment_q);
Karthikeyan Ramasubramanian9024dd82011-12-19 18:44:19 -07001810 if (!temp_ptr) {
1811 pr_err("%s: pkt_fragment_q is empty\n", __func__);
1812 return -EINVAL;
1813 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001814 hdr = (struct rr_header *)temp_ptr->data;
Karthikeyan Ramasubramanian9024dd82011-12-19 18:44:19 -07001815 if (!hdr) {
1816 pr_err("%s: No data inside the skb\n", __func__);
1817 return -EINVAL;
1818 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001819 msg = (union rr_control_msg *)((char *)hdr + IPC_ROUTER_HDR_SIZE);
1820
1821 switch (msg->cmd) {
1822 case IPC_ROUTER_CTRL_CMD_HELLO:
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001823 rc = process_hello_msg(xprt_info, hdr);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001824 break;
1825 case IPC_ROUTER_CTRL_CMD_RESUME_TX:
Karthikeyan Ramasubramaniane2e15fa2013-05-17 18:18:20 -06001826 rc = process_resume_tx_msg(msg, pkt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001827 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001828 case IPC_ROUTER_CTRL_CMD_NEW_SERVER:
Karthikeyan Ramasubramaniane2e15fa2013-05-17 18:18:20 -06001829 rc = process_new_server_msg(xprt_info, msg, pkt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001830 break;
1831 case IPC_ROUTER_CTRL_CMD_REMOVE_SERVER:
Karthikeyan Ramasubramaniane2e15fa2013-05-17 18:18:20 -06001832 rc = process_rmv_server_msg(xprt_info, msg, pkt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001833 break;
1834 case IPC_ROUTER_CTRL_CMD_REMOVE_CLIENT:
Karthikeyan Ramasubramaniane2e15fa2013-05-17 18:18:20 -06001835 rc = process_rmv_client_msg(xprt_info, msg, pkt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001836 break;
1837 case IPC_ROUTER_CTRL_CMD_PING:
1838 /* No action needed for ping messages received */
1839 RR("o PING\n");
1840 break;
1841 default:
1842 RR("o UNKNOWN(%08x)\n", msg->cmd);
1843 rc = -ENOSYS;
1844 }
1845
1846 return rc;
1847}
1848
1849static void do_read_data(struct work_struct *work)
1850{
1851 struct rr_header *hdr;
1852 struct rr_packet *pkt = NULL;
1853 struct msm_ipc_port *port_ptr;
1854 struct sk_buff *head_skb;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001855 struct msm_ipc_router_remote_port *rport_ptr;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001856 uint32_t resume_tx, resume_tx_node_id, resume_tx_port_id;
1857
1858 struct msm_ipc_router_xprt_info *xprt_info =
1859 container_of(work,
1860 struct msm_ipc_router_xprt_info,
1861 read_data);
1862
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001863 while ((pkt = rr_read(xprt_info)) != NULL) {
1864 if (pkt->length < IPC_ROUTER_HDR_SIZE ||
1865 pkt->length > MAX_IPC_PKT_SIZE) {
1866 pr_err("%s: Invalid pkt length %d\n",
1867 __func__, pkt->length);
1868 goto fail_data;
1869 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001870
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001871 head_skb = skb_peek(pkt->pkt_fragment_q);
1872 if (!head_skb) {
1873 pr_err("%s: head_skb is invalid\n", __func__);
1874 goto fail_data;
1875 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001876
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001877 hdr = (struct rr_header *)(head_skb->data);
Karthikeyan Ramasubramanian2bfe8ec2013-03-22 10:47:20 -06001878 RAW("ver=%d type=%d src=%d:%08x crx=%d siz=%d dst=%d:%08x\n",
1879 hdr->version, hdr->type, hdr->src_node_id,
1880 hdr->src_port_id, hdr->confirm_rx, hdr->size,
1881 hdr->dst_node_id, hdr->dst_port_id);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001882
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001883 if (hdr->version != IPC_ROUTER_VERSION) {
1884 pr_err("version %d != %d\n",
1885 hdr->version, IPC_ROUTER_VERSION);
1886 goto fail_data;
1887 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001888
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001889 if ((hdr->dst_node_id != IPC_ROUTER_NID_LOCAL) &&
1890 ((hdr->type == IPC_ROUTER_CTRL_CMD_RESUME_TX) ||
1891 (hdr->type == IPC_ROUTER_CTRL_CMD_DATA))) {
1892 forward_msg(xprt_info, pkt);
1893 release_pkt(pkt);
1894 continue;
1895 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001896
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001897 if ((hdr->dst_port_id == IPC_ROUTER_ADDRESS) ||
1898 (hdr->type == IPC_ROUTER_CTRL_CMD_HELLO)) {
1899 process_control_msg(xprt_info, pkt);
1900 release_pkt(pkt);
1901 continue;
1902 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001903#if defined(CONFIG_MSM_SMD_LOGGING)
1904#if defined(DEBUG)
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001905 if (msm_ipc_router_debug_mask & SMEM_LOG) {
1906 smem_log_event((SMEM_LOG_PROC_ID_APPS |
1907 SMEM_LOG_RPC_ROUTER_EVENT_BASE |
1908 IPC_ROUTER_LOG_EVENT_RX),
1909 (hdr->src_node_id << 24) |
1910 (hdr->src_port_id & 0xffffff),
1911 (hdr->dst_node_id << 24) |
1912 (hdr->dst_port_id & 0xffffff),
1913 (hdr->type << 24) | (hdr->confirm_rx << 16) |
1914 (hdr->size & 0xffff));
1915 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001916#endif
1917#endif
1918
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001919 resume_tx = hdr->confirm_rx;
1920 resume_tx_node_id = hdr->dst_node_id;
1921 resume_tx_port_id = hdr->dst_port_id;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001922
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001923 rport_ptr = msm_ipc_router_lookup_remote_port(hdr->src_node_id,
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06001924 hdr->src_port_id);
1925
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001926 mutex_lock(&local_ports_lock);
1927 port_ptr = msm_ipc_router_lookup_local_port(hdr->dst_port_id);
1928 if (!port_ptr) {
1929 pr_err("%s: No local port id %08x\n", __func__,
1930 hdr->dst_port_id);
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06001931 mutex_unlock(&local_ports_lock);
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001932 release_pkt(pkt);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001933 goto process_done;
1934 }
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001935
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001936 if (!rport_ptr) {
1937 rport_ptr = msm_ipc_router_create_remote_port(
1938 hdr->src_node_id,
1939 hdr->src_port_id);
1940 if (!rport_ptr) {
1941 pr_err("%s: Rmt Prt %08x:%08x create failed\n",
1942 __func__, hdr->src_node_id,
1943 hdr->src_port_id);
1944 mutex_unlock(&local_ports_lock);
1945 goto process_done;
1946 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001947 }
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001948
Karthikeyan Ramasubramanian6dbb00f2013-05-17 15:19:56 -06001949 post_pkt_to_port(port_ptr, pkt, 0);
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06001950 mutex_unlock(&local_ports_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001951
1952process_done:
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001953 if (resume_tx) {
1954 union rr_control_msg msg;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001955
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001956 msg.cmd = IPC_ROUTER_CTRL_CMD_RESUME_TX;
1957 msg.cli.node_id = resume_tx_node_id;
1958 msg.cli.port_id = resume_tx_port_id;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001959
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001960 RR("x RESUME_TX id=%d:%08x\n",
1961 msg.cli.node_id, msg.cli.port_id);
1962 msm_ipc_router_send_control_msg(xprt_info, &msg);
1963 }
1964
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001965 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001966 return;
1967
1968fail_data:
1969 release_pkt(pkt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001970 pr_err("ipc_router has died\n");
1971}
1972
1973int msm_ipc_router_register_server(struct msm_ipc_port *port_ptr,
1974 struct msm_ipc_addr *name)
1975{
1976 struct msm_ipc_server *server;
1977 unsigned long flags;
1978 union rr_control_msg ctl;
1979
1980 if (!port_ptr || !name)
1981 return -EINVAL;
1982
1983 if (name->addrtype != MSM_IPC_ADDR_NAME)
1984 return -EINVAL;
1985
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06001986 mutex_lock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001987 server = msm_ipc_router_lookup_server(name->addr.port_name.service,
1988 name->addr.port_name.instance,
1989 IPC_ROUTER_NID_LOCAL,
1990 port_ptr->this_port.port_id);
1991 if (server) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06001992 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001993 pr_err("%s: Server already present\n", __func__);
1994 return -EINVAL;
1995 }
1996
1997 server = msm_ipc_router_create_server(name->addr.port_name.service,
1998 name->addr.port_name.instance,
1999 IPC_ROUTER_NID_LOCAL,
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002000 port_ptr->this_port.port_id,
2001 NULL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002002 if (!server) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002003 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002004 pr_err("%s: Server Creation failed\n", __func__);
2005 return -EINVAL;
2006 }
2007
2008 ctl.cmd = IPC_ROUTER_CTRL_CMD_NEW_SERVER;
2009 ctl.srv.service = server->name.service;
2010 ctl.srv.instance = server->name.instance;
2011 ctl.srv.node_id = IPC_ROUTER_NID_LOCAL;
2012 ctl.srv.port_id = port_ptr->this_port.port_id;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002013 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002014 broadcast_ctl_msg(&ctl);
2015 spin_lock_irqsave(&port_ptr->port_lock, flags);
2016 port_ptr->type = SERVER_PORT;
Karthikeyan Ramasubramanian96cced72013-05-02 17:25:54 -06002017 port_ptr->mode_info.mode = MULTI_LINK_MODE;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002018 port_ptr->port_name.service = server->name.service;
2019 port_ptr->port_name.instance = server->name.instance;
2020 spin_unlock_irqrestore(&port_ptr->port_lock, flags);
2021 return 0;
2022}
2023
2024int msm_ipc_router_unregister_server(struct msm_ipc_port *port_ptr)
2025{
2026 struct msm_ipc_server *server;
2027 unsigned long flags;
2028 union rr_control_msg ctl;
2029
2030 if (!port_ptr)
2031 return -EINVAL;
2032
2033 if (port_ptr->type != SERVER_PORT) {
2034 pr_err("%s: Trying to unregister a non-server port\n",
2035 __func__);
2036 return -EINVAL;
2037 }
2038
2039 if (port_ptr->this_port.node_id != IPC_ROUTER_NID_LOCAL) {
2040 pr_err("%s: Trying to unregister a remote server locally\n",
2041 __func__);
2042 return -EINVAL;
2043 }
2044
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002045 mutex_lock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002046 server = msm_ipc_router_lookup_server(port_ptr->port_name.service,
2047 port_ptr->port_name.instance,
2048 port_ptr->this_port.node_id,
2049 port_ptr->this_port.port_id);
2050 if (!server) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002051 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002052 pr_err("%s: Server lookup failed\n", __func__);
2053 return -ENODEV;
2054 }
2055
2056 ctl.cmd = IPC_ROUTER_CTRL_CMD_REMOVE_SERVER;
2057 ctl.srv.service = server->name.service;
2058 ctl.srv.instance = server->name.instance;
2059 ctl.srv.node_id = IPC_ROUTER_NID_LOCAL;
2060 ctl.srv.port_id = port_ptr->this_port.port_id;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002061 msm_ipc_router_destroy_server(server, port_ptr->this_port.node_id,
2062 port_ptr->this_port.port_id);
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002063 mutex_unlock(&server_list_lock);
2064 broadcast_ctl_msg(&ctl);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002065 spin_lock_irqsave(&port_ptr->port_lock, flags);
2066 port_ptr->type = CLIENT_PORT;
2067 spin_unlock_irqrestore(&port_ptr->port_lock, flags);
2068 return 0;
2069}
2070
2071static int loopback_data(struct msm_ipc_port *src,
2072 uint32_t port_id,
2073 struct sk_buff_head *data)
2074{
2075 struct sk_buff *head_skb;
2076 struct rr_header *hdr;
2077 struct msm_ipc_port *port_ptr;
2078 struct rr_packet *pkt;
Karthikeyan Ramasubramanianb0e23c52013-02-08 13:07:42 -07002079 int ret_len;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002080
2081 if (!data) {
2082 pr_err("%s: Invalid pkt pointer\n", __func__);
2083 return -EINVAL;
2084 }
2085
2086 pkt = create_pkt(data);
2087 if (!pkt) {
2088 pr_err("%s: New pkt create failed\n", __func__);
2089 return -ENOMEM;
2090 }
2091
2092 head_skb = skb_peek(pkt->pkt_fragment_q);
Karthikeyan Ramasubramanian9024dd82011-12-19 18:44:19 -07002093 if (!head_skb) {
2094 pr_err("%s: pkt_fragment_q is empty\n", __func__);
Brent Hronik1c9a3d42013-04-17 15:10:40 -06002095 release_pkt(pkt);
Karthikeyan Ramasubramanian9024dd82011-12-19 18:44:19 -07002096 return -EINVAL;
2097 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002098 hdr = (struct rr_header *)skb_push(head_skb, IPC_ROUTER_HDR_SIZE);
2099 if (!hdr) {
2100 pr_err("%s: Prepend Header failed\n", __func__);
2101 release_pkt(pkt);
2102 return -ENOMEM;
2103 }
2104 hdr->version = IPC_ROUTER_VERSION;
2105 hdr->type = IPC_ROUTER_CTRL_CMD_DATA;
2106 hdr->src_node_id = src->this_port.node_id;
2107 hdr->src_port_id = src->this_port.port_id;
2108 hdr->size = pkt->length;
2109 hdr->confirm_rx = 0;
2110 hdr->dst_node_id = IPC_ROUTER_NID_LOCAL;
2111 hdr->dst_port_id = port_id;
2112 pkt->length += IPC_ROUTER_HDR_SIZE;
2113
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06002114 mutex_lock(&local_ports_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002115 port_ptr = msm_ipc_router_lookup_local_port(port_id);
2116 if (!port_ptr) {
2117 pr_err("%s: Local port %d not present\n", __func__, port_id);
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06002118 mutex_unlock(&local_ports_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002119 release_pkt(pkt);
2120 return -ENODEV;
2121 }
2122
Karthikeyan Ramasubramanianb0e23c52013-02-08 13:07:42 -07002123 ret_len = pkt->length;
Karthikeyan Ramasubramanian6dbb00f2013-05-17 15:19:56 -06002124 post_pkt_to_port(port_ptr, pkt, 0);
Karthikeyan Ramasubramanian96cced72013-05-02 17:25:54 -06002125 update_comm_mode_info(&src->mode_info, NULL);
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06002126 mutex_unlock(&local_ports_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002127
Karthikeyan Ramasubramanianb0e23c52013-02-08 13:07:42 -07002128 return ret_len;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002129}
2130
2131static int msm_ipc_router_write_pkt(struct msm_ipc_port *src,
2132 struct msm_ipc_router_remote_port *rport_ptr,
2133 struct rr_packet *pkt)
2134{
2135 struct sk_buff *head_skb;
2136 struct rr_header *hdr;
2137 struct msm_ipc_router_xprt_info *xprt_info;
2138 struct msm_ipc_routing_table_entry *rt_entry;
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05302139 struct msm_ipc_resume_tx_port *resume_tx_port;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002140 int ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002141
2142 if (!rport_ptr || !src || !pkt)
2143 return -EINVAL;
2144
2145 head_skb = skb_peek(pkt->pkt_fragment_q);
Karthikeyan Ramasubramanian9024dd82011-12-19 18:44:19 -07002146 if (!head_skb) {
2147 pr_err("%s: pkt_fragment_q is empty\n", __func__);
2148 return -EINVAL;
2149 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002150 hdr = (struct rr_header *)skb_push(head_skb, IPC_ROUTER_HDR_SIZE);
2151 if (!hdr) {
2152 pr_err("%s: Prepend Header failed\n", __func__);
2153 return -ENOMEM;
2154 }
2155 hdr->version = IPC_ROUTER_VERSION;
2156 hdr->type = IPC_ROUTER_CTRL_CMD_DATA;
2157 hdr->src_node_id = src->this_port.node_id;
2158 hdr->src_port_id = src->this_port.port_id;
2159 hdr->size = pkt->length;
2160 hdr->confirm_rx = 0;
2161 hdr->dst_node_id = rport_ptr->node_id;
2162 hdr->dst_port_id = rport_ptr->port_id;
2163 pkt->length += IPC_ROUTER_HDR_SIZE;
2164
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05302165 mutex_lock(&rport_ptr->quota_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002166 if (rport_ptr->restart_state != RESTART_NORMAL) {
2167 mutex_unlock(&rport_ptr->quota_lock);
2168 return -ENETRESET;
2169 }
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05302170 if (rport_ptr->tx_quota_cnt == IPC_ROUTER_DEFAULT_RX_QUOTA) {
2171 if (msm_ipc_router_lookup_resume_tx_port(
2172 rport_ptr, src->this_port.port_id)) {
2173 mutex_unlock(&rport_ptr->quota_lock);
2174 return -EAGAIN;
2175 }
2176 resume_tx_port =
2177 kzalloc(sizeof(struct msm_ipc_resume_tx_port),
2178 GFP_KERNEL);
2179 if (!resume_tx_port) {
2180 pr_err("%s: Resume_Tx port allocation failed\n",
2181 __func__);
2182 mutex_unlock(&rport_ptr->quota_lock);
2183 return -ENOMEM;
2184 }
2185 INIT_LIST_HEAD(&resume_tx_port->list);
2186 resume_tx_port->port_id = src->this_port.port_id;
2187 resume_tx_port->node_id = src->this_port.node_id;
2188 list_add_tail(&resume_tx_port->list,
2189 &rport_ptr->resume_tx_port_list);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002190 mutex_unlock(&rport_ptr->quota_lock);
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05302191 return -EAGAIN;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002192 }
2193 rport_ptr->tx_quota_cnt++;
2194 if (rport_ptr->tx_quota_cnt == IPC_ROUTER_DEFAULT_RX_QUOTA)
2195 hdr->confirm_rx = 1;
2196 mutex_unlock(&rport_ptr->quota_lock);
2197
2198 mutex_lock(&routing_table_lock);
2199 rt_entry = lookup_routing_table(hdr->dst_node_id);
2200 if (!rt_entry || !rt_entry->xprt_info) {
2201 mutex_unlock(&routing_table_lock);
2202 pr_err("%s: Remote node %d not up\n",
2203 __func__, hdr->dst_node_id);
2204 return -ENODEV;
2205 }
2206 mutex_lock(&rt_entry->lock);
2207 xprt_info = rt_entry->xprt_info;
2208 mutex_lock(&xprt_info->tx_lock);
Karthikeyan Ramasubramanian8cec5922012-02-16 17:41:58 -07002209 ret = xprt_info->xprt->write(pkt, pkt->length, xprt_info->xprt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002210 mutex_unlock(&xprt_info->tx_lock);
2211 mutex_unlock(&rt_entry->lock);
2212 mutex_unlock(&routing_table_lock);
2213
2214 if (ret < 0) {
2215 pr_err("%s: Write on XPRT failed\n", __func__);
2216 return ret;
2217 }
Karthikeyan Ramasubramanian96cced72013-05-02 17:25:54 -06002218 update_comm_mode_info(&src->mode_info, xprt_info);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002219
2220 RAW_HDR("[w rr_h] "
2221 "ver=%i,type=%s,src_nid=%08x,src_port_id=%08x,"
2222 "confirm_rx=%i,size=%3i,dst_pid=%08x,dst_cid=%08x\n",
2223 hdr->version, type_to_str(hdr->type),
2224 hdr->src_node_id, hdr->src_port_id,
2225 hdr->confirm_rx, hdr->size,
2226 hdr->dst_node_id, hdr->dst_port_id);
2227
2228#if defined(CONFIG_MSM_SMD_LOGGING)
2229#if defined(DEBUG)
2230 if (msm_ipc_router_debug_mask & SMEM_LOG) {
2231 smem_log_event((SMEM_LOG_PROC_ID_APPS |
2232 SMEM_LOG_RPC_ROUTER_EVENT_BASE |
2233 IPC_ROUTER_LOG_EVENT_TX),
2234 (hdr->src_node_id << 24) |
2235 (hdr->src_port_id & 0xffffff),
2236 (hdr->dst_node_id << 24) |
2237 (hdr->dst_port_id & 0xffffff),
2238 (hdr->type << 24) | (hdr->confirm_rx << 16) |
2239 (hdr->size & 0xffff));
2240 }
2241#endif
2242#endif
2243
2244 return pkt->length;
2245}
2246
2247int msm_ipc_router_send_to(struct msm_ipc_port *src,
2248 struct sk_buff_head *data,
2249 struct msm_ipc_addr *dest)
2250{
2251 uint32_t dst_node_id = 0, dst_port_id = 0;
2252 struct msm_ipc_server *server;
2253 struct msm_ipc_server_port *server_port;
2254 struct msm_ipc_router_remote_port *rport_ptr = NULL;
2255 struct rr_packet *pkt;
2256 int ret;
2257
2258 if (!src || !data || !dest) {
2259 pr_err("%s: Invalid Parameters\n", __func__);
2260 return -EINVAL;
2261 }
2262
2263 /* Resolve Address*/
2264 if (dest->addrtype == MSM_IPC_ADDR_ID) {
2265 dst_node_id = dest->addr.port_addr.node_id;
2266 dst_port_id = dest->addr.port_addr.port_id;
2267 } else if (dest->addrtype == MSM_IPC_ADDR_NAME) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002268 mutex_lock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002269 server = msm_ipc_router_lookup_server(
2270 dest->addr.port_name.service,
2271 dest->addr.port_name.instance,
2272 0, 0);
2273 if (!server) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002274 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002275 pr_err("%s: Destination not reachable\n", __func__);
2276 return -ENODEV;
2277 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002278 server_port = list_first_entry(&server->server_port_list,
2279 struct msm_ipc_server_port,
2280 list);
2281 dst_node_id = server_port->server_addr.node_id;
2282 dst_port_id = server_port->server_addr.port_id;
2283 mutex_unlock(&server_list_lock);
2284 }
2285 if (dst_node_id == IPC_ROUTER_NID_LOCAL) {
2286 ret = loopback_data(src, dst_port_id, data);
2287 return ret;
2288 }
2289
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002290 rport_ptr = msm_ipc_router_lookup_remote_port(dst_node_id,
2291 dst_port_id);
2292 if (!rport_ptr) {
Zaheerulla Meer3d8b0a02013-05-10 15:51:28 +05302293 pr_err("%s: Remote port not found\n", __func__);
2294 return -ENODEV;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002295 }
2296
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -06002297 if (src->check_send_permissions) {
2298 ret = src->check_send_permissions(rport_ptr->sec_rule);
2299 if (ret <= 0) {
2300 pr_err("%s: permission failure for %s\n",
2301 __func__, current->comm);
2302 return -EPERM;
2303 }
2304 }
2305
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002306 pkt = create_pkt(data);
2307 if (!pkt) {
2308 pr_err("%s: Pkt creation failed\n", __func__);
2309 return -ENOMEM;
2310 }
2311
2312 ret = msm_ipc_router_write_pkt(src, rport_ptr, pkt);
2313 release_pkt(pkt);
2314
2315 return ret;
2316}
2317
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06002318int msm_ipc_router_send_msg(struct msm_ipc_port *src,
2319 struct msm_ipc_addr *dest,
2320 void *data, unsigned int data_len)
2321{
2322 struct sk_buff_head *out_skb_head;
2323 int ret;
2324
2325 out_skb_head = msm_ipc_router_buf_to_skb(data, data_len);
2326 if (!out_skb_head) {
2327 pr_err("%s: SKB conversion failed\n", __func__);
2328 return -EFAULT;
2329 }
2330
2331 ret = msm_ipc_router_send_to(src, out_skb_head, dest);
2332 if (ret < 0) {
2333 pr_err("%s: msm_ipc_router_send_to failed - ret: %d\n",
2334 __func__, ret);
2335 msm_ipc_router_free_skb(out_skb_head);
2336 }
2337 return 0;
2338}
2339
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002340int msm_ipc_router_read(struct msm_ipc_port *port_ptr,
2341 struct sk_buff_head **data,
2342 size_t buf_len)
2343{
2344 struct rr_packet *pkt;
2345 int ret;
2346
2347 if (!port_ptr || !data)
2348 return -EINVAL;
2349
2350 mutex_lock(&port_ptr->port_rx_q_lock);
2351 if (list_empty(&port_ptr->port_rx_q)) {
2352 mutex_unlock(&port_ptr->port_rx_q_lock);
2353 return -EAGAIN;
2354 }
2355
2356 pkt = list_first_entry(&port_ptr->port_rx_q, struct rr_packet, list);
2357 if ((buf_len) && ((pkt->length - IPC_ROUTER_HDR_SIZE) > buf_len)) {
2358 mutex_unlock(&port_ptr->port_rx_q_lock);
2359 return -ETOOSMALL;
2360 }
2361 list_del(&pkt->list);
2362 if (list_empty(&port_ptr->port_rx_q))
2363 wake_unlock(&port_ptr->port_rx_wake_lock);
2364 *data = pkt->pkt_fragment_q;
2365 ret = pkt->length;
2366 kfree(pkt);
2367 mutex_unlock(&port_ptr->port_rx_q_lock);
2368
2369 return ret;
2370}
2371
2372int msm_ipc_router_recv_from(struct msm_ipc_port *port_ptr,
2373 struct sk_buff_head **data,
2374 struct msm_ipc_addr *src,
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06002375 long timeout)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002376{
2377 int ret, data_len, align_size;
2378 struct sk_buff *temp_skb;
2379 struct rr_header *hdr = NULL;
2380
2381 if (!port_ptr || !data) {
2382 pr_err("%s: Invalid pointers being passed\n", __func__);
2383 return -EINVAL;
2384 }
2385
2386 *data = NULL;
2387 mutex_lock(&port_ptr->port_rx_q_lock);
2388 while (list_empty(&port_ptr->port_rx_q)) {
2389 mutex_unlock(&port_ptr->port_rx_q_lock);
2390 if (timeout < 0) {
2391 ret = wait_event_interruptible(
2392 port_ptr->port_rx_wait_q,
2393 !list_empty(&port_ptr->port_rx_q));
2394 if (ret)
2395 return ret;
2396 } else if (timeout > 0) {
2397 timeout = wait_event_interruptible_timeout(
2398 port_ptr->port_rx_wait_q,
2399 !list_empty(&port_ptr->port_rx_q),
2400 timeout);
2401 if (timeout < 0)
2402 return -EFAULT;
2403 }
2404 if (timeout == 0)
2405 return -ETIMEDOUT;
2406 mutex_lock(&port_ptr->port_rx_q_lock);
2407 }
2408 mutex_unlock(&port_ptr->port_rx_q_lock);
2409
2410 ret = msm_ipc_router_read(port_ptr, data, 0);
2411 if (ret <= 0 || !(*data))
2412 return ret;
2413
2414 temp_skb = skb_peek(*data);
2415 hdr = (struct rr_header *)(temp_skb->data);
2416 if (src) {
2417 src->addrtype = MSM_IPC_ADDR_ID;
2418 src->addr.port_addr.node_id = hdr->src_node_id;
2419 src->addr.port_addr.port_id = hdr->src_port_id;
2420 }
2421
2422 data_len = hdr->size;
2423 skb_pull(temp_skb, IPC_ROUTER_HDR_SIZE);
2424 align_size = ALIGN_SIZE(data_len);
2425 if (align_size) {
2426 temp_skb = skb_peek_tail(*data);
2427 skb_trim(temp_skb, (temp_skb->len - align_size));
2428 }
2429 return data_len;
2430}
2431
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06002432int msm_ipc_router_read_msg(struct msm_ipc_port *port_ptr,
2433 struct msm_ipc_addr *src,
2434 unsigned char **data,
2435 unsigned int *len)
2436{
2437 struct sk_buff_head *in_skb_head;
2438 int ret;
2439
2440 ret = msm_ipc_router_recv_from(port_ptr, &in_skb_head, src, -1);
2441 if (ret < 0) {
2442 pr_err("%s: msm_ipc_router_recv_from failed - ret: %d\n",
2443 __func__, ret);
2444 return ret;
2445 }
2446
2447 *data = msm_ipc_router_skb_to_buf(in_skb_head, ret);
2448 if (!(*data))
2449 pr_err("%s: Buf conversion failed\n", __func__);
2450
2451 *len = ret;
2452 msm_ipc_router_free_skb(in_skb_head);
2453 return 0;
2454}
2455
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002456struct msm_ipc_port *msm_ipc_router_create_port(
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06002457 void (*notify)(unsigned event, void *priv),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002458 void *priv)
2459{
2460 struct msm_ipc_port *port_ptr;
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06002461 int ret;
2462
2463 ret = wait_for_completion_interruptible(&msm_ipc_local_router_up);
2464 if (ret < 0) {
2465 pr_err("%s: Error waiting for local router\n", __func__);
2466 return NULL;
2467 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002468
2469 port_ptr = msm_ipc_router_create_raw_port(NULL, notify, priv);
2470 if (!port_ptr)
2471 pr_err("%s: port_ptr alloc failed\n", __func__);
2472
2473 return port_ptr;
2474}
2475
2476int msm_ipc_router_close_port(struct msm_ipc_port *port_ptr)
2477{
2478 union rr_control_msg msg;
2479 struct rr_packet *pkt, *temp_pkt;
2480 struct msm_ipc_server *server;
2481
2482 if (!port_ptr)
2483 return -EINVAL;
2484
2485 if (port_ptr->type == SERVER_PORT || port_ptr->type == CLIENT_PORT) {
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06002486 mutex_lock(&local_ports_lock);
2487 list_del(&port_ptr->list);
2488 mutex_unlock(&local_ports_lock);
2489
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002490 if (port_ptr->type == SERVER_PORT) {
2491 msg.cmd = IPC_ROUTER_CTRL_CMD_REMOVE_SERVER;
2492 msg.srv.service = port_ptr->port_name.service;
2493 msg.srv.instance = port_ptr->port_name.instance;
2494 msg.srv.node_id = port_ptr->this_port.node_id;
2495 msg.srv.port_id = port_ptr->this_port.port_id;
2496 RR("x REMOVE_SERVER Name=%d:%08x Id=%d:%08x\n",
2497 msg.srv.service, msg.srv.instance,
2498 msg.srv.node_id, msg.srv.port_id);
Karthikeyan Ramasubramanian5c568fb2013-01-10 17:09:13 -07002499 broadcast_ctl_msg(&msg);
2500 broadcast_ctl_msg_locally(&msg);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002501 }
Karthikeyan Ramasubramanian5c568fb2013-01-10 17:09:13 -07002502
2503 /*
2504 * Server port could have been a client port earlier.
2505 * Send REMOVE_CLIENT message in either case.
2506 */
Karthikeyan Ramasubramanian5c568fb2013-01-10 17:09:13 -07002507 RR("x REMOVE_CLIENT id=%d:%08x\n",
Karthikeyan Ramasubramanian96cced72013-05-02 17:25:54 -06002508 port_ptr->this_port.node_id, port_ptr->this_port.port_id);
2509 msm_ipc_router_send_remove_client(&port_ptr->mode_info,
2510 port_ptr->this_port.node_id,
2511 port_ptr->this_port.port_id);
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06002512 } else if (port_ptr->type == CONTROL_PORT) {
2513 mutex_lock(&control_ports_lock);
2514 list_del(&port_ptr->list);
2515 mutex_unlock(&control_ports_lock);
Karthikeyan Ramasubramanianf7a4b6e2013-01-16 09:00:28 -07002516 } else if (port_ptr->type == IRSC_PORT) {
2517 mutex_lock(&local_ports_lock);
2518 list_del(&port_ptr->list);
2519 mutex_unlock(&local_ports_lock);
2520 signal_irsc_completion();
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002521 }
2522
2523 mutex_lock(&port_ptr->port_rx_q_lock);
2524 list_for_each_entry_safe(pkt, temp_pkt, &port_ptr->port_rx_q, list) {
2525 list_del(&pkt->list);
2526 release_pkt(pkt);
2527 }
2528 mutex_unlock(&port_ptr->port_rx_q_lock);
2529
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002530 if (port_ptr->type == SERVER_PORT) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002531 mutex_lock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002532 server = msm_ipc_router_lookup_server(
2533 port_ptr->port_name.service,
2534 port_ptr->port_name.instance,
2535 port_ptr->this_port.node_id,
2536 port_ptr->this_port.port_id);
2537 if (server)
2538 msm_ipc_router_destroy_server(server,
2539 port_ptr->this_port.node_id,
2540 port_ptr->this_port.port_id);
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002541 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002542 }
2543
Karthikeyan Ramasubramaniandd8c3b52011-11-30 16:26:12 -07002544 wake_lock_destroy(&port_ptr->port_rx_wake_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002545 kfree(port_ptr);
2546 return 0;
2547}
2548
2549int msm_ipc_router_get_curr_pkt_size(struct msm_ipc_port *port_ptr)
2550{
2551 struct rr_packet *pkt;
2552 int rc = 0;
2553
2554 if (!port_ptr)
2555 return -EINVAL;
2556
2557 mutex_lock(&port_ptr->port_rx_q_lock);
2558 if (!list_empty(&port_ptr->port_rx_q)) {
2559 pkt = list_first_entry(&port_ptr->port_rx_q,
2560 struct rr_packet, list);
2561 rc = pkt->length;
2562 }
2563 mutex_unlock(&port_ptr->port_rx_q_lock);
2564
2565 return rc;
2566}
2567
2568int msm_ipc_router_bind_control_port(struct msm_ipc_port *port_ptr)
2569{
2570 if (!port_ptr)
2571 return -EINVAL;
2572
2573 mutex_lock(&local_ports_lock);
2574 list_del(&port_ptr->list);
2575 mutex_unlock(&local_ports_lock);
2576 port_ptr->type = CONTROL_PORT;
2577 mutex_lock(&control_ports_lock);
2578 list_add_tail(&port_ptr->list, &control_ports);
2579 mutex_unlock(&control_ports_lock);
2580
2581 return 0;
2582}
2583
2584int msm_ipc_router_lookup_server_name(struct msm_ipc_port_name *srv_name,
Karthikeyan Ramasubramaniandfde01b2012-06-12 14:25:13 -06002585 struct msm_ipc_server_info *srv_info,
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002586 int num_entries_in_array,
2587 uint32_t lookup_mask)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002588{
2589 struct msm_ipc_server *server;
2590 struct msm_ipc_server_port *server_port;
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002591 int key, i = 0; /*num_entries_found*/
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002592
2593 if (!srv_name) {
2594 pr_err("%s: Invalid srv_name\n", __func__);
2595 return -EINVAL;
2596 }
2597
Karthikeyan Ramasubramaniandfde01b2012-06-12 14:25:13 -06002598 if (num_entries_in_array && !srv_info) {
2599 pr_err("%s: srv_info NULL\n", __func__);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002600 return -EINVAL;
2601 }
2602
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002603 mutex_lock(&server_list_lock);
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002604 if (!lookup_mask)
2605 lookup_mask = 0xFFFFFFFF;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002606 key = (srv_name->service & (SRV_HASH_SIZE - 1));
2607 list_for_each_entry(server, &server_list[key], list) {
2608 if ((server->name.service != srv_name->service) ||
2609 ((server->name.instance & lookup_mask) !=
2610 srv_name->instance))
2611 continue;
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002612
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002613 list_for_each_entry(server_port,
2614 &server->server_port_list, list) {
2615 if (i < num_entries_in_array) {
2616 srv_info[i].node_id =
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002617 server_port->server_addr.node_id;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002618 srv_info[i].port_id =
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002619 server_port->server_addr.port_id;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002620 srv_info[i].service = server->name.service;
2621 srv_info[i].instance = server->name.instance;
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002622 }
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002623 i++;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002624 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002625 }
2626 mutex_unlock(&server_list_lock);
2627
2628 return i;
2629}
2630
2631int msm_ipc_router_close(void)
2632{
2633 struct msm_ipc_router_xprt_info *xprt_info, *tmp_xprt_info;
2634
2635 mutex_lock(&xprt_info_list_lock);
2636 list_for_each_entry_safe(xprt_info, tmp_xprt_info,
2637 &xprt_info_list, list) {
Karthikeyan Ramasubramanian8cec5922012-02-16 17:41:58 -07002638 xprt_info->xprt->close(xprt_info->xprt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002639 list_del(&xprt_info->list);
2640 kfree(xprt_info);
2641 }
2642 mutex_unlock(&xprt_info_list_lock);
2643 return 0;
2644}
2645
2646#if defined(CONFIG_DEBUG_FS)
2647static int dump_routing_table(char *buf, int max)
2648{
2649 int i = 0, j;
2650 struct msm_ipc_routing_table_entry *rt_entry;
2651
2652 for (j = 0; j < RT_HASH_SIZE; j++) {
2653 mutex_lock(&routing_table_lock);
2654 list_for_each_entry(rt_entry, &routing_table[j], list) {
2655 mutex_lock(&rt_entry->lock);
2656 i += scnprintf(buf + i, max - i,
2657 "Node Id: 0x%08x\n", rt_entry->node_id);
Karthikeyan Ramasubramanianbddeec72012-09-10 16:10:24 -06002658 if (rt_entry->node_id == IPC_ROUTER_NID_LOCAL) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002659 i += scnprintf(buf + i, max - i,
2660 "XPRT Name: Loopback\n");
2661 i += scnprintf(buf + i, max - i,
2662 "Next Hop: %d\n", rt_entry->node_id);
2663 } else {
2664 i += scnprintf(buf + i, max - i,
2665 "XPRT Name: %s\n",
2666 rt_entry->xprt_info->xprt->name);
2667 i += scnprintf(buf + i, max - i,
2668 "Next Hop: 0x%08x\n",
2669 rt_entry->xprt_info->remote_node_id);
2670 }
2671 i += scnprintf(buf + i, max - i, "\n");
2672 mutex_unlock(&rt_entry->lock);
2673 }
2674 mutex_unlock(&routing_table_lock);
2675 }
2676
2677 return i;
2678}
2679
2680static int dump_xprt_info(char *buf, int max)
2681{
2682 int i = 0;
2683 struct msm_ipc_router_xprt_info *xprt_info;
2684
2685 mutex_lock(&xprt_info_list_lock);
2686 list_for_each_entry(xprt_info, &xprt_info_list, list) {
2687 i += scnprintf(buf + i, max - i, "XPRT Name: %s\n",
2688 xprt_info->xprt->name);
2689 i += scnprintf(buf + i, max - i, "Link Id: %d\n",
2690 xprt_info->xprt->link_id);
2691 i += scnprintf(buf + i, max - i, "Initialized: %s\n",
2692 (xprt_info->initialized ? "Y" : "N"));
2693 i += scnprintf(buf + i, max - i, "Remote Node Id: 0x%08x\n",
2694 xprt_info->remote_node_id);
2695 i += scnprintf(buf + i, max - i, "\n");
2696 }
2697 mutex_unlock(&xprt_info_list_lock);
2698
2699 return i;
2700}
2701
2702static int dump_servers(char *buf, int max)
2703{
2704 int i = 0, j;
2705 struct msm_ipc_server *server;
2706 struct msm_ipc_server_port *server_port;
2707
2708 mutex_lock(&server_list_lock);
2709 for (j = 0; j < SRV_HASH_SIZE; j++) {
2710 list_for_each_entry(server, &server_list[j], list) {
2711 list_for_each_entry(server_port,
2712 &server->server_port_list,
2713 list) {
2714 i += scnprintf(buf + i, max - i, "Service: "
2715 "0x%08x\n", server->name.service);
2716 i += scnprintf(buf + i, max - i, "Instance: "
2717 "0x%08x\n", server->name.instance);
2718 i += scnprintf(buf + i, max - i,
2719 "Node_id: 0x%08x\n",
2720 server_port->server_addr.node_id);
2721 i += scnprintf(buf + i, max - i,
2722 "Port_id: 0x%08x\n",
2723 server_port->server_addr.port_id);
2724 i += scnprintf(buf + i, max - i, "\n");
2725 }
2726 }
2727 }
2728 mutex_unlock(&server_list_lock);
2729
2730 return i;
2731}
2732
2733static int dump_remote_ports(char *buf, int max)
2734{
2735 int i = 0, j, k;
2736 struct msm_ipc_router_remote_port *rport_ptr;
2737 struct msm_ipc_routing_table_entry *rt_entry;
2738
2739 for (j = 0; j < RT_HASH_SIZE; j++) {
2740 mutex_lock(&routing_table_lock);
2741 list_for_each_entry(rt_entry, &routing_table[j], list) {
2742 mutex_lock(&rt_entry->lock);
2743 for (k = 0; k < RP_HASH_SIZE; k++) {
2744 list_for_each_entry(rport_ptr,
2745 &rt_entry->remote_port_list[k],
2746 list) {
2747 i += scnprintf(buf + i, max - i,
2748 "Node_id: 0x%08x\n",
2749 rport_ptr->node_id);
2750 i += scnprintf(buf + i, max - i,
2751 "Port_id: 0x%08x\n",
2752 rport_ptr->port_id);
2753 i += scnprintf(buf + i, max - i,
2754 "Quota_cnt: %d\n",
2755 rport_ptr->tx_quota_cnt);
2756 i += scnprintf(buf + i, max - i, "\n");
2757 }
2758 }
2759 mutex_unlock(&rt_entry->lock);
2760 }
2761 mutex_unlock(&routing_table_lock);
2762 }
2763
2764 return i;
2765}
2766
2767static int dump_control_ports(char *buf, int max)
2768{
2769 int i = 0;
2770 struct msm_ipc_port *port_ptr;
2771
2772 mutex_lock(&control_ports_lock);
2773 list_for_each_entry(port_ptr, &control_ports, list) {
2774 i += scnprintf(buf + i, max - i, "Node_id: 0x%08x\n",
2775 port_ptr->this_port.node_id);
2776 i += scnprintf(buf + i, max - i, "Port_id: 0x%08x\n",
2777 port_ptr->this_port.port_id);
2778 i += scnprintf(buf + i, max - i, "\n");
2779 }
2780 mutex_unlock(&control_ports_lock);
2781
2782 return i;
2783}
2784
2785static int dump_local_ports(char *buf, int max)
2786{
2787 int i = 0, j;
2788 unsigned long flags;
2789 struct msm_ipc_port *port_ptr;
2790
2791 mutex_lock(&local_ports_lock);
2792 for (j = 0; j < LP_HASH_SIZE; j++) {
2793 list_for_each_entry(port_ptr, &local_ports[j], list) {
2794 spin_lock_irqsave(&port_ptr->port_lock, flags);
2795 i += scnprintf(buf + i, max - i, "Node_id: 0x%08x\n",
2796 port_ptr->this_port.node_id);
2797 i += scnprintf(buf + i, max - i, "Port_id: 0x%08x\n",
2798 port_ptr->this_port.port_id);
2799 i += scnprintf(buf + i, max - i, "# pkts tx'd %d\n",
2800 port_ptr->num_tx);
2801 i += scnprintf(buf + i, max - i, "# pkts rx'd %d\n",
2802 port_ptr->num_rx);
2803 i += scnprintf(buf + i, max - i, "# bytes tx'd %ld\n",
2804 port_ptr->num_tx_bytes);
2805 i += scnprintf(buf + i, max - i, "# bytes rx'd %ld\n",
2806 port_ptr->num_rx_bytes);
2807 spin_unlock_irqrestore(&port_ptr->port_lock, flags);
2808 i += scnprintf(buf + i, max - i, "\n");
2809 }
2810 }
2811 mutex_unlock(&local_ports_lock);
2812
2813 return i;
2814}
2815
2816#define DEBUG_BUFMAX 4096
2817static char debug_buffer[DEBUG_BUFMAX];
2818
2819static ssize_t debug_read(struct file *file, char __user *buf,
2820 size_t count, loff_t *ppos)
2821{
2822 int (*fill)(char *buf, int max) = file->private_data;
2823 int bsize = fill(debug_buffer, DEBUG_BUFMAX);
2824 return simple_read_from_buffer(buf, count, ppos, debug_buffer, bsize);
2825}
2826
2827static int debug_open(struct inode *inode, struct file *file)
2828{
2829 file->private_data = inode->i_private;
2830 return 0;
2831}
2832
2833static const struct file_operations debug_ops = {
2834 .read = debug_read,
2835 .open = debug_open,
2836};
2837
2838static void debug_create(const char *name, mode_t mode,
2839 struct dentry *dent,
2840 int (*fill)(char *buf, int max))
2841{
2842 debugfs_create_file(name, mode, dent, fill, &debug_ops);
2843}
2844
2845static void debugfs_init(void)
2846{
2847 struct dentry *dent;
2848
2849 dent = debugfs_create_dir("msm_ipc_router", 0);
2850 if (IS_ERR(dent))
2851 return;
2852
2853 debug_create("dump_local_ports", 0444, dent,
2854 dump_local_ports);
2855 debug_create("dump_remote_ports", 0444, dent,
2856 dump_remote_ports);
2857 debug_create("dump_control_ports", 0444, dent,
2858 dump_control_ports);
2859 debug_create("dump_servers", 0444, dent,
2860 dump_servers);
2861 debug_create("dump_xprt_info", 0444, dent,
2862 dump_xprt_info);
2863 debug_create("dump_routing_table", 0444, dent,
2864 dump_routing_table);
2865}
2866
2867#else
2868static void debugfs_init(void) {}
2869#endif
2870
2871static int msm_ipc_router_add_xprt(struct msm_ipc_router_xprt *xprt)
2872{
2873 struct msm_ipc_router_xprt_info *xprt_info;
2874 struct msm_ipc_routing_table_entry *rt_entry;
2875
2876 xprt_info = kmalloc(sizeof(struct msm_ipc_router_xprt_info),
2877 GFP_KERNEL);
2878 if (!xprt_info)
2879 return -ENOMEM;
2880
2881 xprt_info->xprt = xprt;
2882 xprt_info->initialized = 0;
2883 xprt_info->remote_node_id = -1;
2884 INIT_LIST_HEAD(&xprt_info->pkt_list);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002885 mutex_init(&xprt_info->rx_lock);
2886 mutex_init(&xprt_info->tx_lock);
2887 wake_lock_init(&xprt_info->wakelock,
2888 WAKE_LOCK_SUSPEND, xprt->name);
2889 xprt_info->need_len = 0;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002890 xprt_info->abort_data_read = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002891 INIT_WORK(&xprt_info->read_data, do_read_data);
2892 INIT_LIST_HEAD(&xprt_info->list);
2893
2894 xprt_info->workqueue = create_singlethread_workqueue(xprt->name);
2895 if (!xprt_info->workqueue) {
2896 kfree(xprt_info);
2897 return -ENOMEM;
2898 }
2899
2900 if (!strcmp(xprt->name, "msm_ipc_router_loopback_xprt")) {
2901 xprt_info->remote_node_id = IPC_ROUTER_NID_LOCAL;
2902 xprt_info->initialized = 1;
2903 }
2904
2905 mutex_lock(&xprt_info_list_lock);
2906 list_add_tail(&xprt_info->list, &xprt_info_list);
2907 mutex_unlock(&xprt_info_list_lock);
2908
2909 mutex_lock(&routing_table_lock);
2910 if (!routing_table_inited) {
2911 init_routing_table();
2912 rt_entry = alloc_routing_table_entry(IPC_ROUTER_NID_LOCAL);
2913 add_routing_table_entry(rt_entry);
2914 routing_table_inited = 1;
2915 }
2916 mutex_unlock(&routing_table_lock);
2917
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002918 xprt->priv = xprt_info;
2919
2920 return 0;
2921}
2922
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002923static void msm_ipc_router_remove_xprt(struct msm_ipc_router_xprt *xprt)
2924{
2925 struct msm_ipc_router_xprt_info *xprt_info;
2926
2927 if (xprt && xprt->priv) {
2928 xprt_info = xprt->priv;
2929
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06002930 mutex_lock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002931 xprt_info->abort_data_read = 1;
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06002932 mutex_unlock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002933
2934 mutex_lock(&xprt_info_list_lock);
2935 list_del(&xprt_info->list);
2936 mutex_unlock(&xprt_info_list_lock);
2937
2938 flush_workqueue(xprt_info->workqueue);
2939 destroy_workqueue(xprt_info->workqueue);
2940 wake_lock_destroy(&xprt_info->wakelock);
2941
2942 xprt->priv = 0;
2943 kfree(xprt_info);
2944 }
2945}
2946
2947
2948struct msm_ipc_router_xprt_work {
2949 struct msm_ipc_router_xprt *xprt;
2950 struct work_struct work;
2951};
2952
2953static void xprt_open_worker(struct work_struct *work)
2954{
2955 struct msm_ipc_router_xprt_work *xprt_work =
2956 container_of(work, struct msm_ipc_router_xprt_work, work);
2957
2958 msm_ipc_router_add_xprt(xprt_work->xprt);
2959 kfree(xprt_work);
2960}
2961
2962static void xprt_close_worker(struct work_struct *work)
2963{
2964 struct msm_ipc_router_xprt_work *xprt_work =
2965 container_of(work, struct msm_ipc_router_xprt_work, work);
2966
2967 modem_reset_cleanup(xprt_work->xprt->priv);
2968 msm_ipc_router_remove_xprt(xprt_work->xprt);
2969
2970 if (atomic_dec_return(&pending_close_count) == 0)
2971 wake_up(&subsystem_restart_wait);
2972
2973 kfree(xprt_work);
2974}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002975
2976void msm_ipc_router_xprt_notify(struct msm_ipc_router_xprt *xprt,
2977 unsigned event,
2978 void *data)
2979{
2980 struct msm_ipc_router_xprt_info *xprt_info = xprt->priv;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002981 struct msm_ipc_router_xprt_work *xprt_work;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002982 struct rr_packet *pkt;
Karthikeyan Ramasubramanian4af9f7c2011-09-30 15:55:18 -06002983 unsigned long ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002984
Karthikeyan Ramasubramanian4af9f7c2011-09-30 15:55:18 -06002985 if (!msm_ipc_router_workqueue) {
2986 ret = wait_for_completion_timeout(&msm_ipc_local_router_up,
2987 IPC_ROUTER_INIT_TIMEOUT);
2988 if (!ret || !msm_ipc_router_workqueue) {
2989 pr_err("%s: IPC Router not initialized\n", __func__);
2990 return;
2991 }
2992 }
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002993
2994 switch (event) {
2995 case IPC_ROUTER_XPRT_EVENT_OPEN:
2996 D("open event for '%s'\n", xprt->name);
2997 xprt_work = kmalloc(sizeof(struct msm_ipc_router_xprt_work),
2998 GFP_ATOMIC);
Karthikeyan Ramasubramanian840bed12013-05-16 15:51:29 -06002999 if (xprt_work) {
3000 xprt_work->xprt = xprt;
3001 INIT_WORK(&xprt_work->work, xprt_open_worker);
3002 queue_work(msm_ipc_router_workqueue, &xprt_work->work);
3003 } else {
3004 pr_err("%s: malloc failure - Couldn't notify OPEN event",
3005 __func__);
3006 }
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06003007 break;
3008
3009 case IPC_ROUTER_XPRT_EVENT_CLOSE:
3010 D("close event for '%s'\n", xprt->name);
3011 atomic_inc(&pending_close_count);
3012 xprt_work = kmalloc(sizeof(struct msm_ipc_router_xprt_work),
3013 GFP_ATOMIC);
Karthikeyan Ramasubramanian840bed12013-05-16 15:51:29 -06003014 if (xprt_work) {
3015 xprt_work->xprt = xprt;
3016 INIT_WORK(&xprt_work->work, xprt_close_worker);
3017 queue_work(msm_ipc_router_workqueue, &xprt_work->work);
3018 } else {
3019 pr_err("%s: malloc failure - Couldn't notify CLOSE event",
3020 __func__);
3021 }
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06003022 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003023 }
3024
3025 if (!data)
3026 return;
3027
3028 while (!xprt_info) {
3029 msleep(100);
3030 xprt_info = xprt->priv;
3031 }
3032
3033 pkt = clone_pkt((struct rr_packet *)data);
3034 if (!pkt)
3035 return;
3036
3037 mutex_lock(&xprt_info->rx_lock);
3038 list_add_tail(&pkt->list, &xprt_info->pkt_list);
3039 wake_lock(&xprt_info->wakelock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003040 mutex_unlock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06003041 queue_work(xprt_info->workqueue, &xprt_info->read_data);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003042}
3043
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06003044static int modem_restart_notifier_cb(struct notifier_block *this,
3045 unsigned long code,
3046 void *data);
3047static struct notifier_block msm_ipc_router_nb = {
3048 .notifier_call = modem_restart_notifier_cb,
3049};
3050
3051static int modem_restart_notifier_cb(struct notifier_block *this,
3052 unsigned long code,
3053 void *data)
3054{
3055 switch (code) {
3056 case SUBSYS_BEFORE_SHUTDOWN:
3057 D("%s: SUBSYS_BEFORE_SHUTDOWN\n", __func__);
3058 break;
3059
3060 case SUBSYS_BEFORE_POWERUP:
3061 D("%s: waiting for RPC restart to complete\n", __func__);
3062 wait_event(subsystem_restart_wait,
3063 atomic_read(&pending_close_count) == 0);
3064 D("%s: finished restart wait\n", __func__);
3065 break;
3066
3067 default:
3068 break;
3069 }
3070
3071 return NOTIFY_DONE;
3072}
3073
3074static void *restart_notifier_handle;
3075static __init int msm_ipc_router_modem_restart_late_init(void)
3076{
3077 restart_notifier_handle = subsys_notif_register_notifier("modem",
3078 &msm_ipc_router_nb);
3079 return 0;
3080}
3081late_initcall(msm_ipc_router_modem_restart_late_init);
3082
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003083static int __init msm_ipc_router_init(void)
3084{
3085 int i, ret;
3086 struct msm_ipc_routing_table_entry *rt_entry;
3087
3088 msm_ipc_router_debug_mask |= SMEM_LOG;
Karthikeyan Ramasubramanian2bfe8ec2013-03-22 10:47:20 -06003089 ipc_rtr_log_ctxt = ipc_log_context_create(IPC_RTR_LOG_PAGES,
3090 "ipc_router");
3091 if (!ipc_rtr_log_ctxt)
3092 pr_err("%s: Unable to create IPC logging for IPC RTR",
3093 __func__);
3094
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06003095 msm_ipc_router_workqueue =
3096 create_singlethread_workqueue("msm_ipc_router");
3097 if (!msm_ipc_router_workqueue)
3098 return -ENOMEM;
3099
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003100 debugfs_init();
3101
3102 for (i = 0; i < SRV_HASH_SIZE; i++)
3103 INIT_LIST_HEAD(&server_list[i]);
3104
3105 for (i = 0; i < LP_HASH_SIZE; i++)
3106 INIT_LIST_HEAD(&local_ports[i]);
3107
3108 mutex_lock(&routing_table_lock);
3109 if (!routing_table_inited) {
3110 init_routing_table();
3111 rt_entry = alloc_routing_table_entry(IPC_ROUTER_NID_LOCAL);
3112 add_routing_table_entry(rt_entry);
3113 routing_table_inited = 1;
3114 }
3115 mutex_unlock(&routing_table_lock);
3116
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06003117 init_waitqueue_head(&subsystem_restart_wait);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003118 ret = msm_ipc_router_init_sockets();
3119 if (ret < 0)
3120 pr_err("%s: Init sockets failed\n", __func__);
3121
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -06003122 ret = msm_ipc_router_security_init();
3123 if (ret < 0)
3124 pr_err("%s: Security Init failed\n", __func__);
3125
Karthikeyan Ramasubramanian4af9f7c2011-09-30 15:55:18 -06003126 complete_all(&msm_ipc_local_router_up);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003127 return ret;
3128}
3129
3130module_init(msm_ipc_router_init);
3131MODULE_DESCRIPTION("MSM IPC Router");
3132MODULE_LICENSE("GPL v2");