| // 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); |