blob: c6204195e9d9659e912b9166e56ea5289ba2e3e9 [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);
118static wait_queue_head_t newserver_wait;
119
120struct msm_ipc_server {
121 struct list_head list;
122 struct msm_ipc_port_name name;
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600123 char pdev_name[32];
124 int next_pdev_id;
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -0600125 int synced_sec_rule;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700126 struct list_head server_port_list;
127};
128
129struct msm_ipc_server_port {
130 struct list_head list;
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600131 struct platform_device pdev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700132 struct msm_ipc_port_addr server_addr;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600133 struct msm_ipc_router_xprt_info *xprt_info;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700134};
135
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530136struct msm_ipc_resume_tx_port {
137 struct list_head list;
138 uint32_t port_id;
139 uint32_t node_id;
140};
141
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700142#define RP_HASH_SIZE 32
143struct msm_ipc_router_remote_port {
144 struct list_head list;
145 uint32_t node_id;
146 uint32_t port_id;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600147 uint32_t restart_state;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700148 uint32_t tx_quota_cnt;
149 struct mutex quota_lock;
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530150 struct list_head resume_tx_port_list;
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -0600151 void *sec_rule;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700152};
153
154struct msm_ipc_router_xprt_info {
155 struct list_head list;
156 struct msm_ipc_router_xprt *xprt;
157 uint32_t remote_node_id;
158 uint32_t initialized;
159 struct list_head pkt_list;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700160 struct wake_lock wakelock;
161 struct mutex rx_lock;
162 struct mutex tx_lock;
163 uint32_t need_len;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600164 uint32_t abort_data_read;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700165 struct work_struct read_data;
166 struct workqueue_struct *workqueue;
167};
168
169#define RT_HASH_SIZE 4
170struct msm_ipc_routing_table_entry {
171 struct list_head list;
172 uint32_t node_id;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600173 uint32_t neighbor_node_id;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700174 struct list_head remote_port_list[RP_HASH_SIZE];
175 struct msm_ipc_router_xprt_info *xprt_info;
176 struct mutex lock;
177 unsigned long num_tx_bytes;
178 unsigned long num_rx_bytes;
179};
180
181static struct list_head routing_table[RT_HASH_SIZE];
182static DEFINE_MUTEX(routing_table_lock);
183static int routing_table_inited;
184
185static LIST_HEAD(msm_ipc_board_dev_list);
186static DEFINE_MUTEX(msm_ipc_board_dev_list_lock);
187
188static void do_read_data(struct work_struct *work);
189
190#define RR_STATE_IDLE 0
191#define RR_STATE_HEADER 1
192#define RR_STATE_BODY 2
193#define RR_STATE_ERROR 3
194
195#define RESTART_NORMAL 0
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600196#define RESTART_PEND 1
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700197
198/* State for remote ep following restart */
199#define RESTART_QUOTA_ABORT 1
200
201static LIST_HEAD(xprt_info_list);
202static DEFINE_MUTEX(xprt_info_list_lock);
203
Karthikeyan Ramasubramanian4af9f7c2011-09-30 15:55:18 -0600204static DECLARE_COMPLETION(msm_ipc_local_router_up);
205#define IPC_ROUTER_INIT_TIMEOUT (10 * HZ)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700206
207static uint32_t next_port_id;
208static DEFINE_MUTEX(next_port_id_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600209static atomic_t pending_close_count = ATOMIC_INIT(0);
210static wait_queue_head_t subsystem_restart_wait;
211static struct workqueue_struct *msm_ipc_router_workqueue;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700212
213enum {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700214 DOWN,
215 UP,
216};
217
218static void init_routing_table(void)
219{
220 int i;
221 for (i = 0; i < RT_HASH_SIZE; i++)
222 INIT_LIST_HEAD(&routing_table[i]);
223}
224
225static struct msm_ipc_routing_table_entry *alloc_routing_table_entry(
226 uint32_t node_id)
227{
228 int i;
229 struct msm_ipc_routing_table_entry *rt_entry;
230
231 rt_entry = kmalloc(sizeof(struct msm_ipc_routing_table_entry),
232 GFP_KERNEL);
233 if (!rt_entry) {
234 pr_err("%s: rt_entry allocation failed for %d\n",
235 __func__, node_id);
236 return NULL;
237 }
238
239 for (i = 0; i < RP_HASH_SIZE; i++)
240 INIT_LIST_HEAD(&rt_entry->remote_port_list[i]);
241
242 mutex_init(&rt_entry->lock);
243 rt_entry->node_id = node_id;
244 rt_entry->xprt_info = NULL;
245 return rt_entry;
246}
247
248/*Please take routing_table_lock before calling this function*/
249static int add_routing_table_entry(
250 struct msm_ipc_routing_table_entry *rt_entry)
251{
252 uint32_t key;
253
254 if (!rt_entry)
255 return -EINVAL;
256
257 key = (rt_entry->node_id % RT_HASH_SIZE);
258 list_add_tail(&rt_entry->list, &routing_table[key]);
259 return 0;
260}
261
262/*Please take routing_table_lock before calling this function*/
263static struct msm_ipc_routing_table_entry *lookup_routing_table(
264 uint32_t node_id)
265{
266 uint32_t key = (node_id % RT_HASH_SIZE);
267 struct msm_ipc_routing_table_entry *rt_entry;
268
269 list_for_each_entry(rt_entry, &routing_table[key], list) {
270 if (rt_entry->node_id == node_id)
271 return rt_entry;
272 }
273 return NULL;
274}
275
276struct rr_packet *rr_read(struct msm_ipc_router_xprt_info *xprt_info)
277{
278 struct rr_packet *temp_pkt;
279
280 if (!xprt_info)
281 return NULL;
282
283 mutex_lock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600284 if (xprt_info->abort_data_read) {
285 mutex_unlock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -0600286 pr_err("%s detected SSR & exiting now\n",
287 xprt_info->xprt->name);
288 return NULL;
289 }
290
291 if (list_empty(&xprt_info->pkt_list)) {
292 mutex_unlock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600293 return NULL;
294 }
295
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700296 temp_pkt = list_first_entry(&xprt_info->pkt_list,
297 struct rr_packet, list);
298 list_del(&temp_pkt->list);
299 if (list_empty(&xprt_info->pkt_list))
300 wake_unlock(&xprt_info->wakelock);
301 mutex_unlock(&xprt_info->rx_lock);
302 return temp_pkt;
303}
304
305struct rr_packet *clone_pkt(struct rr_packet *pkt)
306{
307 struct rr_packet *cloned_pkt;
308 struct sk_buff *temp_skb, *cloned_skb;
309 struct sk_buff_head *pkt_fragment_q;
310
311 cloned_pkt = kzalloc(sizeof(struct rr_packet), GFP_KERNEL);
312 if (!cloned_pkt) {
313 pr_err("%s: failure\n", __func__);
314 return NULL;
315 }
316
317 pkt_fragment_q = kmalloc(sizeof(struct sk_buff_head), GFP_KERNEL);
318 if (!pkt_fragment_q) {
319 pr_err("%s: pkt_frag_q alloc failure\n", __func__);
320 kfree(cloned_pkt);
321 return NULL;
322 }
323 skb_queue_head_init(pkt_fragment_q);
324
325 skb_queue_walk(pkt->pkt_fragment_q, temp_skb) {
326 cloned_skb = skb_clone(temp_skb, GFP_KERNEL);
327 if (!cloned_skb)
328 goto fail_clone;
329 skb_queue_tail(pkt_fragment_q, cloned_skb);
330 }
331 cloned_pkt->pkt_fragment_q = pkt_fragment_q;
332 cloned_pkt->length = pkt->length;
333 return cloned_pkt;
334
335fail_clone:
336 while (!skb_queue_empty(pkt_fragment_q)) {
337 temp_skb = skb_dequeue(pkt_fragment_q);
338 kfree_skb(temp_skb);
339 }
340 kfree(pkt_fragment_q);
341 kfree(cloned_pkt);
342 return NULL;
343}
344
345struct rr_packet *create_pkt(struct sk_buff_head *data)
346{
347 struct rr_packet *pkt;
348 struct sk_buff *temp_skb;
349
350 pkt = kzalloc(sizeof(struct rr_packet), GFP_KERNEL);
351 if (!pkt) {
352 pr_err("%s: failure\n", __func__);
353 return NULL;
354 }
355
356 pkt->pkt_fragment_q = data;
357 skb_queue_walk(pkt->pkt_fragment_q, temp_skb)
358 pkt->length += temp_skb->len;
359 return pkt;
360}
361
362void release_pkt(struct rr_packet *pkt)
363{
364 struct sk_buff *temp_skb;
365
366 if (!pkt)
367 return;
368
369 if (!pkt->pkt_fragment_q) {
370 kfree(pkt);
371 return;
372 }
373
374 while (!skb_queue_empty(pkt->pkt_fragment_q)) {
375 temp_skb = skb_dequeue(pkt->pkt_fragment_q);
376 kfree_skb(temp_skb);
377 }
378 kfree(pkt->pkt_fragment_q);
379 kfree(pkt);
380 return;
381}
382
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600383static struct sk_buff_head *msm_ipc_router_buf_to_skb(void *buf,
384 unsigned int buf_len)
385{
386 struct sk_buff_head *skb_head;
387 struct sk_buff *skb;
388 int first = 1, offset = 0;
389 int skb_size, data_size;
390 void *data;
391
392 skb_head = kmalloc(sizeof(struct sk_buff_head), GFP_KERNEL);
393 if (!skb_head) {
394 pr_err("%s: Couldnot allocate skb_head\n", __func__);
395 return NULL;
396 }
397 skb_queue_head_init(skb_head);
398
399 data_size = buf_len;
400 while (offset != buf_len) {
401 skb_size = data_size;
402 if (first)
403 skb_size += IPC_ROUTER_HDR_SIZE;
404
405 skb = alloc_skb(skb_size, GFP_KERNEL);
406 if (!skb) {
407 if (skb_size <= (PAGE_SIZE/2)) {
408 pr_err("%s: cannot allocate skb\n", __func__);
409 goto buf_to_skb_error;
410 }
411 data_size = data_size / 2;
412 continue;
413 }
414
415 if (first) {
416 skb_reserve(skb, IPC_ROUTER_HDR_SIZE);
417 first = 0;
418 }
419
420 data = skb_put(skb, data_size);
421 memcpy(skb->data, buf + offset, data_size);
422 skb_queue_tail(skb_head, skb);
423 offset += data_size;
424 data_size = buf_len - offset;
425 }
426 return skb_head;
427
428buf_to_skb_error:
429 while (!skb_queue_empty(skb_head)) {
430 skb = skb_dequeue(skb_head);
431 kfree_skb(skb);
432 }
433 kfree(skb_head);
434 return NULL;
435}
436
437static void *msm_ipc_router_skb_to_buf(struct sk_buff_head *skb_head,
438 unsigned int len)
439{
440 struct sk_buff *temp;
441 int offset = 0, buf_len = 0, copy_len;
442 void *buf;
443
444 if (!skb_head) {
445 pr_err("%s: NULL skb_head\n", __func__);
446 return NULL;
447 }
448
449 temp = skb_peek(skb_head);
450 buf_len = len;
451 buf = kmalloc(buf_len, GFP_KERNEL);
452 if (!buf) {
453 pr_err("%s: cannot allocate buf\n", __func__);
454 return NULL;
455 }
456 skb_queue_walk(skb_head, temp) {
457 copy_len = buf_len < temp->len ? buf_len : temp->len;
458 memcpy(buf + offset, temp->data, copy_len);
459 offset += copy_len;
460 buf_len -= copy_len;
461 }
462 return buf;
463}
464
465static void msm_ipc_router_free_skb(struct sk_buff_head *skb_head)
466{
467 struct sk_buff *temp_skb;
468
469 if (!skb_head)
470 return;
471
472 while (!skb_queue_empty(skb_head)) {
473 temp_skb = skb_dequeue(skb_head);
474 kfree_skb(temp_skb);
475 }
476 kfree(skb_head);
477}
478
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700479static int post_control_ports(struct rr_packet *pkt)
480{
481 struct msm_ipc_port *port_ptr;
482 struct rr_packet *cloned_pkt;
483
484 if (!pkt)
485 return -EINVAL;
486
487 mutex_lock(&control_ports_lock);
488 list_for_each_entry(port_ptr, &control_ports, list) {
489 mutex_lock(&port_ptr->port_rx_q_lock);
490 cloned_pkt = clone_pkt(pkt);
491 wake_lock(&port_ptr->port_rx_wake_lock);
492 list_add_tail(&cloned_pkt->list, &port_ptr->port_rx_q);
493 wake_up(&port_ptr->port_rx_wait_q);
494 mutex_unlock(&port_ptr->port_rx_q_lock);
495 }
496 mutex_unlock(&control_ports_lock);
497 return 0;
498}
499
500static uint32_t allocate_port_id(void)
501{
502 uint32_t port_id = 0, prev_port_id, key;
503 struct msm_ipc_port *port_ptr;
504
505 mutex_lock(&next_port_id_lock);
506 prev_port_id = next_port_id;
507 mutex_lock(&local_ports_lock);
508 do {
509 next_port_id++;
510 if ((next_port_id & 0xFFFFFFFE) == 0xFFFFFFFE)
511 next_port_id = 1;
512
513 key = (next_port_id & (LP_HASH_SIZE - 1));
514 if (list_empty(&local_ports[key])) {
515 port_id = next_port_id;
516 break;
517 }
518 list_for_each_entry(port_ptr, &local_ports[key], list) {
519 if (port_ptr->this_port.port_id == next_port_id) {
520 port_id = next_port_id;
521 break;
522 }
523 }
524 if (!port_id) {
525 port_id = next_port_id;
526 break;
527 }
528 port_id = 0;
529 } while (next_port_id != prev_port_id);
530 mutex_unlock(&local_ports_lock);
531 mutex_unlock(&next_port_id_lock);
532
533 return port_id;
534}
535
536void msm_ipc_router_add_local_port(struct msm_ipc_port *port_ptr)
537{
538 uint32_t key;
539
540 if (!port_ptr)
541 return;
542
543 key = (port_ptr->this_port.port_id & (LP_HASH_SIZE - 1));
544 mutex_lock(&local_ports_lock);
545 list_add_tail(&port_ptr->list, &local_ports[key]);
546 mutex_unlock(&local_ports_lock);
547}
548
549struct msm_ipc_port *msm_ipc_router_create_raw_port(void *endpoint,
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600550 void (*notify)(unsigned event, void *priv),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700551 void *priv)
552{
553 struct msm_ipc_port *port_ptr;
554
555 port_ptr = kzalloc(sizeof(struct msm_ipc_port), GFP_KERNEL);
556 if (!port_ptr)
557 return NULL;
558
559 port_ptr->this_port.node_id = IPC_ROUTER_NID_LOCAL;
560 port_ptr->this_port.port_id = allocate_port_id();
561 if (!port_ptr->this_port.port_id) {
562 pr_err("%s: All port ids are in use\n", __func__);
563 kfree(port_ptr);
564 return NULL;
565 }
566
567 spin_lock_init(&port_ptr->port_lock);
568 INIT_LIST_HEAD(&port_ptr->incomplete);
569 mutex_init(&port_ptr->incomplete_lock);
570 INIT_LIST_HEAD(&port_ptr->port_rx_q);
571 mutex_init(&port_ptr->port_rx_q_lock);
572 init_waitqueue_head(&port_ptr->port_rx_wait_q);
Karthikeyan Ramasubramanianb4aeeb92012-03-13 12:31:43 -0600573 snprintf(port_ptr->rx_wakelock_name, MAX_WAKELOCK_NAME_SZ,
Karthikeyan Ramasubramanian6396bdd2013-02-14 13:53:20 -0700574 "ipc%08x_%s",
575 port_ptr->this_port.port_id,
576 current->comm);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700577 wake_lock_init(&port_ptr->port_rx_wake_lock,
Karthikeyan Ramasubramanianb4aeeb92012-03-13 12:31:43 -0600578 WAKE_LOCK_SUSPEND, port_ptr->rx_wakelock_name);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700579
580 port_ptr->endpoint = endpoint;
581 port_ptr->notify = notify;
582 port_ptr->priv = priv;
583
584 msm_ipc_router_add_local_port(port_ptr);
585 return port_ptr;
586}
587
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -0600588/*
589 * Should be called with local_ports_lock locked
590 */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700591static struct msm_ipc_port *msm_ipc_router_lookup_local_port(uint32_t port_id)
592{
593 int key = (port_id & (LP_HASH_SIZE - 1));
594 struct msm_ipc_port *port_ptr;
595
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700596 list_for_each_entry(port_ptr, &local_ports[key], list) {
597 if (port_ptr->this_port.port_id == port_id) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700598 return port_ptr;
599 }
600 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700601 return NULL;
602}
603
604static struct msm_ipc_router_remote_port *msm_ipc_router_lookup_remote_port(
605 uint32_t node_id,
606 uint32_t port_id)
607{
608 struct msm_ipc_router_remote_port *rport_ptr;
609 struct msm_ipc_routing_table_entry *rt_entry;
610 int key = (port_id & (RP_HASH_SIZE - 1));
611
612 mutex_lock(&routing_table_lock);
613 rt_entry = lookup_routing_table(node_id);
614 if (!rt_entry) {
615 mutex_unlock(&routing_table_lock);
616 pr_err("%s: Node is not up\n", __func__);
617 return NULL;
618 }
619
620 mutex_lock(&rt_entry->lock);
621 list_for_each_entry(rport_ptr,
622 &rt_entry->remote_port_list[key], list) {
623 if (rport_ptr->port_id == port_id) {
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600624 if (rport_ptr->restart_state != RESTART_NORMAL)
625 rport_ptr = NULL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700626 mutex_unlock(&rt_entry->lock);
627 mutex_unlock(&routing_table_lock);
628 return rport_ptr;
629 }
630 }
631 mutex_unlock(&rt_entry->lock);
632 mutex_unlock(&routing_table_lock);
633 return NULL;
634}
635
636static struct msm_ipc_router_remote_port *msm_ipc_router_create_remote_port(
637 uint32_t node_id,
638 uint32_t port_id)
639{
640 struct msm_ipc_router_remote_port *rport_ptr;
641 struct msm_ipc_routing_table_entry *rt_entry;
642 int key = (port_id & (RP_HASH_SIZE - 1));
643
644 mutex_lock(&routing_table_lock);
645 rt_entry = lookup_routing_table(node_id);
646 if (!rt_entry) {
647 mutex_unlock(&routing_table_lock);
648 pr_err("%s: Node is not up\n", __func__);
649 return NULL;
650 }
651
652 mutex_lock(&rt_entry->lock);
653 rport_ptr = kmalloc(sizeof(struct msm_ipc_router_remote_port),
654 GFP_KERNEL);
655 if (!rport_ptr) {
656 mutex_unlock(&rt_entry->lock);
657 mutex_unlock(&routing_table_lock);
658 pr_err("%s: Remote port alloc failed\n", __func__);
659 return NULL;
660 }
661 rport_ptr->port_id = port_id;
662 rport_ptr->node_id = node_id;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600663 rport_ptr->restart_state = RESTART_NORMAL;
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -0600664 rport_ptr->sec_rule = NULL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700665 rport_ptr->tx_quota_cnt = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700666 mutex_init(&rport_ptr->quota_lock);
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530667 INIT_LIST_HEAD(&rport_ptr->resume_tx_port_list);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700668 list_add_tail(&rport_ptr->list,
669 &rt_entry->remote_port_list[key]);
670 mutex_unlock(&rt_entry->lock);
671 mutex_unlock(&routing_table_lock);
672 return rport_ptr;
673}
674
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530675/**
676 * msm_ipc_router_free_resume_tx_port() - Free the resume_tx ports
677 * @rport_ptr: Pointer to the remote port.
678 *
679 * This function deletes all the resume_tx ports associated with a remote port
680 * and frees the memory allocated to each resume_tx port.
681 *
682 * Must be called with rport_ptr->quota_lock locked.
683 */
684static void msm_ipc_router_free_resume_tx_port(
685 struct msm_ipc_router_remote_port *rport_ptr)
686{
687 struct msm_ipc_resume_tx_port *rtx_port, *tmp_rtx_port;
688
689 list_for_each_entry_safe(rtx_port, tmp_rtx_port,
690 &rport_ptr->resume_tx_port_list, list) {
691 list_del(&rtx_port->list);
692 kfree(rtx_port);
693 }
694}
695
696/**
697 * msm_ipc_router_lookup_resume_tx_port() - Lookup resume_tx port list
698 * @rport_ptr: Remote port whose resume_tx port list needs to be looked.
699 * @port_id: Port ID which needs to be looked from the list.
700 *
701 * return 1 if the port_id is found in the list, else 0.
702 *
703 * This function is used to lookup the existence of a local port in
704 * remote port's resume_tx list. This function is used to ensure that
705 * the same port is not added to the remote_port's resume_tx list repeatedly.
706 *
707 * Must be called with rport_ptr->quota_lock locked.
708 */
709static int msm_ipc_router_lookup_resume_tx_port(
710 struct msm_ipc_router_remote_port *rport_ptr, uint32_t port_id)
711{
712 struct msm_ipc_resume_tx_port *rtx_port;
713
714 list_for_each_entry(rtx_port, &rport_ptr->resume_tx_port_list, list) {
715 if (port_id == rtx_port->port_id)
716 return 1;
717 }
718 return 0;
719}
720
721/**
722 * post_resume_tx() - Post the resume_tx event
723 * @rport_ptr: Pointer to the remote port
724 * @pkt : The data packet that is received on a resume_tx event
725 *
726 * This function informs about the reception of the resume_tx message from a
727 * remote port pointed by rport_ptr to all the local ports that are in the
728 * resume_tx_ports_list of this remote port. On posting the information, this
729 * function sequentially deletes each entry in the resume_tx_port_list of the
730 * remote port.
731 *
732 * Must be called with rport_ptr->quota_lock locked.
733 */
734static void post_resume_tx(struct msm_ipc_router_remote_port *rport_ptr,
735 struct rr_packet *pkt)
736{
737 struct msm_ipc_resume_tx_port *rtx_port, *tmp_rtx_port;
738 struct msm_ipc_port *local_port;
739 struct rr_packet *cloned_pkt;
740
741 list_for_each_entry_safe(rtx_port, tmp_rtx_port,
742 &rport_ptr->resume_tx_port_list, list) {
743 mutex_lock(&local_ports_lock);
744 local_port =
745 msm_ipc_router_lookup_local_port(rtx_port->port_id);
746 if (local_port) {
747 cloned_pkt = clone_pkt(pkt);
748 if (cloned_pkt) {
749 mutex_lock(&local_port->port_rx_q_lock);
750 list_add_tail(&cloned_pkt->list,
751 &local_port->port_rx_q);
752 wake_up(&local_port->port_rx_wait_q);
753 mutex_unlock(&local_port->port_rx_q_lock);
754 } else {
755 pr_err("%s: Clone_pkt failed for %08x:%08x\n",
756 __func__, local_port->this_port.node_id,
757 local_port->this_port.port_id);
758 }
759 }
760 mutex_unlock(&local_ports_lock);
761 list_del(&rtx_port->list);
762 kfree(rtx_port);
763 }
764}
765
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700766static void msm_ipc_router_destroy_remote_port(
767 struct msm_ipc_router_remote_port *rport_ptr)
768{
769 uint32_t node_id;
770 struct msm_ipc_routing_table_entry *rt_entry;
771
772 if (!rport_ptr)
773 return;
774
775 node_id = rport_ptr->node_id;
776 mutex_lock(&routing_table_lock);
777 rt_entry = lookup_routing_table(node_id);
778 if (!rt_entry) {
779 mutex_unlock(&routing_table_lock);
780 pr_err("%s: Node %d is not up\n", __func__, node_id);
781 return;
782 }
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +0530783 mutex_lock(&rport_ptr->quota_lock);
784 msm_ipc_router_free_resume_tx_port(rport_ptr);
785 mutex_unlock(&rport_ptr->quota_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700786 mutex_lock(&rt_entry->lock);
787 list_del(&rport_ptr->list);
788 kfree(rport_ptr);
789 mutex_unlock(&rt_entry->lock);
790 mutex_unlock(&routing_table_lock);
791 return;
792}
793
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600794/**
795 * msm_ipc_router_lookup_server() - Lookup server information
796 * @service: Service ID of the server info to be looked up.
797 * @instance: Instance ID of the server info to be looked up.
798 * @node_id: Node/Processor ID in which the server is hosted.
799 * @port_id: Port ID within the node in which the server is hosted.
800 *
801 * @return: If found Pointer to server structure, else NULL.
802 *
803 * Note1: Lock the server_list_lock before accessing this function.
804 * Note2: If the <node_id:port_id> are <0:0>, then the lookup is restricted
805 * to <service:instance>. Used only when a client wants to send a
806 * message to any QMI server.
807 */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700808static struct msm_ipc_server *msm_ipc_router_lookup_server(
809 uint32_t service,
810 uint32_t instance,
811 uint32_t node_id,
812 uint32_t port_id)
813{
814 struct msm_ipc_server *server;
815 struct msm_ipc_server_port *server_port;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600816 int key = (service & (SRV_HASH_SIZE - 1));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700817
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700818 list_for_each_entry(server, &server_list[key], list) {
819 if ((server->name.service != service) ||
820 (server->name.instance != instance))
821 continue;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600822 if ((node_id == 0) && (port_id == 0))
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700823 return server;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700824 list_for_each_entry(server_port, &server->server_port_list,
825 list) {
826 if ((server_port->server_addr.node_id == node_id) &&
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600827 (server_port->server_addr.port_id == port_id))
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700828 return server;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700829 }
830 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700831 return NULL;
832}
833
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600834static void dummy_release(struct device *dev)
835{
836}
837
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600838/**
839 * msm_ipc_router_create_server() - Add server info to hash table
840 * @service: Service ID of the server info to be created.
841 * @instance: Instance ID of the server info to be created.
842 * @node_id: Node/Processor ID in which the server is hosted.
843 * @port_id: Port ID within the node in which the server is hosted.
844 * @xprt_info: XPRT through which the node hosting the server is reached.
845 *
846 * @return: Pointer to server structure on success, else NULL.
847 *
848 * This function adds the server info to the hash table. If the same
849 * server(i.e. <service_id:instance_id>) is hosted in different nodes,
850 * they are maintained as list of "server_port" under "server" structure.
851 * Note: Lock the server_list_lock before accessing this function.
852 */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700853static struct msm_ipc_server *msm_ipc_router_create_server(
854 uint32_t service,
855 uint32_t instance,
856 uint32_t node_id,
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600857 uint32_t port_id,
858 struct msm_ipc_router_xprt_info *xprt_info)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700859{
860 struct msm_ipc_server *server = NULL;
861 struct msm_ipc_server_port *server_port;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600862 int key = (service & (SRV_HASH_SIZE - 1));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700863
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700864 list_for_each_entry(server, &server_list[key], list) {
865 if ((server->name.service == service) &&
866 (server->name.instance == instance))
867 goto create_srv_port;
868 }
869
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600870 server = kzalloc(sizeof(struct msm_ipc_server), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700871 if (!server) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700872 pr_err("%s: Server allocation failed\n", __func__);
873 return NULL;
874 }
875 server->name.service = service;
876 server->name.instance = instance;
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -0600877 server->synced_sec_rule = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700878 INIT_LIST_HEAD(&server->server_port_list);
879 list_add_tail(&server->list, &server_list[key]);
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600880 scnprintf(server->pdev_name, sizeof(server->pdev_name),
881 "QMI%08x:%08x", service, instance);
882 server->next_pdev_id = 1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700883
884create_srv_port:
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600885 server_port = kzalloc(sizeof(struct msm_ipc_server_port), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700886 if (!server_port) {
887 if (list_empty(&server->server_port_list)) {
888 list_del(&server->list);
889 kfree(server);
890 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700891 pr_err("%s: Server Port allocation failed\n", __func__);
892 return NULL;
893 }
894 server_port->server_addr.node_id = node_id;
895 server_port->server_addr.port_id = port_id;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -0600896 server_port->xprt_info = xprt_info;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700897 list_add_tail(&server_port->list, &server->server_port_list);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700898
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600899 server_port->pdev.name = server->pdev_name;
900 server_port->pdev.id = server->next_pdev_id++;
901 server_port->pdev.dev.release = dummy_release;
902 platform_device_register(&server_port->pdev);
903
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700904 return server;
905}
906
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -0600907/**
908 * msm_ipc_router_destroy_server() - Remove server info from hash table
909 * @server: Server info to be removed.
910 * @node_id: Node/Processor ID in which the server is hosted.
911 * @port_id: Port ID within the node in which the server is hosted.
912 *
913 * This function removes the server_port identified using <node_id:port_id>
914 * from the server structure. If the server_port list under server structure
915 * is empty after removal, then remove the server structure from the server
916 * hash table.
917 * Note: Lock the server_list_lock before accessing this function.
918 */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700919static void msm_ipc_router_destroy_server(struct msm_ipc_server *server,
920 uint32_t node_id, uint32_t port_id)
921{
922 struct msm_ipc_server_port *server_port;
923
924 if (!server)
925 return;
926
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700927 list_for_each_entry(server_port, &server->server_port_list, list) {
928 if ((server_port->server_addr.node_id == node_id) &&
929 (server_port->server_addr.port_id == port_id))
930 break;
931 }
932 if (server_port) {
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -0600933 platform_device_unregister(&server_port->pdev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700934 list_del(&server_port->list);
935 kfree(server_port);
936 }
937 if (list_empty(&server->server_port_list)) {
938 list_del(&server->list);
939 kfree(server);
940 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700941 return;
942}
943
944static int msm_ipc_router_send_control_msg(
945 struct msm_ipc_router_xprt_info *xprt_info,
946 union rr_control_msg *msg)
947{
948 struct rr_packet *pkt;
949 struct sk_buff *ipc_rtr_pkt;
950 struct rr_header *hdr;
951 int pkt_size;
952 void *data;
953 struct sk_buff_head *pkt_fragment_q;
954 int ret;
955
956 if (!xprt_info || ((msg->cmd != IPC_ROUTER_CTRL_CMD_HELLO) &&
957 !xprt_info->initialized)) {
958 pr_err("%s: xprt_info not initialized\n", __func__);
959 return -EINVAL;
960 }
961
962 if (xprt_info->remote_node_id == IPC_ROUTER_NID_LOCAL)
963 return 0;
964
965 pkt = kzalloc(sizeof(struct rr_packet), GFP_KERNEL);
966 if (!pkt) {
967 pr_err("%s: pkt alloc failed\n", __func__);
968 return -ENOMEM;
969 }
970
971 pkt_fragment_q = kmalloc(sizeof(struct sk_buff_head), GFP_KERNEL);
972 if (!pkt_fragment_q) {
973 pr_err("%s: pkt_fragment_q alloc failed\n", __func__);
974 kfree(pkt);
975 return -ENOMEM;
976 }
977 skb_queue_head_init(pkt_fragment_q);
978
979 pkt_size = IPC_ROUTER_HDR_SIZE + sizeof(*msg);
980 ipc_rtr_pkt = alloc_skb(pkt_size, GFP_KERNEL);
981 if (!ipc_rtr_pkt) {
982 pr_err("%s: ipc_rtr_pkt alloc failed\n", __func__);
983 kfree(pkt_fragment_q);
984 kfree(pkt);
985 return -ENOMEM;
986 }
987
988 skb_reserve(ipc_rtr_pkt, IPC_ROUTER_HDR_SIZE);
989 data = skb_put(ipc_rtr_pkt, sizeof(*msg));
990 memcpy(data, msg, sizeof(*msg));
991 hdr = (struct rr_header *)skb_push(ipc_rtr_pkt, IPC_ROUTER_HDR_SIZE);
992 if (!hdr) {
993 pr_err("%s: skb_push failed\n", __func__);
994 kfree_skb(ipc_rtr_pkt);
995 kfree(pkt_fragment_q);
996 kfree(pkt);
997 return -ENOMEM;
998 }
999
1000 hdr->version = IPC_ROUTER_VERSION;
1001 hdr->type = msg->cmd;
1002 hdr->src_node_id = IPC_ROUTER_NID_LOCAL;
1003 hdr->src_port_id = IPC_ROUTER_ADDRESS;
1004 hdr->confirm_rx = 0;
1005 hdr->size = sizeof(*msg);
1006 hdr->dst_node_id = xprt_info->remote_node_id;
1007 hdr->dst_port_id = IPC_ROUTER_ADDRESS;
1008 skb_queue_tail(pkt_fragment_q, ipc_rtr_pkt);
1009 pkt->pkt_fragment_q = pkt_fragment_q;
1010 pkt->length = pkt_size;
1011
1012 mutex_lock(&xprt_info->tx_lock);
Karthikeyan Ramasubramanian8cec5922012-02-16 17:41:58 -07001013 ret = xprt_info->xprt->write(pkt, pkt_size, xprt_info->xprt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001014 mutex_unlock(&xprt_info->tx_lock);
1015
1016 release_pkt(pkt);
1017 return ret;
1018}
1019
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001020static int msm_ipc_router_send_server_list(uint32_t node_id,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001021 struct msm_ipc_router_xprt_info *xprt_info)
1022{
1023 union rr_control_msg ctl;
1024 struct msm_ipc_server *server;
1025 struct msm_ipc_server_port *server_port;
1026 int i;
1027
1028 if (!xprt_info || !xprt_info->initialized) {
1029 pr_err("%s: Xprt info not initialized\n", __func__);
1030 return -EINVAL;
1031 }
1032
1033 ctl.cmd = IPC_ROUTER_CTRL_CMD_NEW_SERVER;
1034
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001035 for (i = 0; i < SRV_HASH_SIZE; i++) {
1036 list_for_each_entry(server, &server_list[i], list) {
1037 ctl.srv.service = server->name.service;
1038 ctl.srv.instance = server->name.instance;
1039 list_for_each_entry(server_port,
1040 &server->server_port_list, list) {
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001041 if (server_port->server_addr.node_id !=
1042 node_id)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001043 continue;
1044
1045 ctl.srv.node_id =
1046 server_port->server_addr.node_id;
1047 ctl.srv.port_id =
1048 server_port->server_addr.port_id;
1049 msm_ipc_router_send_control_msg(xprt_info,
1050 &ctl);
1051 }
1052 }
1053 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001054
1055 return 0;
1056}
1057
1058#if defined(DEBUG)
1059static char *type_to_str(int i)
1060{
1061 switch (i) {
1062 case IPC_ROUTER_CTRL_CMD_DATA:
1063 return "data ";
1064 case IPC_ROUTER_CTRL_CMD_HELLO:
1065 return "hello ";
1066 case IPC_ROUTER_CTRL_CMD_BYE:
1067 return "bye ";
1068 case IPC_ROUTER_CTRL_CMD_NEW_SERVER:
1069 return "new_srvr";
1070 case IPC_ROUTER_CTRL_CMD_REMOVE_SERVER:
1071 return "rmv_srvr";
1072 case IPC_ROUTER_CTRL_CMD_REMOVE_CLIENT:
1073 return "rmv_clnt";
1074 case IPC_ROUTER_CTRL_CMD_RESUME_TX:
1075 return "resum_tx";
1076 case IPC_ROUTER_CTRL_CMD_EXIT:
1077 return "cmd_exit";
1078 default:
1079 return "invalid";
1080 }
1081}
1082#endif
1083
1084static int broadcast_ctl_msg_locally(union rr_control_msg *msg)
1085{
1086 struct rr_packet *pkt;
1087 struct sk_buff *ipc_rtr_pkt;
1088 struct rr_header *hdr;
1089 int pkt_size;
1090 void *data;
1091 struct sk_buff_head *pkt_fragment_q;
1092 int ret;
1093
1094 pkt = kzalloc(sizeof(struct rr_packet), GFP_KERNEL);
1095 if (!pkt) {
1096 pr_err("%s: pkt alloc failed\n", __func__);
1097 return -ENOMEM;
1098 }
1099
1100 pkt_fragment_q = kmalloc(sizeof(struct sk_buff_head), GFP_KERNEL);
1101 if (!pkt_fragment_q) {
1102 pr_err("%s: pkt_fragment_q alloc failed\n", __func__);
1103 kfree(pkt);
1104 return -ENOMEM;
1105 }
1106 skb_queue_head_init(pkt_fragment_q);
1107
1108 pkt_size = IPC_ROUTER_HDR_SIZE + sizeof(*msg);
1109 ipc_rtr_pkt = alloc_skb(pkt_size, GFP_KERNEL);
1110 if (!ipc_rtr_pkt) {
1111 pr_err("%s: ipc_rtr_pkt alloc failed\n", __func__);
1112 kfree(pkt_fragment_q);
1113 kfree(pkt);
1114 return -ENOMEM;
1115 }
1116
1117 skb_reserve(ipc_rtr_pkt, IPC_ROUTER_HDR_SIZE);
1118 data = skb_put(ipc_rtr_pkt, sizeof(*msg));
1119 memcpy(data, msg, sizeof(*msg));
1120 hdr = (struct rr_header *)skb_push(ipc_rtr_pkt, IPC_ROUTER_HDR_SIZE);
1121 if (!hdr) {
1122 pr_err("%s: skb_push failed\n", __func__);
1123 kfree_skb(ipc_rtr_pkt);
1124 kfree(pkt_fragment_q);
1125 kfree(pkt);
1126 return -ENOMEM;
1127 }
1128 hdr->version = IPC_ROUTER_VERSION;
1129 hdr->type = msg->cmd;
1130 hdr->src_node_id = IPC_ROUTER_NID_LOCAL;
1131 hdr->src_port_id = IPC_ROUTER_ADDRESS;
1132 hdr->confirm_rx = 0;
1133 hdr->size = sizeof(*msg);
1134 hdr->dst_node_id = IPC_ROUTER_NID_LOCAL;
1135 hdr->dst_port_id = IPC_ROUTER_ADDRESS;
1136 skb_queue_tail(pkt_fragment_q, ipc_rtr_pkt);
1137 pkt->pkt_fragment_q = pkt_fragment_q;
1138 pkt->length = pkt_size;
1139
1140 ret = post_control_ports(pkt);
1141 release_pkt(pkt);
1142 return ret;
1143}
1144
1145static int broadcast_ctl_msg(union rr_control_msg *ctl)
1146{
1147 struct msm_ipc_router_xprt_info *xprt_info;
1148
1149 mutex_lock(&xprt_info_list_lock);
1150 list_for_each_entry(xprt_info, &xprt_info_list, list) {
1151 msm_ipc_router_send_control_msg(xprt_info, ctl);
1152 }
1153 mutex_unlock(&xprt_info_list_lock);
1154
1155 return 0;
1156}
1157
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001158static int relay_ctl_msg(struct msm_ipc_router_xprt_info *xprt_info,
1159 union rr_control_msg *ctl)
1160{
1161 struct msm_ipc_router_xprt_info *fwd_xprt_info;
1162
1163 if (!xprt_info || !ctl)
1164 return -EINVAL;
1165
1166 mutex_lock(&xprt_info_list_lock);
1167 list_for_each_entry(fwd_xprt_info, &xprt_info_list, list) {
1168 if (xprt_info->xprt->link_id != fwd_xprt_info->xprt->link_id)
1169 msm_ipc_router_send_control_msg(fwd_xprt_info, ctl);
1170 }
1171 mutex_unlock(&xprt_info_list_lock);
1172
1173 return 0;
1174}
1175
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001176static int relay_msg(struct msm_ipc_router_xprt_info *xprt_info,
1177 struct rr_packet *pkt)
1178{
1179 struct msm_ipc_router_xprt_info *fwd_xprt_info;
1180
1181 if (!xprt_info || !pkt)
1182 return -EINVAL;
1183
1184 mutex_lock(&xprt_info_list_lock);
1185 list_for_each_entry(fwd_xprt_info, &xprt_info_list, list) {
1186 mutex_lock(&fwd_xprt_info->tx_lock);
1187 if (xprt_info->xprt->link_id != fwd_xprt_info->xprt->link_id)
Karthikeyan Ramasubramanian8cec5922012-02-16 17:41:58 -07001188 fwd_xprt_info->xprt->write(pkt, pkt->length,
1189 fwd_xprt_info->xprt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001190 mutex_unlock(&fwd_xprt_info->tx_lock);
1191 }
1192 mutex_unlock(&xprt_info_list_lock);
1193 return 0;
1194}
1195
1196static int forward_msg(struct msm_ipc_router_xprt_info *xprt_info,
1197 struct rr_packet *pkt)
1198{
1199 uint32_t dst_node_id;
1200 struct sk_buff *head_pkt;
1201 struct rr_header *hdr;
1202 struct msm_ipc_router_xprt_info *fwd_xprt_info;
1203 struct msm_ipc_routing_table_entry *rt_entry;
1204
1205 if (!xprt_info || !pkt)
1206 return -EINVAL;
1207
1208 head_pkt = skb_peek(pkt->pkt_fragment_q);
1209 if (!head_pkt)
1210 return -EINVAL;
1211
1212 hdr = (struct rr_header *)head_pkt->data;
1213 dst_node_id = hdr->dst_node_id;
1214 mutex_lock(&routing_table_lock);
1215 rt_entry = lookup_routing_table(dst_node_id);
1216 if (!(rt_entry) || !(rt_entry->xprt_info)) {
1217 mutex_unlock(&routing_table_lock);
1218 pr_err("%s: Routing table not initialized\n", __func__);
1219 return -ENODEV;
1220 }
1221
1222 mutex_lock(&rt_entry->lock);
1223 fwd_xprt_info = rt_entry->xprt_info;
1224 mutex_lock(&fwd_xprt_info->tx_lock);
1225 if (xprt_info->remote_node_id == fwd_xprt_info->remote_node_id) {
1226 mutex_unlock(&fwd_xprt_info->tx_lock);
1227 mutex_unlock(&rt_entry->lock);
1228 mutex_unlock(&routing_table_lock);
1229 pr_err("%s: Discarding Command to route back\n", __func__);
1230 return -EINVAL;
1231 }
1232
1233 if (xprt_info->xprt->link_id == fwd_xprt_info->xprt->link_id) {
1234 mutex_unlock(&fwd_xprt_info->tx_lock);
1235 mutex_unlock(&rt_entry->lock);
1236 mutex_unlock(&routing_table_lock);
1237 pr_err("%s: DST in the same cluster\n", __func__);
1238 return 0;
1239 }
Karthikeyan Ramasubramanian8cec5922012-02-16 17:41:58 -07001240 fwd_xprt_info->xprt->write(pkt, pkt->length, fwd_xprt_info->xprt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001241 mutex_unlock(&fwd_xprt_info->tx_lock);
1242 mutex_unlock(&rt_entry->lock);
1243 mutex_unlock(&routing_table_lock);
1244
1245 return 0;
1246}
1247
Karthikeyan Ramasubramanian96cced72013-05-02 17:25:54 -06001248static int msm_ipc_router_send_remove_client(struct comm_mode_info *mode_info,
1249 uint32_t node_id, uint32_t port_id)
1250{
1251 union rr_control_msg msg;
1252 struct msm_ipc_router_xprt_info *tmp_xprt_info;
1253 int mode;
1254 void *xprt_info;
1255 int rc = 0;
1256
1257 if (!mode_info) {
1258 pr_err("%s: NULL mode_info\n", __func__);
1259 return -EINVAL;
1260 }
1261 mode = mode_info->mode;
1262 xprt_info = mode_info->xprt_info;
1263
1264 msg.cmd = IPC_ROUTER_CTRL_CMD_REMOVE_CLIENT;
1265 msg.cli.node_id = node_id;
1266 msg.cli.port_id = port_id;
1267
1268 if ((mode == SINGLE_LINK_MODE) && xprt_info) {
1269 mutex_lock(&xprt_info_list_lock);
1270 list_for_each_entry(tmp_xprt_info, &xprt_info_list, list) {
1271 if (tmp_xprt_info != xprt_info)
1272 continue;
1273 msm_ipc_router_send_control_msg(tmp_xprt_info, &msg);
1274 break;
1275 }
1276 mutex_unlock(&xprt_info_list_lock);
1277 } else if ((mode == SINGLE_LINK_MODE) && !xprt_info) {
1278 broadcast_ctl_msg_locally(&msg);
1279 } else if (mode == MULTI_LINK_MODE) {
1280 broadcast_ctl_msg(&msg);
1281 broadcast_ctl_msg_locally(&msg);
1282 } else if (mode != NULL_MODE) {
1283 pr_err("%s: Invalid mode(%d) + xprt_inf(%p) for %08x:%08x\n",
1284 __func__, mode, xprt_info, node_id, port_id);
1285 rc = -EINVAL;
1286 }
1287 return rc;
1288}
1289
1290static void update_comm_mode_info(struct comm_mode_info *mode_info,
1291 struct msm_ipc_router_xprt_info *xprt_info)
1292{
1293 if (!mode_info) {
1294 pr_err("%s: NULL mode_info\n", __func__);
1295 return;
1296 }
1297
1298 if (mode_info->mode == NULL_MODE) {
1299 mode_info->xprt_info = xprt_info;
1300 mode_info->mode = SINGLE_LINK_MODE;
1301 } else if (mode_info->mode == SINGLE_LINK_MODE &&
1302 mode_info->xprt_info != xprt_info) {
1303 mode_info->mode = MULTI_LINK_MODE;
1304 }
1305
1306 return;
1307}
1308
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001309static void reset_remote_port_info(uint32_t node_id, uint32_t port_id)
1310{
1311 struct msm_ipc_router_remote_port *rport_ptr;
1312
1313 rport_ptr = msm_ipc_router_lookup_remote_port(node_id, port_id);
1314 if (!rport_ptr) {
1315 pr_err("%s: No such remote port %08x:%08x\n",
1316 __func__, node_id, port_id);
1317 return;
1318 }
1319 mutex_lock(&rport_ptr->quota_lock);
1320 rport_ptr->restart_state = RESTART_PEND;
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05301321 msm_ipc_router_free_resume_tx_port(rport_ptr);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001322 mutex_unlock(&rport_ptr->quota_lock);
1323 return;
1324}
1325
1326static void msm_ipc_cleanup_remote_server_info(
1327 struct msm_ipc_router_xprt_info *xprt_info)
1328{
1329 struct msm_ipc_server *svr, *tmp_svr;
1330 struct msm_ipc_server_port *svr_port, *tmp_svr_port;
1331 int i;
1332 union rr_control_msg ctl;
1333
1334 if (!xprt_info) {
1335 pr_err("%s: Invalid xprt_info\n", __func__);
1336 return;
1337 }
1338
1339 ctl.cmd = IPC_ROUTER_CTRL_CMD_REMOVE_SERVER;
1340 mutex_lock(&server_list_lock);
1341 for (i = 0; i < SRV_HASH_SIZE; i++) {
1342 list_for_each_entry_safe(svr, tmp_svr, &server_list[i], list) {
1343 ctl.srv.service = svr->name.service;
1344 ctl.srv.instance = svr->name.instance;
1345 list_for_each_entry_safe(svr_port, tmp_svr_port,
1346 &svr->server_port_list, list) {
1347 if (svr_port->xprt_info != xprt_info)
1348 continue;
1349 D("Remove server %08x:%08x - %08x:%08x",
1350 ctl.srv.service, ctl.srv.instance,
1351 svr_port->server_addr.node_id,
1352 svr_port->server_addr.port_id);
1353 reset_remote_port_info(
1354 svr_port->server_addr.node_id,
1355 svr_port->server_addr.port_id);
1356 ctl.srv.node_id = svr_port->server_addr.node_id;
1357 ctl.srv.port_id = svr_port->server_addr.port_id;
1358 relay_ctl_msg(xprt_info, &ctl);
1359 broadcast_ctl_msg_locally(&ctl);
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06001360 platform_device_unregister(&svr_port->pdev);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001361 list_del(&svr_port->list);
1362 kfree(svr_port);
1363 }
1364 if (list_empty(&svr->server_port_list)) {
1365 list_del(&svr->list);
1366 kfree(svr);
1367 }
1368 }
1369 }
1370 mutex_unlock(&server_list_lock);
1371}
1372
1373static void msm_ipc_cleanup_remote_client_info(
1374 struct msm_ipc_router_xprt_info *xprt_info)
1375{
1376 struct msm_ipc_routing_table_entry *rt_entry;
1377 struct msm_ipc_router_remote_port *rport_ptr;
1378 int i, j;
1379 union rr_control_msg ctl;
1380
1381 if (!xprt_info) {
1382 pr_err("%s: Invalid xprt_info\n", __func__);
1383 return;
1384 }
1385
1386 ctl.cmd = IPC_ROUTER_CTRL_CMD_REMOVE_CLIENT;
1387 mutex_lock(&routing_table_lock);
1388 for (i = 0; i < RT_HASH_SIZE; i++) {
1389 list_for_each_entry(rt_entry, &routing_table[i], list) {
1390 mutex_lock(&rt_entry->lock);
1391 if (rt_entry->xprt_info != xprt_info) {
1392 mutex_unlock(&rt_entry->lock);
1393 continue;
1394 }
1395 for (j = 0; j < RP_HASH_SIZE; j++) {
1396 list_for_each_entry(rport_ptr,
1397 &rt_entry->remote_port_list[j], list) {
1398 if (rport_ptr->restart_state ==
1399 RESTART_PEND)
1400 continue;
1401 mutex_lock(&rport_ptr->quota_lock);
1402 rport_ptr->restart_state = RESTART_PEND;
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05301403 msm_ipc_router_free_resume_tx_port(
1404 rport_ptr);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001405 mutex_unlock(&rport_ptr->quota_lock);
1406 ctl.cli.node_id = rport_ptr->node_id;
1407 ctl.cli.port_id = rport_ptr->port_id;
1408 broadcast_ctl_msg_locally(&ctl);
1409 }
1410 }
1411 mutex_unlock(&rt_entry->lock);
1412 }
1413 }
1414 mutex_unlock(&routing_table_lock);
1415}
1416
1417static void msm_ipc_cleanup_remote_port_info(uint32_t node_id)
1418{
1419 struct msm_ipc_routing_table_entry *rt_entry, *tmp_rt_entry;
1420 struct msm_ipc_router_remote_port *rport_ptr, *tmp_rport_ptr;
1421 int i, j;
1422
1423 mutex_lock(&routing_table_lock);
1424 for (i = 0; i < RT_HASH_SIZE; i++) {
1425 list_for_each_entry_safe(rt_entry, tmp_rt_entry,
1426 &routing_table[i], list) {
1427 mutex_lock(&rt_entry->lock);
1428 if (rt_entry->neighbor_node_id != node_id) {
1429 mutex_unlock(&rt_entry->lock);
1430 continue;
1431 }
1432 for (j = 0; j < RP_HASH_SIZE; j++) {
1433 list_for_each_entry_safe(rport_ptr,
1434 tmp_rport_ptr,
1435 &rt_entry->remote_port_list[j], list) {
1436 list_del(&rport_ptr->list);
1437 kfree(rport_ptr);
1438 }
1439 }
1440 mutex_unlock(&rt_entry->lock);
1441 }
1442 }
1443 mutex_unlock(&routing_table_lock);
1444}
1445
1446static void msm_ipc_cleanup_routing_table(
1447 struct msm_ipc_router_xprt_info *xprt_info)
1448{
1449 int i;
1450 struct msm_ipc_routing_table_entry *rt_entry;
1451
1452 if (!xprt_info) {
1453 pr_err("%s: Invalid xprt_info\n", __func__);
1454 return;
1455 }
1456
1457 mutex_lock(&routing_table_lock);
1458 for (i = 0; i < RT_HASH_SIZE; i++) {
1459 list_for_each_entry(rt_entry, &routing_table[i], list) {
1460 mutex_lock(&rt_entry->lock);
1461 if (rt_entry->xprt_info == xprt_info)
1462 rt_entry->xprt_info = NULL;
1463 mutex_unlock(&rt_entry->lock);
1464 }
1465 }
1466 mutex_unlock(&routing_table_lock);
1467}
1468
1469static void modem_reset_cleanup(struct msm_ipc_router_xprt_info *xprt_info)
1470{
1471
1472 if (!xprt_info) {
1473 pr_err("%s: Invalid xprt_info\n", __func__);
1474 return;
1475 }
1476
1477 msm_ipc_cleanup_remote_server_info(xprt_info);
1478 msm_ipc_cleanup_remote_client_info(xprt_info);
1479 msm_ipc_cleanup_routing_table(xprt_info);
1480}
1481
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -06001482/**
1483 * sync_sec_rule() - Synchrnoize the security rule into the server structure
1484 * @server: Server structure where the rule has to be synchronized.
1485 * @rule: Security tule to be synchronized.
1486 *
1487 * This function is used to update the server structure with the security
1488 * rule configured for the <service:instance> corresponding to that server.
1489 */
1490static void sync_sec_rule(struct msm_ipc_server *server, void *rule)
1491{
1492 struct msm_ipc_server_port *server_port;
1493 struct msm_ipc_router_remote_port *rport_ptr = NULL;
1494
1495 list_for_each_entry(server_port, &server->server_port_list, list) {
1496 rport_ptr = msm_ipc_router_lookup_remote_port(
1497 server_port->server_addr.node_id,
1498 server_port->server_addr.port_id);
1499 if (!rport_ptr)
1500 continue;
1501 rport_ptr->sec_rule = rule;
1502 }
1503 server->synced_sec_rule = 1;
1504}
1505
1506/**
1507 * msm_ipc_sync_sec_rule() - Sync the security rule to the service
1508 * @service: Service for which the rule has to be synchronized.
1509 * @instance: Instance for which the rule has to be synchronized.
1510 * @rule: Security rule to be synchronized.
1511 *
1512 * This function is used to syncrhonize the security rule with the server
1513 * hash table, if the user-space script configures the rule after the service
1514 * has come up. This function is used to synchronize the security rule to a
1515 * specific service and optionally a specific instance.
1516 */
1517void msm_ipc_sync_sec_rule(uint32_t service, uint32_t instance, void *rule)
1518{
1519 int key = (service & (SRV_HASH_SIZE - 1));
1520 struct msm_ipc_server *server;
1521
1522 mutex_lock(&server_list_lock);
1523 list_for_each_entry(server, &server_list[key], list) {
1524 if (server->name.service != service)
1525 continue;
1526
1527 if (server->name.instance != instance &&
1528 instance != ALL_INSTANCE)
1529 continue;
1530
1531 /*
1532 * If the rule applies to all instances and if the specific
1533 * instance of a service has a rule synchronized already,
1534 * do not apply the rule for that specific instance.
1535 */
1536 if (instance == ALL_INSTANCE && server->synced_sec_rule)
1537 continue;
1538
1539 sync_sec_rule(server, rule);
1540 }
1541 mutex_unlock(&server_list_lock);
1542}
1543
1544/**
1545 * msm_ipc_sync_default_sec_rule() - Default security rule to all services
1546 * @rule: Security rule to be synchronized.
1547 *
1548 * This function is used to syncrhonize the security rule with the server
1549 * hash table, if the user-space script configures the rule after the service
1550 * has come up. This function is used to synchronize the security rule that
1551 * applies to all services, if the concerned service do not have any rule
1552 * defined.
1553 */
1554void msm_ipc_sync_default_sec_rule(void *rule)
1555{
1556 int key;
1557 struct msm_ipc_server *server;
1558
1559 mutex_lock(&server_list_lock);
1560 for (key = 0; key < SRV_HASH_SIZE; key++) {
1561 list_for_each_entry(server, &server_list[key], list) {
1562 if (server->synced_sec_rule)
1563 continue;
1564
1565 sync_sec_rule(server, rule);
1566 }
1567 }
1568 mutex_unlock(&server_list_lock);
1569}
1570
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001571static int process_hello_msg(struct msm_ipc_router_xprt_info *xprt_info,
1572 struct rr_header *hdr)
1573{
1574 int i, rc = 0;
1575 union rr_control_msg ctl;
1576 struct msm_ipc_routing_table_entry *rt_entry;
1577
1578 if (!hdr)
1579 return -EINVAL;
1580
1581 RR("o HELLO NID %d\n", hdr->src_node_id);
1582
1583 xprt_info->remote_node_id = hdr->src_node_id;
1584 /*
1585 * Find the entry from Routing Table corresponding to Node ID.
1586 * Under SSR, an entry will be found. When the system boots up
1587 * for the 1st time, an entry will not be found and hence allocate
1588 * an entry. Update the entry with the Node ID that it corresponds
1589 * to and the XPRT through which it can be reached.
1590 */
1591 mutex_lock(&routing_table_lock);
1592 rt_entry = lookup_routing_table(hdr->src_node_id);
1593 if (!rt_entry) {
1594 rt_entry = alloc_routing_table_entry(hdr->src_node_id);
1595 if (!rt_entry) {
1596 mutex_unlock(&routing_table_lock);
1597 pr_err("%s: rt_entry allocation failed\n", __func__);
1598 return -ENOMEM;
1599 }
1600 add_routing_table_entry(rt_entry);
1601 }
1602 mutex_lock(&rt_entry->lock);
1603 rt_entry->neighbor_node_id = xprt_info->remote_node_id;
1604 rt_entry->xprt_info = xprt_info;
1605 mutex_unlock(&rt_entry->lock);
1606 mutex_unlock(&routing_table_lock);
1607
1608 /* Cleanup any remote ports, if the node is coming out of reset */
1609 msm_ipc_cleanup_remote_port_info(xprt_info->remote_node_id);
1610
1611 /* Send a reply HELLO message */
1612 memset(&ctl, 0, sizeof(ctl));
1613 ctl.cmd = IPC_ROUTER_CTRL_CMD_HELLO;
1614 rc = msm_ipc_router_send_control_msg(xprt_info, &ctl);
1615 if (rc < 0) {
1616 pr_err("%s: Error sending reply HELLO message\n", __func__);
1617 return rc;
1618 }
1619 xprt_info->initialized = 1;
1620
1621 /*
1622 * Send list of servers from the local node and from nodes
1623 * outside the mesh network in which this XPRT is part of.
1624 */
Karthikeyan Ramasubramanian05066a82012-10-31 15:12:08 -06001625 mutex_lock(&server_list_lock);
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001626 mutex_lock(&routing_table_lock);
1627 for (i = 0; i < RT_HASH_SIZE; i++) {
1628 list_for_each_entry(rt_entry, &routing_table[i], list) {
1629 if ((rt_entry->node_id != IPC_ROUTER_NID_LOCAL) &&
Karthikeyan Ramasubramanianc4c0aaa2013-01-30 14:17:57 -07001630 (!rt_entry->xprt_info ||
1631 (rt_entry->xprt_info->xprt->link_id ==
1632 xprt_info->xprt->link_id)))
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001633 continue;
1634 rc = msm_ipc_router_send_server_list(rt_entry->node_id,
1635 xprt_info);
1636 if (rc < 0) {
1637 mutex_unlock(&routing_table_lock);
Karthikeyan Ramasubramanian05066a82012-10-31 15:12:08 -06001638 mutex_unlock(&server_list_lock);
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001639 return rc;
1640 }
1641 }
1642 }
1643 mutex_unlock(&routing_table_lock);
Karthikeyan Ramasubramanian05066a82012-10-31 15:12:08 -06001644 mutex_unlock(&server_list_lock);
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001645 RR("HELLO message processed\n");
1646 return rc;
1647}
1648
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001649static int process_control_msg(struct msm_ipc_router_xprt_info *xprt_info,
1650 struct rr_packet *pkt)
1651{
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001652 union rr_control_msg *msg;
1653 struct msm_ipc_router_remote_port *rport_ptr;
1654 int rc = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001655 struct sk_buff *temp_ptr;
1656 struct rr_header *hdr;
1657 struct msm_ipc_server *server;
1658 struct msm_ipc_routing_table_entry *rt_entry;
1659
1660 if (pkt->length != (IPC_ROUTER_HDR_SIZE + sizeof(*msg))) {
1661 pr_err("%s: r2r msg size %d != %d\n", __func__, pkt->length,
1662 (IPC_ROUTER_HDR_SIZE + sizeof(*msg)));
1663 return -EINVAL;
1664 }
1665
1666 temp_ptr = skb_peek(pkt->pkt_fragment_q);
Karthikeyan Ramasubramanian9024dd82011-12-19 18:44:19 -07001667 if (!temp_ptr) {
1668 pr_err("%s: pkt_fragment_q is empty\n", __func__);
1669 return -EINVAL;
1670 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001671 hdr = (struct rr_header *)temp_ptr->data;
Karthikeyan Ramasubramanian9024dd82011-12-19 18:44:19 -07001672 if (!hdr) {
1673 pr_err("%s: No data inside the skb\n", __func__);
1674 return -EINVAL;
1675 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001676 msg = (union rr_control_msg *)((char *)hdr + IPC_ROUTER_HDR_SIZE);
1677
1678 switch (msg->cmd) {
1679 case IPC_ROUTER_CTRL_CMD_HELLO:
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001680 rc = process_hello_msg(xprt_info, hdr);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001681 break;
Karthikeyan Ramasubramaniana2b7fdb2012-10-23 13:12:44 -06001682
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001683 case IPC_ROUTER_CTRL_CMD_RESUME_TX:
1684 RR("o RESUME_TX id=%d:%08x\n",
1685 msg->cli.node_id, msg->cli.port_id);
1686
1687 rport_ptr = msm_ipc_router_lookup_remote_port(msg->cli.node_id,
1688 msg->cli.port_id);
1689 if (!rport_ptr) {
1690 pr_err("%s: Unable to resume client\n", __func__);
1691 break;
1692 }
1693 mutex_lock(&rport_ptr->quota_lock);
1694 rport_ptr->tx_quota_cnt = 0;
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05301695 post_resume_tx(rport_ptr, pkt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001696 mutex_unlock(&rport_ptr->quota_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001697 break;
1698
1699 case IPC_ROUTER_CTRL_CMD_NEW_SERVER:
1700 if (msg->srv.instance == 0) {
1701 pr_err(
1702 "rpcrouter: Server create rejected, version = 0, "
1703 "service = %08x\n", msg->srv.service);
1704 break;
1705 }
1706
1707 RR("o NEW_SERVER id=%d:%08x service=%08x:%08x\n",
1708 msg->srv.node_id, msg->srv.port_id,
1709 msg->srv.service, msg->srv.instance);
1710
1711 mutex_lock(&routing_table_lock);
1712 rt_entry = lookup_routing_table(msg->srv.node_id);
1713 if (!rt_entry) {
1714 rt_entry = alloc_routing_table_entry(msg->srv.node_id);
1715 if (!rt_entry) {
1716 mutex_unlock(&routing_table_lock);
1717 pr_err("%s: rt_entry allocation failed\n",
1718 __func__);
1719 return -ENOMEM;
1720 }
1721 mutex_lock(&rt_entry->lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001722 rt_entry->neighbor_node_id = xprt_info->remote_node_id;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001723 rt_entry->xprt_info = xprt_info;
1724 mutex_unlock(&rt_entry->lock);
1725 add_routing_table_entry(rt_entry);
1726 }
1727 mutex_unlock(&routing_table_lock);
1728
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06001729 mutex_lock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001730 server = msm_ipc_router_lookup_server(msg->srv.service,
1731 msg->srv.instance,
1732 msg->srv.node_id,
1733 msg->srv.port_id);
1734 if (!server) {
1735 server = msm_ipc_router_create_server(
1736 msg->srv.service, msg->srv.instance,
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001737 msg->srv.node_id, msg->srv.port_id, xprt_info);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001738 if (!server) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06001739 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001740 pr_err("%s: Server Create failed\n", __func__);
1741 return -ENOMEM;
1742 }
1743
1744 if (!msm_ipc_router_lookup_remote_port(
1745 msg->srv.node_id, msg->srv.port_id)) {
1746 rport_ptr = msm_ipc_router_create_remote_port(
1747 msg->srv.node_id, msg->srv.port_id);
1748 if (!rport_ptr)
1749 pr_err("%s: Remote port create "
1750 "failed\n", __func__);
Brent Hronik1c9a3d42013-04-17 15:10:40 -06001751 else
1752 rport_ptr->sec_rule =
1753 msm_ipc_get_security_rule(
1754 msg->srv.service,
1755 msg->srv.instance);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001756 }
1757 wake_up(&newserver_wait);
1758 }
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06001759 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001760
1761 relay_msg(xprt_info, pkt);
1762 post_control_ports(pkt);
1763 break;
1764 case IPC_ROUTER_CTRL_CMD_REMOVE_SERVER:
1765 RR("o REMOVE_SERVER service=%08x:%d\n",
1766 msg->srv.service, msg->srv.instance);
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06001767 mutex_lock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001768 server = msm_ipc_router_lookup_server(msg->srv.service,
1769 msg->srv.instance,
1770 msg->srv.node_id,
1771 msg->srv.port_id);
1772 if (server) {
1773 msm_ipc_router_destroy_server(server,
1774 msg->srv.node_id,
1775 msg->srv.port_id);
1776 relay_msg(xprt_info, pkt);
1777 post_control_ports(pkt);
1778 }
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06001779 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001780 break;
1781 case IPC_ROUTER_CTRL_CMD_REMOVE_CLIENT:
1782 RR("o REMOVE_CLIENT id=%d:%08x\n",
1783 msg->cli.node_id, msg->cli.port_id);
1784 rport_ptr = msm_ipc_router_lookup_remote_port(msg->cli.node_id,
1785 msg->cli.port_id);
1786 if (rport_ptr)
1787 msm_ipc_router_destroy_remote_port(rport_ptr);
1788
1789 relay_msg(xprt_info, pkt);
1790 post_control_ports(pkt);
1791 break;
1792 case IPC_ROUTER_CTRL_CMD_PING:
1793 /* No action needed for ping messages received */
1794 RR("o PING\n");
1795 break;
1796 default:
1797 RR("o UNKNOWN(%08x)\n", msg->cmd);
1798 rc = -ENOSYS;
1799 }
1800
1801 return rc;
1802}
1803
1804static void do_read_data(struct work_struct *work)
1805{
1806 struct rr_header *hdr;
1807 struct rr_packet *pkt = NULL;
1808 struct msm_ipc_port *port_ptr;
1809 struct sk_buff *head_skb;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001810 struct msm_ipc_router_remote_port *rport_ptr;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001811 uint32_t resume_tx, resume_tx_node_id, resume_tx_port_id;
1812
1813 struct msm_ipc_router_xprt_info *xprt_info =
1814 container_of(work,
1815 struct msm_ipc_router_xprt_info,
1816 read_data);
1817
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001818 while ((pkt = rr_read(xprt_info)) != NULL) {
1819 if (pkt->length < IPC_ROUTER_HDR_SIZE ||
1820 pkt->length > MAX_IPC_PKT_SIZE) {
1821 pr_err("%s: Invalid pkt length %d\n",
1822 __func__, pkt->length);
1823 goto fail_data;
1824 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001825
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001826 head_skb = skb_peek(pkt->pkt_fragment_q);
1827 if (!head_skb) {
1828 pr_err("%s: head_skb is invalid\n", __func__);
1829 goto fail_data;
1830 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001831
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001832 hdr = (struct rr_header *)(head_skb->data);
Karthikeyan Ramasubramanian2bfe8ec2013-03-22 10:47:20 -06001833 RAW("ver=%d type=%d src=%d:%08x crx=%d siz=%d dst=%d:%08x\n",
1834 hdr->version, hdr->type, hdr->src_node_id,
1835 hdr->src_port_id, hdr->confirm_rx, hdr->size,
1836 hdr->dst_node_id, hdr->dst_port_id);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001837
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001838 if (hdr->version != IPC_ROUTER_VERSION) {
1839 pr_err("version %d != %d\n",
1840 hdr->version, IPC_ROUTER_VERSION);
1841 goto fail_data;
1842 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001843
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001844 if ((hdr->dst_node_id != IPC_ROUTER_NID_LOCAL) &&
1845 ((hdr->type == IPC_ROUTER_CTRL_CMD_RESUME_TX) ||
1846 (hdr->type == IPC_ROUTER_CTRL_CMD_DATA))) {
1847 forward_msg(xprt_info, pkt);
1848 release_pkt(pkt);
1849 continue;
1850 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001851
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001852 if ((hdr->dst_port_id == IPC_ROUTER_ADDRESS) ||
1853 (hdr->type == IPC_ROUTER_CTRL_CMD_HELLO)) {
1854 process_control_msg(xprt_info, pkt);
1855 release_pkt(pkt);
1856 continue;
1857 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001858#if defined(CONFIG_MSM_SMD_LOGGING)
1859#if defined(DEBUG)
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001860 if (msm_ipc_router_debug_mask & SMEM_LOG) {
1861 smem_log_event((SMEM_LOG_PROC_ID_APPS |
1862 SMEM_LOG_RPC_ROUTER_EVENT_BASE |
1863 IPC_ROUTER_LOG_EVENT_RX),
1864 (hdr->src_node_id << 24) |
1865 (hdr->src_port_id & 0xffffff),
1866 (hdr->dst_node_id << 24) |
1867 (hdr->dst_port_id & 0xffffff),
1868 (hdr->type << 24) | (hdr->confirm_rx << 16) |
1869 (hdr->size & 0xffff));
1870 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001871#endif
1872#endif
1873
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001874 resume_tx = hdr->confirm_rx;
1875 resume_tx_node_id = hdr->dst_node_id;
1876 resume_tx_port_id = hdr->dst_port_id;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001877
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001878 rport_ptr = msm_ipc_router_lookup_remote_port(hdr->src_node_id,
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06001879 hdr->src_port_id);
1880
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001881 mutex_lock(&local_ports_lock);
1882 port_ptr = msm_ipc_router_lookup_local_port(hdr->dst_port_id);
1883 if (!port_ptr) {
1884 pr_err("%s: No local port id %08x\n", __func__,
1885 hdr->dst_port_id);
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06001886 mutex_unlock(&local_ports_lock);
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001887 release_pkt(pkt);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001888 goto process_done;
1889 }
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001890
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001891 if (!rport_ptr) {
1892 rport_ptr = msm_ipc_router_create_remote_port(
1893 hdr->src_node_id,
1894 hdr->src_port_id);
1895 if (!rport_ptr) {
1896 pr_err("%s: Rmt Prt %08x:%08x create failed\n",
1897 __func__, hdr->src_node_id,
1898 hdr->src_port_id);
1899 mutex_unlock(&local_ports_lock);
1900 goto process_done;
1901 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001902 }
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001903
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06001904 mutex_lock(&port_ptr->port_rx_q_lock);
1905 wake_lock(&port_ptr->port_rx_wake_lock);
1906 list_add_tail(&pkt->list, &port_ptr->port_rx_q);
1907 wake_up(&port_ptr->port_rx_wait_q);
1908 if (port_ptr->notify)
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001909 port_ptr->notify(MSM_IPC_ROUTER_READ_CB,
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06001910 port_ptr->priv);
1911 mutex_unlock(&port_ptr->port_rx_q_lock);
1912 mutex_unlock(&local_ports_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001913
1914process_done:
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001915 if (resume_tx) {
1916 union rr_control_msg msg;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001917
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001918 msg.cmd = IPC_ROUTER_CTRL_CMD_RESUME_TX;
1919 msg.cli.node_id = resume_tx_node_id;
1920 msg.cli.port_id = resume_tx_port_id;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001921
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06001922 RR("x RESUME_TX id=%d:%08x\n",
1923 msg.cli.node_id, msg.cli.port_id);
1924 msm_ipc_router_send_control_msg(xprt_info, &msg);
1925 }
1926
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001927 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001928 return;
1929
1930fail_data:
1931 release_pkt(pkt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001932 pr_err("ipc_router has died\n");
1933}
1934
1935int msm_ipc_router_register_server(struct msm_ipc_port *port_ptr,
1936 struct msm_ipc_addr *name)
1937{
1938 struct msm_ipc_server *server;
1939 unsigned long flags;
1940 union rr_control_msg ctl;
1941
1942 if (!port_ptr || !name)
1943 return -EINVAL;
1944
1945 if (name->addrtype != MSM_IPC_ADDR_NAME)
1946 return -EINVAL;
1947
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06001948 mutex_lock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001949 server = msm_ipc_router_lookup_server(name->addr.port_name.service,
1950 name->addr.port_name.instance,
1951 IPC_ROUTER_NID_LOCAL,
1952 port_ptr->this_port.port_id);
1953 if (server) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06001954 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001955 pr_err("%s: Server already present\n", __func__);
1956 return -EINVAL;
1957 }
1958
1959 server = msm_ipc_router_create_server(name->addr.port_name.service,
1960 name->addr.port_name.instance,
1961 IPC_ROUTER_NID_LOCAL,
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06001962 port_ptr->this_port.port_id,
1963 NULL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001964 if (!server) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06001965 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001966 pr_err("%s: Server Creation failed\n", __func__);
1967 return -EINVAL;
1968 }
1969
1970 ctl.cmd = IPC_ROUTER_CTRL_CMD_NEW_SERVER;
1971 ctl.srv.service = server->name.service;
1972 ctl.srv.instance = server->name.instance;
1973 ctl.srv.node_id = IPC_ROUTER_NID_LOCAL;
1974 ctl.srv.port_id = port_ptr->this_port.port_id;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06001975 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001976 broadcast_ctl_msg(&ctl);
1977 spin_lock_irqsave(&port_ptr->port_lock, flags);
1978 port_ptr->type = SERVER_PORT;
Karthikeyan Ramasubramanian96cced72013-05-02 17:25:54 -06001979 port_ptr->mode_info.mode = MULTI_LINK_MODE;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001980 port_ptr->port_name.service = server->name.service;
1981 port_ptr->port_name.instance = server->name.instance;
1982 spin_unlock_irqrestore(&port_ptr->port_lock, flags);
1983 return 0;
1984}
1985
1986int msm_ipc_router_unregister_server(struct msm_ipc_port *port_ptr)
1987{
1988 struct msm_ipc_server *server;
1989 unsigned long flags;
1990 union rr_control_msg ctl;
1991
1992 if (!port_ptr)
1993 return -EINVAL;
1994
1995 if (port_ptr->type != SERVER_PORT) {
1996 pr_err("%s: Trying to unregister a non-server port\n",
1997 __func__);
1998 return -EINVAL;
1999 }
2000
2001 if (port_ptr->this_port.node_id != IPC_ROUTER_NID_LOCAL) {
2002 pr_err("%s: Trying to unregister a remote server locally\n",
2003 __func__);
2004 return -EINVAL;
2005 }
2006
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002007 mutex_lock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002008 server = msm_ipc_router_lookup_server(port_ptr->port_name.service,
2009 port_ptr->port_name.instance,
2010 port_ptr->this_port.node_id,
2011 port_ptr->this_port.port_id);
2012 if (!server) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002013 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002014 pr_err("%s: Server lookup failed\n", __func__);
2015 return -ENODEV;
2016 }
2017
2018 ctl.cmd = IPC_ROUTER_CTRL_CMD_REMOVE_SERVER;
2019 ctl.srv.service = server->name.service;
2020 ctl.srv.instance = server->name.instance;
2021 ctl.srv.node_id = IPC_ROUTER_NID_LOCAL;
2022 ctl.srv.port_id = port_ptr->this_port.port_id;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002023 msm_ipc_router_destroy_server(server, port_ptr->this_port.node_id,
2024 port_ptr->this_port.port_id);
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002025 mutex_unlock(&server_list_lock);
2026 broadcast_ctl_msg(&ctl);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002027 spin_lock_irqsave(&port_ptr->port_lock, flags);
2028 port_ptr->type = CLIENT_PORT;
2029 spin_unlock_irqrestore(&port_ptr->port_lock, flags);
2030 return 0;
2031}
2032
2033static int loopback_data(struct msm_ipc_port *src,
2034 uint32_t port_id,
2035 struct sk_buff_head *data)
2036{
2037 struct sk_buff *head_skb;
2038 struct rr_header *hdr;
2039 struct msm_ipc_port *port_ptr;
2040 struct rr_packet *pkt;
Karthikeyan Ramasubramanianb0e23c52013-02-08 13:07:42 -07002041 int ret_len;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002042
2043 if (!data) {
2044 pr_err("%s: Invalid pkt pointer\n", __func__);
2045 return -EINVAL;
2046 }
2047
2048 pkt = create_pkt(data);
2049 if (!pkt) {
2050 pr_err("%s: New pkt create failed\n", __func__);
2051 return -ENOMEM;
2052 }
2053
2054 head_skb = skb_peek(pkt->pkt_fragment_q);
Karthikeyan Ramasubramanian9024dd82011-12-19 18:44:19 -07002055 if (!head_skb) {
2056 pr_err("%s: pkt_fragment_q is empty\n", __func__);
Brent Hronik1c9a3d42013-04-17 15:10:40 -06002057 release_pkt(pkt);
Karthikeyan Ramasubramanian9024dd82011-12-19 18:44:19 -07002058 return -EINVAL;
2059 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002060 hdr = (struct rr_header *)skb_push(head_skb, IPC_ROUTER_HDR_SIZE);
2061 if (!hdr) {
2062 pr_err("%s: Prepend Header failed\n", __func__);
2063 release_pkt(pkt);
2064 return -ENOMEM;
2065 }
2066 hdr->version = IPC_ROUTER_VERSION;
2067 hdr->type = IPC_ROUTER_CTRL_CMD_DATA;
2068 hdr->src_node_id = src->this_port.node_id;
2069 hdr->src_port_id = src->this_port.port_id;
2070 hdr->size = pkt->length;
2071 hdr->confirm_rx = 0;
2072 hdr->dst_node_id = IPC_ROUTER_NID_LOCAL;
2073 hdr->dst_port_id = port_id;
2074 pkt->length += IPC_ROUTER_HDR_SIZE;
2075
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06002076 mutex_lock(&local_ports_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002077 port_ptr = msm_ipc_router_lookup_local_port(port_id);
2078 if (!port_ptr) {
2079 pr_err("%s: Local port %d not present\n", __func__, port_id);
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06002080 mutex_unlock(&local_ports_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002081 release_pkt(pkt);
2082 return -ENODEV;
2083 }
2084
2085 mutex_lock(&port_ptr->port_rx_q_lock);
2086 wake_lock(&port_ptr->port_rx_wake_lock);
2087 list_add_tail(&pkt->list, &port_ptr->port_rx_q);
Karthikeyan Ramasubramanianb0e23c52013-02-08 13:07:42 -07002088 ret_len = pkt->length;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002089 wake_up(&port_ptr->port_rx_wait_q);
2090 mutex_unlock(&port_ptr->port_rx_q_lock);
Karthikeyan Ramasubramanian96cced72013-05-02 17:25:54 -06002091 update_comm_mode_info(&src->mode_info, NULL);
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06002092 mutex_unlock(&local_ports_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002093
Karthikeyan Ramasubramanianb0e23c52013-02-08 13:07:42 -07002094 return ret_len;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002095}
2096
2097static int msm_ipc_router_write_pkt(struct msm_ipc_port *src,
2098 struct msm_ipc_router_remote_port *rport_ptr,
2099 struct rr_packet *pkt)
2100{
2101 struct sk_buff *head_skb;
2102 struct rr_header *hdr;
2103 struct msm_ipc_router_xprt_info *xprt_info;
2104 struct msm_ipc_routing_table_entry *rt_entry;
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05302105 struct msm_ipc_resume_tx_port *resume_tx_port;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002106 int ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002107
2108 if (!rport_ptr || !src || !pkt)
2109 return -EINVAL;
2110
2111 head_skb = skb_peek(pkt->pkt_fragment_q);
Karthikeyan Ramasubramanian9024dd82011-12-19 18:44:19 -07002112 if (!head_skb) {
2113 pr_err("%s: pkt_fragment_q is empty\n", __func__);
2114 return -EINVAL;
2115 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002116 hdr = (struct rr_header *)skb_push(head_skb, IPC_ROUTER_HDR_SIZE);
2117 if (!hdr) {
2118 pr_err("%s: Prepend Header failed\n", __func__);
2119 return -ENOMEM;
2120 }
2121 hdr->version = IPC_ROUTER_VERSION;
2122 hdr->type = IPC_ROUTER_CTRL_CMD_DATA;
2123 hdr->src_node_id = src->this_port.node_id;
2124 hdr->src_port_id = src->this_port.port_id;
2125 hdr->size = pkt->length;
2126 hdr->confirm_rx = 0;
2127 hdr->dst_node_id = rport_ptr->node_id;
2128 hdr->dst_port_id = rport_ptr->port_id;
2129 pkt->length += IPC_ROUTER_HDR_SIZE;
2130
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05302131 mutex_lock(&rport_ptr->quota_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002132 if (rport_ptr->restart_state != RESTART_NORMAL) {
2133 mutex_unlock(&rport_ptr->quota_lock);
2134 return -ENETRESET;
2135 }
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05302136 if (rport_ptr->tx_quota_cnt == IPC_ROUTER_DEFAULT_RX_QUOTA) {
2137 if (msm_ipc_router_lookup_resume_tx_port(
2138 rport_ptr, src->this_port.port_id)) {
2139 mutex_unlock(&rport_ptr->quota_lock);
2140 return -EAGAIN;
2141 }
2142 resume_tx_port =
2143 kzalloc(sizeof(struct msm_ipc_resume_tx_port),
2144 GFP_KERNEL);
2145 if (!resume_tx_port) {
2146 pr_err("%s: Resume_Tx port allocation failed\n",
2147 __func__);
2148 mutex_unlock(&rport_ptr->quota_lock);
2149 return -ENOMEM;
2150 }
2151 INIT_LIST_HEAD(&resume_tx_port->list);
2152 resume_tx_port->port_id = src->this_port.port_id;
2153 resume_tx_port->node_id = src->this_port.node_id;
2154 list_add_tail(&resume_tx_port->list,
2155 &rport_ptr->resume_tx_port_list);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002156 mutex_unlock(&rport_ptr->quota_lock);
Zaheerulla Meere3f6c3a2013-04-17 01:16:47 +05302157 return -EAGAIN;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002158 }
2159 rport_ptr->tx_quota_cnt++;
2160 if (rport_ptr->tx_quota_cnt == IPC_ROUTER_DEFAULT_RX_QUOTA)
2161 hdr->confirm_rx = 1;
2162 mutex_unlock(&rport_ptr->quota_lock);
2163
2164 mutex_lock(&routing_table_lock);
2165 rt_entry = lookup_routing_table(hdr->dst_node_id);
2166 if (!rt_entry || !rt_entry->xprt_info) {
2167 mutex_unlock(&routing_table_lock);
2168 pr_err("%s: Remote node %d not up\n",
2169 __func__, hdr->dst_node_id);
2170 return -ENODEV;
2171 }
2172 mutex_lock(&rt_entry->lock);
2173 xprt_info = rt_entry->xprt_info;
2174 mutex_lock(&xprt_info->tx_lock);
Karthikeyan Ramasubramanian8cec5922012-02-16 17:41:58 -07002175 ret = xprt_info->xprt->write(pkt, pkt->length, xprt_info->xprt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002176 mutex_unlock(&xprt_info->tx_lock);
2177 mutex_unlock(&rt_entry->lock);
2178 mutex_unlock(&routing_table_lock);
2179
2180 if (ret < 0) {
2181 pr_err("%s: Write on XPRT failed\n", __func__);
2182 return ret;
2183 }
Karthikeyan Ramasubramanian96cced72013-05-02 17:25:54 -06002184 update_comm_mode_info(&src->mode_info, xprt_info);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002185
2186 RAW_HDR("[w rr_h] "
2187 "ver=%i,type=%s,src_nid=%08x,src_port_id=%08x,"
2188 "confirm_rx=%i,size=%3i,dst_pid=%08x,dst_cid=%08x\n",
2189 hdr->version, type_to_str(hdr->type),
2190 hdr->src_node_id, hdr->src_port_id,
2191 hdr->confirm_rx, hdr->size,
2192 hdr->dst_node_id, hdr->dst_port_id);
2193
2194#if defined(CONFIG_MSM_SMD_LOGGING)
2195#if defined(DEBUG)
2196 if (msm_ipc_router_debug_mask & SMEM_LOG) {
2197 smem_log_event((SMEM_LOG_PROC_ID_APPS |
2198 SMEM_LOG_RPC_ROUTER_EVENT_BASE |
2199 IPC_ROUTER_LOG_EVENT_TX),
2200 (hdr->src_node_id << 24) |
2201 (hdr->src_port_id & 0xffffff),
2202 (hdr->dst_node_id << 24) |
2203 (hdr->dst_port_id & 0xffffff),
2204 (hdr->type << 24) | (hdr->confirm_rx << 16) |
2205 (hdr->size & 0xffff));
2206 }
2207#endif
2208#endif
2209
2210 return pkt->length;
2211}
2212
2213int msm_ipc_router_send_to(struct msm_ipc_port *src,
2214 struct sk_buff_head *data,
2215 struct msm_ipc_addr *dest)
2216{
2217 uint32_t dst_node_id = 0, dst_port_id = 0;
2218 struct msm_ipc_server *server;
2219 struct msm_ipc_server_port *server_port;
2220 struct msm_ipc_router_remote_port *rport_ptr = NULL;
2221 struct rr_packet *pkt;
2222 int ret;
2223
2224 if (!src || !data || !dest) {
2225 pr_err("%s: Invalid Parameters\n", __func__);
2226 return -EINVAL;
2227 }
2228
2229 /* Resolve Address*/
2230 if (dest->addrtype == MSM_IPC_ADDR_ID) {
2231 dst_node_id = dest->addr.port_addr.node_id;
2232 dst_port_id = dest->addr.port_addr.port_id;
2233 } else if (dest->addrtype == MSM_IPC_ADDR_NAME) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002234 mutex_lock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002235 server = msm_ipc_router_lookup_server(
2236 dest->addr.port_name.service,
2237 dest->addr.port_name.instance,
2238 0, 0);
2239 if (!server) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002240 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002241 pr_err("%s: Destination not reachable\n", __func__);
2242 return -ENODEV;
2243 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002244 server_port = list_first_entry(&server->server_port_list,
2245 struct msm_ipc_server_port,
2246 list);
2247 dst_node_id = server_port->server_addr.node_id;
2248 dst_port_id = server_port->server_addr.port_id;
2249 mutex_unlock(&server_list_lock);
2250 }
2251 if (dst_node_id == IPC_ROUTER_NID_LOCAL) {
2252 ret = loopback_data(src, dst_port_id, data);
2253 return ret;
2254 }
2255
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002256 rport_ptr = msm_ipc_router_lookup_remote_port(dst_node_id,
2257 dst_port_id);
2258 if (!rport_ptr) {
Zaheerulla Meer3d8b0a02013-05-10 15:51:28 +05302259 pr_err("%s: Remote port not found\n", __func__);
2260 return -ENODEV;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002261 }
2262
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -06002263 if (src->check_send_permissions) {
2264 ret = src->check_send_permissions(rport_ptr->sec_rule);
2265 if (ret <= 0) {
2266 pr_err("%s: permission failure for %s\n",
2267 __func__, current->comm);
2268 return -EPERM;
2269 }
2270 }
2271
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002272 pkt = create_pkt(data);
2273 if (!pkt) {
2274 pr_err("%s: Pkt creation failed\n", __func__);
2275 return -ENOMEM;
2276 }
2277
2278 ret = msm_ipc_router_write_pkt(src, rport_ptr, pkt);
2279 release_pkt(pkt);
2280
2281 return ret;
2282}
2283
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06002284int msm_ipc_router_send_msg(struct msm_ipc_port *src,
2285 struct msm_ipc_addr *dest,
2286 void *data, unsigned int data_len)
2287{
2288 struct sk_buff_head *out_skb_head;
2289 int ret;
2290
2291 out_skb_head = msm_ipc_router_buf_to_skb(data, data_len);
2292 if (!out_skb_head) {
2293 pr_err("%s: SKB conversion failed\n", __func__);
2294 return -EFAULT;
2295 }
2296
2297 ret = msm_ipc_router_send_to(src, out_skb_head, dest);
2298 if (ret < 0) {
2299 pr_err("%s: msm_ipc_router_send_to failed - ret: %d\n",
2300 __func__, ret);
2301 msm_ipc_router_free_skb(out_skb_head);
2302 }
2303 return 0;
2304}
2305
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002306int msm_ipc_router_read(struct msm_ipc_port *port_ptr,
2307 struct sk_buff_head **data,
2308 size_t buf_len)
2309{
2310 struct rr_packet *pkt;
2311 int ret;
2312
2313 if (!port_ptr || !data)
2314 return -EINVAL;
2315
2316 mutex_lock(&port_ptr->port_rx_q_lock);
2317 if (list_empty(&port_ptr->port_rx_q)) {
2318 mutex_unlock(&port_ptr->port_rx_q_lock);
2319 return -EAGAIN;
2320 }
2321
2322 pkt = list_first_entry(&port_ptr->port_rx_q, struct rr_packet, list);
2323 if ((buf_len) && ((pkt->length - IPC_ROUTER_HDR_SIZE) > buf_len)) {
2324 mutex_unlock(&port_ptr->port_rx_q_lock);
2325 return -ETOOSMALL;
2326 }
2327 list_del(&pkt->list);
2328 if (list_empty(&port_ptr->port_rx_q))
2329 wake_unlock(&port_ptr->port_rx_wake_lock);
2330 *data = pkt->pkt_fragment_q;
2331 ret = pkt->length;
2332 kfree(pkt);
2333 mutex_unlock(&port_ptr->port_rx_q_lock);
2334
2335 return ret;
2336}
2337
2338int msm_ipc_router_recv_from(struct msm_ipc_port *port_ptr,
2339 struct sk_buff_head **data,
2340 struct msm_ipc_addr *src,
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06002341 long timeout)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002342{
2343 int ret, data_len, align_size;
2344 struct sk_buff *temp_skb;
2345 struct rr_header *hdr = NULL;
2346
2347 if (!port_ptr || !data) {
2348 pr_err("%s: Invalid pointers being passed\n", __func__);
2349 return -EINVAL;
2350 }
2351
2352 *data = NULL;
2353 mutex_lock(&port_ptr->port_rx_q_lock);
2354 while (list_empty(&port_ptr->port_rx_q)) {
2355 mutex_unlock(&port_ptr->port_rx_q_lock);
2356 if (timeout < 0) {
2357 ret = wait_event_interruptible(
2358 port_ptr->port_rx_wait_q,
2359 !list_empty(&port_ptr->port_rx_q));
2360 if (ret)
2361 return ret;
2362 } else if (timeout > 0) {
2363 timeout = wait_event_interruptible_timeout(
2364 port_ptr->port_rx_wait_q,
2365 !list_empty(&port_ptr->port_rx_q),
2366 timeout);
2367 if (timeout < 0)
2368 return -EFAULT;
2369 }
2370 if (timeout == 0)
2371 return -ETIMEDOUT;
2372 mutex_lock(&port_ptr->port_rx_q_lock);
2373 }
2374 mutex_unlock(&port_ptr->port_rx_q_lock);
2375
2376 ret = msm_ipc_router_read(port_ptr, data, 0);
2377 if (ret <= 0 || !(*data))
2378 return ret;
2379
2380 temp_skb = skb_peek(*data);
2381 hdr = (struct rr_header *)(temp_skb->data);
2382 if (src) {
2383 src->addrtype = MSM_IPC_ADDR_ID;
2384 src->addr.port_addr.node_id = hdr->src_node_id;
2385 src->addr.port_addr.port_id = hdr->src_port_id;
2386 }
2387
2388 data_len = hdr->size;
2389 skb_pull(temp_skb, IPC_ROUTER_HDR_SIZE);
2390 align_size = ALIGN_SIZE(data_len);
2391 if (align_size) {
2392 temp_skb = skb_peek_tail(*data);
2393 skb_trim(temp_skb, (temp_skb->len - align_size));
2394 }
2395 return data_len;
2396}
2397
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06002398int msm_ipc_router_read_msg(struct msm_ipc_port *port_ptr,
2399 struct msm_ipc_addr *src,
2400 unsigned char **data,
2401 unsigned int *len)
2402{
2403 struct sk_buff_head *in_skb_head;
2404 int ret;
2405
2406 ret = msm_ipc_router_recv_from(port_ptr, &in_skb_head, src, -1);
2407 if (ret < 0) {
2408 pr_err("%s: msm_ipc_router_recv_from failed - ret: %d\n",
2409 __func__, ret);
2410 return ret;
2411 }
2412
2413 *data = msm_ipc_router_skb_to_buf(in_skb_head, ret);
2414 if (!(*data))
2415 pr_err("%s: Buf conversion failed\n", __func__);
2416
2417 *len = ret;
2418 msm_ipc_router_free_skb(in_skb_head);
2419 return 0;
2420}
2421
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002422struct msm_ipc_port *msm_ipc_router_create_port(
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06002423 void (*notify)(unsigned event, void *priv),
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002424 void *priv)
2425{
2426 struct msm_ipc_port *port_ptr;
Karthikeyan Ramasubramanianefc493b2012-07-12 10:25:49 -06002427 int ret;
2428
2429 ret = wait_for_completion_interruptible(&msm_ipc_local_router_up);
2430 if (ret < 0) {
2431 pr_err("%s: Error waiting for local router\n", __func__);
2432 return NULL;
2433 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002434
2435 port_ptr = msm_ipc_router_create_raw_port(NULL, notify, priv);
2436 if (!port_ptr)
2437 pr_err("%s: port_ptr alloc failed\n", __func__);
2438
2439 return port_ptr;
2440}
2441
2442int msm_ipc_router_close_port(struct msm_ipc_port *port_ptr)
2443{
2444 union rr_control_msg msg;
2445 struct rr_packet *pkt, *temp_pkt;
2446 struct msm_ipc_server *server;
2447
2448 if (!port_ptr)
2449 return -EINVAL;
2450
2451 if (port_ptr->type == SERVER_PORT || port_ptr->type == CLIENT_PORT) {
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06002452 mutex_lock(&local_ports_lock);
2453 list_del(&port_ptr->list);
2454 mutex_unlock(&local_ports_lock);
2455
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002456 if (port_ptr->type == SERVER_PORT) {
2457 msg.cmd = IPC_ROUTER_CTRL_CMD_REMOVE_SERVER;
2458 msg.srv.service = port_ptr->port_name.service;
2459 msg.srv.instance = port_ptr->port_name.instance;
2460 msg.srv.node_id = port_ptr->this_port.node_id;
2461 msg.srv.port_id = port_ptr->this_port.port_id;
2462 RR("x REMOVE_SERVER Name=%d:%08x Id=%d:%08x\n",
2463 msg.srv.service, msg.srv.instance,
2464 msg.srv.node_id, msg.srv.port_id);
Karthikeyan Ramasubramanian5c568fb2013-01-10 17:09:13 -07002465 broadcast_ctl_msg(&msg);
2466 broadcast_ctl_msg_locally(&msg);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002467 }
Karthikeyan Ramasubramanian5c568fb2013-01-10 17:09:13 -07002468
2469 /*
2470 * Server port could have been a client port earlier.
2471 * Send REMOVE_CLIENT message in either case.
2472 */
Karthikeyan Ramasubramanian5c568fb2013-01-10 17:09:13 -07002473 RR("x REMOVE_CLIENT id=%d:%08x\n",
Karthikeyan Ramasubramanian96cced72013-05-02 17:25:54 -06002474 port_ptr->this_port.node_id, port_ptr->this_port.port_id);
2475 msm_ipc_router_send_remove_client(&port_ptr->mode_info,
2476 port_ptr->this_port.node_id,
2477 port_ptr->this_port.port_id);
Karthikeyan Ramasubramanianbe9954b2012-06-13 12:59:54 -06002478 } else if (port_ptr->type == CONTROL_PORT) {
2479 mutex_lock(&control_ports_lock);
2480 list_del(&port_ptr->list);
2481 mutex_unlock(&control_ports_lock);
Karthikeyan Ramasubramanianf7a4b6e2013-01-16 09:00:28 -07002482 } else if (port_ptr->type == IRSC_PORT) {
2483 mutex_lock(&local_ports_lock);
2484 list_del(&port_ptr->list);
2485 mutex_unlock(&local_ports_lock);
2486 signal_irsc_completion();
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002487 }
2488
2489 mutex_lock(&port_ptr->port_rx_q_lock);
2490 list_for_each_entry_safe(pkt, temp_pkt, &port_ptr->port_rx_q, list) {
2491 list_del(&pkt->list);
2492 release_pkt(pkt);
2493 }
2494 mutex_unlock(&port_ptr->port_rx_q_lock);
2495
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002496 if (port_ptr->type == SERVER_PORT) {
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002497 mutex_lock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002498 server = msm_ipc_router_lookup_server(
2499 port_ptr->port_name.service,
2500 port_ptr->port_name.instance,
2501 port_ptr->this_port.node_id,
2502 port_ptr->this_port.port_id);
2503 if (server)
2504 msm_ipc_router_destroy_server(server,
2505 port_ptr->this_port.node_id,
2506 port_ptr->this_port.port_id);
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002507 mutex_unlock(&server_list_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002508 }
2509
Karthikeyan Ramasubramaniandd8c3b52011-11-30 16:26:12 -07002510 wake_lock_destroy(&port_ptr->port_rx_wake_lock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002511 kfree(port_ptr);
2512 return 0;
2513}
2514
2515int msm_ipc_router_get_curr_pkt_size(struct msm_ipc_port *port_ptr)
2516{
2517 struct rr_packet *pkt;
2518 int rc = 0;
2519
2520 if (!port_ptr)
2521 return -EINVAL;
2522
2523 mutex_lock(&port_ptr->port_rx_q_lock);
2524 if (!list_empty(&port_ptr->port_rx_q)) {
2525 pkt = list_first_entry(&port_ptr->port_rx_q,
2526 struct rr_packet, list);
2527 rc = pkt->length;
2528 }
2529 mutex_unlock(&port_ptr->port_rx_q_lock);
2530
2531 return rc;
2532}
2533
2534int msm_ipc_router_bind_control_port(struct msm_ipc_port *port_ptr)
2535{
2536 if (!port_ptr)
2537 return -EINVAL;
2538
2539 mutex_lock(&local_ports_lock);
2540 list_del(&port_ptr->list);
2541 mutex_unlock(&local_ports_lock);
2542 port_ptr->type = CONTROL_PORT;
2543 mutex_lock(&control_ports_lock);
2544 list_add_tail(&port_ptr->list, &control_ports);
2545 mutex_unlock(&control_ports_lock);
2546
2547 return 0;
2548}
2549
2550int msm_ipc_router_lookup_server_name(struct msm_ipc_port_name *srv_name,
Karthikeyan Ramasubramaniandfde01b2012-06-12 14:25:13 -06002551 struct msm_ipc_server_info *srv_info,
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002552 int num_entries_in_array,
2553 uint32_t lookup_mask)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002554{
2555 struct msm_ipc_server *server;
2556 struct msm_ipc_server_port *server_port;
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002557 int key, i = 0; /*num_entries_found*/
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002558
2559 if (!srv_name) {
2560 pr_err("%s: Invalid srv_name\n", __func__);
2561 return -EINVAL;
2562 }
2563
Karthikeyan Ramasubramaniandfde01b2012-06-12 14:25:13 -06002564 if (num_entries_in_array && !srv_info) {
2565 pr_err("%s: srv_info NULL\n", __func__);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002566 return -EINVAL;
2567 }
2568
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002569 mutex_lock(&server_list_lock);
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002570 if (!lookup_mask)
2571 lookup_mask = 0xFFFFFFFF;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002572 key = (srv_name->service & (SRV_HASH_SIZE - 1));
2573 list_for_each_entry(server, &server_list[key], list) {
2574 if ((server->name.service != srv_name->service) ||
2575 ((server->name.instance & lookup_mask) !=
2576 srv_name->instance))
2577 continue;
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002578
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002579 list_for_each_entry(server_port,
2580 &server->server_port_list, list) {
2581 if (i < num_entries_in_array) {
2582 srv_info[i].node_id =
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002583 server_port->server_addr.node_id;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002584 srv_info[i].port_id =
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002585 server_port->server_addr.port_id;
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002586 srv_info[i].service = server->name.service;
2587 srv_info[i].instance = server->name.instance;
Karthikeyan Ramasubramaniancc450c92011-07-27 14:38:15 -06002588 }
Karthikeyan Ramasubramanianbb8306b2012-10-25 15:40:45 -06002589 i++;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002590 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002591 }
2592 mutex_unlock(&server_list_lock);
2593
2594 return i;
2595}
2596
2597int msm_ipc_router_close(void)
2598{
2599 struct msm_ipc_router_xprt_info *xprt_info, *tmp_xprt_info;
2600
2601 mutex_lock(&xprt_info_list_lock);
2602 list_for_each_entry_safe(xprt_info, tmp_xprt_info,
2603 &xprt_info_list, list) {
Karthikeyan Ramasubramanian8cec5922012-02-16 17:41:58 -07002604 xprt_info->xprt->close(xprt_info->xprt);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002605 list_del(&xprt_info->list);
2606 kfree(xprt_info);
2607 }
2608 mutex_unlock(&xprt_info_list_lock);
2609 return 0;
2610}
2611
2612#if defined(CONFIG_DEBUG_FS)
2613static int dump_routing_table(char *buf, int max)
2614{
2615 int i = 0, j;
2616 struct msm_ipc_routing_table_entry *rt_entry;
2617
2618 for (j = 0; j < RT_HASH_SIZE; j++) {
2619 mutex_lock(&routing_table_lock);
2620 list_for_each_entry(rt_entry, &routing_table[j], list) {
2621 mutex_lock(&rt_entry->lock);
2622 i += scnprintf(buf + i, max - i,
2623 "Node Id: 0x%08x\n", rt_entry->node_id);
Karthikeyan Ramasubramanianbddeec72012-09-10 16:10:24 -06002624 if (rt_entry->node_id == IPC_ROUTER_NID_LOCAL) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002625 i += scnprintf(buf + i, max - i,
2626 "XPRT Name: Loopback\n");
2627 i += scnprintf(buf + i, max - i,
2628 "Next Hop: %d\n", rt_entry->node_id);
2629 } else {
2630 i += scnprintf(buf + i, max - i,
2631 "XPRT Name: %s\n",
2632 rt_entry->xprt_info->xprt->name);
2633 i += scnprintf(buf + i, max - i,
2634 "Next Hop: 0x%08x\n",
2635 rt_entry->xprt_info->remote_node_id);
2636 }
2637 i += scnprintf(buf + i, max - i, "\n");
2638 mutex_unlock(&rt_entry->lock);
2639 }
2640 mutex_unlock(&routing_table_lock);
2641 }
2642
2643 return i;
2644}
2645
2646static int dump_xprt_info(char *buf, int max)
2647{
2648 int i = 0;
2649 struct msm_ipc_router_xprt_info *xprt_info;
2650
2651 mutex_lock(&xprt_info_list_lock);
2652 list_for_each_entry(xprt_info, &xprt_info_list, list) {
2653 i += scnprintf(buf + i, max - i, "XPRT Name: %s\n",
2654 xprt_info->xprt->name);
2655 i += scnprintf(buf + i, max - i, "Link Id: %d\n",
2656 xprt_info->xprt->link_id);
2657 i += scnprintf(buf + i, max - i, "Initialized: %s\n",
2658 (xprt_info->initialized ? "Y" : "N"));
2659 i += scnprintf(buf + i, max - i, "Remote Node Id: 0x%08x\n",
2660 xprt_info->remote_node_id);
2661 i += scnprintf(buf + i, max - i, "\n");
2662 }
2663 mutex_unlock(&xprt_info_list_lock);
2664
2665 return i;
2666}
2667
2668static int dump_servers(char *buf, int max)
2669{
2670 int i = 0, j;
2671 struct msm_ipc_server *server;
2672 struct msm_ipc_server_port *server_port;
2673
2674 mutex_lock(&server_list_lock);
2675 for (j = 0; j < SRV_HASH_SIZE; j++) {
2676 list_for_each_entry(server, &server_list[j], list) {
2677 list_for_each_entry(server_port,
2678 &server->server_port_list,
2679 list) {
2680 i += scnprintf(buf + i, max - i, "Service: "
2681 "0x%08x\n", server->name.service);
2682 i += scnprintf(buf + i, max - i, "Instance: "
2683 "0x%08x\n", server->name.instance);
2684 i += scnprintf(buf + i, max - i,
2685 "Node_id: 0x%08x\n",
2686 server_port->server_addr.node_id);
2687 i += scnprintf(buf + i, max - i,
2688 "Port_id: 0x%08x\n",
2689 server_port->server_addr.port_id);
2690 i += scnprintf(buf + i, max - i, "\n");
2691 }
2692 }
2693 }
2694 mutex_unlock(&server_list_lock);
2695
2696 return i;
2697}
2698
2699static int dump_remote_ports(char *buf, int max)
2700{
2701 int i = 0, j, k;
2702 struct msm_ipc_router_remote_port *rport_ptr;
2703 struct msm_ipc_routing_table_entry *rt_entry;
2704
2705 for (j = 0; j < RT_HASH_SIZE; j++) {
2706 mutex_lock(&routing_table_lock);
2707 list_for_each_entry(rt_entry, &routing_table[j], list) {
2708 mutex_lock(&rt_entry->lock);
2709 for (k = 0; k < RP_HASH_SIZE; k++) {
2710 list_for_each_entry(rport_ptr,
2711 &rt_entry->remote_port_list[k],
2712 list) {
2713 i += scnprintf(buf + i, max - i,
2714 "Node_id: 0x%08x\n",
2715 rport_ptr->node_id);
2716 i += scnprintf(buf + i, max - i,
2717 "Port_id: 0x%08x\n",
2718 rport_ptr->port_id);
2719 i += scnprintf(buf + i, max - i,
2720 "Quota_cnt: %d\n",
2721 rport_ptr->tx_quota_cnt);
2722 i += scnprintf(buf + i, max - i, "\n");
2723 }
2724 }
2725 mutex_unlock(&rt_entry->lock);
2726 }
2727 mutex_unlock(&routing_table_lock);
2728 }
2729
2730 return i;
2731}
2732
2733static int dump_control_ports(char *buf, int max)
2734{
2735 int i = 0;
2736 struct msm_ipc_port *port_ptr;
2737
2738 mutex_lock(&control_ports_lock);
2739 list_for_each_entry(port_ptr, &control_ports, list) {
2740 i += scnprintf(buf + i, max - i, "Node_id: 0x%08x\n",
2741 port_ptr->this_port.node_id);
2742 i += scnprintf(buf + i, max - i, "Port_id: 0x%08x\n",
2743 port_ptr->this_port.port_id);
2744 i += scnprintf(buf + i, max - i, "\n");
2745 }
2746 mutex_unlock(&control_ports_lock);
2747
2748 return i;
2749}
2750
2751static int dump_local_ports(char *buf, int max)
2752{
2753 int i = 0, j;
2754 unsigned long flags;
2755 struct msm_ipc_port *port_ptr;
2756
2757 mutex_lock(&local_ports_lock);
2758 for (j = 0; j < LP_HASH_SIZE; j++) {
2759 list_for_each_entry(port_ptr, &local_ports[j], list) {
2760 spin_lock_irqsave(&port_ptr->port_lock, flags);
2761 i += scnprintf(buf + i, max - i, "Node_id: 0x%08x\n",
2762 port_ptr->this_port.node_id);
2763 i += scnprintf(buf + i, max - i, "Port_id: 0x%08x\n",
2764 port_ptr->this_port.port_id);
2765 i += scnprintf(buf + i, max - i, "# pkts tx'd %d\n",
2766 port_ptr->num_tx);
2767 i += scnprintf(buf + i, max - i, "# pkts rx'd %d\n",
2768 port_ptr->num_rx);
2769 i += scnprintf(buf + i, max - i, "# bytes tx'd %ld\n",
2770 port_ptr->num_tx_bytes);
2771 i += scnprintf(buf + i, max - i, "# bytes rx'd %ld\n",
2772 port_ptr->num_rx_bytes);
2773 spin_unlock_irqrestore(&port_ptr->port_lock, flags);
2774 i += scnprintf(buf + i, max - i, "\n");
2775 }
2776 }
2777 mutex_unlock(&local_ports_lock);
2778
2779 return i;
2780}
2781
2782#define DEBUG_BUFMAX 4096
2783static char debug_buffer[DEBUG_BUFMAX];
2784
2785static ssize_t debug_read(struct file *file, char __user *buf,
2786 size_t count, loff_t *ppos)
2787{
2788 int (*fill)(char *buf, int max) = file->private_data;
2789 int bsize = fill(debug_buffer, DEBUG_BUFMAX);
2790 return simple_read_from_buffer(buf, count, ppos, debug_buffer, bsize);
2791}
2792
2793static int debug_open(struct inode *inode, struct file *file)
2794{
2795 file->private_data = inode->i_private;
2796 return 0;
2797}
2798
2799static const struct file_operations debug_ops = {
2800 .read = debug_read,
2801 .open = debug_open,
2802};
2803
2804static void debug_create(const char *name, mode_t mode,
2805 struct dentry *dent,
2806 int (*fill)(char *buf, int max))
2807{
2808 debugfs_create_file(name, mode, dent, fill, &debug_ops);
2809}
2810
2811static void debugfs_init(void)
2812{
2813 struct dentry *dent;
2814
2815 dent = debugfs_create_dir("msm_ipc_router", 0);
2816 if (IS_ERR(dent))
2817 return;
2818
2819 debug_create("dump_local_ports", 0444, dent,
2820 dump_local_ports);
2821 debug_create("dump_remote_ports", 0444, dent,
2822 dump_remote_ports);
2823 debug_create("dump_control_ports", 0444, dent,
2824 dump_control_ports);
2825 debug_create("dump_servers", 0444, dent,
2826 dump_servers);
2827 debug_create("dump_xprt_info", 0444, dent,
2828 dump_xprt_info);
2829 debug_create("dump_routing_table", 0444, dent,
2830 dump_routing_table);
2831}
2832
2833#else
2834static void debugfs_init(void) {}
2835#endif
2836
2837static int msm_ipc_router_add_xprt(struct msm_ipc_router_xprt *xprt)
2838{
2839 struct msm_ipc_router_xprt_info *xprt_info;
2840 struct msm_ipc_routing_table_entry *rt_entry;
2841
2842 xprt_info = kmalloc(sizeof(struct msm_ipc_router_xprt_info),
2843 GFP_KERNEL);
2844 if (!xprt_info)
2845 return -ENOMEM;
2846
2847 xprt_info->xprt = xprt;
2848 xprt_info->initialized = 0;
2849 xprt_info->remote_node_id = -1;
2850 INIT_LIST_HEAD(&xprt_info->pkt_list);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002851 mutex_init(&xprt_info->rx_lock);
2852 mutex_init(&xprt_info->tx_lock);
2853 wake_lock_init(&xprt_info->wakelock,
2854 WAKE_LOCK_SUSPEND, xprt->name);
2855 xprt_info->need_len = 0;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002856 xprt_info->abort_data_read = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002857 INIT_WORK(&xprt_info->read_data, do_read_data);
2858 INIT_LIST_HEAD(&xprt_info->list);
2859
2860 xprt_info->workqueue = create_singlethread_workqueue(xprt->name);
2861 if (!xprt_info->workqueue) {
2862 kfree(xprt_info);
2863 return -ENOMEM;
2864 }
2865
2866 if (!strcmp(xprt->name, "msm_ipc_router_loopback_xprt")) {
2867 xprt_info->remote_node_id = IPC_ROUTER_NID_LOCAL;
2868 xprt_info->initialized = 1;
2869 }
2870
2871 mutex_lock(&xprt_info_list_lock);
2872 list_add_tail(&xprt_info->list, &xprt_info_list);
2873 mutex_unlock(&xprt_info_list_lock);
2874
2875 mutex_lock(&routing_table_lock);
2876 if (!routing_table_inited) {
2877 init_routing_table();
2878 rt_entry = alloc_routing_table_entry(IPC_ROUTER_NID_LOCAL);
2879 add_routing_table_entry(rt_entry);
2880 routing_table_inited = 1;
2881 }
2882 mutex_unlock(&routing_table_lock);
2883
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002884 xprt->priv = xprt_info;
2885
2886 return 0;
2887}
2888
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002889static void msm_ipc_router_remove_xprt(struct msm_ipc_router_xprt *xprt)
2890{
2891 struct msm_ipc_router_xprt_info *xprt_info;
2892
2893 if (xprt && xprt->priv) {
2894 xprt_info = xprt->priv;
2895
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06002896 mutex_lock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002897 xprt_info->abort_data_read = 1;
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06002898 mutex_unlock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002899
2900 mutex_lock(&xprt_info_list_lock);
2901 list_del(&xprt_info->list);
2902 mutex_unlock(&xprt_info_list_lock);
2903
2904 flush_workqueue(xprt_info->workqueue);
2905 destroy_workqueue(xprt_info->workqueue);
2906 wake_lock_destroy(&xprt_info->wakelock);
2907
2908 xprt->priv = 0;
2909 kfree(xprt_info);
2910 }
2911}
2912
2913
2914struct msm_ipc_router_xprt_work {
2915 struct msm_ipc_router_xprt *xprt;
2916 struct work_struct work;
2917};
2918
2919static void xprt_open_worker(struct work_struct *work)
2920{
2921 struct msm_ipc_router_xprt_work *xprt_work =
2922 container_of(work, struct msm_ipc_router_xprt_work, work);
2923
2924 msm_ipc_router_add_xprt(xprt_work->xprt);
2925 kfree(xprt_work);
2926}
2927
2928static void xprt_close_worker(struct work_struct *work)
2929{
2930 struct msm_ipc_router_xprt_work *xprt_work =
2931 container_of(work, struct msm_ipc_router_xprt_work, work);
2932
2933 modem_reset_cleanup(xprt_work->xprt->priv);
2934 msm_ipc_router_remove_xprt(xprt_work->xprt);
2935
2936 if (atomic_dec_return(&pending_close_count) == 0)
2937 wake_up(&subsystem_restart_wait);
2938
2939 kfree(xprt_work);
2940}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002941
2942void msm_ipc_router_xprt_notify(struct msm_ipc_router_xprt *xprt,
2943 unsigned event,
2944 void *data)
2945{
2946 struct msm_ipc_router_xprt_info *xprt_info = xprt->priv;
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002947 struct msm_ipc_router_xprt_work *xprt_work;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002948 struct rr_packet *pkt;
Karthikeyan Ramasubramanian4af9f7c2011-09-30 15:55:18 -06002949 unsigned long ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002950
Karthikeyan Ramasubramanian4af9f7c2011-09-30 15:55:18 -06002951 if (!msm_ipc_router_workqueue) {
2952 ret = wait_for_completion_timeout(&msm_ipc_local_router_up,
2953 IPC_ROUTER_INIT_TIMEOUT);
2954 if (!ret || !msm_ipc_router_workqueue) {
2955 pr_err("%s: IPC Router not initialized\n", __func__);
2956 return;
2957 }
2958 }
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002959
2960 switch (event) {
2961 case IPC_ROUTER_XPRT_EVENT_OPEN:
2962 D("open event for '%s'\n", xprt->name);
2963 xprt_work = kmalloc(sizeof(struct msm_ipc_router_xprt_work),
2964 GFP_ATOMIC);
Karthikeyan Ramasubramanian840bed12013-05-16 15:51:29 -06002965 if (xprt_work) {
2966 xprt_work->xprt = xprt;
2967 INIT_WORK(&xprt_work->work, xprt_open_worker);
2968 queue_work(msm_ipc_router_workqueue, &xprt_work->work);
2969 } else {
2970 pr_err("%s: malloc failure - Couldn't notify OPEN event",
2971 __func__);
2972 }
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002973 break;
2974
2975 case IPC_ROUTER_XPRT_EVENT_CLOSE:
2976 D("close event for '%s'\n", xprt->name);
2977 atomic_inc(&pending_close_count);
2978 xprt_work = kmalloc(sizeof(struct msm_ipc_router_xprt_work),
2979 GFP_ATOMIC);
Karthikeyan Ramasubramanian840bed12013-05-16 15:51:29 -06002980 if (xprt_work) {
2981 xprt_work->xprt = xprt;
2982 INIT_WORK(&xprt_work->work, xprt_close_worker);
2983 queue_work(msm_ipc_router_workqueue, &xprt_work->work);
2984 } else {
2985 pr_err("%s: malloc failure - Couldn't notify CLOSE event",
2986 __func__);
2987 }
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06002988 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002989 }
2990
2991 if (!data)
2992 return;
2993
2994 while (!xprt_info) {
2995 msleep(100);
2996 xprt_info = xprt->priv;
2997 }
2998
2999 pkt = clone_pkt((struct rr_packet *)data);
3000 if (!pkt)
3001 return;
3002
3003 mutex_lock(&xprt_info->rx_lock);
3004 list_add_tail(&pkt->list, &xprt_info->pkt_list);
3005 wake_lock(&xprt_info->wakelock);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003006 mutex_unlock(&xprt_info->rx_lock);
Karthikeyan Ramasubramanian872ecd82012-07-25 11:07:48 -06003007 queue_work(xprt_info->workqueue, &xprt_info->read_data);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003008}
3009
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06003010static int modem_restart_notifier_cb(struct notifier_block *this,
3011 unsigned long code,
3012 void *data);
3013static struct notifier_block msm_ipc_router_nb = {
3014 .notifier_call = modem_restart_notifier_cb,
3015};
3016
3017static int modem_restart_notifier_cb(struct notifier_block *this,
3018 unsigned long code,
3019 void *data)
3020{
3021 switch (code) {
3022 case SUBSYS_BEFORE_SHUTDOWN:
3023 D("%s: SUBSYS_BEFORE_SHUTDOWN\n", __func__);
3024 break;
3025
3026 case SUBSYS_BEFORE_POWERUP:
3027 D("%s: waiting for RPC restart to complete\n", __func__);
3028 wait_event(subsystem_restart_wait,
3029 atomic_read(&pending_close_count) == 0);
3030 D("%s: finished restart wait\n", __func__);
3031 break;
3032
3033 default:
3034 break;
3035 }
3036
3037 return NOTIFY_DONE;
3038}
3039
3040static void *restart_notifier_handle;
3041static __init int msm_ipc_router_modem_restart_late_init(void)
3042{
3043 restart_notifier_handle = subsys_notif_register_notifier("modem",
3044 &msm_ipc_router_nb);
3045 return 0;
3046}
3047late_initcall(msm_ipc_router_modem_restart_late_init);
3048
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003049static int __init msm_ipc_router_init(void)
3050{
3051 int i, ret;
3052 struct msm_ipc_routing_table_entry *rt_entry;
3053
3054 msm_ipc_router_debug_mask |= SMEM_LOG;
Karthikeyan Ramasubramanian2bfe8ec2013-03-22 10:47:20 -06003055 ipc_rtr_log_ctxt = ipc_log_context_create(IPC_RTR_LOG_PAGES,
3056 "ipc_router");
3057 if (!ipc_rtr_log_ctxt)
3058 pr_err("%s: Unable to create IPC logging for IPC RTR",
3059 __func__);
3060
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06003061 msm_ipc_router_workqueue =
3062 create_singlethread_workqueue("msm_ipc_router");
3063 if (!msm_ipc_router_workqueue)
3064 return -ENOMEM;
3065
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003066 debugfs_init();
3067
3068 for (i = 0; i < SRV_HASH_SIZE; i++)
3069 INIT_LIST_HEAD(&server_list[i]);
3070
3071 for (i = 0; i < LP_HASH_SIZE; i++)
3072 INIT_LIST_HEAD(&local_ports[i]);
3073
3074 mutex_lock(&routing_table_lock);
3075 if (!routing_table_inited) {
3076 init_routing_table();
3077 rt_entry = alloc_routing_table_entry(IPC_ROUTER_NID_LOCAL);
3078 add_routing_table_entry(rt_entry);
3079 routing_table_inited = 1;
3080 }
3081 mutex_unlock(&routing_table_lock);
3082
3083 init_waitqueue_head(&newserver_wait);
Karthikeyan Ramasubramanianff6fbae2011-06-09 11:13:19 -06003084 init_waitqueue_head(&subsystem_restart_wait);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003085 ret = msm_ipc_router_init_sockets();
3086 if (ret < 0)
3087 pr_err("%s: Init sockets failed\n", __func__);
3088
Karthikeyan Ramasubramanian5b502d3642012-09-23 22:23:36 -06003089 ret = msm_ipc_router_security_init();
3090 if (ret < 0)
3091 pr_err("%s: Security Init failed\n", __func__);
3092
Karthikeyan Ramasubramanian4af9f7c2011-09-30 15:55:18 -06003093 complete_all(&msm_ipc_local_router_up);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07003094 return ret;
3095}
3096
3097module_init(msm_ipc_router_init);
3098MODULE_DESCRIPTION("MSM IPC Router");
3099MODULE_LICENSE("GPL v2");