/* * Copyright (c) 2012 The Linux Foundation. All rights reserved.* */ /* Copyright (c) 2010-2012, The Linux Foundation. All rights reserved. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 and * only version 2 as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. */ #define pr_fmt(fmt) "AXI: %s(): " fmt, __func__ #include #include #include #include #include #include #include #include #include "msm_bus_core.h" enum { SLAVE_NODE, MASTER_NODE, CLK_NODE, }; enum { DISABLE, ENABLE, }; struct msm_bus_fabric { struct msm_bus_fabric_device fabdev; int ahb; void *cdata[NUM_CTX]; bool arb_dirty; bool clk_dirty; struct radix_tree_root fab_tree; int num_nodes; struct list_head gateways; struct msm_bus_inode_info info; struct msm_bus_fabric_registration *pdata; void *hw_data; }; #define to_msm_bus_fabric(d) container_of(d, \ struct msm_bus_fabric, d) /** * msm_bus_fabric_add_node() - Add a node to the fabric structure * @fabric: Fabric device to which the node should be added * @info: The node to be added */ static int msm_bus_fabric_add_node(struct msm_bus_fabric *fabric, struct msm_bus_inode_info *info) { int status = -ENOMEM, ctx; MSM_BUS_DBG("msm_bus_fabric_add_node: ID %d Gw: %d\n", info->node_info->priv_id, info->node_info->gateway); status = radix_tree_preload(GFP_ATOMIC); if (status) goto out; status = radix_tree_insert(&fabric->fab_tree, info->node_info->priv_id, info); radix_tree_preload_end(); if (IS_SLAVE(info->node_info->priv_id)) radix_tree_tag_set(&fabric->fab_tree, info->node_info->priv_id, SLAVE_NODE); for (ctx = 0; ctx < NUM_CTX; ctx++) { if (info->node_info->slaveclk[ctx]) { radix_tree_tag_set(&fabric->fab_tree, info->node_info->priv_id, CLK_NODE); break; } info->nodeclk[ctx].enable = false; info->nodeclk[ctx].dirty = false; } out: return status; } /** * msm_bus_add_fab() - Add a fabric (gateway) to the current fabric * @fabric: Fabric device to which the gateway info should be added * @info: Gateway node to be added to the fabric */ static int msm_bus_fabric_add_fab(struct msm_bus_fabric *fabric, struct msm_bus_inode_info *info) { struct msm_bus_fabnodeinfo *fabnodeinfo; MSM_BUS_DBG("msm_bus_fabric_add_fab: ID %d Gw: %d\n", info->node_info->priv_id, info->node_info->gateway); fabnodeinfo = kzalloc(sizeof(struct msm_bus_fabnodeinfo), GFP_KERNEL); if (fabnodeinfo == NULL) { MSM_FAB_ERR("msm_bus_fabric_add_fab: " "No Node Info\n"); MSM_FAB_ERR("axi: Cannot register fabric!\n"); return -ENOMEM; } fabnodeinfo->info = info; fabnodeinfo->info->num_pnodes = -1; list_add_tail(&fabnodeinfo->list, &fabric->gateways); return 0; } /** * register_fabric_info() - Create the internal fabric structure and * build the topology tree from platform specific data * @pdev: Platform device for getting base addresses * @fabric: Fabric to which the gateways, nodes should be added * * This function is called from probe. Iterates over the platform data, * and builds the topology */ static int register_fabric_info(struct platform_device *pdev, struct msm_bus_fabric *fabric) { int i = 0, ret = 0, err = 0; MSM_BUS_DBG("id:%d pdata-id: %d len: %d\n", fabric->fabdev.id, fabric->pdata->id, fabric->pdata->len); fabric->hw_data = fabric->fabdev.hw_algo.allocate_hw_data(pdev, fabric->pdata); if (ZERO_OR_NULL_PTR(fabric->hw_data) && fabric->pdata->ahb == 0) { MSM_BUS_ERR("Couldn't allocate hw_data for fab: %d\n", fabric->fabdev.id); goto error; } for (i = 0; i < fabric->pdata->len; i++) { struct msm_bus_inode_info *info; int ctx, j; info = kzalloc(sizeof(struct msm_bus_inode_info), GFP_KERNEL); if (info == NULL) { MSM_BUS_ERR("Error allocating info\n"); return -ENOMEM; } info->node_info = fabric->pdata->info + i; info->commit_index = -1; info->num_pnodes = -1; for (ctx = 0; ctx < NUM_CTX; ctx++) { if (info->node_info->slaveclk[ctx]) { info->nodeclk[ctx].clk = clk_get_sys("msm_bus", info->node_info->slaveclk[ctx]); if (IS_ERR(info->nodeclk[ctx].clk)) { MSM_BUS_ERR("Couldn't get clk %s\n", info->node_info->slaveclk[ctx]); err = -EINVAL; } info->nodeclk[ctx].enable = false; info->nodeclk[ctx].dirty = false; } if (info->node_info->memclk[ctx]) { info->memclk[ctx].clk = clk_get_sys("msm_bus", info->node_info->memclk[ctx]); if (IS_ERR(info->memclk[ctx].clk)) { MSM_BUS_ERR("Couldn't get clk %s\n", info->node_info->memclk[ctx]); err = -EINVAL; } info->memclk[ctx].enable = false; info->memclk[ctx].dirty = false; } } ret = info->node_info->gateway ? msm_bus_fabric_add_fab(fabric, info) : msm_bus_fabric_add_node(fabric, info); if (ret) { MSM_BUS_ERR("Unable to add node info, ret: %d\n", ret); kfree(info); goto error; } if (fabric->fabdev.hw_algo.node_init == NULL) continue; for (j = 0; j < NUM_CTX; j++) clk_prepare_enable(fabric->info.nodeclk[j].clk); fabric->fabdev.hw_algo.node_init(fabric->hw_data, info); if (ret) { MSM_BUS_ERR("Unable to init node info, ret: %d\n", ret); kfree(info); } for (j = 0; j < NUM_CTX; j++) clk_disable_unprepare(fabric->info.nodeclk[j].clk); } MSM_BUS_DBG("Fabric: %d nmasters: %d nslaves: %d\n" " ntieredslaves: %d, rpm_enabled: %d\n", fabric->fabdev.id, fabric->pdata->nmasters, fabric->pdata->nslaves, fabric->pdata->ntieredslaves, fabric->pdata->rpm_enabled); MSM_BUS_DBG("msm_bus_register_fabric_info i: %d\n", i); fabric->num_nodes = fabric->pdata->len; error: fabric->num_nodes = i; msm_bus_dbg_commit_data(fabric->fabdev.name, NULL, 0, 0, 0, MSM_BUS_DBG_REGISTER); return ret | err; } /** * msm_bus_fabric_update_clks() - Set the clocks for fabrics and slaves * @fabric: Fabric for which the clocks need to be updated * @slave: The node for which the clocks need to be updated * @index: The index for which the current clocks are set * @curr_clk_hz:Current clock value * @req_clk_hz: Requested clock value * @bwsum: Bandwidth Sum * @clk_flag: Flag determining whether fabric clock or the slave clock has to * be set. If clk_flag is set, fabric clock is set, else slave clock is set. */ static int msm_bus_fabric_update_clks(struct msm_bus_fabric_device *fabdev, struct msm_bus_inode_info *slave, int index, uint64_t curr_clk_hz, uint64_t req_clk_hz, uint64_t bwsum_hz, int clk_flag, int ctx, unsigned int cl_active_flag) { int i, status = 0; uint64_t max_pclk = 0, rate; uint64_t *pclk = NULL; struct msm_bus_fabric *fabric = to_msm_bus_fabric(fabdev); struct nodeclk *nodeclk; /** * Integration for clock rates is not required if context is not * same as client's active-only flag */ if (ctx != cl_active_flag) goto skip_set_clks; /* Maximum for this gateway */ for (i = 0; i <= slave->num_pnodes; i++) { if (i == index && (req_clk_hz < curr_clk_hz)) continue; slave->pnode[i].sel_clk = &slave->pnode[i].clk[ctx]; max_pclk = max(max_pclk, *slave->pnode[i].sel_clk); } *slave->link_info.sel_clk = max(max_pclk, max(bwsum_hz, req_clk_hz)); /* Is this gateway or slave? */ if (clk_flag && (!fabric->ahb)) { struct msm_bus_fabnodeinfo *fabgw = NULL; struct msm_bus_inode_info *info = NULL; /* Maximum of all gateways set at fabric */ list_for_each_entry(fabgw, &fabric->gateways, list) { info = fabgw->info; if (!info) continue; info->link_info.sel_clk = &info->link_info.clk[ctx]; max_pclk = max(max_pclk, *info->link_info.sel_clk); } MSM_BUS_DBG("max_pclk from gateways: %llu\n", max_pclk); /* Maximum of all slave clocks. */ for (i = 0; i < fabric->pdata->len; i++) { if (fabric->pdata->info[i].gateway || (fabric->pdata->info[i].id < SLAVE_ID_KEY)) continue; info = radix_tree_lookup(&fabric->fab_tree, fabric->pdata->info[i].priv_id); if (!info) continue; info->link_info.sel_clk = &info->link_info.clk[ctx]; max_pclk = max(max_pclk, *info->link_info.sel_clk); } MSM_BUS_DBG("max_pclk from slaves & gws: %llu\n", max_pclk); fabric->info.link_info.sel_clk = &fabric->info.link_info.clk[ctx]; pclk = fabric->info.link_info.sel_clk; } else { slave->link_info.sel_clk = &slave->link_info.clk[ctx]; pclk = slave->link_info.sel_clk; } *pclk = max(max_pclk, max(bwsum_hz, req_clk_hz)); if (!fabric->pdata->rpm_enabled) goto skip_set_clks; if (clk_flag) { nodeclk = &fabric->info.nodeclk[ctx]; if (nodeclk->clk) { MSM_BUS_DBG("clks: id: %d set-clk: %llu bws_hz:%llu\n", fabric->fabdev.id, *pclk, bwsum_hz); if (nodeclk->rate != *pclk) { nodeclk->dirty = true; nodeclk->rate = *pclk; } fabric->clk_dirty = true; } } else { nodeclk = &slave->nodeclk[ctx]; if (nodeclk->clk) { rate = *pclk; MSM_BUS_DBG("clks: id: %d set-clk: %llu bws_hz: %llu\n", slave->node_info->priv_id, rate, bwsum_hz); if (nodeclk->rate != rate) { nodeclk->dirty = true; nodeclk->rate = rate; } } if (!status && slave->memclk[ctx].clk) { rate = *slave->link_info.sel_clk; if (slave->memclk[ctx].rate != rate) { slave->memclk[ctx].rate = rate; slave->memclk[ctx].dirty = true; } slave->memclk[ctx].rate = rate; fabric->clk_dirty = true; } } skip_set_clks: return status; } void msm_bus_fabric_update_bw(struct msm_bus_fabric_device *fabdev, struct msm_bus_inode_info *hop, struct msm_bus_inode_info *info, int64_t add_bw, int *master_tiers, int ctx) { struct msm_bus_fabric *fabric = to_msm_bus_fabric(fabdev); void *sel_cdata; int i; sel_cdata = fabric->cdata[ctx]; /* If it's an ahb fabric, don't calculate arb values */ if (fabric->ahb) { MSM_BUS_DBG("AHB fabric, skipping bw calculation\n"); return; } if (!add_bw) { MSM_BUS_DBG("No bandwidth delta. Skipping commit\n"); return; } for (i = 0; i < NUM_CTX; i++) clk_prepare_enable(fabric->info.nodeclk[i].clk); fabdev->hw_algo.update_bw(hop, info, fabric->pdata, sel_cdata, master_tiers, add_bw); for (i = 0; i < NUM_CTX; i++) clk_disable_unprepare(fabric->info.nodeclk[i].clk); fabric->arb_dirty = true; } static int msm_bus_fabric_clk_set(int enable, struct msm_bus_inode_info *info) { int i, status = 0; long rounded_rate; for (i = 0; i < NUM_CTX; i++) { if (info->nodeclk[i].dirty) { if (info->nodeclk[i].rate != 0) { rounded_rate = clk_round_rate(info-> nodeclk[i].clk, info->nodeclk[i].rate); status = clk_set_rate(info->nodeclk[i].clk, rounded_rate); MSM_BUS_DBG("AXI: node: %d set_rate: %ld\n", info->node_info->id, rounded_rate); } if (enable && !(info->nodeclk[i].enable)) { clk_prepare_enable(info->nodeclk[i].clk); info->nodeclk[i].dirty = false; info->nodeclk[i].enable = true; } else if ((info->nodeclk[i].rate == 0) && (!enable) && (info->nodeclk[i].enable)) { clk_disable_unprepare(info->nodeclk[i].clk); info->nodeclk[i].dirty = false; info->nodeclk[i].enable = false; } } if (info->memclk[i].dirty) { if (info->nodeclk[i].rate != 0) { rounded_rate = clk_round_rate(info-> memclk[i].clk, info->memclk[i].rate); status = clk_set_rate(info->memclk[i].clk, rounded_rate); MSM_BUS_DBG("AXI: node: %d set_rate: %ld\n", info->node_info->id, rounded_rate); } if (enable && !(info->memclk[i].enable)) { clk_prepare_enable(info->memclk[i].clk); info->memclk[i].dirty = false; info->memclk[i].enable = true; } else if (info->memclk[i].rate == 0 && (!enable) && (info->memclk[i].enable)) { clk_disable_unprepare(info->memclk[i].clk); info->memclk[i].dirty = false; info->memclk[i].enable = false; } } } return status; } /** * msm_bus_fabric_clk_commit() - Call clock enable and update clock * values. */ static int msm_bus_fabric_clk_commit(int enable, struct msm_bus_fabric *fabric) { unsigned int i, nfound = 0, status = 0; struct msm_bus_inode_info *info[fabric->pdata->nslaves]; if (fabric->clk_dirty == true) status = msm_bus_fabric_clk_set(enable, &fabric->info); if (status) MSM_BUS_WARN("Error setting clocks on fabric: %d\n", fabric->fabdev.id); nfound = radix_tree_gang_lookup_tag(&fabric->fab_tree, (void **)&info, fabric->fabdev.id, fabric->pdata->nslaves, CLK_NODE); if (nfound == 0) { MSM_BUS_DBG("No clock nodes found for fabric: %d\n", fabric->fabdev.id); goto out; } for (i = 0; i < nfound; i++) { status = msm_bus_fabric_clk_set(enable, info[i]); if (status) MSM_BUS_WARN("Error setting clocks for node: %d\n", info[i]->node_info->id); } out: return status; } /** * msm_bus_fabric_hw_commit() - Commit the arbitration data to Hardware. * @fabric: Fabric for which the data should be committed * */ static int msm_bus_fabric_hw_commit(struct msm_bus_fabric_device *fabdev) { int status = 0; struct msm_bus_fabric *fabric = to_msm_bus_fabric(fabdev); /* * For a non-zero bandwidth request, clocks should be enabled before * sending the arbitration data to RPM, but should be disabled only * after commiting the data. */ status = msm_bus_fabric_clk_commit(ENABLE, fabric); if (status) MSM_BUS_DBG("Error setting clocks on fabric: %d\n", fabric->fabdev.id); if (!fabric->arb_dirty) { MSM_BUS_DBG("Not committing as fabric not arb_dirty\n"); goto skip_arb; } status = fabdev->hw_algo.commit(fabric->pdata, fabric->hw_data, (void **)fabric->cdata); if (status) MSM_BUS_DBG("Error committing arb data for fabric: %d\n", fabric->fabdev.id); fabric->arb_dirty = false; skip_arb: /* * If the bandwidth request is 0 for a fabric, the clocks * should be disabled after arbitration data is committed. */ status = msm_bus_fabric_clk_commit(DISABLE, fabric); if (status) MSM_BUS_WARN("Error disabling clocks on fabric: %d\n", fabric->fabdev.id); fabric->clk_dirty = false; return status; } /** * msm_bus_fabric_port_halt() - Used to halt a master port * @fabric: Fabric on which the current master node is present * @portid: Port id of the master */ int msm_bus_fabric_port_halt(struct msm_bus_fabric_device *fabdev, int iid) { struct msm_bus_inode_info *info = NULL; uint8_t mport; uint32_t haltid = 0; struct msm_bus_fabric *fabric = to_msm_bus_fabric(fabdev); info = fabdev->algo->find_node(fabdev, iid); if (!info) { MSM_BUS_ERR("Error: Info not found for id: %u", iid); return -EINVAL; } haltid = fabric->pdata->haltid; mport = info->node_info->masterp[0]; return fabdev->hw_algo.port_halt(haltid, mport); } /** * msm_bus_fabric_port_unhalt() - Used to unhalt a master port * @fabric: Fabric on which the current master node is present * @portid: Port id of the master */ int msm_bus_fabric_port_unhalt(struct msm_bus_fabric_device *fabdev, int iid) { struct msm_bus_inode_info *info = NULL; uint8_t mport; uint32_t haltid = 0; struct msm_bus_fabric *fabric = to_msm_bus_fabric(fabdev); info = fabdev->algo->find_node(fabdev, iid); if (!info) { MSM_BUS_ERR("Error: Info not found for id: %u", iid); return -EINVAL; } haltid = fabric->pdata->haltid; mport = info->node_info->masterp[0]; return fabdev->hw_algo.port_unhalt(haltid, mport); } /** * msm_bus_fabric_find_gw_node() - This function finds the gateway node * attached on a given fabric * @id: ID of the gateway node * @fabric: Fabric to find the gateway node on * Function returns: Pointer to the gateway node */ static struct msm_bus_inode_info *msm_bus_fabric_find_gw_node(struct msm_bus_fabric_device * fabdev, int id) { struct msm_bus_inode_info *info = NULL; struct msm_bus_fabnodeinfo *fab; struct msm_bus_fabric *fabric; if (!fabdev) { MSM_BUS_ERR("No fabric device found!\n"); return NULL; } fabric = to_msm_bus_fabric(fabdev); if (!fabric || IS_ERR(fabric)) { MSM_BUS_ERR("No fabric type found!\n"); return NULL; } list_for_each_entry(fab, &fabric->gateways, list) { if (fab->info->node_info->priv_id == id) { info = fab->info; break; } } return info; } static struct msm_bus_inode_info *msm_bus_fabric_find_node(struct msm_bus_fabric_device * fabdev, int id) { struct msm_bus_inode_info *info = NULL; struct msm_bus_fabric *fabric = to_msm_bus_fabric(fabdev); info = radix_tree_lookup(&fabric->fab_tree, id); if (!info) MSM_BUS_ERR("Null info found for id %d\n", id); return info; } static struct list_head *msm_bus_fabric_get_gw_list(struct msm_bus_fabric_device *fabdev) { struct msm_bus_fabric *fabric = to_msm_bus_fabric(fabdev); if (!fabric || IS_ERR(fabric)) { MSM_BUS_ERR("No fabric found from fabdev\n"); return NULL; } return &fabric->gateways; } static struct msm_bus_fab_algorithm msm_bus_algo = { .update_clks = msm_bus_fabric_update_clks, .update_bw = msm_bus_fabric_update_bw, .port_halt = msm_bus_fabric_port_halt, .port_unhalt = msm_bus_fabric_port_unhalt, .commit = msm_bus_fabric_hw_commit, .find_node = msm_bus_fabric_find_node, .find_gw_node = msm_bus_fabric_find_gw_node, .get_gw_list = msm_bus_fabric_get_gw_list, }; static int msm_bus_fabric_hw_init(struct msm_bus_fabric_registration *pdata, struct msm_bus_hw_algorithm *hw_algo) { int ret = 0; #ifdef CONFIG_MSM_RPM switch (pdata->hw_sel) { case MSM_BUS_NOC: msm_bus_noc_hw_init(pdata, hw_algo); break; case MSM_BUS_BIMC: msm_bus_bimc_hw_init(pdata, hw_algo); break; default: ret = msm_bus_rpm_hw_init(pdata, hw_algo); if (ret) { MSM_BUS_ERR("RPM initialization failed\n"); ret = -EINVAL; } break; } #endif return ret; } static int msm_bus_fabric_probe(struct platform_device *pdev) { int ctx, ret = 0; struct msm_bus_fabric *fabric; struct msm_bus_fabric_registration *pdata; fabric = kzalloc(sizeof(struct msm_bus_fabric), GFP_KERNEL); if (!fabric) { MSM_BUS_ERR("Fabric alloc failed\n"); return -ENOMEM; } INIT_LIST_HEAD(&fabric->gateways); INIT_RADIX_TREE(&fabric->fab_tree, GFP_ATOMIC); fabric->num_nodes = 0; fabric->fabdev.id = pdev->id; fabric->fabdev.visited = false; fabric->info.node_info = kzalloc(sizeof(struct msm_bus_node_info), GFP_KERNEL); if (ZERO_OR_NULL_PTR(fabric->info.node_info)) { MSM_BUS_ERR("Fabric node info alloc failed\n"); kfree(fabric); return -ENOMEM; } fabric->info.node_info->priv_id = fabric->fabdev.id; fabric->info.node_info->id = fabric->fabdev.id; fabric->info.num_pnodes = -1; fabric->info.link_info.clk[DUAL_CTX] = 0; fabric->info.link_info.bw[DUAL_CTX] = 0; fabric->info.link_info.clk[ACTIVE_CTX] = 0; fabric->info.link_info.bw[ACTIVE_CTX] = 0; fabric->fabdev.id = pdev->id; pdata = (struct msm_bus_fabric_registration *)pdev->dev.platform_data; fabric->fabdev.name = pdata->name; fabric->fabdev.algo = &msm_bus_algo; ret = msm_bus_fabric_hw_init(pdata, &fabric->fabdev.hw_algo); if (ret) { MSM_BUS_ERR("Error initializing hardware for fabric: %d\n", fabric->fabdev.id); goto err; } fabric->ahb = pdata->ahb; fabric->pdata = pdata; fabric->pdata->board_algo->assign_iids(fabric->pdata, fabric->fabdev.id); fabric->fabdev.board_algo = fabric->pdata->board_algo; /* * clk and bw for fabric->info will contain the max bw and clk * it will allow. This info will come from the boards file. */ ret = msm_bus_fabric_device_register(&fabric->fabdev); if (ret) { MSM_BUS_ERR("Error registering fabric %d ret %d\n", fabric->fabdev.id, ret); goto err; } for (ctx = 0; ctx < NUM_CTX; ctx++) { if (pdata->fabclk[ctx]) { fabric->info.nodeclk[ctx].clk = clk_get( &fabric->fabdev.dev, pdata->fabclk[ctx]); if (IS_ERR(fabric->info.nodeclk[ctx].clk)) { MSM_BUS_ERR("Couldn't get clock %s\n", pdata->fabclk[ctx]); ret = -EINVAL; goto err; } fabric->info.nodeclk[ctx].enable = false; fabric->info.nodeclk[ctx].dirty = false; } } /* Find num. of slaves, masters, populate gateways, radix tree */ ret = register_fabric_info(pdev, fabric); if (ret) { MSM_BUS_ERR("Could not register fabric %d info, ret: %d\n", fabric->fabdev.id, ret); goto err; } if (!fabric->ahb) { /* Allocate memory for commit data */ for (ctx = 0; ctx < NUM_CTX; ctx++) { ret = fabric->fabdev.hw_algo.allocate_commit_data( fabric->pdata, &fabric->cdata[ctx], ctx); if (ret) { MSM_BUS_ERR("Failed to alloc commit data for " "fab: %d, ret = %d\n", fabric->fabdev.id, ret); goto err; } } } return ret; err: kfree(fabric->info.node_info); kfree(fabric); return ret; } static int msm_bus_fabric_remove(struct platform_device *pdev) { struct msm_bus_fabric_device *fabdev = NULL; struct msm_bus_fabric *fabric; int i; int ret = 0; fabdev = platform_get_drvdata(pdev); msm_bus_fabric_device_unregister(fabdev); fabric = to_msm_bus_fabric(fabdev); msm_bus_dbg_commit_data(fabric->fabdev.name, NULL, 0, 0, 0, MSM_BUS_DBG_UNREGISTER); for (i = 0; i < fabric->pdata->nmasters; i++) radix_tree_delete(&fabric->fab_tree, fabric->fabdev.id + i); for (i = (fabric->fabdev.id + SLAVE_ID_KEY); i < fabric->pdata->nslaves; i++) radix_tree_delete(&fabric->fab_tree, i); if (!fabric->ahb) { fabdev->hw_algo.free_commit_data(fabric->cdata[DUAL_CTX]); fabdev->hw_algo.free_commit_data(fabric->cdata[ACTIVE_CTX]); } kfree(fabric->info.node_info); kfree(fabric->hw_data); kfree(fabric); return ret; } static struct platform_driver msm_bus_fabric_driver = { .probe = msm_bus_fabric_probe, .remove = msm_bus_fabric_remove, .driver = { .name = "msm_bus_fabric", .owner = THIS_MODULE, }, }; static int __init msm_bus_fabric_init_driver(void) { MSM_BUS_ERR("msm_bus_fabric_init_driver\n"); return platform_driver_register(&msm_bus_fabric_driver); } postcore_initcall(msm_bus_fabric_init_driver);