blob: e38c53e405cbf59dad9f2f1ad61498fda0d34f05 [file] [log] [blame]
David Dai0d014432016-11-10 12:57:44 -08001/* Copyright (c) 2014-2017, The Linux Foundation. All rights reserved.
David Dai04ce4202016-09-26 16:24:13 -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#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
David Daia55b46e2017-05-08 10:31:31 -070027#define MSM_BUS_MAS_ALC 144
28#define MSM_BUS_RSC_APPS 8000
29#define MSM_BUS_RSC_DISP 8001
30#define BCM_TCS_CMD_ACV_APPS 0x8
David Daibee66232017-03-31 19:05:39 -070031
David Dai04ce4202016-09-26 16:24:13 -070032struct bus_search_type {
33 struct list_head link;
34 struct list_head node_list;
35};
36
37struct handle_type {
38 int num_entries;
39 struct msm_bus_client **cl_list;
40};
41
42static struct handle_type handle_list;
43static LIST_HEAD(input_list);
44static LIST_HEAD(apply_list);
45static LIST_HEAD(commit_list);
David Dai62c04a62017-04-12 19:05:49 -070046static LIST_HEAD(late_init_clist);
David Dai9d66f3f2016-11-23 15:33:06 -080047static LIST_HEAD(query_list);
David Dai04ce4202016-09-26 16:24:13 -070048
49DEFINE_RT_MUTEX(msm_bus_adhoc_lock);
50
51static bool chk_bl_list(struct list_head *black_list, unsigned int id)
52{
53 struct msm_bus_node_device_type *bus_node = NULL;
54
55 list_for_each_entry(bus_node, black_list, link) {
56 if (bus_node->node_info->id == id)
57 return true;
58 }
59 return false;
60}
61
62static void copy_remaining_nodes(struct list_head *edge_list, struct list_head
63 *traverse_list, struct list_head *route_list)
64{
65 struct bus_search_type *search_node;
66
67 if (list_empty(edge_list) && list_empty(traverse_list))
68 return;
69
70 search_node = kzalloc(sizeof(struct bus_search_type), GFP_KERNEL);
71 INIT_LIST_HEAD(&search_node->node_list);
72 list_splice_init(edge_list, traverse_list);
73 list_splice_init(traverse_list, &search_node->node_list);
74 list_add_tail(&search_node->link, route_list);
75}
76
77/*
78 * Duplicate instantiaion from msm_bus_arb.c. Todo there needs to be a
79 * "util" file for these common func/macros.
80 *
81 */
David Dai607fbc42017-03-02 16:22:36 -080082uint64_t msm_bus_div64(uint64_t num, unsigned int base)
David Dai04ce4202016-09-26 16:24:13 -070083{
David Dai607fbc42017-03-02 16:22:36 -080084 uint64_t *n = &num;
David Dai04ce4202016-09-26 16:24:13 -070085
David Dai607fbc42017-03-02 16:22:36 -080086 if ((num > 0) && (num < base))
David Dai04ce4202016-09-26 16:24:13 -070087 return 1;
88
David Dai607fbc42017-03-02 16:22:36 -080089 switch (base) {
David Dai04ce4202016-09-26 16:24:13 -070090 case 0:
91 WARN(1, "AXI: Divide by 0 attempted\n");
David Dai607fbc42017-03-02 16:22:36 -080092 case 1: return num;
93 case 2: return (num >> 1);
94 case 4: return (num >> 2);
95 case 8: return (num >> 3);
96 case 16: return (num >> 4);
97 case 32: return (num >> 5);
David Dai04ce4202016-09-26 16:24:13 -070098 }
99
David Dai607fbc42017-03-02 16:22:36 -0800100 do_div(*n, base);
101 return *n;
David Dai04ce4202016-09-26 16:24:13 -0700102}
103
104int msm_bus_device_match_adhoc(struct device *dev, void *id)
105{
106 int ret = 0;
107 struct msm_bus_node_device_type *bnode = to_msm_bus_node(dev);
108
109 if (bnode)
110 ret = (bnode->node_info->id == *(unsigned int *)id);
111 else
112 ret = 0;
113
114 return ret;
115}
116
117static void bcm_add_bus_req(struct device *dev)
118{
119 struct msm_bus_node_device_type *cur_dev = NULL;
120 struct msm_bus_node_device_type *bcm_dev = NULL;
121 struct link_node *lnode;
122 int lnode_idx = -1;
123 int max_num_lnodes = 0;
124 int i;
125
126 cur_dev = to_msm_bus_node(dev);
127 if (!cur_dev) {
128 MSM_BUS_ERR("%s: Null device ptr", __func__);
129 goto exit_bcm_add_bus_req;
130 }
131
132 if (!cur_dev->node_info->num_bcm_devs)
133 goto exit_bcm_add_bus_req;
134
135 for (i = 0; i < cur_dev->node_info->num_bcm_devs; i++) {
David Daia55b46e2017-05-08 10:31:31 -0700136 if (cur_dev->node_info->bcm_req_idx[i] != -1)
137 continue;
David Dai04ce4202016-09-26 16:24:13 -0700138 bcm_dev = to_msm_bus_node(cur_dev->node_info->bcm_devs[i]);
139 max_num_lnodes = bcm_dev->bcmdev->num_bus_devs;
David Dai04ce4202016-09-26 16:24:13 -0700140 if (!bcm_dev->num_lnodes) {
141 bcm_dev->lnode_list = devm_kzalloc(dev,
142 sizeof(struct link_node) * max_num_lnodes,
143 GFP_KERNEL);
144 if (!bcm_dev->lnode_list)
145 goto exit_bcm_add_bus_req;
146
147 lnode = bcm_dev->lnode_list;
148 bcm_dev->num_lnodes = max_num_lnodes;
149 lnode_idx = 0;
150 } else {
151 int i;
152
153 for (i = 0; i < bcm_dev->num_lnodes; i++) {
154 if (!bcm_dev->lnode_list[i].in_use)
155 break;
156 }
157
158 if (i < bcm_dev->num_lnodes) {
159 lnode = &bcm_dev->lnode_list[i];
160 lnode_idx = i;
161 } else {
162 struct link_node *realloc_list;
163 size_t cur_size = sizeof(struct link_node) *
164 bcm_dev->num_lnodes;
165
166 bcm_dev->num_lnodes += NUM_LNODES;
167 realloc_list = msm_bus_realloc_devmem(
168 dev,
169 bcm_dev->lnode_list,
170 cur_size,
171 sizeof(struct link_node) *
172 bcm_dev->num_lnodes,
173 GFP_KERNEL);
174
175 if (!realloc_list)
176 goto exit_bcm_add_bus_req;
177
178 bcm_dev->lnode_list = realloc_list;
179 lnode = &bcm_dev->lnode_list[i];
180 lnode_idx = i;
181 }
182 }
183
184 lnode->in_use = 1;
185 lnode->bus_dev_id = cur_dev->node_info->id;
David Daia55b46e2017-05-08 10:31:31 -0700186 cur_dev->node_info->bcm_req_idx[i] = lnode_idx;
David Dai04ce4202016-09-26 16:24:13 -0700187 memset(lnode->lnode_ib, 0, sizeof(uint64_t) * NUM_CTX);
188 memset(lnode->lnode_ab, 0, sizeof(uint64_t) * NUM_CTX);
David Dai04ce4202016-09-26 16:24:13 -0700189 }
190
191exit_bcm_add_bus_req:
192 return;
193}
194
195static int gen_lnode(struct device *dev,
196 int next_hop, int prev_idx, const char *cl_name)
197{
198 struct link_node *lnode;
199 struct msm_bus_node_device_type *cur_dev = NULL;
200 int lnode_idx = -1;
201
202 if (!dev)
203 goto exit_gen_lnode;
204
205 cur_dev = to_msm_bus_node(dev);
206 if (!cur_dev) {
207 MSM_BUS_ERR("%s: Null device ptr", __func__);
208 goto exit_gen_lnode;
209 }
210
211 if (!cur_dev->num_lnodes) {
212 cur_dev->lnode_list = devm_kzalloc(dev,
213 sizeof(struct link_node) * NUM_LNODES,
214 GFP_KERNEL);
215 if (!cur_dev->lnode_list)
216 goto exit_gen_lnode;
217
218 lnode = cur_dev->lnode_list;
219 cur_dev->num_lnodes = NUM_LNODES;
220 lnode_idx = 0;
221 } else {
222 int i;
223
224 for (i = 0; i < cur_dev->num_lnodes; i++) {
225 if (!cur_dev->lnode_list[i].in_use)
226 break;
227 }
228
229 if (i < cur_dev->num_lnodes) {
230 lnode = &cur_dev->lnode_list[i];
231 lnode_idx = i;
232 } else {
233 struct link_node *realloc_list;
234 size_t cur_size = sizeof(struct link_node) *
235 cur_dev->num_lnodes;
236
237 cur_dev->num_lnodes += NUM_LNODES;
238 realloc_list = msm_bus_realloc_devmem(
239 dev,
240 cur_dev->lnode_list,
241 cur_size,
242 sizeof(struct link_node) *
243 cur_dev->num_lnodes, GFP_KERNEL);
244
245 if (!realloc_list)
246 goto exit_gen_lnode;
247
248 cur_dev->lnode_list = realloc_list;
249 lnode = &cur_dev->lnode_list[i];
250 lnode_idx = i;
251 }
252 }
253
254 lnode->in_use = 1;
255 lnode->cl_name = cl_name;
256 if (next_hop == cur_dev->node_info->id) {
257 lnode->next = -1;
258 lnode->next_dev = NULL;
259 } else {
260 lnode->next = prev_idx;
261 lnode->next_dev = bus_find_device(&msm_bus_type, NULL,
262 (void *) &next_hop,
263 msm_bus_device_match_adhoc);
264 }
265
266 memset(lnode->lnode_ib, 0, sizeof(uint64_t) * NUM_CTX);
267 memset(lnode->lnode_ab, 0, sizeof(uint64_t) * NUM_CTX);
268
269exit_gen_lnode:
270 return lnode_idx;
271}
272
273static int remove_lnode(struct msm_bus_node_device_type *cur_dev,
274 int lnode_idx)
275{
276 int ret = 0;
277
278 if (!cur_dev) {
279 MSM_BUS_ERR("%s: Null device ptr", __func__);
280 ret = -ENODEV;
281 goto exit_remove_lnode;
282 }
283
284 if (lnode_idx != -1) {
285 if (!cur_dev->num_lnodes ||
286 (lnode_idx > (cur_dev->num_lnodes - 1))) {
287 MSM_BUS_ERR("%s: Invalid Idx %d, num_lnodes %d",
288 __func__, lnode_idx, cur_dev->num_lnodes);
289 ret = -ENODEV;
290 goto exit_remove_lnode;
291 }
292
293 cur_dev->lnode_list[lnode_idx].next = -1;
294 cur_dev->lnode_list[lnode_idx].next_dev = NULL;
295 cur_dev->lnode_list[lnode_idx].in_use = 0;
296 cur_dev->lnode_list[lnode_idx].cl_name = NULL;
297 }
298
299exit_remove_lnode:
300 return ret;
301}
302
303static int prune_path(struct list_head *route_list, int dest, int src,
304 struct list_head *black_list, int found,
305 const char *cl_name)
306{
307 struct bus_search_type *search_node, *temp_search_node;
308 struct msm_bus_node_device_type *bus_node;
309 struct list_head *bl_list;
310 struct list_head *temp_bl_list;
311 int search_dev_id = dest;
312 struct device *dest_dev = bus_find_device(&msm_bus_type, NULL,
313 (void *) &dest,
314 msm_bus_device_match_adhoc);
315 int lnode_hop = -1;
316
317 if (!found)
318 goto reset_links;
319
320 if (!dest_dev) {
321 MSM_BUS_ERR("%s: Can't find dest dev %d", __func__, dest);
322 goto exit_prune_path;
323 }
David Dai04ce4202016-09-26 16:24:13 -0700324
325 lnode_hop = gen_lnode(dest_dev, search_dev_id, lnode_hop, cl_name);
326 bcm_add_bus_req(dest_dev);
327
328 list_for_each_entry_reverse(search_node, route_list, link) {
329 list_for_each_entry(bus_node, &search_node->node_list, link) {
330 unsigned int i;
331
332 for (i = 0; i < bus_node->node_info->num_connections;
333 i++) {
334 if (bus_node->node_info->connections[i] ==
335 search_dev_id) {
336 dest_dev = bus_find_device(
337 &msm_bus_type,
338 NULL,
339 (void *)
340 &bus_node->node_info->
341 id,
342 msm_bus_device_match_adhoc);
343
344 if (!dest_dev) {
345 lnode_hop = -1;
346 goto reset_links;
347 }
348
349 lnode_hop = gen_lnode(dest_dev,
350 search_dev_id,
351 lnode_hop, cl_name);
352 bcm_add_bus_req(dest_dev);
353 search_dev_id =
354 bus_node->node_info->id;
355 break;
356 }
357 }
358 }
359 }
360reset_links:
361 list_for_each_entry_safe(search_node, temp_search_node, route_list,
362 link) {
363 list_for_each_entry(bus_node, &search_node->node_list, link)
364 bus_node->node_info->is_traversed = false;
365
366 list_del(&search_node->link);
367 kfree(search_node);
368 }
369
370 list_for_each_safe(bl_list, temp_bl_list, black_list)
371 list_del(bl_list);
372
373exit_prune_path:
374 return lnode_hop;
375}
376
377static void setup_bl_list(struct msm_bus_node_device_type *node,
378 struct list_head *black_list)
379{
380 unsigned int i;
381
382 for (i = 0; i < node->node_info->num_blist; i++) {
383 struct msm_bus_node_device_type *bdev;
384
385 bdev = to_msm_bus_node(node->node_info->black_connections[i]);
386 list_add_tail(&bdev->link, black_list);
387 }
388}
389
390static int getpath(struct device *src_dev, int dest, const char *cl_name)
391{
392 struct list_head traverse_list;
393 struct list_head edge_list;
394 struct list_head route_list;
395 struct list_head black_list;
396 struct msm_bus_node_device_type *src_node;
397 struct bus_search_type *search_node;
398 int found = 0;
399 int depth_index = 0;
400 int first_hop = -1;
401 int src;
402
403 INIT_LIST_HEAD(&traverse_list);
404 INIT_LIST_HEAD(&edge_list);
405 INIT_LIST_HEAD(&route_list);
406 INIT_LIST_HEAD(&black_list);
407
408 if (!src_dev) {
409 MSM_BUS_ERR("%s: Cannot locate src dev ", __func__);
410 goto exit_getpath;
411 }
412
413 src_node = to_msm_bus_node(src_dev);
414 if (!src_node) {
415 MSM_BUS_ERR("%s:Fatal, Source node not found", __func__);
416 goto exit_getpath;
417 }
418 src = src_node->node_info->id;
419 list_add_tail(&src_node->link, &traverse_list);
420
421 while ((!found && !list_empty(&traverse_list))) {
422 struct msm_bus_node_device_type *bus_node = NULL;
423 /* Locate dest_id in the traverse list */
424 list_for_each_entry(bus_node, &traverse_list, link) {
425 if (bus_node->node_info->id == dest) {
426 found = 1;
427 break;
428 }
429 }
430
431 if (!found) {
432 unsigned int i;
433 /* Setup the new edge list */
434 list_for_each_entry(bus_node, &traverse_list, link) {
435 /* Setup list of black-listed nodes */
436 setup_bl_list(bus_node, &black_list);
437
438 for (i = 0; i < bus_node->node_info->
439 num_connections; i++) {
440 bool skip;
441 struct msm_bus_node_device_type
442 *node_conn;
443 node_conn =
444 to_msm_bus_node(bus_node->node_info->
445 dev_connections[i]);
446 if (node_conn->node_info->
447 is_traversed) {
448 MSM_BUS_ERR("Circ Path %d\n",
449 node_conn->node_info->id);
450 goto reset_traversed;
451 }
452 skip = chk_bl_list(&black_list,
453 bus_node->node_info->
454 connections[i]);
455 if (!skip) {
456 list_add_tail(&node_conn->link,
457 &edge_list);
458 node_conn->node_info->
459 is_traversed = true;
460 }
461 }
462 }
463
464 /* Keep tabs of the previous search list */
465 search_node = kzalloc(sizeof(struct bus_search_type),
466 GFP_KERNEL);
467 INIT_LIST_HEAD(&search_node->node_list);
468 list_splice_init(&traverse_list,
469 &search_node->node_list);
470 /* Add the previous search list to a route list */
471 list_add_tail(&search_node->link, &route_list);
472 /* Advancing the list depth */
473 depth_index++;
474 list_splice_init(&edge_list, &traverse_list);
475 }
476 }
477reset_traversed:
478 copy_remaining_nodes(&edge_list, &traverse_list, &route_list);
479 first_hop = prune_path(&route_list, dest, src, &black_list, found,
480 cl_name);
481
482exit_getpath:
483 return first_hop;
484}
485
David Daia55b46e2017-05-08 10:31:31 -0700486static void bcm_update_acv_req(struct msm_bus_node_device_type *cur_rsc,
487 uint64_t max_ab, uint64_t max_ib,
488 uint64_t *vec_a, uint64_t *vec_b,
489 uint32_t *acv, int ctx)
490{
491 uint32_t acv_bmsk = 0;
492 /*
493 * Base ACV voting on current RSC until mapping is set up in commanddb
494 * that allows us to vote ACV based on master.
495 */
496
497 if (cur_rsc->node_info->id == MSM_BUS_RSC_APPS)
498 acv_bmsk = BCM_TCS_CMD_ACV_APPS;
499
500 if (max_ab == 0 && max_ib == 0)
501 *acv = *acv & ~acv_bmsk;
502 else
503 *acv = *acv | acv_bmsk;
504 *vec_a = 0;
505 *vec_b = *acv;
506}
507
David Dai04ce4202016-09-26 16:24:13 -0700508static void bcm_update_bus_req(struct device *dev, int ctx)
509{
510 struct msm_bus_node_device_type *cur_dev = NULL;
511 struct msm_bus_node_device_type *bcm_dev = NULL;
David Daia55b46e2017-05-08 10:31:31 -0700512 struct msm_bus_node_device_type *cur_rsc = NULL;
513
514 int i, j;
David Dai04ce4202016-09-26 16:24:13 -0700515 uint64_t max_ib = 0;
516 uint64_t max_ab = 0;
517 int lnode_idx = 0;
518
519 cur_dev = to_msm_bus_node(dev);
520 if (!cur_dev) {
521 MSM_BUS_ERR("%s: Null device ptr", __func__);
522 goto exit_bcm_update_bus_req;
523 }
524
525 if (!cur_dev->node_info->num_bcm_devs)
526 goto exit_bcm_update_bus_req;
527
528 for (i = 0; i < cur_dev->node_info->num_bcm_devs; i++) {
529 bcm_dev = to_msm_bus_node(cur_dev->node_info->bcm_devs[i]);
530
531 if (!bcm_dev)
532 goto exit_bcm_update_bus_req;
533
David Daia55b46e2017-05-08 10:31:31 -0700534 lnode_idx = cur_dev->node_info->bcm_req_idx[i];
David Dai04ce4202016-09-26 16:24:13 -0700535 bcm_dev->lnode_list[lnode_idx].lnode_ib[ctx] =
David Dai607fbc42017-03-02 16:22:36 -0800536 msm_bus_div64(cur_dev->node_bw[ctx].max_ib *
537 (uint64_t)bcm_dev->bcmdev->width,
538 cur_dev->node_info->agg_params.buswidth);
David Dai04ce4202016-09-26 16:24:13 -0700539
540 bcm_dev->lnode_list[lnode_idx].lnode_ab[ctx] =
David Dai607fbc42017-03-02 16:22:36 -0800541 msm_bus_div64(cur_dev->node_bw[ctx].sum_ab *
542 (uint64_t)bcm_dev->bcmdev->width,
543 cur_dev->node_info->agg_params.buswidth *
544 cur_dev->node_info->agg_params.num_aggports);
David Dai04ce4202016-09-26 16:24:13 -0700545
David Daia55b46e2017-05-08 10:31:31 -0700546 for (j = 0; j < bcm_dev->num_lnodes; j++) {
David Dai04ce4202016-09-26 16:24:13 -0700547 if (ctx == ACTIVE_CTX) {
548 max_ib = max(max_ib,
David Daia55b46e2017-05-08 10:31:31 -0700549 max(bcm_dev->lnode_list[j].lnode_ib[ACTIVE_CTX],
550 bcm_dev->lnode_list[j].lnode_ib[DUAL_CTX]));
David Dai04ce4202016-09-26 16:24:13 -0700551 max_ab = max(max_ab,
David Daia55b46e2017-05-08 10:31:31 -0700552 bcm_dev->lnode_list[j].lnode_ab[ACTIVE_CTX] +
553 bcm_dev->lnode_list[j].lnode_ab[DUAL_CTX]);
David Dai04ce4202016-09-26 16:24:13 -0700554 } else {
555 max_ib = max(max_ib,
David Daia55b46e2017-05-08 10:31:31 -0700556 bcm_dev->lnode_list[j].lnode_ib[ctx]);
David Dai04ce4202016-09-26 16:24:13 -0700557 max_ab = max(max_ab,
David Daia55b46e2017-05-08 10:31:31 -0700558 bcm_dev->lnode_list[j].lnode_ab[ctx]);
David Dai04ce4202016-09-26 16:24:13 -0700559 }
560 }
David Dai04ce4202016-09-26 16:24:13 -0700561 bcm_dev->node_bw[ctx].max_ab = max_ab;
562 bcm_dev->node_bw[ctx].max_ib = max_ib;
David Daibee66232017-03-31 19:05:39 -0700563
564 max_ab = msm_bus_div64(max_ab, bcm_dev->bcmdev->unit_size);
565 max_ib = msm_bus_div64(max_ib, bcm_dev->bcmdev->unit_size);
566
David Daia55b46e2017-05-08 10:31:31 -0700567 if (bcm_dev->node_info->id == MSM_BUS_BCM_ACV) {
568 cur_rsc = to_msm_bus_node(bcm_dev->node_info->
569 rsc_devs[0]);
570 bcm_update_acv_req(cur_rsc, max_ab, max_ib,
571 &bcm_dev->node_vec[ctx].vec_a,
572 &bcm_dev->node_vec[ctx].vec_b,
573 &cur_rsc->rscdev->acv[ctx], ctx);
574
575 } else {
576 bcm_dev->node_vec[ctx].vec_a = max_ab;
577 bcm_dev->node_vec[ctx].vec_b = max_ib;
578 }
David Dai04ce4202016-09-26 16:24:13 -0700579 }
580exit_bcm_update_bus_req:
581 return;
582}
583
David Dai9d66f3f2016-11-23 15:33:06 -0800584static void bcm_query_bus_req(struct device *dev, int ctx)
585{
586 struct msm_bus_node_device_type *cur_dev = NULL;
587 struct msm_bus_node_device_type *bcm_dev = NULL;
David Daia55b46e2017-05-08 10:31:31 -0700588 struct msm_bus_node_device_type *cur_rsc = NULL;
589 int i, j;
David Dai9d66f3f2016-11-23 15:33:06 -0800590 uint64_t max_query_ib = 0;
591 uint64_t max_query_ab = 0;
592 int lnode_idx = 0;
593
594 cur_dev = to_msm_bus_node(dev);
595 if (!cur_dev) {
596 MSM_BUS_ERR("%s: Null device ptr", __func__);
597 goto exit_bcm_query_bus_req;
598 }
599
600 if (!cur_dev->node_info->num_bcm_devs)
601 goto exit_bcm_query_bus_req;
602
603 for (i = 0; i < cur_dev->node_info->num_bcm_devs; i++) {
604 bcm_dev = to_msm_bus_node(cur_dev->node_info->bcm_devs[i]);
605
606 if (!bcm_dev)
607 goto exit_bcm_query_bus_req;
608
David Daia55b46e2017-05-08 10:31:31 -0700609 lnode_idx = cur_dev->node_info->bcm_req_idx[i];
David Dai9d66f3f2016-11-23 15:33:06 -0800610 bcm_dev->lnode_list[lnode_idx].lnode_query_ib[ctx] =
David Dai607fbc42017-03-02 16:22:36 -0800611 msm_bus_div64(cur_dev->node_bw[ctx].max_query_ib *
612 (uint64_t)bcm_dev->bcmdev->width,
613 cur_dev->node_info->agg_params.buswidth);
David Dai9d66f3f2016-11-23 15:33:06 -0800614
615 bcm_dev->lnode_list[lnode_idx].lnode_query_ab[ctx] =
David Dai607fbc42017-03-02 16:22:36 -0800616 msm_bus_div64(cur_dev->node_bw[ctx].sum_query_ab *
617 (uint64_t)bcm_dev->bcmdev->width,
David Dai489b0d72017-03-14 13:05:01 -0700618 cur_dev->node_info->agg_params.num_aggports *
David Dai607fbc42017-03-02 16:22:36 -0800619 cur_dev->node_info->agg_params.buswidth);
David Dai9d66f3f2016-11-23 15:33:06 -0800620
David Daia55b46e2017-05-08 10:31:31 -0700621 for (j = 0; j < bcm_dev->num_lnodes; j++) {
David Dai9d66f3f2016-11-23 15:33:06 -0800622 if (ctx == ACTIVE_CTX) {
623 max_query_ib = max(max_query_ib,
David Daia55b46e2017-05-08 10:31:31 -0700624 max(bcm_dev->lnode_list[j].
David Dai9d66f3f2016-11-23 15:33:06 -0800625 lnode_query_ib[ACTIVE_CTX],
David Daia55b46e2017-05-08 10:31:31 -0700626 bcm_dev->lnode_list[j].
David Dai9d66f3f2016-11-23 15:33:06 -0800627 lnode_query_ib[DUAL_CTX]));
628
629 max_query_ab = max(max_query_ab,
David Daia55b46e2017-05-08 10:31:31 -0700630 bcm_dev->lnode_list[j].
David Dai9d66f3f2016-11-23 15:33:06 -0800631 lnode_query_ab[ACTIVE_CTX] +
David Daia55b46e2017-05-08 10:31:31 -0700632 bcm_dev->lnode_list[j].
David Dai9d66f3f2016-11-23 15:33:06 -0800633 lnode_query_ab[DUAL_CTX]);
634 } else {
635 max_query_ib = max(max_query_ib,
David Daia55b46e2017-05-08 10:31:31 -0700636 bcm_dev->lnode_list[j].
David Dai9d66f3f2016-11-23 15:33:06 -0800637 lnode_query_ib[ctx]);
638 max_query_ab = max(max_query_ab,
David Daia55b46e2017-05-08 10:31:31 -0700639 bcm_dev->lnode_list[j].
David Dai9d66f3f2016-11-23 15:33:06 -0800640 lnode_query_ab[ctx]);
641 }
642 }
643
David Daibee66232017-03-31 19:05:39 -0700644 max_query_ab = msm_bus_div64(max_query_ab,
645 bcm_dev->bcmdev->unit_size);
646 max_query_ib = msm_bus_div64(max_query_ib,
647 bcm_dev->bcmdev->unit_size);
648
David Daia55b46e2017-05-08 10:31:31 -0700649 if (bcm_dev->node_info->id == MSM_BUS_BCM_ACV) {
650 cur_rsc = to_msm_bus_node(bcm_dev->node_info->
651 rsc_devs[0]);
652 bcm_update_acv_req(cur_rsc, max_query_ab, max_query_ib,
653 &bcm_dev->node_vec[ctx].query_vec_a,
654 &bcm_dev->node_vec[ctx].query_vec_b,
655 &cur_rsc->rscdev->query_acv[ctx], ctx);
656 } else {
657 bcm_dev->node_vec[ctx].query_vec_a = max_query_ab;
658 bcm_dev->node_vec[ctx].query_vec_b = max_query_ib;
659 }
660
David Dai9d66f3f2016-11-23 15:33:06 -0800661 bcm_dev->node_bw[ctx].max_query_ab = max_query_ab;
662 bcm_dev->node_bw[ctx].max_query_ib = max_query_ib;
David Dai9d66f3f2016-11-23 15:33:06 -0800663 }
664exit_bcm_query_bus_req:
665 return;
666}
667
David Daibee66232017-03-31 19:05:39 -0700668static void bcm_update_alc_req(struct msm_bus_node_device_type *dev, int ctx)
669{
670 struct msm_bus_node_device_type *bcm_dev = NULL;
671 int i;
672 uint64_t max_alc = 0;
David Dai9d66f3f2016-11-23 15:33:06 -0800673
David Daibee66232017-03-31 19:05:39 -0700674 if (!dev || !to_msm_bus_node(dev->node_info->bus_device)) {
675 MSM_BUS_ERR("Bus node pointer is Invalid");
676 goto exit_bcm_update_alc_req;
677 }
678
679 for (i = 0; i < dev->num_lnodes; i++)
680 max_alc = max(max_alc, dev->lnode_list[i].alc_idx[ctx]);
681
682 dev->node_bw[ctx].max_alc = max_alc;
683
684 bcm_dev = to_msm_bus_node(dev->node_info->bcm_devs[0]);
685
686 if (ctx == ACTIVE_CTX) {
687 max_alc = max(max_alc,
688 max(dev->node_bw[ACTIVE_CTX].max_alc,
689 dev->node_bw[DUAL_CTX].max_alc));
690 } else {
691 max_alc = dev->node_bw[ctx].max_alc;
692 }
693
694 bcm_dev->node_bw[ctx].max_alc = max_alc;
695 bcm_dev->node_vec[ctx].vec_a = max_alc;
696 bcm_dev->node_vec[ctx].vec_b = 0;
697
698exit_bcm_update_alc_req:
699 return;
700}
David Dai9d66f3f2016-11-23 15:33:06 -0800701
David Dai04ce4202016-09-26 16:24:13 -0700702int bcm_remove_handoff_req(struct device *dev, void *data)
703{
David Dai62c04a62017-04-12 19:05:49 -0700704 struct msm_bus_node_device_type *bus_dev = NULL;
705 struct msm_bus_node_device_type *cur_bcm = NULL;
706 struct msm_bus_node_device_type *cur_rsc = NULL;
David Dai04ce4202016-09-26 16:24:13 -0700707 int ret = 0;
708
David Dai62c04a62017-04-12 19:05:49 -0700709 bus_dev = to_msm_bus_node(dev);
710 if (bus_dev->node_info->is_bcm_dev ||
711 bus_dev->node_info->is_fab_dev ||
712 bus_dev->node_info->is_rsc_dev)
David Dai04ce4202016-09-26 16:24:13 -0700713 goto exit_bcm_remove_handoff_req;
714
David Dai62c04a62017-04-12 19:05:49 -0700715 if (bus_dev->node_info->num_bcm_devs) {
716 cur_bcm = to_msm_bus_node(bus_dev->node_info->bcm_devs[0]);
717 if (cur_bcm->node_info->num_rsc_devs) {
718 cur_rsc =
719 to_msm_bus_node(cur_bcm->node_info->rsc_devs[0]);
720 if (cur_rsc->node_info->id != MSM_BUS_RSC_APPS)
721 goto exit_bcm_remove_handoff_req;
722 }
David Dai04ce4202016-09-26 16:24:13 -0700723 }
724
David Dai62c04a62017-04-12 19:05:49 -0700725 if (!bus_dev->dirty) {
726 list_add_tail(&bus_dev->link, &late_init_clist);
727 bus_dev->dirty = true;
728 }
David Dai04ce4202016-09-26 16:24:13 -0700729
730exit_bcm_remove_handoff_req:
David Dai04ce4202016-09-26 16:24:13 -0700731 return ret;
732}
733
David Dai04ce4202016-09-26 16:24:13 -0700734static void aggregate_bus_req(struct msm_bus_node_device_type *bus_dev,
735 int ctx)
736{
737 int i;
738 uint64_t max_ib = 0;
739 uint64_t sum_ab = 0;
740
741 if (!bus_dev || !to_msm_bus_node(bus_dev->node_info->bus_device)) {
742 MSM_BUS_ERR("Bus node pointer is Invalid");
743 goto exit_agg_bus_req;
744 }
745
746 for (i = 0; i < bus_dev->num_lnodes; i++) {
747 max_ib = max(max_ib, bus_dev->lnode_list[i].lnode_ib[ctx]);
748 sum_ab += bus_dev->lnode_list[i].lnode_ab[ctx];
749 }
750
751 bus_dev->node_bw[ctx].sum_ab = sum_ab;
752 bus_dev->node_bw[ctx].max_ib = max_ib;
753
754exit_agg_bus_req:
755 return;
756}
757
David Dai9d66f3f2016-11-23 15:33:06 -0800758static void aggregate_bus_query_req(struct msm_bus_node_device_type *bus_dev,
759 int ctx)
760{
761 int i;
762 uint64_t max_ib = 0;
763 uint64_t sum_ab = 0;
764
765 if (!bus_dev || !to_msm_bus_node(bus_dev->node_info->bus_device)) {
766 MSM_BUS_ERR("Bus node pointer is Invalid");
767 goto exit_agg_bus_req;
768 }
769
770 for (i = 0; i < bus_dev->num_lnodes; i++) {
771 max_ib = max(max_ib,
772 bus_dev->lnode_list[i].lnode_query_ib[ctx]);
773 sum_ab += bus_dev->lnode_list[i].lnode_query_ab[ctx];
774 }
775
David Dai9d66f3f2016-11-23 15:33:06 -0800776 bus_dev->node_bw[ctx].sum_query_ab = sum_ab;
777 bus_dev->node_bw[ctx].max_query_ib = max_ib;
778
779exit_agg_bus_req:
780 return;
781}
David Dai04ce4202016-09-26 16:24:13 -0700782
783static void del_inp_list(struct list_head *list)
784{
785 struct rule_update_path_info *rule_node;
786 struct rule_update_path_info *rule_node_tmp;
787
788 list_for_each_entry_safe(rule_node, rule_node_tmp, list, link) {
789 list_del(&rule_node->link);
790 rule_node->added = false;
791 }
792}
793
794static void del_op_list(struct list_head *list)
795{
796 struct rule_apply_rcm_info *rule;
797 struct rule_apply_rcm_info *rule_tmp;
798
799 list_for_each_entry_safe(rule, rule_tmp, list, link)
800 list_del(&rule->link);
801}
802
803static int msm_bus_apply_rules(struct list_head *list, bool after_clk_commit)
804{
805 struct rule_apply_rcm_info *rule;
806 struct device *dev = NULL;
807 struct msm_bus_node_device_type *dev_info = NULL;
808 int ret = 0;
809
810 list_for_each_entry(rule, list, link) {
811 if (!rule)
812 continue;
813
814 if (rule && (rule->after_clk_commit != after_clk_commit))
815 continue;
816
817 dev = bus_find_device(&msm_bus_type, NULL,
818 (void *) &rule->id,
819 msm_bus_device_match_adhoc);
820
821 if (!dev) {
822 MSM_BUS_ERR("Can't find dev node for %d", rule->id);
823 continue;
824 }
825 dev_info = to_msm_bus_node(dev);
826
827 ret = msm_bus_enable_limiter(dev_info, rule->throttle,
828 rule->lim_bw);
829 if (ret)
830 MSM_BUS_ERR("Failed to set limiter for %d", rule->id);
831 }
832
833 return ret;
834}
835
836static void commit_data(void)
837{
838 bool rules_registered = msm_rule_are_rules_registered();
839
840 if (rules_registered) {
841 msm_rules_update_path(&input_list, &apply_list);
842 msm_bus_apply_rules(&apply_list, false);
843 }
844
845 msm_bus_commit_data(&commit_list);
846
847 if (rules_registered) {
848 msm_bus_apply_rules(&apply_list, true);
849 del_inp_list(&input_list);
850 del_op_list(&apply_list);
851 }
852 INIT_LIST_HEAD(&input_list);
853 INIT_LIST_HEAD(&apply_list);
854 INIT_LIST_HEAD(&commit_list);
855}
856
David Daibe2827a2017-06-05 16:20:32 -0700857int commit_late_init_data(void)
David Dai62c04a62017-04-12 19:05:49 -0700858{
David Daibe2827a2017-06-05 16:20:32 -0700859 int rc;
David Dai62c04a62017-04-12 19:05:49 -0700860 rt_mutex_lock(&msm_bus_adhoc_lock);
David Daibe2827a2017-06-05 16:20:32 -0700861 rc = bus_for_each_dev(&msm_bus_type, NULL, NULL,
862 bcm_remove_handoff_req);
David Dai62c04a62017-04-12 19:05:49 -0700863
864 msm_bus_commit_data(&late_init_clist);
865 INIT_LIST_HEAD(&late_init_clist);
866
867 rt_mutex_unlock(&msm_bus_adhoc_lock);
David Daibe2827a2017-06-05 16:20:32 -0700868 return rc;
David Dai62c04a62017-04-12 19:05:49 -0700869}
870
871
872
David Dai04ce4202016-09-26 16:24:13 -0700873static void add_node_to_clist(struct msm_bus_node_device_type *node)
874{
875 struct msm_bus_node_device_type *node_parent =
876 to_msm_bus_node(node->node_info->bus_device);
877
878 if (!node->dirty) {
879 list_add_tail(&node->link, &commit_list);
880 node->dirty = true;
881 }
882
883 if (!node_parent->dirty) {
884 list_add_tail(&node_parent->link, &commit_list);
885 node_parent->dirty = true;
886 }
887}
888
David Dai9d66f3f2016-11-23 15:33:06 -0800889static void add_node_to_query_list(struct msm_bus_node_device_type *node)
890{
891 if (!node->query_dirty) {
892 list_add_tail(&node->query_link, &query_list);
893 node->query_dirty = true;
894 }
895}
896
David Dai04ce4202016-09-26 16:24:13 -0700897static int update_path(struct device *src_dev, int dest, uint64_t act_req_ib,
898 uint64_t act_req_bw, uint64_t slp_req_ib,
899 uint64_t slp_req_bw, uint64_t cur_ib, uint64_t cur_bw,
900 int src_idx, int ctx)
901{
902 struct device *next_dev = NULL;
903 struct link_node *lnode = NULL;
904 struct msm_bus_node_device_type *dev_info = NULL;
905 int curr_idx;
906 int ret = 0;
907 struct rule_update_path_info *rule_node;
908 bool rules_registered = msm_rule_are_rules_registered();
909
910 if (IS_ERR_OR_NULL(src_dev)) {
911 MSM_BUS_ERR("%s: No source device", __func__);
912 ret = -ENODEV;
913 goto exit_update_path;
914 }
915
916 next_dev = src_dev;
917
918 if (src_idx < 0) {
919 MSM_BUS_ERR("%s: Invalid lnode idx %d", __func__, src_idx);
920 ret = -ENXIO;
921 goto exit_update_path;
922 }
923 curr_idx = src_idx;
924
925 while (next_dev) {
926 int i;
927
928 dev_info = to_msm_bus_node(next_dev);
929
930 if (curr_idx >= dev_info->num_lnodes) {
931 MSM_BUS_ERR("%s: Invalid lnode Idx %d num lnodes %d",
932 __func__, curr_idx, dev_info->num_lnodes);
933 ret = -ENXIO;
934 goto exit_update_path;
935 }
936
937 lnode = &dev_info->lnode_list[curr_idx];
938 if (!lnode) {
939 MSM_BUS_ERR("%s: Invalid lnode ptr lnode %d",
940 __func__, curr_idx);
941 ret = -ENXIO;
942 goto exit_update_path;
943 }
944 lnode->lnode_ib[ACTIVE_CTX] = act_req_ib;
945 lnode->lnode_ab[ACTIVE_CTX] = act_req_bw;
946 lnode->lnode_ib[DUAL_CTX] = slp_req_ib;
947 lnode->lnode_ab[DUAL_CTX] = slp_req_bw;
948
949 for (i = 0; i < NUM_CTX; i++) {
950 aggregate_bus_req(dev_info, i);
951 bcm_update_bus_req(next_dev, i);
952 }
953
954 add_node_to_clist(dev_info);
955
956 if (rules_registered) {
957 rule_node = &dev_info->node_info->rule;
958 rule_node->id = dev_info->node_info->id;
959 rule_node->ib = dev_info->node_bw[ACTIVE_CTX].max_ib;
960 rule_node->ab = dev_info->node_bw[ACTIVE_CTX].sum_ab;
961 rule_node->clk =
962 dev_info->node_bw[ACTIVE_CTX].cur_clk_hz;
963 if (!rule_node->added) {
964 list_add_tail(&rule_node->link, &input_list);
965 rule_node->added = true;
966 }
967 }
968
969 next_dev = lnode->next_dev;
970 curr_idx = lnode->next;
971 }
972
973exit_update_path:
974 return ret;
975}
976
David Daibee66232017-03-31 19:05:39 -0700977static int update_alc_vote(struct device *alc_dev, uint64_t act_req_fa_lat,
978 uint64_t act_req_idle_time, uint64_t slp_req_fa_lat,
979 uint64_t slp_req_idle_time, uint64_t cur_fa_lat,
980 uint64_t cur_idle_time, int idx, int ctx)
981{
982 struct link_node *lnode = NULL;
983 struct msm_bus_node_device_type *dev_info = NULL;
984 int curr_idx, i;
985 int ret = 0;
986
987 if (IS_ERR_OR_NULL(alc_dev)) {
988 MSM_BUS_ERR("%s: No source device", __func__);
989 ret = -ENODEV;
990 goto exit_update_alc_vote;
991 }
992
993 if (idx < 0) {
994 MSM_BUS_ERR("%s: Invalid lnode idx %d", __func__, idx);
995 ret = -ENXIO;
996 goto exit_update_alc_vote;
997 }
998
999 dev_info = to_msm_bus_node(alc_dev);
1000 curr_idx = idx;
1001
1002 if (curr_idx >= dev_info->num_lnodes) {
1003 MSM_BUS_ERR("%s: Invalid lnode Idx %d num lnodes %d",
1004 __func__, curr_idx, dev_info->num_lnodes);
1005 ret = -ENXIO;
1006 goto exit_update_alc_vote;
1007 }
1008
1009 lnode = &dev_info->lnode_list[curr_idx];
1010 if (!lnode) {
1011 MSM_BUS_ERR("%s: Invalid lnode ptr lnode %d",
1012 __func__, curr_idx);
1013 ret = -ENXIO;
1014 goto exit_update_alc_vote;
1015 }
1016
1017 /*
1018 * Add aggregation and mapping logic once LUT is avail.
1019 * Use default values for time being.
1020 */
1021 lnode->alc_idx[ACTIVE_CTX] = 12;
1022 lnode->alc_idx[DUAL_CTX] = 0;
1023
1024 for (i = 0; i < NUM_CTX; i++)
1025 bcm_update_alc_req(dev_info, i);
1026
1027 add_node_to_clist(dev_info);
1028
1029exit_update_alc_vote:
1030 return ret;
1031}
1032
1033
David Dai9d66f3f2016-11-23 15:33:06 -08001034static int query_path(struct device *src_dev, int dest, uint64_t act_req_ib,
1035 uint64_t act_req_bw, uint64_t slp_req_ib,
1036 uint64_t slp_req_bw, uint64_t cur_ib, uint64_t cur_bw,
1037 int src_idx)
1038{
1039 struct device *next_dev = NULL;
1040 struct link_node *lnode = NULL;
1041 struct msm_bus_node_device_type *dev_info = NULL;
1042 int curr_idx;
1043 int ret = 0;
1044
1045 if (IS_ERR_OR_NULL(src_dev)) {
1046 MSM_BUS_ERR("%s: No source device", __func__);
1047 ret = -ENODEV;
1048 goto exit_query_path;
1049 }
1050
1051 next_dev = src_dev;
1052
1053 if (src_idx < 0) {
1054 MSM_BUS_ERR("%s: Invalid lnode idx %d", __func__, src_idx);
1055 ret = -ENXIO;
1056 goto exit_query_path;
1057 }
1058 curr_idx = src_idx;
1059
1060 while (next_dev) {
1061 int i;
1062
1063 dev_info = to_msm_bus_node(next_dev);
1064
1065 if (curr_idx >= dev_info->num_lnodes) {
1066 MSM_BUS_ERR("%s: Invalid lnode Idx %d num lnodes %d",
1067 __func__, curr_idx, dev_info->num_lnodes);
1068 ret = -ENXIO;
1069 goto exit_query_path;
1070 }
1071
1072 lnode = &dev_info->lnode_list[curr_idx];
1073 if (!lnode) {
1074 MSM_BUS_ERR("%s: Invalid lnode ptr lnode %d",
1075 __func__, curr_idx);
1076 ret = -ENXIO;
1077 goto exit_query_path;
1078 }
1079 lnode->lnode_query_ib[ACTIVE_CTX] = act_req_ib;
1080 lnode->lnode_query_ab[ACTIVE_CTX] = act_req_bw;
1081 lnode->lnode_query_ib[DUAL_CTX] = slp_req_ib;
1082 lnode->lnode_query_ab[DUAL_CTX] = slp_req_bw;
1083
1084 for (i = 0; i < NUM_CTX; i++) {
1085 aggregate_bus_query_req(dev_info, i);
1086 bcm_query_bus_req(next_dev, i);
1087 }
1088
1089 add_node_to_query_list(dev_info);
1090
1091 next_dev = lnode->next_dev;
1092 curr_idx = lnode->next;
1093 }
1094
1095exit_query_path:
1096 return ret;
1097}
1098
David Dai04ce4202016-09-26 16:24:13 -07001099static int remove_path(struct device *src_dev, int dst, uint64_t cur_ib,
1100 uint64_t cur_ab, int src_idx, int active_only)
1101{
1102 struct device *next_dev = NULL;
1103 struct link_node *lnode = NULL;
1104 struct msm_bus_node_device_type *dev_info = NULL;
1105 int ret = 0;
1106 int cur_idx = src_idx;
1107 int next_idx;
1108
1109 /* Update the current path to zero out all request from
1110 * this cient on all paths
1111 */
1112 if (!src_dev) {
1113 MSM_BUS_ERR("%s: Can't find source device", __func__);
1114 ret = -ENODEV;
1115 goto exit_remove_path;
1116 }
1117
1118 ret = update_path(src_dev, dst, 0, 0, 0, 0, cur_ib, cur_ab, src_idx,
1119 active_only);
1120 if (ret) {
1121 MSM_BUS_ERR("%s: Error zeroing out path ctx %d",
1122 __func__, ACTIVE_CTX);
1123 goto exit_remove_path;
1124 }
1125
1126 next_dev = src_dev;
1127
1128 while (next_dev) {
1129 dev_info = to_msm_bus_node(next_dev);
1130 lnode = &dev_info->lnode_list[cur_idx];
1131 next_idx = lnode->next;
1132 next_dev = lnode->next_dev;
1133 remove_lnode(dev_info, cur_idx);
1134 cur_idx = next_idx;
1135 }
1136
1137exit_remove_path:
1138 return ret;
1139}
1140
1141static void getpath_debug(int src, int curr, int active_only)
1142{
1143 struct device *dev_node;
1144 struct device *dev_it;
1145 unsigned int hop = 1;
1146 int idx;
1147 struct msm_bus_node_device_type *devinfo;
1148 int i;
1149
1150 dev_node = bus_find_device(&msm_bus_type, NULL,
1151 (void *) &src,
1152 msm_bus_device_match_adhoc);
1153
1154 if (!dev_node) {
1155 MSM_BUS_ERR("SRC NOT FOUND %d", src);
1156 return;
1157 }
1158
1159 idx = curr;
1160 devinfo = to_msm_bus_node(dev_node);
1161 dev_it = dev_node;
1162
1163 MSM_BUS_ERR("Route list Src %d", src);
1164 while (dev_it) {
1165 struct msm_bus_node_device_type *busdev =
1166 to_msm_bus_node(devinfo->node_info->bus_device);
1167
1168 MSM_BUS_ERR("Hop[%d] at Device %d ctx %d", hop,
1169 devinfo->node_info->id, active_only);
1170
1171 for (i = 0; i < NUM_CTX; i++) {
1172 MSM_BUS_ERR("dev info sel ib %llu",
1173 devinfo->node_bw[i].cur_clk_hz);
1174 MSM_BUS_ERR("dev info sel ab %llu",
1175 devinfo->node_bw[i].sum_ab);
1176 }
1177
1178 dev_it = devinfo->lnode_list[idx].next_dev;
1179 idx = devinfo->lnode_list[idx].next;
1180 if (dev_it)
1181 devinfo = to_msm_bus_node(dev_it);
1182
1183 MSM_BUS_ERR("Bus Device %d", busdev->node_info->id);
1184 MSM_BUS_ERR("Bus Clock %llu", busdev->clk[active_only].rate);
1185
1186 if (idx < 0)
1187 break;
1188 hop++;
1189 }
1190}
1191
1192static void unregister_client_adhoc(uint32_t cl)
1193{
1194 int i;
1195 struct msm_bus_scale_pdata *pdata;
1196 int lnode, src, curr, dest;
1197 uint64_t cur_clk, cur_bw;
1198 struct msm_bus_client *client;
1199 struct device *src_dev;
1200
1201 rt_mutex_lock(&msm_bus_adhoc_lock);
1202 if (!cl) {
1203 MSM_BUS_ERR("%s: Null cl handle passed unregister\n",
1204 __func__);
1205 goto exit_unregister_client;
1206 }
1207 client = handle_list.cl_list[cl];
1208 pdata = client->pdata;
1209 if (!pdata) {
1210 MSM_BUS_ERR("%s: Null pdata passed to unregister\n",
1211 __func__);
1212 goto exit_unregister_client;
1213 }
1214
1215 curr = client->curr;
1216 if (curr >= pdata->num_usecases) {
1217 MSM_BUS_ERR("Invalid index Defaulting curr to 0");
1218 curr = 0;
1219 }
1220
David Dai04ce4202016-09-26 16:24:13 -07001221 for (i = 0; i < pdata->usecase->num_paths; i++) {
1222 src = client->pdata->usecase[curr].vectors[i].src;
1223 dest = client->pdata->usecase[curr].vectors[i].dst;
1224
1225 lnode = client->src_pnode[i];
1226 src_dev = client->src_devs[i];
1227 cur_clk = client->pdata->usecase[curr].vectors[i].ib;
1228 cur_bw = client->pdata->usecase[curr].vectors[i].ab;
1229 remove_path(src_dev, dest, cur_clk, cur_bw, lnode,
1230 pdata->active_only);
1231 }
1232 commit_data();
1233 msm_bus_dbg_client_data(client->pdata, MSM_BUS_DBG_UNREGISTER, cl);
1234 kfree(client->src_pnode);
1235 kfree(client->src_devs);
1236 kfree(client);
1237 handle_list.cl_list[cl] = NULL;
1238exit_unregister_client:
1239 rt_mutex_unlock(&msm_bus_adhoc_lock);
1240}
1241
1242static int alloc_handle_lst(int size)
1243{
1244 int ret = 0;
1245 struct msm_bus_client **t_cl_list;
1246
1247 if (!handle_list.num_entries) {
1248 t_cl_list = kzalloc(sizeof(struct msm_bus_client *)
1249 * NUM_CL_HANDLES, GFP_KERNEL);
1250 if (ZERO_OR_NULL_PTR(t_cl_list)) {
1251 ret = -ENOMEM;
1252 MSM_BUS_ERR("%s: Failed to allocate handles list",
1253 __func__);
1254 goto exit_alloc_handle_lst;
1255 }
1256 handle_list.cl_list = t_cl_list;
1257 handle_list.num_entries += NUM_CL_HANDLES;
1258 } else {
1259 t_cl_list = krealloc(handle_list.cl_list,
1260 sizeof(struct msm_bus_client *) *
1261 (handle_list.num_entries + NUM_CL_HANDLES),
1262 GFP_KERNEL);
1263 if (ZERO_OR_NULL_PTR(t_cl_list)) {
1264 ret = -ENOMEM;
1265 MSM_BUS_ERR("%s: Failed to allocate handles list",
1266 __func__);
1267 goto exit_alloc_handle_lst;
1268 }
1269
1270 handle_list.cl_list = t_cl_list;
1271 memset(&handle_list.cl_list[handle_list.num_entries], 0,
1272 NUM_CL_HANDLES * sizeof(struct msm_bus_client *));
1273 handle_list.num_entries += NUM_CL_HANDLES;
1274 }
1275exit_alloc_handle_lst:
1276 return ret;
1277}
1278
1279static uint32_t gen_handle(struct msm_bus_client *client)
1280{
1281 uint32_t handle = 0;
1282 int i;
1283 int ret = 0;
1284
1285 for (i = 0; i < handle_list.num_entries; i++) {
1286 if (i && !handle_list.cl_list[i]) {
1287 handle = i;
1288 break;
1289 }
1290 }
1291
1292 if (!handle) {
1293 ret = alloc_handle_lst(NUM_CL_HANDLES);
1294
1295 if (ret) {
1296 MSM_BUS_ERR("%s: Failed to allocate handle list",
1297 __func__);
1298 goto exit_gen_handle;
1299 }
1300 handle = i + 1;
1301 }
1302 handle_list.cl_list[handle] = client;
1303exit_gen_handle:
1304 return handle;
1305}
1306
1307static uint32_t register_client_adhoc(struct msm_bus_scale_pdata *pdata)
1308{
1309 int src, dest;
1310 int i;
1311 struct msm_bus_client *client = NULL;
1312 int *lnode;
1313 struct device *dev;
1314 uint32_t handle = 0;
1315
1316 rt_mutex_lock(&msm_bus_adhoc_lock);
1317 client = kzalloc(sizeof(struct msm_bus_client), GFP_KERNEL);
1318 if (!client) {
1319 MSM_BUS_ERR("%s: Error allocating client data", __func__);
1320 goto exit_register_client;
1321 }
1322 client->pdata = pdata;
1323
David Daibee66232017-03-31 19:05:39 -07001324 if (pdata->alc) {
1325 client->curr = -1;
1326 lnode = kzalloc(sizeof(int), GFP_KERNEL);
1327
1328 if (ZERO_OR_NULL_PTR(lnode)) {
1329 MSM_BUS_ERR("%s: Error allocating lnode!", __func__);
1330 goto exit_lnode_malloc_fail;
1331 }
1332 client->src_pnode = lnode;
1333
1334 client->src_devs = kzalloc(sizeof(struct device *),
1335 GFP_KERNEL);
1336 if (IS_ERR_OR_NULL(client->src_devs)) {
1337 MSM_BUS_ERR("%s: Error allocating src_dev!", __func__);
1338 goto exit_src_dev_malloc_fail;
1339 }
1340 src = MSM_BUS_MAS_ALC;
1341 dev = bus_find_device(&msm_bus_type, NULL,
1342 (void *) &src,
1343 msm_bus_device_match_adhoc);
1344 if (IS_ERR_OR_NULL(dev)) {
1345 MSM_BUS_ERR("%s:Failed to find alc device",
1346 __func__);
1347 goto exit_invalid_data;
1348 }
1349 gen_lnode(dev, MSM_BUS_MAS_ALC, 0, pdata->name);
1350 bcm_add_bus_req(dev);
1351
1352 client->src_devs[0] = dev;
1353
1354 handle = gen_handle(client);
1355 goto exit_register_client;
1356 }
1357
David Dai04ce4202016-09-26 16:24:13 -07001358 lnode = kcalloc(pdata->usecase->num_paths, sizeof(int), GFP_KERNEL);
1359 if (ZERO_OR_NULL_PTR(lnode)) {
1360 MSM_BUS_ERR("%s: Error allocating pathnode ptr!", __func__);
1361 goto exit_lnode_malloc_fail;
1362 }
1363 client->src_pnode = lnode;
1364
1365 client->src_devs = kcalloc(pdata->usecase->num_paths,
1366 sizeof(struct device *), GFP_KERNEL);
1367 if (IS_ERR_OR_NULL(client->src_devs)) {
1368 MSM_BUS_ERR("%s: Error allocating pathnode ptr!", __func__);
1369 goto exit_src_dev_malloc_fail;
1370 }
1371 client->curr = -1;
1372
1373 for (i = 0; i < pdata->usecase->num_paths; i++) {
1374 src = pdata->usecase->vectors[i].src;
1375 dest = pdata->usecase->vectors[i].dst;
1376
1377 if ((src < 0) || (dest < 0)) {
1378 MSM_BUS_ERR("%s:Invalid src/dst.src %d dest %d",
1379 __func__, src, dest);
1380 goto exit_invalid_data;
1381 }
1382 dev = bus_find_device(&msm_bus_type, NULL,
1383 (void *) &src,
1384 msm_bus_device_match_adhoc);
1385 if (IS_ERR_OR_NULL(dev)) {
1386 MSM_BUS_ERR("%s:Failed to find path.src %d dest %d",
1387 __func__, src, dest);
1388 goto exit_invalid_data;
1389 }
1390 client->src_devs[i] = dev;
1391
1392 MSM_BUS_ERR("%s:find path.src %d dest %d",
1393 __func__, src, dest);
1394
1395 lnode[i] = getpath(dev, dest, client->pdata->name);
1396 if (lnode[i] < 0) {
1397 MSM_BUS_ERR("%s:Failed to find path.src %d dest %d",
1398 __func__, src, dest);
1399 goto exit_invalid_data;
1400 }
1401 }
1402
1403 handle = gen_handle(client);
1404 msm_bus_dbg_client_data(client->pdata, MSM_BUS_DBG_REGISTER,
1405 handle);
1406 MSM_BUS_ERR("%s:Client handle %d %s", __func__, handle,
1407 client->pdata->name);
1408 rt_mutex_unlock(&msm_bus_adhoc_lock);
1409 return handle;
1410exit_invalid_data:
1411 kfree(client->src_devs);
1412exit_src_dev_malloc_fail:
1413 kfree(lnode);
1414exit_lnode_malloc_fail:
1415 kfree(client);
1416exit_register_client:
1417 rt_mutex_unlock(&msm_bus_adhoc_lock);
1418 return handle;
1419}
1420
1421static int update_client_paths(struct msm_bus_client *client, bool log_trns,
1422 unsigned int idx)
1423{
1424 int lnode, src, dest, cur_idx;
1425 uint64_t req_clk, req_bw, curr_clk, curr_bw, slp_clk, slp_bw;
1426 int i, ret = 0;
1427 struct msm_bus_scale_pdata *pdata;
1428 struct device *src_dev;
1429
1430 if (!client) {
1431 MSM_BUS_ERR("Client handle Null");
1432 ret = -ENXIO;
1433 goto exit_update_client_paths;
1434 }
1435
1436 pdata = client->pdata;
1437 if (!pdata) {
1438 MSM_BUS_ERR("Client pdata Null");
1439 ret = -ENXIO;
1440 goto exit_update_client_paths;
1441 }
1442
1443 cur_idx = client->curr;
1444 client->curr = idx;
1445 for (i = 0; i < pdata->usecase->num_paths; i++) {
1446 src = pdata->usecase[idx].vectors[i].src;
1447 dest = pdata->usecase[idx].vectors[i].dst;
1448
1449 lnode = client->src_pnode[i];
1450 src_dev = client->src_devs[i];
1451 req_clk = client->pdata->usecase[idx].vectors[i].ib;
1452 req_bw = client->pdata->usecase[idx].vectors[i].ab;
1453 if (cur_idx < 0) {
1454 curr_clk = 0;
1455 curr_bw = 0;
1456 } else {
1457 curr_clk =
1458 client->pdata->usecase[cur_idx].vectors[i].ib;
1459 curr_bw = client->pdata->usecase[cur_idx].vectors[i].ab;
1460 MSM_BUS_DBG("%s:ab: %llu ib: %llu\n", __func__,
1461 curr_bw, curr_clk);
1462 }
1463
1464 if (pdata->active_only) {
1465 slp_clk = 0;
1466 slp_bw = 0;
1467 } else {
1468 slp_clk = req_clk;
1469 slp_bw = req_bw;
David Dai0d014432016-11-10 12:57:44 -08001470 req_clk = 0;
1471 req_bw = 0;
David Dai04ce4202016-09-26 16:24:13 -07001472 }
1473
1474 ret = update_path(src_dev, dest, req_clk, req_bw, slp_clk,
1475 slp_bw, curr_clk, curr_bw, lnode, pdata->active_only);
1476
1477 if (ret) {
1478 MSM_BUS_ERR("%s: Update path failed! %d ctx %d\n",
1479 __func__, ret, pdata->active_only);
1480 goto exit_update_client_paths;
1481 }
1482
1483 if (log_trns)
1484 getpath_debug(src, lnode, pdata->active_only);
1485 }
1486 commit_data();
1487exit_update_client_paths:
1488 return ret;
1489}
1490
David Daibee66232017-03-31 19:05:39 -07001491static int update_client_alc(struct msm_bus_client *client, bool log_trns,
1492 unsigned int idx)
1493{
1494 int lnode, cur_idx;
1495 uint64_t req_idle_time, req_fal, dual_idle_time, dual_fal,
1496 cur_idle_time, cur_fal;
1497 int ret = 0;
1498 struct msm_bus_scale_pdata *pdata;
1499 struct device *src_dev;
1500
1501 if (!client) {
1502 MSM_BUS_ERR("Client handle Null");
1503 ret = -ENXIO;
1504 goto exit_update_client_alc;
1505 }
1506
1507 pdata = client->pdata;
1508 if (!pdata) {
1509 MSM_BUS_ERR("Client pdata Null");
1510 ret = -ENXIO;
1511 goto exit_update_client_alc;
1512 }
1513
1514 cur_idx = client->curr;
1515 client->curr = idx;
1516 req_fal = pdata->usecase_lat[idx].fal_ns;
1517 req_idle_time = pdata->usecase_lat[idx].idle_t_ns;
1518 lnode = client->src_pnode[0];
1519 src_dev = client->src_devs[0];
1520
1521 if (pdata->active_only) {
1522 dual_fal = 0;
1523 dual_idle_time = 0;
1524 } else {
1525 dual_fal = req_fal;
1526 dual_idle_time = req_idle_time;
1527 }
1528
1529 ret = update_alc_vote(src_dev, req_fal, req_idle_time, dual_fal,
1530 dual_idle_time, cur_fal, cur_idle_time, lnode,
1531 pdata->active_only);
1532
1533 if (ret) {
1534 MSM_BUS_ERR("%s: Update path failed! %d ctx %d\n",
1535 __func__, ret, pdata->active_only);
1536 goto exit_update_client_alc;
1537 }
1538 commit_data();
1539exit_update_client_alc:
1540 return ret;
1541}
1542
David Dai9d66f3f2016-11-23 15:33:06 -08001543static int query_usecase(struct msm_bus_client *client, bool log_trns,
1544 unsigned int idx,
1545 struct msm_bus_tcs_usecase *tcs_usecase)
1546{
1547 int lnode, src, dest, cur_idx;
David Dai489b0d72017-03-14 13:05:01 -07001548 uint64_t req_clk, req_bw, curr_clk, curr_bw;
David Dai9d66f3f2016-11-23 15:33:06 -08001549 int i, ret = 0;
1550 struct msm_bus_scale_pdata *pdata;
1551 struct device *src_dev;
1552 struct msm_bus_node_device_type *node = NULL;
1553 struct msm_bus_node_device_type *node_tmp = NULL;
1554
1555 if (!client) {
1556 MSM_BUS_ERR("Client handle Null");
1557 ret = -ENXIO;
1558 goto exit_query_usecase;
1559 }
1560
1561 pdata = client->pdata;
1562 if (!pdata) {
1563 MSM_BUS_ERR("Client pdata Null");
1564 ret = -ENXIO;
1565 goto exit_query_usecase;
1566 }
1567
1568 cur_idx = client->curr;
1569 client->curr = idx;
1570 for (i = 0; i < pdata->usecase->num_paths; i++) {
1571 src = pdata->usecase[idx].vectors[i].src;
1572 dest = pdata->usecase[idx].vectors[i].dst;
1573
1574 lnode = client->src_pnode[i];
1575 src_dev = client->src_devs[i];
1576 req_clk = client->pdata->usecase[idx].vectors[i].ib;
1577 req_bw = client->pdata->usecase[idx].vectors[i].ab;
1578 if (cur_idx < 0) {
1579 curr_clk = 0;
1580 curr_bw = 0;
1581 } else {
1582 curr_clk =
1583 client->pdata->usecase[cur_idx].vectors[i].ib;
1584 curr_bw = client->pdata->usecase[cur_idx].vectors[i].ab;
1585 MSM_BUS_DBG("%s:ab: %llu ib: %llu\n", __func__,
1586 curr_bw, curr_clk);
1587 }
1588
David Dai489b0d72017-03-14 13:05:01 -07001589 ret = query_path(src_dev, dest, req_clk, req_bw, 0,
1590 0, curr_clk, curr_bw, lnode);
David Dai9d66f3f2016-11-23 15:33:06 -08001591
1592 if (ret) {
1593 MSM_BUS_ERR("%s: Query path failed! %d ctx %d\n",
1594 __func__, ret, pdata->active_only);
1595 goto exit_query_usecase;
1596 }
1597 }
1598 msm_bus_query_gen(&query_list, tcs_usecase);
1599 INIT_LIST_HEAD(&query_list);
1600
1601 for (i = 0; i < pdata->usecase->num_paths; i++) {
1602 src = pdata->usecase[idx].vectors[i].src;
1603 dest = pdata->usecase[idx].vectors[i].dst;
1604
1605 lnode = client->src_pnode[i];
1606 src_dev = client->src_devs[i];
1607
1608 ret = query_path(src_dev, dest, 0, 0, 0, 0,
1609 curr_clk, curr_bw, lnode);
1610
1611 if (ret) {
1612 MSM_BUS_ERR("%s: Clear query path failed! %d ctx %d\n",
1613 __func__, ret, pdata->active_only);
1614 goto exit_query_usecase;
1615 }
1616 }
1617
1618 list_for_each_entry_safe(node, node_tmp, &query_list, query_link) {
1619 node->query_dirty = false;
1620 list_del_init(&node->query_link);
1621 }
1622
1623 INIT_LIST_HEAD(&query_list);
1624
1625exit_query_usecase:
1626 return ret;
1627}
1628
David Dai04ce4202016-09-26 16:24:13 -07001629static int update_context(uint32_t cl, bool active_only,
1630 unsigned int ctx_idx)
1631{
1632 int ret = 0;
1633 struct msm_bus_scale_pdata *pdata;
1634 struct msm_bus_client *client;
1635
1636 rt_mutex_lock(&msm_bus_adhoc_lock);
1637 if (!cl) {
1638 MSM_BUS_ERR("%s: Invalid client handle %d", __func__, cl);
1639 ret = -ENXIO;
1640 goto exit_update_context;
1641 }
1642
1643 client = handle_list.cl_list[cl];
1644 if (!client) {
1645 ret = -ENXIO;
1646 goto exit_update_context;
1647 }
1648
1649 pdata = client->pdata;
1650 if (!pdata) {
1651 ret = -ENXIO;
1652 goto exit_update_context;
1653 }
1654 if (pdata->active_only == active_only) {
1655 MSM_BUS_ERR("No change in context(%d==%d), skip\n",
1656 pdata->active_only, active_only);
1657 ret = -ENXIO;
1658 goto exit_update_context;
1659 }
1660
1661 if (ctx_idx >= pdata->num_usecases) {
1662 MSM_BUS_ERR("Client %u passed invalid index: %d\n",
1663 cl, ctx_idx);
1664 ret = -ENXIO;
1665 goto exit_update_context;
1666 }
1667
1668 pdata->active_only = active_only;
1669
1670 msm_bus_dbg_client_data(client->pdata, ctx_idx, cl);
1671 ret = update_client_paths(client, false, ctx_idx);
1672 if (ret) {
1673 pr_err("%s: Err updating path\n", __func__);
1674 goto exit_update_context;
1675 }
1676
1677// trace_bus_update_request_end(pdata->name);
1678
1679exit_update_context:
1680 rt_mutex_unlock(&msm_bus_adhoc_lock);
1681 return ret;
1682}
1683
1684static int update_request_adhoc(uint32_t cl, unsigned int index)
1685{
1686 int ret = 0;
1687 struct msm_bus_scale_pdata *pdata;
1688 struct msm_bus_client *client;
1689 const char *test_cl = "Null";
1690 bool log_transaction = false;
1691
1692 rt_mutex_lock(&msm_bus_adhoc_lock);
1693
1694 if (!cl) {
1695 MSM_BUS_ERR("%s: Invalid client handle %d", __func__, cl);
1696 ret = -ENXIO;
1697 goto exit_update_request;
1698 }
1699
1700 client = handle_list.cl_list[cl];
1701 if (!client) {
1702 MSM_BUS_ERR("%s: Invalid client pointer ", __func__);
1703 ret = -ENXIO;
1704 goto exit_update_request;
1705 }
1706
1707 pdata = client->pdata;
1708 if (!pdata) {
1709 MSM_BUS_ERR("%s: Client data Null.[client didn't register]",
1710 __func__);
1711 ret = -ENXIO;
1712 goto exit_update_request;
1713 }
1714
1715 if (index >= pdata->num_usecases) {
1716 MSM_BUS_ERR("Client %u passed invalid index: %d\n",
1717 cl, index);
1718 ret = -ENXIO;
1719 goto exit_update_request;
1720 }
1721
1722 if (client->curr == index) {
1723 MSM_BUS_DBG("%s: Not updating client request idx %d unchanged",
1724 __func__, index);
1725 goto exit_update_request;
1726 }
1727
1728 if (!strcmp(test_cl, pdata->name))
1729 log_transaction = true;
1730
1731 MSM_BUS_DBG("%s: cl: %u index: %d curr: %d num_paths: %d\n", __func__,
1732 cl, index, client->curr, client->pdata->usecase->num_paths);
David Daibee66232017-03-31 19:05:39 -07001733
1734 if (pdata->alc)
1735 ret = update_client_alc(client, log_transaction, index);
1736 else {
1737 msm_bus_dbg_client_data(client->pdata, index, cl);
1738 ret = update_client_paths(client, log_transaction, index);
1739 }
David Dai04ce4202016-09-26 16:24:13 -07001740 if (ret) {
1741 pr_err("%s: Err updating path\n", __func__);
1742 goto exit_update_request;
1743 }
1744
1745// trace_bus_update_request_end(pdata->name);
1746
1747exit_update_request:
1748 rt_mutex_unlock(&msm_bus_adhoc_lock);
1749 return ret;
1750}
1751
David Dai9d66f3f2016-11-23 15:33:06 -08001752static int query_client_usecase(struct msm_bus_tcs_usecase *tcs_usecase,
1753 uint32_t cl, unsigned int index)
1754{
1755 int ret = 0;
1756 struct msm_bus_scale_pdata *pdata;
1757 struct msm_bus_client *client;
1758 const char *test_cl = "Null";
1759 bool log_transaction = false;
1760
1761 rt_mutex_lock(&msm_bus_adhoc_lock);
1762
1763 if (!cl) {
1764 MSM_BUS_ERR("%s: Invalid client handle %d", __func__, cl);
1765 ret = -ENXIO;
1766 goto exit_query_client_usecase;
1767 }
1768
1769 client = handle_list.cl_list[cl];
1770 if (!client) {
1771 MSM_BUS_ERR("%s: Invalid client pointer ", __func__);
1772 ret = -ENXIO;
1773 goto exit_query_client_usecase;
1774 }
1775
1776 pdata = client->pdata;
1777 if (!pdata) {
1778 MSM_BUS_ERR("%s: Client data Null.[client didn't register]",
1779 __func__);
1780 ret = -ENXIO;
1781 goto exit_query_client_usecase;
1782 }
1783
1784 if (index >= pdata->num_usecases) {
1785 MSM_BUS_ERR("Client %u passed invalid index: %d\n",
1786 cl, index);
1787 ret = -ENXIO;
1788 goto exit_query_client_usecase;
1789 }
1790
1791 if (!strcmp(test_cl, pdata->name))
1792 log_transaction = true;
1793
1794 MSM_BUS_DBG("%s: cl: %u index: %d curr: %d num_paths: %d\n", __func__,
1795 cl, index, client->curr, client->pdata->usecase->num_paths);
1796 ret = query_usecase(client, log_transaction, index, tcs_usecase);
1797 if (ret) {
1798 pr_err("%s: Err updating path\n", __func__);
1799 goto exit_query_client_usecase;
1800 }
1801
1802// trace_bus_update_request_end(pdata->name);
1803
1804exit_query_client_usecase:
1805 rt_mutex_unlock(&msm_bus_adhoc_lock);
1806 return ret;
1807}
1808
1809static int query_client_usecase_all(struct msm_bus_tcs_handle *tcs_handle,
1810 uint32_t cl)
1811{
1812 int ret = 0;
1813 struct msm_bus_scale_pdata *pdata;
1814 struct msm_bus_client *client;
1815 const char *test_cl = "Null";
1816 bool log_transaction = false;
1817 int i = 0;
1818
1819 rt_mutex_lock(&msm_bus_adhoc_lock);
1820
1821 if (!cl) {
1822 MSM_BUS_ERR("%s: Invalid client handle %d", __func__, cl);
1823 ret = -ENXIO;
1824 goto exit_query_client_usecase_all;
1825 }
1826
1827 client = handle_list.cl_list[cl];
1828 if (!client) {
1829 MSM_BUS_ERR("%s: Invalid client pointer ", __func__);
1830 ret = -ENXIO;
1831 goto exit_query_client_usecase_all;
1832 }
1833
1834 pdata = client->pdata;
1835 if (!pdata) {
1836 MSM_BUS_ERR("%s: Client data Null.[client didn't register]",
1837 __func__);
1838 ret = -ENXIO;
1839 goto exit_query_client_usecase_all;
1840 }
1841
1842 if (!strcmp(test_cl, pdata->name))
1843 log_transaction = true;
1844
1845 MSM_BUS_ERR("%s: query_start", __func__);
1846 for (i = 0; i < pdata->num_usecases; i++)
1847 query_usecase(client, log_transaction, i,
1848 &tcs_handle->usecases[i]);
1849 tcs_handle->num_usecases = pdata->num_usecases;
1850
1851 if (ret) {
1852 pr_err("%s: Err updating path\n", __func__);
1853 goto exit_query_client_usecase_all;
1854 }
1855
1856// trace_bus_update_request_end(pdata->name);
1857
1858exit_query_client_usecase_all:
1859 rt_mutex_unlock(&msm_bus_adhoc_lock);
1860 return ret;
1861}
1862
David Dai04ce4202016-09-26 16:24:13 -07001863static void free_cl_mem(struct msm_bus_client_handle *cl)
1864{
1865 if (cl) {
1866 kfree(cl->name);
1867 kfree(cl);
1868 cl = NULL;
1869 }
1870}
1871
1872static int update_bw_adhoc(struct msm_bus_client_handle *cl, u64 ab, u64 ib)
1873{
1874 int ret = 0;
1875 char *test_cl = "test-client";
1876 bool log_transaction = false;
1877 u64 slp_ib, slp_ab;
1878
1879 rt_mutex_lock(&msm_bus_adhoc_lock);
1880
1881 if (!cl) {
1882 MSM_BUS_ERR("%s: Invalid client handle %p", __func__, cl);
1883 ret = -ENXIO;
1884 goto exit_update_request;
1885 }
1886
1887 if (!strcmp(test_cl, cl->name))
1888 log_transaction = true;
1889
1890 msm_bus_dbg_rec_transaction(cl, ab, ib);
1891
1892 if ((cl->cur_act_ib == ib) && (cl->cur_act_ab == ab)) {
1893 MSM_BUS_DBG("%s:no change in request", cl->name);
1894 goto exit_update_request;
1895 }
1896
1897 if (cl->active_only) {
1898 slp_ib = 0;
1899 slp_ab = 0;
1900 } else {
1901 slp_ib = ib;
1902 slp_ab = ab;
1903 }
1904
1905 ret = update_path(cl->mas_dev, cl->slv, ib, ab, slp_ib, slp_ab,
1906 cl->cur_act_ib, cl->cur_act_ab, cl->first_hop, cl->active_only);
1907
1908 if (ret) {
1909 MSM_BUS_ERR("%s: Update path failed! %d active_only %d\n",
1910 __func__, ret, cl->active_only);
1911 goto exit_update_request;
1912 }
1913
1914 commit_data();
1915 cl->cur_act_ib = ib;
1916 cl->cur_act_ab = ab;
1917 cl->cur_slp_ib = slp_ib;
1918 cl->cur_slp_ab = slp_ab;
1919
1920 if (log_transaction)
1921 getpath_debug(cl->mas, cl->first_hop, cl->active_only);
1922// trace_bus_update_request_end(cl->name);
1923exit_update_request:
1924 rt_mutex_unlock(&msm_bus_adhoc_lock);
1925
1926 return ret;
1927}
1928
1929static int update_bw_context(struct msm_bus_client_handle *cl, u64 act_ab,
1930 u64 act_ib, u64 slp_ib, u64 slp_ab)
1931{
1932 int ret = 0;
1933
1934 rt_mutex_lock(&msm_bus_adhoc_lock);
1935 if (!cl) {
1936 MSM_BUS_ERR("Invalid client handle %p", cl);
1937 ret = -ENXIO;
1938 goto exit_change_context;
1939 }
1940
1941 if ((cl->cur_act_ib == act_ib) &&
1942 (cl->cur_act_ab == act_ab) &&
1943 (cl->cur_slp_ib == slp_ib) &&
1944 (cl->cur_slp_ab == slp_ab)) {
1945 MSM_BUS_ERR("No change in vote");
1946 goto exit_change_context;
1947 }
1948
1949 if (!slp_ab && !slp_ib)
1950 cl->active_only = true;
1951 msm_bus_dbg_rec_transaction(cl, cl->cur_act_ab, cl->cur_slp_ib);
1952 ret = update_path(cl->mas_dev, cl->slv, act_ib, act_ab, slp_ib, slp_ab,
1953 cl->cur_act_ab, cl->cur_act_ab, cl->first_hop,
1954 cl->active_only);
1955 if (ret) {
1956 MSM_BUS_ERR("%s: Update path failed! %d active_only %d\n",
1957 __func__, ret, cl->active_only);
1958 goto exit_change_context;
1959 }
1960 commit_data();
1961 cl->cur_act_ib = act_ib;
1962 cl->cur_act_ab = act_ab;
1963 cl->cur_slp_ib = slp_ib;
1964 cl->cur_slp_ab = slp_ab;
1965// trace_bus_update_request_end(cl->name);
1966exit_change_context:
1967 rt_mutex_unlock(&msm_bus_adhoc_lock);
1968 return ret;
1969}
1970
1971static void unregister_adhoc(struct msm_bus_client_handle *cl)
1972{
1973 rt_mutex_lock(&msm_bus_adhoc_lock);
1974 if (!cl) {
1975 MSM_BUS_ERR("%s: Null cl handle passed unregister\n",
1976 __func__);
1977 goto exit_unregister_client;
1978 }
1979
1980 MSM_BUS_DBG("%s: Unregistering client %p", __func__, cl);
1981
1982 remove_path(cl->mas_dev, cl->slv, cl->cur_act_ib, cl->cur_act_ab,
1983 cl->first_hop, cl->active_only);
1984 commit_data();
1985 msm_bus_dbg_remove_client(cl);
1986 kfree(cl);
David Dai0d014432016-11-10 12:57:44 -08001987 MSM_BUS_DBG("%s: Unregistered client", __func__);
David Dai04ce4202016-09-26 16:24:13 -07001988exit_unregister_client:
1989 rt_mutex_unlock(&msm_bus_adhoc_lock);
1990}
1991
1992static struct msm_bus_client_handle*
1993register_adhoc(uint32_t mas, uint32_t slv, char *name, bool active_only)
1994{
1995 struct msm_bus_client_handle *client = NULL;
1996 int len = 0;
1997
1998 rt_mutex_lock(&msm_bus_adhoc_lock);
1999
2000 if (!(mas && slv && name)) {
2001 pr_err("%s: Error: src dst name num_paths are required",
2002 __func__);
2003 goto exit_register;
2004 }
2005
2006 client = kzalloc(sizeof(struct msm_bus_client_handle), GFP_KERNEL);
2007 if (!client) {
2008 MSM_BUS_ERR("%s: Error allocating client data", __func__);
2009 goto exit_register;
2010 }
2011
2012 len = strnlen(name, MAX_STR_CL);
2013 client->name = kzalloc((len + 1), GFP_KERNEL);
2014 if (!client->name) {
2015 MSM_BUS_ERR("%s: Error allocating client name buf", __func__);
2016 free_cl_mem(client);
2017 goto exit_register;
2018 }
2019 strlcpy(client->name, name, MAX_STR_CL);
2020 client->active_only = active_only;
2021
2022 client->mas = mas;
2023 client->slv = slv;
2024
2025 client->mas_dev = bus_find_device(&msm_bus_type, NULL,
2026 (void *) &mas,
2027 msm_bus_device_match_adhoc);
2028 if (IS_ERR_OR_NULL(client->mas_dev)) {
2029 MSM_BUS_ERR("%s:Failed to find path.src %d dest %d",
2030 __func__, client->mas, client->slv);
2031 free_cl_mem(client);
2032 goto exit_register;
2033 }
2034
2035 client->first_hop = getpath(client->mas_dev, client->slv, client->name);
2036 if (client->first_hop < 0) {
2037 MSM_BUS_ERR("%s:Failed to find path.src %d dest %d",
2038 __func__, client->mas, client->slv);
2039 free_cl_mem(client);
2040 goto exit_register;
2041 }
2042
2043 MSM_BUS_DBG("%s:Client handle %p %s", __func__, client,
2044 client->name);
2045 msm_bus_dbg_add_client(client);
2046exit_register:
2047 rt_mutex_unlock(&msm_bus_adhoc_lock);
2048 return client;
2049}
2050/**
2051 * msm_bus_arb_setops_adhoc() : Setup the bus arbitration ops
2052 * @ arb_ops: pointer to the arb ops.
2053 */
2054void msm_bus_arb_setops_adhoc(struct msm_bus_arb_ops *arb_ops)
2055{
2056 arb_ops->register_client = register_client_adhoc;
2057 arb_ops->update_request = update_request_adhoc;
2058 arb_ops->unregister_client = unregister_client_adhoc;
2059 arb_ops->update_context = update_context;
2060
2061 arb_ops->register_cl = register_adhoc;
2062 arb_ops->unregister = unregister_adhoc;
2063 arb_ops->update_bw = update_bw_adhoc;
2064 arb_ops->update_bw_context = update_bw_context;
David Dai9d66f3f2016-11-23 15:33:06 -08002065 arb_ops->query_usecase = query_client_usecase;
2066 arb_ops->query_usecase_all = query_client_usecase_all;
David Dai04ce4202016-09-26 16:24:13 -07002067}