mirror of
https://github.com/torvalds/linux.git
synced 2024-11-29 15:41:36 +00:00
11bec9cba4
In error case in bridge_platform_create after calling
platform_device_add()/platform_device_add_data()/
platform_device_add_resources(), release the failed
'pdev' or it will be leak, call platform_device_put()
to fix this problem.
Besides, 'pdev' is divided into 'pdev_wd' and 'pdev_bd',
use platform_device_unregister() to release sgi_w1
resources when xtalk-bridge registration fails.
Fixes: 5dc76a96e9
("MIPS: PCI: use information from 1-wire PROM for IOC3 detection")
Signed-off-by: Lin Yujun <linyujun809@huawei.com>
Signed-off-by: Thomas Bogendoerfer <tsbogend@alpha.franken.de>
243 lines
6.1 KiB
C
243 lines
6.1 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright (C) 1999, 2000 Ralf Baechle (ralf@gnu.org)
|
|
* Copyright (C) 1999, 2000 Silcon Graphics, Inc.
|
|
* Copyright (C) 2004 Christoph Hellwig.
|
|
*
|
|
* Generic XTALK initialization code
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/smp.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/platform_data/sgi-w1.h>
|
|
#include <linux/platform_data/xtalk-bridge.h>
|
|
#include <asm/sn/addrs.h>
|
|
#include <asm/sn/types.h>
|
|
#include <asm/sn/klconfig.h>
|
|
#include <asm/pci/bridge.h>
|
|
#include <asm/xtalk/xtalk.h>
|
|
|
|
|
|
#define XBOW_WIDGET_PART_NUM 0x0
|
|
#define XXBOW_WIDGET_PART_NUM 0xd000 /* Xbow in Xbridge */
|
|
#define BASE_XBOW_PORT 8 /* Lowest external port */
|
|
|
|
static void bridge_platform_create(nasid_t nasid, int widget, int masterwid)
|
|
{
|
|
struct xtalk_bridge_platform_data *bd;
|
|
struct sgi_w1_platform_data *wd;
|
|
struct platform_device *pdev_wd;
|
|
struct platform_device *pdev_bd;
|
|
struct resource w1_res;
|
|
unsigned long offset;
|
|
|
|
offset = NODE_OFFSET(nasid);
|
|
|
|
wd = kzalloc(sizeof(*wd), GFP_KERNEL);
|
|
if (!wd) {
|
|
pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
|
|
return;
|
|
}
|
|
|
|
snprintf(wd->dev_id, sizeof(wd->dev_id), "bridge-%012lx",
|
|
offset + (widget << SWIN_SIZE_BITS));
|
|
|
|
memset(&w1_res, 0, sizeof(w1_res));
|
|
w1_res.start = offset + (widget << SWIN_SIZE_BITS) +
|
|
offsetof(struct bridge_regs, b_nic);
|
|
w1_res.end = w1_res.start + 3;
|
|
w1_res.flags = IORESOURCE_MEM;
|
|
|
|
pdev_wd = platform_device_alloc("sgi_w1", PLATFORM_DEVID_AUTO);
|
|
if (!pdev_wd) {
|
|
pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
|
|
goto err_kfree_wd;
|
|
}
|
|
if (platform_device_add_resources(pdev_wd, &w1_res, 1)) {
|
|
pr_warn("xtalk:n%d/%x bridge failed to add platform resources.\n", nasid, widget);
|
|
goto err_put_pdev_wd;
|
|
}
|
|
if (platform_device_add_data(pdev_wd, wd, sizeof(*wd))) {
|
|
pr_warn("xtalk:n%d/%x bridge failed to add platform data.\n", nasid, widget);
|
|
goto err_put_pdev_wd;
|
|
}
|
|
if (platform_device_add(pdev_wd)) {
|
|
pr_warn("xtalk:n%d/%x bridge failed to add platform device.\n", nasid, widget);
|
|
goto err_put_pdev_wd;
|
|
}
|
|
/* platform_device_add_data() duplicates the data */
|
|
kfree(wd);
|
|
|
|
bd = kzalloc(sizeof(*bd), GFP_KERNEL);
|
|
if (!bd) {
|
|
pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
|
|
goto err_unregister_pdev_wd;
|
|
}
|
|
pdev_bd = platform_device_alloc("xtalk-bridge", PLATFORM_DEVID_AUTO);
|
|
if (!pdev_bd) {
|
|
pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
|
|
goto err_kfree_bd;
|
|
}
|
|
|
|
|
|
bd->bridge_addr = RAW_NODE_SWIN_BASE(nasid, widget);
|
|
bd->intr_addr = BIT_ULL(47) + 0x01800000 + PI_INT_PEND_MOD;
|
|
bd->nasid = nasid;
|
|
bd->masterwid = masterwid;
|
|
|
|
bd->mem.name = "Bridge PCI MEM";
|
|
bd->mem.start = offset + (widget << SWIN_SIZE_BITS) + BRIDGE_DEVIO0;
|
|
bd->mem.end = offset + (widget << SWIN_SIZE_BITS) + SWIN_SIZE - 1;
|
|
bd->mem.flags = IORESOURCE_MEM;
|
|
bd->mem_offset = offset;
|
|
|
|
bd->io.name = "Bridge PCI IO";
|
|
bd->io.start = offset + (widget << SWIN_SIZE_BITS) + BRIDGE_DEVIO0;
|
|
bd->io.end = offset + (widget << SWIN_SIZE_BITS) + SWIN_SIZE - 1;
|
|
bd->io.flags = IORESOURCE_IO;
|
|
bd->io_offset = offset;
|
|
|
|
if (platform_device_add_data(pdev_bd, bd, sizeof(*bd))) {
|
|
pr_warn("xtalk:n%d/%x bridge failed to add platform data.\n", nasid, widget);
|
|
goto err_put_pdev_bd;
|
|
}
|
|
if (platform_device_add(pdev_bd)) {
|
|
pr_warn("xtalk:n%d/%x bridge failed to add platform device.\n", nasid, widget);
|
|
goto err_put_pdev_bd;
|
|
}
|
|
/* platform_device_add_data() duplicates the data */
|
|
kfree(bd);
|
|
pr_info("xtalk:n%d/%x bridge widget\n", nasid, widget);
|
|
return;
|
|
|
|
err_put_pdev_bd:
|
|
platform_device_put(pdev_bd);
|
|
err_kfree_bd:
|
|
kfree(bd);
|
|
err_unregister_pdev_wd:
|
|
platform_device_unregister(pdev_wd);
|
|
return;
|
|
err_put_pdev_wd:
|
|
platform_device_put(pdev_wd);
|
|
err_kfree_wd:
|
|
kfree(wd);
|
|
return;
|
|
}
|
|
|
|
static int probe_one_port(nasid_t nasid, int widget, int masterwid)
|
|
{
|
|
widgetreg_t widget_id;
|
|
xwidget_part_num_t partnum;
|
|
|
|
widget_id = *(volatile widgetreg_t *)
|
|
(RAW_NODE_SWIN_BASE(nasid, widget) + WIDGET_ID);
|
|
partnum = XWIDGET_PART_NUM(widget_id);
|
|
|
|
switch (partnum) {
|
|
case BRIDGE_WIDGET_PART_NUM:
|
|
case XBRIDGE_WIDGET_PART_NUM:
|
|
bridge_platform_create(nasid, widget, masterwid);
|
|
break;
|
|
default:
|
|
pr_info("xtalk:n%d/%d unknown widget (0x%x)\n",
|
|
nasid, widget, partnum);
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int xbow_probe(nasid_t nasid)
|
|
{
|
|
lboard_t *brd;
|
|
klxbow_t *xbow_p;
|
|
unsigned masterwid, i;
|
|
|
|
/*
|
|
* found xbow, so may have multiple bridges
|
|
* need to probe xbow
|
|
*/
|
|
brd = find_lboard((lboard_t *)KL_CONFIG_INFO(nasid), KLTYPE_MIDPLANE8);
|
|
if (!brd)
|
|
return -ENODEV;
|
|
|
|
xbow_p = (klxbow_t *)find_component(brd, NULL, KLSTRUCT_XBOW);
|
|
if (!xbow_p)
|
|
return -ENODEV;
|
|
|
|
/*
|
|
* Okay, here's a xbow. Let's arbitrate and find
|
|
* out if we should initialize it. Set enabled
|
|
* hub connected at highest or lowest widget as
|
|
* master.
|
|
*/
|
|
#ifdef WIDGET_A
|
|
i = HUB_WIDGET_ID_MAX + 1;
|
|
do {
|
|
i--;
|
|
} while ((!XBOW_PORT_TYPE_HUB(xbow_p, i)) ||
|
|
(!XBOW_PORT_IS_ENABLED(xbow_p, i)));
|
|
#else
|
|
i = HUB_WIDGET_ID_MIN - 1;
|
|
do {
|
|
i++;
|
|
} while ((!XBOW_PORT_TYPE_HUB(xbow_p, i)) ||
|
|
(!XBOW_PORT_IS_ENABLED(xbow_p, i)));
|
|
#endif
|
|
|
|
masterwid = i;
|
|
if (nasid != XBOW_PORT_NASID(xbow_p, i))
|
|
return 1;
|
|
|
|
for (i = HUB_WIDGET_ID_MIN; i <= HUB_WIDGET_ID_MAX; i++) {
|
|
if (XBOW_PORT_IS_ENABLED(xbow_p, i) &&
|
|
XBOW_PORT_TYPE_IO(xbow_p, i))
|
|
probe_one_port(nasid, i, masterwid);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void xtalk_probe_node(nasid_t nasid)
|
|
{
|
|
volatile u64 hubreg;
|
|
xwidget_part_num_t partnum;
|
|
widgetreg_t widget_id;
|
|
|
|
hubreg = REMOTE_HUB_L(nasid, IIO_LLP_CSR);
|
|
|
|
/* check whether the link is up */
|
|
if (!(hubreg & IIO_LLP_CSR_IS_UP))
|
|
return;
|
|
|
|
widget_id = *(volatile widgetreg_t *)
|
|
(RAW_NODE_SWIN_BASE(nasid, 0x0) + WIDGET_ID);
|
|
partnum = XWIDGET_PART_NUM(widget_id);
|
|
|
|
switch (partnum) {
|
|
case BRIDGE_WIDGET_PART_NUM:
|
|
bridge_platform_create(nasid, 0x8, 0xa);
|
|
break;
|
|
case XBOW_WIDGET_PART_NUM:
|
|
case XXBOW_WIDGET_PART_NUM:
|
|
pr_info("xtalk:n%d/0 xbow widget\n", nasid);
|
|
xbow_probe(nasid);
|
|
break;
|
|
default:
|
|
pr_info("xtalk:n%d/0 unknown widget (0x%x)\n", nasid, partnum);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static int __init xtalk_init(void)
|
|
{
|
|
nasid_t nasid;
|
|
|
|
for_each_online_node(nasid)
|
|
xtalk_probe_node(nasid);
|
|
|
|
return 0;
|
|
}
|
|
arch_initcall(xtalk_init);
|