162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Copyright (C) 1999, 2000 Ralf Baechle (ralf@gnu.org)
462306a36Sopenharmony_ci * Copyright (C) 1999, 2000 Silcon Graphics, Inc.
562306a36Sopenharmony_ci * Copyright (C) 2004 Christoph Hellwig.
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Generic XTALK initialization code
862306a36Sopenharmony_ci */
962306a36Sopenharmony_ci
1062306a36Sopenharmony_ci#include <linux/kernel.h>
1162306a36Sopenharmony_ci#include <linux/smp.h>
1262306a36Sopenharmony_ci#include <linux/platform_device.h>
1362306a36Sopenharmony_ci#include <linux/platform_data/sgi-w1.h>
1462306a36Sopenharmony_ci#include <linux/platform_data/xtalk-bridge.h>
1562306a36Sopenharmony_ci#include <asm/sn/addrs.h>
1662306a36Sopenharmony_ci#include <asm/sn/types.h>
1762306a36Sopenharmony_ci#include <asm/sn/klconfig.h>
1862306a36Sopenharmony_ci#include <asm/pci/bridge.h>
1962306a36Sopenharmony_ci#include <asm/xtalk/xtalk.h>
2062306a36Sopenharmony_ci
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_ci#define XBOW_WIDGET_PART_NUM	0x0
2362306a36Sopenharmony_ci#define XXBOW_WIDGET_PART_NUM	0xd000	/* Xbow in Xbridge */
2462306a36Sopenharmony_ci#define BASE_XBOW_PORT		8     /* Lowest external port */
2562306a36Sopenharmony_ci
2662306a36Sopenharmony_cistatic void bridge_platform_create(nasid_t nasid, int widget, int masterwid)
2762306a36Sopenharmony_ci{
2862306a36Sopenharmony_ci	struct xtalk_bridge_platform_data *bd;
2962306a36Sopenharmony_ci	struct sgi_w1_platform_data *wd;
3062306a36Sopenharmony_ci	struct platform_device *pdev_wd;
3162306a36Sopenharmony_ci	struct platform_device *pdev_bd;
3262306a36Sopenharmony_ci	struct resource w1_res;
3362306a36Sopenharmony_ci	unsigned long offset;
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_ci	offset = NODE_OFFSET(nasid);
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci	wd = kzalloc(sizeof(*wd), GFP_KERNEL);
3862306a36Sopenharmony_ci	if (!wd) {
3962306a36Sopenharmony_ci		pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
4062306a36Sopenharmony_ci		return;
4162306a36Sopenharmony_ci	}
4262306a36Sopenharmony_ci
4362306a36Sopenharmony_ci	snprintf(wd->dev_id, sizeof(wd->dev_id), "bridge-%012lx",
4462306a36Sopenharmony_ci		 offset + (widget << SWIN_SIZE_BITS));
4562306a36Sopenharmony_ci
4662306a36Sopenharmony_ci	memset(&w1_res, 0, sizeof(w1_res));
4762306a36Sopenharmony_ci	w1_res.start = offset + (widget << SWIN_SIZE_BITS) +
4862306a36Sopenharmony_ci				offsetof(struct bridge_regs, b_nic);
4962306a36Sopenharmony_ci	w1_res.end = w1_res.start + 3;
5062306a36Sopenharmony_ci	w1_res.flags = IORESOURCE_MEM;
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci	pdev_wd = platform_device_alloc("sgi_w1", PLATFORM_DEVID_AUTO);
5362306a36Sopenharmony_ci	if (!pdev_wd) {
5462306a36Sopenharmony_ci		pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
5562306a36Sopenharmony_ci		goto err_kfree_wd;
5662306a36Sopenharmony_ci	}
5762306a36Sopenharmony_ci	if (platform_device_add_resources(pdev_wd, &w1_res, 1)) {
5862306a36Sopenharmony_ci		pr_warn("xtalk:n%d/%x bridge failed to add platform resources.\n", nasid, widget);
5962306a36Sopenharmony_ci		goto err_put_pdev_wd;
6062306a36Sopenharmony_ci	}
6162306a36Sopenharmony_ci	if (platform_device_add_data(pdev_wd, wd, sizeof(*wd))) {
6262306a36Sopenharmony_ci		pr_warn("xtalk:n%d/%x bridge failed to add platform data.\n", nasid, widget);
6362306a36Sopenharmony_ci		goto err_put_pdev_wd;
6462306a36Sopenharmony_ci	}
6562306a36Sopenharmony_ci	if (platform_device_add(pdev_wd)) {
6662306a36Sopenharmony_ci		pr_warn("xtalk:n%d/%x bridge failed to add platform device.\n", nasid, widget);
6762306a36Sopenharmony_ci		goto err_put_pdev_wd;
6862306a36Sopenharmony_ci	}
6962306a36Sopenharmony_ci	/* platform_device_add_data() duplicates the data */
7062306a36Sopenharmony_ci	kfree(wd);
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_ci	bd = kzalloc(sizeof(*bd), GFP_KERNEL);
7362306a36Sopenharmony_ci	if (!bd) {
7462306a36Sopenharmony_ci		pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
7562306a36Sopenharmony_ci		goto err_unregister_pdev_wd;
7662306a36Sopenharmony_ci	}
7762306a36Sopenharmony_ci	pdev_bd = platform_device_alloc("xtalk-bridge", PLATFORM_DEVID_AUTO);
7862306a36Sopenharmony_ci	if (!pdev_bd) {
7962306a36Sopenharmony_ci		pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
8062306a36Sopenharmony_ci		goto err_kfree_bd;
8162306a36Sopenharmony_ci	}
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_ci
8462306a36Sopenharmony_ci	bd->bridge_addr = RAW_NODE_SWIN_BASE(nasid, widget);
8562306a36Sopenharmony_ci	bd->intr_addr	= BIT_ULL(47) + 0x01800000 + PI_INT_PEND_MOD;
8662306a36Sopenharmony_ci	bd->nasid	= nasid;
8762306a36Sopenharmony_ci	bd->masterwid	= masterwid;
8862306a36Sopenharmony_ci
8962306a36Sopenharmony_ci	bd->mem.name	= "Bridge PCI MEM";
9062306a36Sopenharmony_ci	bd->mem.start	= offset + (widget << SWIN_SIZE_BITS) + BRIDGE_DEVIO0;
9162306a36Sopenharmony_ci	bd->mem.end	= offset + (widget << SWIN_SIZE_BITS) + SWIN_SIZE - 1;
9262306a36Sopenharmony_ci	bd->mem.flags	= IORESOURCE_MEM;
9362306a36Sopenharmony_ci	bd->mem_offset	= offset;
9462306a36Sopenharmony_ci
9562306a36Sopenharmony_ci	bd->io.name	= "Bridge PCI IO";
9662306a36Sopenharmony_ci	bd->io.start	= offset + (widget << SWIN_SIZE_BITS) + BRIDGE_DEVIO0;
9762306a36Sopenharmony_ci	bd->io.end	= offset + (widget << SWIN_SIZE_BITS) + SWIN_SIZE - 1;
9862306a36Sopenharmony_ci	bd->io.flags	= IORESOURCE_IO;
9962306a36Sopenharmony_ci	bd->io_offset	= offset;
10062306a36Sopenharmony_ci
10162306a36Sopenharmony_ci	if (platform_device_add_data(pdev_bd, bd, sizeof(*bd))) {
10262306a36Sopenharmony_ci		pr_warn("xtalk:n%d/%x bridge failed to add platform data.\n", nasid, widget);
10362306a36Sopenharmony_ci		goto err_put_pdev_bd;
10462306a36Sopenharmony_ci	}
10562306a36Sopenharmony_ci	if (platform_device_add(pdev_bd)) {
10662306a36Sopenharmony_ci		pr_warn("xtalk:n%d/%x bridge failed to add platform device.\n", nasid, widget);
10762306a36Sopenharmony_ci		goto err_put_pdev_bd;
10862306a36Sopenharmony_ci	}
10962306a36Sopenharmony_ci	/* platform_device_add_data() duplicates the data */
11062306a36Sopenharmony_ci	kfree(bd);
11162306a36Sopenharmony_ci	pr_info("xtalk:n%d/%x bridge widget\n", nasid, widget);
11262306a36Sopenharmony_ci	return;
11362306a36Sopenharmony_ci
11462306a36Sopenharmony_cierr_put_pdev_bd:
11562306a36Sopenharmony_ci	platform_device_put(pdev_bd);
11662306a36Sopenharmony_cierr_kfree_bd:
11762306a36Sopenharmony_ci	kfree(bd);
11862306a36Sopenharmony_cierr_unregister_pdev_wd:
11962306a36Sopenharmony_ci	platform_device_unregister(pdev_wd);
12062306a36Sopenharmony_ci	return;
12162306a36Sopenharmony_cierr_put_pdev_wd:
12262306a36Sopenharmony_ci	platform_device_put(pdev_wd);
12362306a36Sopenharmony_cierr_kfree_wd:
12462306a36Sopenharmony_ci	kfree(wd);
12562306a36Sopenharmony_ci	return;
12662306a36Sopenharmony_ci}
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_cistatic int probe_one_port(nasid_t nasid, int widget, int masterwid)
12962306a36Sopenharmony_ci{
13062306a36Sopenharmony_ci	widgetreg_t		widget_id;
13162306a36Sopenharmony_ci	xwidget_part_num_t	partnum;
13262306a36Sopenharmony_ci
13362306a36Sopenharmony_ci	widget_id = *(volatile widgetreg_t *)
13462306a36Sopenharmony_ci		(RAW_NODE_SWIN_BASE(nasid, widget) + WIDGET_ID);
13562306a36Sopenharmony_ci	partnum = XWIDGET_PART_NUM(widget_id);
13662306a36Sopenharmony_ci
13762306a36Sopenharmony_ci	switch (partnum) {
13862306a36Sopenharmony_ci	case BRIDGE_WIDGET_PART_NUM:
13962306a36Sopenharmony_ci	case XBRIDGE_WIDGET_PART_NUM:
14062306a36Sopenharmony_ci		bridge_platform_create(nasid, widget, masterwid);
14162306a36Sopenharmony_ci		break;
14262306a36Sopenharmony_ci	default:
14362306a36Sopenharmony_ci		pr_info("xtalk:n%d/%d unknown widget (0x%x)\n",
14462306a36Sopenharmony_ci			nasid, widget, partnum);
14562306a36Sopenharmony_ci		break;
14662306a36Sopenharmony_ci	}
14762306a36Sopenharmony_ci
14862306a36Sopenharmony_ci	return 0;
14962306a36Sopenharmony_ci}
15062306a36Sopenharmony_ci
15162306a36Sopenharmony_cistatic int xbow_probe(nasid_t nasid)
15262306a36Sopenharmony_ci{
15362306a36Sopenharmony_ci	lboard_t *brd;
15462306a36Sopenharmony_ci	klxbow_t *xbow_p;
15562306a36Sopenharmony_ci	unsigned masterwid, i;
15662306a36Sopenharmony_ci
15762306a36Sopenharmony_ci	/*
15862306a36Sopenharmony_ci	 * found xbow, so may have multiple bridges
15962306a36Sopenharmony_ci	 * need to probe xbow
16062306a36Sopenharmony_ci	 */
16162306a36Sopenharmony_ci	brd = find_lboard((lboard_t *)KL_CONFIG_INFO(nasid), KLTYPE_MIDPLANE8);
16262306a36Sopenharmony_ci	if (!brd)
16362306a36Sopenharmony_ci		return -ENODEV;
16462306a36Sopenharmony_ci
16562306a36Sopenharmony_ci	xbow_p = (klxbow_t *)find_component(brd, NULL, KLSTRUCT_XBOW);
16662306a36Sopenharmony_ci	if (!xbow_p)
16762306a36Sopenharmony_ci		return -ENODEV;
16862306a36Sopenharmony_ci
16962306a36Sopenharmony_ci	/*
17062306a36Sopenharmony_ci	 * Okay, here's a xbow. Let's arbitrate and find
17162306a36Sopenharmony_ci	 * out if we should initialize it. Set enabled
17262306a36Sopenharmony_ci	 * hub connected at highest or lowest widget as
17362306a36Sopenharmony_ci	 * master.
17462306a36Sopenharmony_ci	 */
17562306a36Sopenharmony_ci#ifdef WIDGET_A
17662306a36Sopenharmony_ci	i = HUB_WIDGET_ID_MAX + 1;
17762306a36Sopenharmony_ci	do {
17862306a36Sopenharmony_ci		i--;
17962306a36Sopenharmony_ci	} while ((!XBOW_PORT_TYPE_HUB(xbow_p, i)) ||
18062306a36Sopenharmony_ci		 (!XBOW_PORT_IS_ENABLED(xbow_p, i)));
18162306a36Sopenharmony_ci#else
18262306a36Sopenharmony_ci	i = HUB_WIDGET_ID_MIN - 1;
18362306a36Sopenharmony_ci	do {
18462306a36Sopenharmony_ci		i++;
18562306a36Sopenharmony_ci	} while ((!XBOW_PORT_TYPE_HUB(xbow_p, i)) ||
18662306a36Sopenharmony_ci		 (!XBOW_PORT_IS_ENABLED(xbow_p, i)));
18762306a36Sopenharmony_ci#endif
18862306a36Sopenharmony_ci
18962306a36Sopenharmony_ci	masterwid = i;
19062306a36Sopenharmony_ci	if (nasid != XBOW_PORT_NASID(xbow_p, i))
19162306a36Sopenharmony_ci		return 1;
19262306a36Sopenharmony_ci
19362306a36Sopenharmony_ci	for (i = HUB_WIDGET_ID_MIN; i <= HUB_WIDGET_ID_MAX; i++) {
19462306a36Sopenharmony_ci		if (XBOW_PORT_IS_ENABLED(xbow_p, i) &&
19562306a36Sopenharmony_ci		    XBOW_PORT_TYPE_IO(xbow_p, i))
19662306a36Sopenharmony_ci			probe_one_port(nasid, i, masterwid);
19762306a36Sopenharmony_ci	}
19862306a36Sopenharmony_ci
19962306a36Sopenharmony_ci	return 0;
20062306a36Sopenharmony_ci}
20162306a36Sopenharmony_ci
20262306a36Sopenharmony_cistatic void xtalk_probe_node(nasid_t nasid)
20362306a36Sopenharmony_ci{
20462306a36Sopenharmony_ci	volatile u64		hubreg;
20562306a36Sopenharmony_ci	xwidget_part_num_t	partnum;
20662306a36Sopenharmony_ci	widgetreg_t		widget_id;
20762306a36Sopenharmony_ci
20862306a36Sopenharmony_ci	hubreg = REMOTE_HUB_L(nasid, IIO_LLP_CSR);
20962306a36Sopenharmony_ci
21062306a36Sopenharmony_ci	/* check whether the link is up */
21162306a36Sopenharmony_ci	if (!(hubreg & IIO_LLP_CSR_IS_UP))
21262306a36Sopenharmony_ci		return;
21362306a36Sopenharmony_ci
21462306a36Sopenharmony_ci	widget_id = *(volatile widgetreg_t *)
21562306a36Sopenharmony_ci		       (RAW_NODE_SWIN_BASE(nasid, 0x0) + WIDGET_ID);
21662306a36Sopenharmony_ci	partnum = XWIDGET_PART_NUM(widget_id);
21762306a36Sopenharmony_ci
21862306a36Sopenharmony_ci	switch (partnum) {
21962306a36Sopenharmony_ci	case BRIDGE_WIDGET_PART_NUM:
22062306a36Sopenharmony_ci		bridge_platform_create(nasid, 0x8, 0xa);
22162306a36Sopenharmony_ci		break;
22262306a36Sopenharmony_ci	case XBOW_WIDGET_PART_NUM:
22362306a36Sopenharmony_ci	case XXBOW_WIDGET_PART_NUM:
22462306a36Sopenharmony_ci		pr_info("xtalk:n%d/0 xbow widget\n", nasid);
22562306a36Sopenharmony_ci		xbow_probe(nasid);
22662306a36Sopenharmony_ci		break;
22762306a36Sopenharmony_ci	default:
22862306a36Sopenharmony_ci		pr_info("xtalk:n%d/0 unknown widget (0x%x)\n", nasid, partnum);
22962306a36Sopenharmony_ci		break;
23062306a36Sopenharmony_ci	}
23162306a36Sopenharmony_ci}
23262306a36Sopenharmony_ci
23362306a36Sopenharmony_cistatic int __init xtalk_init(void)
23462306a36Sopenharmony_ci{
23562306a36Sopenharmony_ci	nasid_t nasid;
23662306a36Sopenharmony_ci
23762306a36Sopenharmony_ci	for_each_online_node(nasid)
23862306a36Sopenharmony_ci		xtalk_probe_node(nasid);
23962306a36Sopenharmony_ci
24062306a36Sopenharmony_ci	return 0;
24162306a36Sopenharmony_ci}
24262306a36Sopenharmony_ciarch_initcall(xtalk_init);
243