blob: 9c864032fbfd8abbbe201e9e02e38a0d04eda4c7 [file] [log] [blame]
David Dai04ce4202016-09-26 16:24:13 -07001/* Copyright (c) 2014-2016, The Linux Foundation. All rights reserved.
2 *
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#include <linux/kernel.h>
13#include <linux/init.h>
14#include <linux/list.h>
15#include <linux/module.h>
16#include <linux/slab.h>
17#include <linux/rtmutex.h>
18#include <linux/clk.h>
19#include <linux/msm-bus.h>
20#include "msm_bus_core.h"
21#include "msm_bus_rpmh.h"
22
23#define NUM_CL_HANDLES 50
24#define NUM_LNODES 3
25#define MAX_STR_CL 50
26
27struct bus_search_type {
28 struct list_head link;
29 struct list_head node_list;
30};
31
32struct handle_type {
33 int num_entries;
34 struct msm_bus_client **cl_list;
35};
36
37static struct handle_type handle_list;
38static LIST_HEAD(input_list);
39static LIST_HEAD(apply_list);
40static LIST_HEAD(commit_list);
41
42DEFINE_RT_MUTEX(msm_bus_adhoc_lock);
43
44static bool chk_bl_list(struct list_head *black_list, unsigned int id)
45{
46 struct msm_bus_node_device_type *bus_node = NULL;
47
48 list_for_each_entry(bus_node, black_list, link) {
49 if (bus_node->node_info->id == id)
50 return true;
51 }
52 return false;
53}
54
55static void copy_remaining_nodes(struct list_head *edge_list, struct list_head
56 *traverse_list, struct list_head *route_list)
57{
58 struct bus_search_type *search_node;
59
60 if (list_empty(edge_list) && list_empty(traverse_list))
61 return;
62
63 search_node = kzalloc(sizeof(struct bus_search_type), GFP_KERNEL);
64 INIT_LIST_HEAD(&search_node->node_list);
65 list_splice_init(edge_list, traverse_list);
66 list_splice_init(traverse_list, &search_node->node_list);
67 list_add_tail(&search_node->link, route_list);
68}
69
70/*
71 * Duplicate instantiaion from msm_bus_arb.c. Todo there needs to be a
72 * "util" file for these common func/macros.
73 *
74 */
75uint64_t msm_bus_div64(unsigned int w, uint64_t bw)
76{
77 uint64_t *b = &bw;
78
79 if ((bw > 0) && (bw < w))
80 return 1;
81
82 switch (w) {
83 case 0:
84 WARN(1, "AXI: Divide by 0 attempted\n");
85 case 1: return bw;
86 case 2: return (bw >> 1);
87 case 4: return (bw >> 2);
88 case 8: return (bw >> 3);
89 case 16: return (bw >> 4);
90 case 32: return (bw >> 5);
91 }
92
93 do_div(*b, w);
94 return *b;
95}
96
97int msm_bus_device_match_adhoc(struct device *dev, void *id)
98{
99 int ret = 0;
100 struct msm_bus_node_device_type *bnode = to_msm_bus_node(dev);
101
102 if (bnode)
103 ret = (bnode->node_info->id == *(unsigned int *)id);
104 else
105 ret = 0;
106
107 return ret;
108}
109
110static void bcm_add_bus_req(struct device *dev)
111{
112 struct msm_bus_node_device_type *cur_dev = NULL;
113 struct msm_bus_node_device_type *bcm_dev = NULL;
114 struct link_node *lnode;
115 int lnode_idx = -1;
116 int max_num_lnodes = 0;
117 int i;
118
119 cur_dev = to_msm_bus_node(dev);
120 if (!cur_dev) {
121 MSM_BUS_ERR("%s: Null device ptr", __func__);
122 goto exit_bcm_add_bus_req;
123 }
124
125 if (!cur_dev->node_info->num_bcm_devs)
126 goto exit_bcm_add_bus_req;
127
128 for (i = 0; i < cur_dev->node_info->num_bcm_devs; i++) {
129 bcm_dev = to_msm_bus_node(cur_dev->node_info->bcm_devs[i]);
130 max_num_lnodes = bcm_dev->bcmdev->num_bus_devs;
131
132 if (!bcm_dev->num_lnodes) {
133 bcm_dev->lnode_list = devm_kzalloc(dev,
134 sizeof(struct link_node) * max_num_lnodes,
135 GFP_KERNEL);
136 if (!bcm_dev->lnode_list)
137 goto exit_bcm_add_bus_req;
138
139 lnode = bcm_dev->lnode_list;
140 bcm_dev->num_lnodes = max_num_lnodes;
141 lnode_idx = 0;
142 } else {
143 int i;
144
145 for (i = 0; i < bcm_dev->num_lnodes; i++) {
146 if (!bcm_dev->lnode_list[i].in_use)
147 break;
148 }
149
150 if (i < bcm_dev->num_lnodes) {
151 lnode = &bcm_dev->lnode_list[i];
152 lnode_idx = i;
153 } else {
154 struct link_node *realloc_list;
155 size_t cur_size = sizeof(struct link_node) *
156 bcm_dev->num_lnodes;
157
158 bcm_dev->num_lnodes += NUM_LNODES;
159 realloc_list = msm_bus_realloc_devmem(
160 dev,
161 bcm_dev->lnode_list,
162 cur_size,
163 sizeof(struct link_node) *
164 bcm_dev->num_lnodes,
165 GFP_KERNEL);
166
167 if (!realloc_list)
168 goto exit_bcm_add_bus_req;
169
170 bcm_dev->lnode_list = realloc_list;
171 lnode = &bcm_dev->lnode_list[i];
172 lnode_idx = i;
173 }
174 }
175
176 lnode->in_use = 1;
177 lnode->bus_dev_id = cur_dev->node_info->id;
178 cur_dev->node_info->bcm_req_idx = lnode_idx;
179 memset(lnode->lnode_ib, 0, sizeof(uint64_t) * NUM_CTX);
180 memset(lnode->lnode_ab, 0, sizeof(uint64_t) * NUM_CTX);
181 MSM_BUS_ERR("%s: Added %d entry to bcm %d @ %d\n", __func__,
182 lnode->bus_dev_id, bcm_dev->node_info->id, lnode_idx);
183 }
184
185exit_bcm_add_bus_req:
186 return;
187}
188
189static int gen_lnode(struct device *dev,
190 int next_hop, int prev_idx, const char *cl_name)
191{
192 struct link_node *lnode;
193 struct msm_bus_node_device_type *cur_dev = NULL;
194 int lnode_idx = -1;
195
196 if (!dev)
197 goto exit_gen_lnode;
198
199 cur_dev = to_msm_bus_node(dev);
200 if (!cur_dev) {
201 MSM_BUS_ERR("%s: Null device ptr", __func__);
202 goto exit_gen_lnode;
203 }
204
205 if (!cur_dev->num_lnodes) {
206 cur_dev->lnode_list = devm_kzalloc(dev,
207 sizeof(struct link_node) * NUM_LNODES,
208 GFP_KERNEL);
209 if (!cur_dev->lnode_list)
210 goto exit_gen_lnode;
211
212 lnode = cur_dev->lnode_list;
213 cur_dev->num_lnodes = NUM_LNODES;
214 lnode_idx = 0;
215 } else {
216 int i;
217
218 for (i = 0; i < cur_dev->num_lnodes; i++) {
219 if (!cur_dev->lnode_list[i].in_use)
220 break;
221 }
222
223 if (i < cur_dev->num_lnodes) {
224 lnode = &cur_dev->lnode_list[i];
225 lnode_idx = i;
226 } else {
227 struct link_node *realloc_list;
228 size_t cur_size = sizeof(struct link_node) *
229 cur_dev->num_lnodes;
230
231 cur_dev->num_lnodes += NUM_LNODES;
232 realloc_list = msm_bus_realloc_devmem(
233 dev,
234 cur_dev->lnode_list,
235 cur_size,
236 sizeof(struct link_node) *
237 cur_dev->num_lnodes, GFP_KERNEL);
238
239 if (!realloc_list)
240 goto exit_gen_lnode;
241
242 cur_dev->lnode_list = realloc_list;
243 lnode = &cur_dev->lnode_list[i];
244 lnode_idx = i;
245 }
246 }
247
248 lnode->in_use = 1;
249 lnode->cl_name = cl_name;
250 if (next_hop == cur_dev->node_info->id) {
251 lnode->next = -1;
252 lnode->next_dev = NULL;
253 } else {
254 lnode->next = prev_idx;
255 lnode->next_dev = bus_find_device(&msm_bus_type, NULL,
256 (void *) &next_hop,
257 msm_bus_device_match_adhoc);
258 }
259
260 memset(lnode->lnode_ib, 0, sizeof(uint64_t) * NUM_CTX);
261 memset(lnode->lnode_ab, 0, sizeof(uint64_t) * NUM_CTX);
262
263exit_gen_lnode:
264 return lnode_idx;
265}
266
267static int remove_lnode(struct msm_bus_node_device_type *cur_dev,
268 int lnode_idx)
269{
270 int ret = 0;
271
272 if (!cur_dev) {
273 MSM_BUS_ERR("%s: Null device ptr", __func__);
274 ret = -ENODEV;
275 goto exit_remove_lnode;
276 }
277
278 if (lnode_idx != -1) {
279 if (!cur_dev->num_lnodes ||
280 (lnode_idx > (cur_dev->num_lnodes - 1))) {
281 MSM_BUS_ERR("%s: Invalid Idx %d, num_lnodes %d",
282 __func__, lnode_idx, cur_dev->num_lnodes);
283 ret = -ENODEV;
284 goto exit_remove_lnode;
285 }
286
287 cur_dev->lnode_list[lnode_idx].next = -1;
288 cur_dev->lnode_list[lnode_idx].next_dev = NULL;
289 cur_dev->lnode_list[lnode_idx].in_use = 0;
290 cur_dev->lnode_list[lnode_idx].cl_name = NULL;
291 }
292
293exit_remove_lnode:
294 return ret;
295}
296
297static int prune_path(struct list_head *route_list, int dest, int src,
298 struct list_head *black_list, int found,
299 const char *cl_name)
300{
301 struct bus_search_type *search_node, *temp_search_node;
302 struct msm_bus_node_device_type *bus_node;
303 struct list_head *bl_list;
304 struct list_head *temp_bl_list;
305 int search_dev_id = dest;
306 struct device *dest_dev = bus_find_device(&msm_bus_type, NULL,
307 (void *) &dest,
308 msm_bus_device_match_adhoc);
309 int lnode_hop = -1;
310
311 if (!found)
312 goto reset_links;
313
314 if (!dest_dev) {
315 MSM_BUS_ERR("%s: Can't find dest dev %d", __func__, dest);
316 goto exit_prune_path;
317 }
318 MSM_BUS_ERR("%s: dest dev %d", __func__, dest);
319
320 lnode_hop = gen_lnode(dest_dev, search_dev_id, lnode_hop, cl_name);
321 bcm_add_bus_req(dest_dev);
322
323 list_for_each_entry_reverse(search_node, route_list, link) {
324 list_for_each_entry(bus_node, &search_node->node_list, link) {
325 unsigned int i;
326
327 for (i = 0; i < bus_node->node_info->num_connections;
328 i++) {
329 if (bus_node->node_info->connections[i] ==
330 search_dev_id) {
331 dest_dev = bus_find_device(
332 &msm_bus_type,
333 NULL,
334 (void *)
335 &bus_node->node_info->
336 id,
337 msm_bus_device_match_adhoc);
338
339 if (!dest_dev) {
340 lnode_hop = -1;
341 goto reset_links;
342 }
343
344 lnode_hop = gen_lnode(dest_dev,
345 search_dev_id,
346 lnode_hop, cl_name);
347 bcm_add_bus_req(dest_dev);
348 search_dev_id =
349 bus_node->node_info->id;
350 break;
351 }
352 }
353 }
354 }
355reset_links:
356 list_for_each_entry_safe(search_node, temp_search_node, route_list,
357 link) {
358 list_for_each_entry(bus_node, &search_node->node_list, link)
359 bus_node->node_info->is_traversed = false;
360
361 list_del(&search_node->link);
362 kfree(search_node);
363 }
364
365 list_for_each_safe(bl_list, temp_bl_list, black_list)
366 list_del(bl_list);
367
368exit_prune_path:
369 return lnode_hop;
370}
371
372static void setup_bl_list(struct msm_bus_node_device_type *node,
373 struct list_head *black_list)
374{
375 unsigned int i;
376
377 for (i = 0; i < node->node_info->num_blist; i++) {
378 struct msm_bus_node_device_type *bdev;
379
380 bdev = to_msm_bus_node(node->node_info->black_connections[i]);
381 list_add_tail(&bdev->link, black_list);
382 }
383}
384
385static int getpath(struct device *src_dev, int dest, const char *cl_name)
386{
387 struct list_head traverse_list;
388 struct list_head edge_list;
389 struct list_head route_list;
390 struct list_head black_list;
391 struct msm_bus_node_device_type *src_node;
392 struct bus_search_type *search_node;
393 int found = 0;
394 int depth_index = 0;
395 int first_hop = -1;
396 int src;
397
398 INIT_LIST_HEAD(&traverse_list);
399 INIT_LIST_HEAD(&edge_list);
400 INIT_LIST_HEAD(&route_list);
401 INIT_LIST_HEAD(&black_list);
402
403 if (!src_dev) {
404 MSM_BUS_ERR("%s: Cannot locate src dev ", __func__);
405 goto exit_getpath;
406 }
407
408 src_node = to_msm_bus_node(src_dev);
409 if (!src_node) {
410 MSM_BUS_ERR("%s:Fatal, Source node not found", __func__);
411 goto exit_getpath;
412 }
413 src = src_node->node_info->id;
414 list_add_tail(&src_node->link, &traverse_list);
415
416 while ((!found && !list_empty(&traverse_list))) {
417 struct msm_bus_node_device_type *bus_node = NULL;
418 /* Locate dest_id in the traverse list */
419 list_for_each_entry(bus_node, &traverse_list, link) {
420 if (bus_node->node_info->id == dest) {
421 found = 1;
422 break;
423 }
424 }
425
426 if (!found) {
427 unsigned int i;
428 /* Setup the new edge list */
429 list_for_each_entry(bus_node, &traverse_list, link) {
430 /* Setup list of black-listed nodes */
431 setup_bl_list(bus_node, &black_list);
432
433 for (i = 0; i < bus_node->node_info->
434 num_connections; i++) {
435 bool skip;
436 struct msm_bus_node_device_type
437 *node_conn;
438 node_conn =
439 to_msm_bus_node(bus_node->node_info->
440 dev_connections[i]);
441 if (node_conn->node_info->
442 is_traversed) {
443 MSM_BUS_ERR("Circ Path %d\n",
444 node_conn->node_info->id);
445 goto reset_traversed;
446 }
447 skip = chk_bl_list(&black_list,
448 bus_node->node_info->
449 connections[i]);
450 if (!skip) {
451 list_add_tail(&node_conn->link,
452 &edge_list);
453 node_conn->node_info->
454 is_traversed = true;
455 }
456 }
457 }
458
459 /* Keep tabs of the previous search list */
460 search_node = kzalloc(sizeof(struct bus_search_type),
461 GFP_KERNEL);
462 INIT_LIST_HEAD(&search_node->node_list);
463 list_splice_init(&traverse_list,
464 &search_node->node_list);
465 /* Add the previous search list to a route list */
466 list_add_tail(&search_node->link, &route_list);
467 /* Advancing the list depth */
468 depth_index++;
469 list_splice_init(&edge_list, &traverse_list);
470 }
471 }
472reset_traversed:
473 copy_remaining_nodes(&edge_list, &traverse_list, &route_list);
474 first_hop = prune_path(&route_list, dest, src, &black_list, found,
475 cl_name);
476
477exit_getpath:
478 return first_hop;
479}
480
481static void bcm_update_bus_req(struct device *dev, int ctx)
482{
483 struct msm_bus_node_device_type *cur_dev = NULL;
484 struct msm_bus_node_device_type *bcm_dev = NULL;
485 int i;
486 uint64_t max_ib = 0;
487 uint64_t max_ab = 0;
488 int lnode_idx = 0;
489
490 cur_dev = to_msm_bus_node(dev);
491 if (!cur_dev) {
492 MSM_BUS_ERR("%s: Null device ptr", __func__);
493 goto exit_bcm_update_bus_req;
494 }
495
496 if (!cur_dev->node_info->num_bcm_devs)
497 goto exit_bcm_update_bus_req;
498
499 for (i = 0; i < cur_dev->node_info->num_bcm_devs; i++) {
500 bcm_dev = to_msm_bus_node(cur_dev->node_info->bcm_devs[i]);
501
502 if (!bcm_dev)
503 goto exit_bcm_update_bus_req;
504
505 lnode_idx = cur_dev->node_info->bcm_req_idx;
506 bcm_dev->lnode_list[lnode_idx].lnode_ib[ctx] =
507 msm_bus_div64(cur_dev->node_info->agg_params.buswidth,
508 cur_dev->node_bw[ctx].max_ib *
509 (uint64_t)bcm_dev->bcmdev->width);
510
511 bcm_dev->lnode_list[lnode_idx].lnode_ab[ctx] =
512 msm_bus_div64(cur_dev->node_info->agg_params.buswidth,
513 cur_dev->node_bw[ctx].max_ab *
514 (uint64_t)bcm_dev->bcmdev->width);
515
516 for (i = 0; i < bcm_dev->num_lnodes; i++) {
517 if (ctx == ACTIVE_CTX) {
518 max_ib = max(max_ib,
519 max(bcm_dev->lnode_list[i].lnode_ib[ACTIVE_CTX],
520 bcm_dev->lnode_list[i].lnode_ib[DUAL_CTX]));
521
522 max_ab = max(max_ab,
523 bcm_dev->lnode_list[i].lnode_ab[ACTIVE_CTX] +
524 bcm_dev->lnode_list[i].lnode_ab[DUAL_CTX]);
525 } else {
526 max_ib = max(max_ib,
527 bcm_dev->lnode_list[i].lnode_ib[ctx]);
528 max_ab = max(max_ab,
529 bcm_dev->lnode_list[i].lnode_ab[ctx]);
530 }
531 }
532
533 bcm_dev->node_bw[ctx].max_ab = max_ab;
534 bcm_dev->node_bw[ctx].max_ib = max_ib;
535 }
536exit_bcm_update_bus_req:
537 return;
538}
539
540int bcm_remove_handoff_req(struct device *dev, void *data)
541{
542 struct msm_bus_node_device_type *bcm_dev = NULL;
543 int i;
544 uint64_t max_ib = 0;
545 uint64_t max_ab = 0;
546 int ret = 0;
547
548 rt_mutex_lock(&msm_bus_adhoc_lock);
549
550 bcm_dev = to_msm_bus_node(dev);
551 if (!bcm_dev) {
552 MSM_BUS_ERR("%s: Null device ptr", __func__);
553 goto exit_bcm_remove_handoff_req;
554 }
555
556 if (!bcm_dev->node_info->is_bcm_dev)
557 goto exit_bcm_remove_handoff_req;
558
559 bcm_dev->bcmdev->init_ab = 0;
560 bcm_dev->bcmdev->init_ib = 0;
561
562 for (i = 0; i < bcm_dev->num_lnodes; i++) {
563 max_ib = max(max_ib,
564 bcm_dev->lnode_list[i].lnode_ib[0]);
565 max_ab = max(max_ab,
566 bcm_dev->lnode_list[i].lnode_ab[0]);
567 }
568
569 bcm_dev->node_bw[0].max_ab = max_ab;
570 bcm_dev->node_bw[0].max_ib = max_ib;
571
572exit_bcm_remove_handoff_req:
573 rt_mutex_unlock(&msm_bus_adhoc_lock);
574 return ret;
575}
576
577
578
579static void aggregate_bus_req(struct msm_bus_node_device_type *bus_dev,
580 int ctx)
581{
582 int i;
583 uint64_t max_ib = 0;
584 uint64_t sum_ab = 0;
585
586 if (!bus_dev || !to_msm_bus_node(bus_dev->node_info->bus_device)) {
587 MSM_BUS_ERR("Bus node pointer is Invalid");
588 goto exit_agg_bus_req;
589 }
590
591 for (i = 0; i < bus_dev->num_lnodes; i++) {
592 max_ib = max(max_ib, bus_dev->lnode_list[i].lnode_ib[ctx]);
593 sum_ab += bus_dev->lnode_list[i].lnode_ab[ctx];
594 }
595
596 bus_dev->node_bw[ctx].sum_ab = sum_ab;
597 bus_dev->node_bw[ctx].max_ib = max_ib;
598
599exit_agg_bus_req:
600 return;
601}
602
603
604static void del_inp_list(struct list_head *list)
605{
606 struct rule_update_path_info *rule_node;
607 struct rule_update_path_info *rule_node_tmp;
608
609 list_for_each_entry_safe(rule_node, rule_node_tmp, list, link) {
610 list_del(&rule_node->link);
611 rule_node->added = false;
612 }
613}
614
615static void del_op_list(struct list_head *list)
616{
617 struct rule_apply_rcm_info *rule;
618 struct rule_apply_rcm_info *rule_tmp;
619
620 list_for_each_entry_safe(rule, rule_tmp, list, link)
621 list_del(&rule->link);
622}
623
624static int msm_bus_apply_rules(struct list_head *list, bool after_clk_commit)
625{
626 struct rule_apply_rcm_info *rule;
627 struct device *dev = NULL;
628 struct msm_bus_node_device_type *dev_info = NULL;
629 int ret = 0;
630
631 list_for_each_entry(rule, list, link) {
632 if (!rule)
633 continue;
634
635 if (rule && (rule->after_clk_commit != after_clk_commit))
636 continue;
637
638 dev = bus_find_device(&msm_bus_type, NULL,
639 (void *) &rule->id,
640 msm_bus_device_match_adhoc);
641
642 if (!dev) {
643 MSM_BUS_ERR("Can't find dev node for %d", rule->id);
644 continue;
645 }
646 dev_info = to_msm_bus_node(dev);
647
648 ret = msm_bus_enable_limiter(dev_info, rule->throttle,
649 rule->lim_bw);
650 if (ret)
651 MSM_BUS_ERR("Failed to set limiter for %d", rule->id);
652 }
653
654 return ret;
655}
656
657static void commit_data(void)
658{
659 bool rules_registered = msm_rule_are_rules_registered();
660
661 if (rules_registered) {
662 msm_rules_update_path(&input_list, &apply_list);
663 msm_bus_apply_rules(&apply_list, false);
664 }
665
666 msm_bus_commit_data(&commit_list);
667
668 if (rules_registered) {
669 msm_bus_apply_rules(&apply_list, true);
670 del_inp_list(&input_list);
671 del_op_list(&apply_list);
672 }
673 INIT_LIST_HEAD(&input_list);
674 INIT_LIST_HEAD(&apply_list);
675 INIT_LIST_HEAD(&commit_list);
676}
677
678static void add_node_to_clist(struct msm_bus_node_device_type *node)
679{
680 struct msm_bus_node_device_type *node_parent =
681 to_msm_bus_node(node->node_info->bus_device);
682
683 if (!node->dirty) {
684 list_add_tail(&node->link, &commit_list);
685 node->dirty = true;
686 }
687
688 if (!node_parent->dirty) {
689 list_add_tail(&node_parent->link, &commit_list);
690 node_parent->dirty = true;
691 }
692}
693
694static int update_path(struct device *src_dev, int dest, uint64_t act_req_ib,
695 uint64_t act_req_bw, uint64_t slp_req_ib,
696 uint64_t slp_req_bw, uint64_t cur_ib, uint64_t cur_bw,
697 int src_idx, int ctx)
698{
699 struct device *next_dev = NULL;
700 struct link_node *lnode = NULL;
701 struct msm_bus_node_device_type *dev_info = NULL;
702 int curr_idx;
703 int ret = 0;
704 struct rule_update_path_info *rule_node;
705 bool rules_registered = msm_rule_are_rules_registered();
706
707 if (IS_ERR_OR_NULL(src_dev)) {
708 MSM_BUS_ERR("%s: No source device", __func__);
709 ret = -ENODEV;
710 goto exit_update_path;
711 }
712
713 next_dev = src_dev;
714
715 if (src_idx < 0) {
716 MSM_BUS_ERR("%s: Invalid lnode idx %d", __func__, src_idx);
717 ret = -ENXIO;
718 goto exit_update_path;
719 }
720 curr_idx = src_idx;
721
722 while (next_dev) {
723 int i;
724
725 dev_info = to_msm_bus_node(next_dev);
726
727 if (curr_idx >= dev_info->num_lnodes) {
728 MSM_BUS_ERR("%s: Invalid lnode Idx %d num lnodes %d",
729 __func__, curr_idx, dev_info->num_lnodes);
730 ret = -ENXIO;
731 goto exit_update_path;
732 }
733
734 lnode = &dev_info->lnode_list[curr_idx];
735 if (!lnode) {
736 MSM_BUS_ERR("%s: Invalid lnode ptr lnode %d",
737 __func__, curr_idx);
738 ret = -ENXIO;
739 goto exit_update_path;
740 }
741 lnode->lnode_ib[ACTIVE_CTX] = act_req_ib;
742 lnode->lnode_ab[ACTIVE_CTX] = act_req_bw;
743 lnode->lnode_ib[DUAL_CTX] = slp_req_ib;
744 lnode->lnode_ab[DUAL_CTX] = slp_req_bw;
745
746 for (i = 0; i < NUM_CTX; i++) {
747 aggregate_bus_req(dev_info, i);
748 bcm_update_bus_req(next_dev, i);
749 }
750
751 add_node_to_clist(dev_info);
752
753 if (rules_registered) {
754 rule_node = &dev_info->node_info->rule;
755 rule_node->id = dev_info->node_info->id;
756 rule_node->ib = dev_info->node_bw[ACTIVE_CTX].max_ib;
757 rule_node->ab = dev_info->node_bw[ACTIVE_CTX].sum_ab;
758 rule_node->clk =
759 dev_info->node_bw[ACTIVE_CTX].cur_clk_hz;
760 if (!rule_node->added) {
761 list_add_tail(&rule_node->link, &input_list);
762 rule_node->added = true;
763 }
764 }
765
766 next_dev = lnode->next_dev;
767 curr_idx = lnode->next;
768 }
769
770exit_update_path:
771 return ret;
772}
773
774static int remove_path(struct device *src_dev, int dst, uint64_t cur_ib,
775 uint64_t cur_ab, int src_idx, int active_only)
776{
777 struct device *next_dev = NULL;
778 struct link_node *lnode = NULL;
779 struct msm_bus_node_device_type *dev_info = NULL;
780 int ret = 0;
781 int cur_idx = src_idx;
782 int next_idx;
783
784 /* Update the current path to zero out all request from
785 * this cient on all paths
786 */
787 if (!src_dev) {
788 MSM_BUS_ERR("%s: Can't find source device", __func__);
789 ret = -ENODEV;
790 goto exit_remove_path;
791 }
792
793 ret = update_path(src_dev, dst, 0, 0, 0, 0, cur_ib, cur_ab, src_idx,
794 active_only);
795 if (ret) {
796 MSM_BUS_ERR("%s: Error zeroing out path ctx %d",
797 __func__, ACTIVE_CTX);
798 goto exit_remove_path;
799 }
800
801 next_dev = src_dev;
802
803 while (next_dev) {
804 dev_info = to_msm_bus_node(next_dev);
805 lnode = &dev_info->lnode_list[cur_idx];
806 next_idx = lnode->next;
807 next_dev = lnode->next_dev;
808 remove_lnode(dev_info, cur_idx);
809 cur_idx = next_idx;
810 }
811
812exit_remove_path:
813 return ret;
814}
815
816static void getpath_debug(int src, int curr, int active_only)
817{
818 struct device *dev_node;
819 struct device *dev_it;
820 unsigned int hop = 1;
821 int idx;
822 struct msm_bus_node_device_type *devinfo;
823 int i;
824
825 dev_node = bus_find_device(&msm_bus_type, NULL,
826 (void *) &src,
827 msm_bus_device_match_adhoc);
828
829 if (!dev_node) {
830 MSM_BUS_ERR("SRC NOT FOUND %d", src);
831 return;
832 }
833
834 idx = curr;
835 devinfo = to_msm_bus_node(dev_node);
836 dev_it = dev_node;
837
838 MSM_BUS_ERR("Route list Src %d", src);
839 while (dev_it) {
840 struct msm_bus_node_device_type *busdev =
841 to_msm_bus_node(devinfo->node_info->bus_device);
842
843 MSM_BUS_ERR("Hop[%d] at Device %d ctx %d", hop,
844 devinfo->node_info->id, active_only);
845
846 for (i = 0; i < NUM_CTX; i++) {
847 MSM_BUS_ERR("dev info sel ib %llu",
848 devinfo->node_bw[i].cur_clk_hz);
849 MSM_BUS_ERR("dev info sel ab %llu",
850 devinfo->node_bw[i].sum_ab);
851 }
852
853 dev_it = devinfo->lnode_list[idx].next_dev;
854 idx = devinfo->lnode_list[idx].next;
855 if (dev_it)
856 devinfo = to_msm_bus_node(dev_it);
857
858 MSM_BUS_ERR("Bus Device %d", busdev->node_info->id);
859 MSM_BUS_ERR("Bus Clock %llu", busdev->clk[active_only].rate);
860
861 if (idx < 0)
862 break;
863 hop++;
864 }
865}
866
867static void unregister_client_adhoc(uint32_t cl)
868{
869 int i;
870 struct msm_bus_scale_pdata *pdata;
871 int lnode, src, curr, dest;
872 uint64_t cur_clk, cur_bw;
873 struct msm_bus_client *client;
874 struct device *src_dev;
875
876 rt_mutex_lock(&msm_bus_adhoc_lock);
877 if (!cl) {
878 MSM_BUS_ERR("%s: Null cl handle passed unregister\n",
879 __func__);
880 goto exit_unregister_client;
881 }
882 client = handle_list.cl_list[cl];
883 pdata = client->pdata;
884 if (!pdata) {
885 MSM_BUS_ERR("%s: Null pdata passed to unregister\n",
886 __func__);
887 goto exit_unregister_client;
888 }
889
890 curr = client->curr;
891 if (curr >= pdata->num_usecases) {
892 MSM_BUS_ERR("Invalid index Defaulting curr to 0");
893 curr = 0;
894 }
895
896 MSM_BUS_DBG("%s: Unregistering client %p", __func__, client);
897
898 for (i = 0; i < pdata->usecase->num_paths; i++) {
899 src = client->pdata->usecase[curr].vectors[i].src;
900 dest = client->pdata->usecase[curr].vectors[i].dst;
901
902 lnode = client->src_pnode[i];
903 src_dev = client->src_devs[i];
904 cur_clk = client->pdata->usecase[curr].vectors[i].ib;
905 cur_bw = client->pdata->usecase[curr].vectors[i].ab;
906 remove_path(src_dev, dest, cur_clk, cur_bw, lnode,
907 pdata->active_only);
908 }
909 commit_data();
910 msm_bus_dbg_client_data(client->pdata, MSM_BUS_DBG_UNREGISTER, cl);
911 kfree(client->src_pnode);
912 kfree(client->src_devs);
913 kfree(client);
914 handle_list.cl_list[cl] = NULL;
915exit_unregister_client:
916 rt_mutex_unlock(&msm_bus_adhoc_lock);
917}
918
919static int alloc_handle_lst(int size)
920{
921 int ret = 0;
922 struct msm_bus_client **t_cl_list;
923
924 if (!handle_list.num_entries) {
925 t_cl_list = kzalloc(sizeof(struct msm_bus_client *)
926 * NUM_CL_HANDLES, GFP_KERNEL);
927 if (ZERO_OR_NULL_PTR(t_cl_list)) {
928 ret = -ENOMEM;
929 MSM_BUS_ERR("%s: Failed to allocate handles list",
930 __func__);
931 goto exit_alloc_handle_lst;
932 }
933 handle_list.cl_list = t_cl_list;
934 handle_list.num_entries += NUM_CL_HANDLES;
935 } else {
936 t_cl_list = krealloc(handle_list.cl_list,
937 sizeof(struct msm_bus_client *) *
938 (handle_list.num_entries + NUM_CL_HANDLES),
939 GFP_KERNEL);
940 if (ZERO_OR_NULL_PTR(t_cl_list)) {
941 ret = -ENOMEM;
942 MSM_BUS_ERR("%s: Failed to allocate handles list",
943 __func__);
944 goto exit_alloc_handle_lst;
945 }
946
947 handle_list.cl_list = t_cl_list;
948 memset(&handle_list.cl_list[handle_list.num_entries], 0,
949 NUM_CL_HANDLES * sizeof(struct msm_bus_client *));
950 handle_list.num_entries += NUM_CL_HANDLES;
951 }
952exit_alloc_handle_lst:
953 return ret;
954}
955
956static uint32_t gen_handle(struct msm_bus_client *client)
957{
958 uint32_t handle = 0;
959 int i;
960 int ret = 0;
961
962 for (i = 0; i < handle_list.num_entries; i++) {
963 if (i && !handle_list.cl_list[i]) {
964 handle = i;
965 break;
966 }
967 }
968
969 if (!handle) {
970 ret = alloc_handle_lst(NUM_CL_HANDLES);
971
972 if (ret) {
973 MSM_BUS_ERR("%s: Failed to allocate handle list",
974 __func__);
975 goto exit_gen_handle;
976 }
977 handle = i + 1;
978 }
979 handle_list.cl_list[handle] = client;
980exit_gen_handle:
981 return handle;
982}
983
984static uint32_t register_client_adhoc(struct msm_bus_scale_pdata *pdata)
985{
986 int src, dest;
987 int i;
988 struct msm_bus_client *client = NULL;
989 int *lnode;
990 struct device *dev;
991 uint32_t handle = 0;
992
993 rt_mutex_lock(&msm_bus_adhoc_lock);
994 client = kzalloc(sizeof(struct msm_bus_client), GFP_KERNEL);
995 if (!client) {
996 MSM_BUS_ERR("%s: Error allocating client data", __func__);
997 goto exit_register_client;
998 }
999 client->pdata = pdata;
1000
1001 lnode = kcalloc(pdata->usecase->num_paths, sizeof(int), GFP_KERNEL);
1002 if (ZERO_OR_NULL_PTR(lnode)) {
1003 MSM_BUS_ERR("%s: Error allocating pathnode ptr!", __func__);
1004 goto exit_lnode_malloc_fail;
1005 }
1006 client->src_pnode = lnode;
1007
1008 client->src_devs = kcalloc(pdata->usecase->num_paths,
1009 sizeof(struct device *), GFP_KERNEL);
1010 if (IS_ERR_OR_NULL(client->src_devs)) {
1011 MSM_BUS_ERR("%s: Error allocating pathnode ptr!", __func__);
1012 goto exit_src_dev_malloc_fail;
1013 }
1014 client->curr = -1;
1015
1016 for (i = 0; i < pdata->usecase->num_paths; i++) {
1017 src = pdata->usecase->vectors[i].src;
1018 dest = pdata->usecase->vectors[i].dst;
1019
1020 if ((src < 0) || (dest < 0)) {
1021 MSM_BUS_ERR("%s:Invalid src/dst.src %d dest %d",
1022 __func__, src, dest);
1023 goto exit_invalid_data;
1024 }
1025 dev = bus_find_device(&msm_bus_type, NULL,
1026 (void *) &src,
1027 msm_bus_device_match_adhoc);
1028 if (IS_ERR_OR_NULL(dev)) {
1029 MSM_BUS_ERR("%s:Failed to find path.src %d dest %d",
1030 __func__, src, dest);
1031 goto exit_invalid_data;
1032 }
1033 client->src_devs[i] = dev;
1034
1035 MSM_BUS_ERR("%s:find path.src %d dest %d",
1036 __func__, src, dest);
1037
1038 lnode[i] = getpath(dev, dest, client->pdata->name);
1039 if (lnode[i] < 0) {
1040 MSM_BUS_ERR("%s:Failed to find path.src %d dest %d",
1041 __func__, src, dest);
1042 goto exit_invalid_data;
1043 }
1044 }
1045
1046 handle = gen_handle(client);
1047 msm_bus_dbg_client_data(client->pdata, MSM_BUS_DBG_REGISTER,
1048 handle);
1049 MSM_BUS_ERR("%s:Client handle %d %s", __func__, handle,
1050 client->pdata->name);
1051 rt_mutex_unlock(&msm_bus_adhoc_lock);
1052 return handle;
1053exit_invalid_data:
1054 kfree(client->src_devs);
1055exit_src_dev_malloc_fail:
1056 kfree(lnode);
1057exit_lnode_malloc_fail:
1058 kfree(client);
1059exit_register_client:
1060 rt_mutex_unlock(&msm_bus_adhoc_lock);
1061 return handle;
1062}
1063
1064static int update_client_paths(struct msm_bus_client *client, bool log_trns,
1065 unsigned int idx)
1066{
1067 int lnode, src, dest, cur_idx;
1068 uint64_t req_clk, req_bw, curr_clk, curr_bw, slp_clk, slp_bw;
1069 int i, ret = 0;
1070 struct msm_bus_scale_pdata *pdata;
1071 struct device *src_dev;
1072
1073 if (!client) {
1074 MSM_BUS_ERR("Client handle Null");
1075 ret = -ENXIO;
1076 goto exit_update_client_paths;
1077 }
1078
1079 pdata = client->pdata;
1080 if (!pdata) {
1081 MSM_BUS_ERR("Client pdata Null");
1082 ret = -ENXIO;
1083 goto exit_update_client_paths;
1084 }
1085
1086 cur_idx = client->curr;
1087 client->curr = idx;
1088 for (i = 0; i < pdata->usecase->num_paths; i++) {
1089 src = pdata->usecase[idx].vectors[i].src;
1090 dest = pdata->usecase[idx].vectors[i].dst;
1091
1092 lnode = client->src_pnode[i];
1093 src_dev = client->src_devs[i];
1094 req_clk = client->pdata->usecase[idx].vectors[i].ib;
1095 req_bw = client->pdata->usecase[idx].vectors[i].ab;
1096 if (cur_idx < 0) {
1097 curr_clk = 0;
1098 curr_bw = 0;
1099 } else {
1100 curr_clk =
1101 client->pdata->usecase[cur_idx].vectors[i].ib;
1102 curr_bw = client->pdata->usecase[cur_idx].vectors[i].ab;
1103 MSM_BUS_DBG("%s:ab: %llu ib: %llu\n", __func__,
1104 curr_bw, curr_clk);
1105 }
1106
1107 if (pdata->active_only) {
1108 slp_clk = 0;
1109 slp_bw = 0;
1110 } else {
1111 slp_clk = req_clk;
1112 slp_bw = req_bw;
1113 }
1114
1115 ret = update_path(src_dev, dest, req_clk, req_bw, slp_clk,
1116 slp_bw, curr_clk, curr_bw, lnode, pdata->active_only);
1117
1118 if (ret) {
1119 MSM_BUS_ERR("%s: Update path failed! %d ctx %d\n",
1120 __func__, ret, pdata->active_only);
1121 goto exit_update_client_paths;
1122 }
1123
1124 if (log_trns)
1125 getpath_debug(src, lnode, pdata->active_only);
1126 }
1127 commit_data();
1128exit_update_client_paths:
1129 return ret;
1130}
1131
1132static int update_context(uint32_t cl, bool active_only,
1133 unsigned int ctx_idx)
1134{
1135 int ret = 0;
1136 struct msm_bus_scale_pdata *pdata;
1137 struct msm_bus_client *client;
1138
1139 rt_mutex_lock(&msm_bus_adhoc_lock);
1140 if (!cl) {
1141 MSM_BUS_ERR("%s: Invalid client handle %d", __func__, cl);
1142 ret = -ENXIO;
1143 goto exit_update_context;
1144 }
1145
1146 client = handle_list.cl_list[cl];
1147 if (!client) {
1148 ret = -ENXIO;
1149 goto exit_update_context;
1150 }
1151
1152 pdata = client->pdata;
1153 if (!pdata) {
1154 ret = -ENXIO;
1155 goto exit_update_context;
1156 }
1157 if (pdata->active_only == active_only) {
1158 MSM_BUS_ERR("No change in context(%d==%d), skip\n",
1159 pdata->active_only, active_only);
1160 ret = -ENXIO;
1161 goto exit_update_context;
1162 }
1163
1164 if (ctx_idx >= pdata->num_usecases) {
1165 MSM_BUS_ERR("Client %u passed invalid index: %d\n",
1166 cl, ctx_idx);
1167 ret = -ENXIO;
1168 goto exit_update_context;
1169 }
1170
1171 pdata->active_only = active_only;
1172
1173 msm_bus_dbg_client_data(client->pdata, ctx_idx, cl);
1174 ret = update_client_paths(client, false, ctx_idx);
1175 if (ret) {
1176 pr_err("%s: Err updating path\n", __func__);
1177 goto exit_update_context;
1178 }
1179
1180// trace_bus_update_request_end(pdata->name);
1181
1182exit_update_context:
1183 rt_mutex_unlock(&msm_bus_adhoc_lock);
1184 return ret;
1185}
1186
1187static int update_request_adhoc(uint32_t cl, unsigned int index)
1188{
1189 int ret = 0;
1190 struct msm_bus_scale_pdata *pdata;
1191 struct msm_bus_client *client;
1192 const char *test_cl = "Null";
1193 bool log_transaction = false;
1194
1195 rt_mutex_lock(&msm_bus_adhoc_lock);
1196
1197 if (!cl) {
1198 MSM_BUS_ERR("%s: Invalid client handle %d", __func__, cl);
1199 ret = -ENXIO;
1200 goto exit_update_request;
1201 }
1202
1203 client = handle_list.cl_list[cl];
1204 if (!client) {
1205 MSM_BUS_ERR("%s: Invalid client pointer ", __func__);
1206 ret = -ENXIO;
1207 goto exit_update_request;
1208 }
1209
1210 pdata = client->pdata;
1211 if (!pdata) {
1212 MSM_BUS_ERR("%s: Client data Null.[client didn't register]",
1213 __func__);
1214 ret = -ENXIO;
1215 goto exit_update_request;
1216 }
1217
1218 if (index >= pdata->num_usecases) {
1219 MSM_BUS_ERR("Client %u passed invalid index: %d\n",
1220 cl, index);
1221 ret = -ENXIO;
1222 goto exit_update_request;
1223 }
1224
1225 if (client->curr == index) {
1226 MSM_BUS_DBG("%s: Not updating client request idx %d unchanged",
1227 __func__, index);
1228 goto exit_update_request;
1229 }
1230
1231 if (!strcmp(test_cl, pdata->name))
1232 log_transaction = true;
1233
1234 MSM_BUS_DBG("%s: cl: %u index: %d curr: %d num_paths: %d\n", __func__,
1235 cl, index, client->curr, client->pdata->usecase->num_paths);
1236 msm_bus_dbg_client_data(client->pdata, index, cl);
1237 ret = update_client_paths(client, log_transaction, index);
1238 if (ret) {
1239 pr_err("%s: Err updating path\n", __func__);
1240 goto exit_update_request;
1241 }
1242
1243// trace_bus_update_request_end(pdata->name);
1244
1245exit_update_request:
1246 rt_mutex_unlock(&msm_bus_adhoc_lock);
1247 return ret;
1248}
1249
1250static void free_cl_mem(struct msm_bus_client_handle *cl)
1251{
1252 if (cl) {
1253 kfree(cl->name);
1254 kfree(cl);
1255 cl = NULL;
1256 }
1257}
1258
1259static int update_bw_adhoc(struct msm_bus_client_handle *cl, u64 ab, u64 ib)
1260{
1261 int ret = 0;
1262 char *test_cl = "test-client";
1263 bool log_transaction = false;
1264 u64 slp_ib, slp_ab;
1265
1266 rt_mutex_lock(&msm_bus_adhoc_lock);
1267
1268 if (!cl) {
1269 MSM_BUS_ERR("%s: Invalid client handle %p", __func__, cl);
1270 ret = -ENXIO;
1271 goto exit_update_request;
1272 }
1273
1274 if (!strcmp(test_cl, cl->name))
1275 log_transaction = true;
1276
1277 msm_bus_dbg_rec_transaction(cl, ab, ib);
1278
1279 if ((cl->cur_act_ib == ib) && (cl->cur_act_ab == ab)) {
1280 MSM_BUS_DBG("%s:no change in request", cl->name);
1281 goto exit_update_request;
1282 }
1283
1284 if (cl->active_only) {
1285 slp_ib = 0;
1286 slp_ab = 0;
1287 } else {
1288 slp_ib = ib;
1289 slp_ab = ab;
1290 }
1291
1292 ret = update_path(cl->mas_dev, cl->slv, ib, ab, slp_ib, slp_ab,
1293 cl->cur_act_ib, cl->cur_act_ab, cl->first_hop, cl->active_only);
1294
1295 if (ret) {
1296 MSM_BUS_ERR("%s: Update path failed! %d active_only %d\n",
1297 __func__, ret, cl->active_only);
1298 goto exit_update_request;
1299 }
1300
1301 commit_data();
1302 cl->cur_act_ib = ib;
1303 cl->cur_act_ab = ab;
1304 cl->cur_slp_ib = slp_ib;
1305 cl->cur_slp_ab = slp_ab;
1306
1307 if (log_transaction)
1308 getpath_debug(cl->mas, cl->first_hop, cl->active_only);
1309// trace_bus_update_request_end(cl->name);
1310exit_update_request:
1311 rt_mutex_unlock(&msm_bus_adhoc_lock);
1312
1313 return ret;
1314}
1315
1316static int update_bw_context(struct msm_bus_client_handle *cl, u64 act_ab,
1317 u64 act_ib, u64 slp_ib, u64 slp_ab)
1318{
1319 int ret = 0;
1320
1321 rt_mutex_lock(&msm_bus_adhoc_lock);
1322 if (!cl) {
1323 MSM_BUS_ERR("Invalid client handle %p", cl);
1324 ret = -ENXIO;
1325 goto exit_change_context;
1326 }
1327
1328 if ((cl->cur_act_ib == act_ib) &&
1329 (cl->cur_act_ab == act_ab) &&
1330 (cl->cur_slp_ib == slp_ib) &&
1331 (cl->cur_slp_ab == slp_ab)) {
1332 MSM_BUS_ERR("No change in vote");
1333 goto exit_change_context;
1334 }
1335
1336 if (!slp_ab && !slp_ib)
1337 cl->active_only = true;
1338 msm_bus_dbg_rec_transaction(cl, cl->cur_act_ab, cl->cur_slp_ib);
1339 ret = update_path(cl->mas_dev, cl->slv, act_ib, act_ab, slp_ib, slp_ab,
1340 cl->cur_act_ab, cl->cur_act_ab, cl->first_hop,
1341 cl->active_only);
1342 if (ret) {
1343 MSM_BUS_ERR("%s: Update path failed! %d active_only %d\n",
1344 __func__, ret, cl->active_only);
1345 goto exit_change_context;
1346 }
1347 commit_data();
1348 cl->cur_act_ib = act_ib;
1349 cl->cur_act_ab = act_ab;
1350 cl->cur_slp_ib = slp_ib;
1351 cl->cur_slp_ab = slp_ab;
1352// trace_bus_update_request_end(cl->name);
1353exit_change_context:
1354 rt_mutex_unlock(&msm_bus_adhoc_lock);
1355 return ret;
1356}
1357
1358static void unregister_adhoc(struct msm_bus_client_handle *cl)
1359{
1360 rt_mutex_lock(&msm_bus_adhoc_lock);
1361 if (!cl) {
1362 MSM_BUS_ERR("%s: Null cl handle passed unregister\n",
1363 __func__);
1364 goto exit_unregister_client;
1365 }
1366
1367 MSM_BUS_DBG("%s: Unregistering client %p", __func__, cl);
1368
1369 remove_path(cl->mas_dev, cl->slv, cl->cur_act_ib, cl->cur_act_ab,
1370 cl->first_hop, cl->active_only);
1371 commit_data();
1372 msm_bus_dbg_remove_client(cl);
1373 kfree(cl);
1374exit_unregister_client:
1375 rt_mutex_unlock(&msm_bus_adhoc_lock);
1376}
1377
1378static struct msm_bus_client_handle*
1379register_adhoc(uint32_t mas, uint32_t slv, char *name, bool active_only)
1380{
1381 struct msm_bus_client_handle *client = NULL;
1382 int len = 0;
1383
1384 rt_mutex_lock(&msm_bus_adhoc_lock);
1385
1386 if (!(mas && slv && name)) {
1387 pr_err("%s: Error: src dst name num_paths are required",
1388 __func__);
1389 goto exit_register;
1390 }
1391
1392 client = kzalloc(sizeof(struct msm_bus_client_handle), GFP_KERNEL);
1393 if (!client) {
1394 MSM_BUS_ERR("%s: Error allocating client data", __func__);
1395 goto exit_register;
1396 }
1397
1398 len = strnlen(name, MAX_STR_CL);
1399 client->name = kzalloc((len + 1), GFP_KERNEL);
1400 if (!client->name) {
1401 MSM_BUS_ERR("%s: Error allocating client name buf", __func__);
1402 free_cl_mem(client);
1403 goto exit_register;
1404 }
1405 strlcpy(client->name, name, MAX_STR_CL);
1406 client->active_only = active_only;
1407
1408 client->mas = mas;
1409 client->slv = slv;
1410
1411 client->mas_dev = bus_find_device(&msm_bus_type, NULL,
1412 (void *) &mas,
1413 msm_bus_device_match_adhoc);
1414 if (IS_ERR_OR_NULL(client->mas_dev)) {
1415 MSM_BUS_ERR("%s:Failed to find path.src %d dest %d",
1416 __func__, client->mas, client->slv);
1417 free_cl_mem(client);
1418 goto exit_register;
1419 }
1420
1421 client->first_hop = getpath(client->mas_dev, client->slv, client->name);
1422 if (client->first_hop < 0) {
1423 MSM_BUS_ERR("%s:Failed to find path.src %d dest %d",
1424 __func__, client->mas, client->slv);
1425 free_cl_mem(client);
1426 goto exit_register;
1427 }
1428
1429 MSM_BUS_DBG("%s:Client handle %p %s", __func__, client,
1430 client->name);
1431 msm_bus_dbg_add_client(client);
1432exit_register:
1433 rt_mutex_unlock(&msm_bus_adhoc_lock);
1434 return client;
1435}
1436/**
1437 * msm_bus_arb_setops_adhoc() : Setup the bus arbitration ops
1438 * @ arb_ops: pointer to the arb ops.
1439 */
1440void msm_bus_arb_setops_adhoc(struct msm_bus_arb_ops *arb_ops)
1441{
1442 arb_ops->register_client = register_client_adhoc;
1443 arb_ops->update_request = update_request_adhoc;
1444 arb_ops->unregister_client = unregister_client_adhoc;
1445 arb_ops->update_context = update_context;
1446
1447 arb_ops->register_cl = register_adhoc;
1448 arb_ops->unregister = unregister_adhoc;
1449 arb_ops->update_bw = update_bw_adhoc;
1450 arb_ops->update_bw_context = update_bw_context;
1451}