xref: /linux/arch/mips/sgi-ip27/ip27-xtalk.c (revision c8bfe3fad4f86a029da7157bae9699c816f0c309)
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Copyright (C) 1999, 2000 Ralf Baechle (ralf@gnu.org)
4  * Copyright (C) 1999, 2000 Silcon Graphics, Inc.
5  * Copyright (C) 2004 Christoph Hellwig.
6  *
7  * Generic XTALK initialization code
8  */
9 
10 #include <linux/kernel.h>
11 #include <linux/smp.h>
12 #include <linux/platform_device.h>
13 #include <linux/platform_data/sgi-w1.h>
14 #include <linux/platform_data/xtalk-bridge.h>
15 #include <asm/sn/addrs.h>
16 #include <asm/sn/types.h>
17 #include <asm/sn/klconfig.h>
18 #include <asm/pci/bridge.h>
19 #include <asm/xtalk/xtalk.h>
20 
21 
22 #define XBOW_WIDGET_PART_NUM	0x0
23 #define XXBOW_WIDGET_PART_NUM	0xd000	/* Xbow in Xbridge */
24 #define BASE_XBOW_PORT		8     /* Lowest external port */
25 
26 static void bridge_platform_create(nasid_t nasid, int widget, int masterwid)
27 {
28 	struct xtalk_bridge_platform_data *bd;
29 	struct sgi_w1_platform_data *wd;
30 	struct platform_device *pdev_wd;
31 	struct platform_device *pdev_bd;
32 	struct resource w1_res;
33 	unsigned long offset;
34 
35 	offset = NODE_OFFSET(nasid);
36 
37 	wd = kzalloc(sizeof(*wd), GFP_KERNEL);
38 	if (!wd) {
39 		pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
40 		return;
41 	}
42 
43 	snprintf(wd->dev_id, sizeof(wd->dev_id), "bridge-%012lx",
44 		 offset + (widget << SWIN_SIZE_BITS));
45 
46 	memset(&w1_res, 0, sizeof(w1_res));
47 	w1_res.start = offset + (widget << SWIN_SIZE_BITS) +
48 				offsetof(struct bridge_regs, b_nic);
49 	w1_res.end = w1_res.start + 3;
50 	w1_res.flags = IORESOURCE_MEM;
51 
52 	pdev_wd = platform_device_alloc("sgi_w1", PLATFORM_DEVID_AUTO);
53 	if (!pdev_wd) {
54 		pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
55 		goto err_kfree_wd;
56 	}
57 	if (platform_device_add_resources(pdev_wd, &w1_res, 1)) {
58 		pr_warn("xtalk:n%d/%x bridge failed to add platform resources.\n", nasid, widget);
59 		goto err_put_pdev_wd;
60 	}
61 	if (platform_device_add_data(pdev_wd, wd, sizeof(*wd))) {
62 		pr_warn("xtalk:n%d/%x bridge failed to add platform data.\n", nasid, widget);
63 		goto err_put_pdev_wd;
64 	}
65 	if (platform_device_add(pdev_wd)) {
66 		pr_warn("xtalk:n%d/%x bridge failed to add platform device.\n", nasid, widget);
67 		goto err_put_pdev_wd;
68 	}
69 	/* platform_device_add_data() duplicates the data */
70 	kfree(wd);
71 
72 	bd = kzalloc(sizeof(*bd), GFP_KERNEL);
73 	if (!bd) {
74 		pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
75 		goto err_unregister_pdev_wd;
76 	}
77 	pdev_bd = platform_device_alloc("xtalk-bridge", PLATFORM_DEVID_AUTO);
78 	if (!pdev_bd) {
79 		pr_warn("xtalk:n%d/%x bridge create out of memory\n", nasid, widget);
80 		goto err_kfree_bd;
81 	}
82 
83 
84 	bd->bridge_addr = RAW_NODE_SWIN_BASE(nasid, widget);
85 	bd->intr_addr	= BIT_ULL(47) + 0x01800000 + PI_INT_PEND_MOD;
86 	bd->nasid	= nasid;
87 	bd->masterwid	= masterwid;
88 
89 	bd->mem.name	= "Bridge PCI MEM";
90 	bd->mem.start	= offset + (widget << SWIN_SIZE_BITS) + BRIDGE_DEVIO0;
91 	bd->mem.end	= offset + (widget << SWIN_SIZE_BITS) + SWIN_SIZE - 1;
92 	bd->mem.flags	= IORESOURCE_MEM;
93 	bd->mem_offset	= offset;
94 
95 	bd->io.name	= "Bridge PCI IO";
96 	bd->io.start	= offset + (widget << SWIN_SIZE_BITS) + BRIDGE_DEVIO0;
97 	bd->io.end	= offset + (widget << SWIN_SIZE_BITS) + SWIN_SIZE - 1;
98 	bd->io.flags	= IORESOURCE_IO;
99 	bd->io_offset	= offset;
100 
101 	if (platform_device_add_data(pdev_bd, bd, sizeof(*bd))) {
102 		pr_warn("xtalk:n%d/%x bridge failed to add platform data.\n", nasid, widget);
103 		goto err_put_pdev_bd;
104 	}
105 	if (platform_device_add(pdev_bd)) {
106 		pr_warn("xtalk:n%d/%x bridge failed to add platform device.\n", nasid, widget);
107 		goto err_put_pdev_bd;
108 	}
109 	/* platform_device_add_data() duplicates the data */
110 	kfree(bd);
111 	pr_info("xtalk:n%d/%x bridge widget\n", nasid, widget);
112 	return;
113 
114 err_put_pdev_bd:
115 	platform_device_put(pdev_bd);
116 err_kfree_bd:
117 	kfree(bd);
118 err_unregister_pdev_wd:
119 	platform_device_unregister(pdev_wd);
120 	return;
121 err_put_pdev_wd:
122 	platform_device_put(pdev_wd);
123 err_kfree_wd:
124 	kfree(wd);
125 	return;
126 }
127 
128 static int probe_one_port(nasid_t nasid, int widget, int masterwid)
129 {
130 	widgetreg_t		widget_id;
131 	xwidget_part_num_t	partnum;
132 
133 	widget_id = *(volatile widgetreg_t *)
134 		(RAW_NODE_SWIN_BASE(nasid, widget) + WIDGET_ID);
135 	partnum = XWIDGET_PART_NUM(widget_id);
136 
137 	switch (partnum) {
138 	case BRIDGE_WIDGET_PART_NUM:
139 	case XBRIDGE_WIDGET_PART_NUM:
140 		bridge_platform_create(nasid, widget, masterwid);
141 		break;
142 	default:
143 		pr_info("xtalk:n%d/%d unknown widget (0x%x)\n",
144 			nasid, widget, partnum);
145 		break;
146 	}
147 
148 	return 0;
149 }
150 
151 static int xbow_probe(nasid_t nasid)
152 {
153 	lboard_t *brd;
154 	klxbow_t *xbow_p;
155 	unsigned masterwid, i;
156 
157 	/*
158 	 * found xbow, so may have multiple bridges
159 	 * need to probe xbow
160 	 */
161 	brd = find_lboard((lboard_t *)KL_CONFIG_INFO(nasid), KLTYPE_MIDPLANE8);
162 	if (!brd)
163 		return -ENODEV;
164 
165 	xbow_p = (klxbow_t *)find_component(brd, NULL, KLSTRUCT_XBOW);
166 	if (!xbow_p)
167 		return -ENODEV;
168 
169 	/*
170 	 * Okay, here's a xbow. Let's arbitrate and find
171 	 * out if we should initialize it. Set enabled
172 	 * hub connected at highest or lowest widget as
173 	 * master.
174 	 */
175 #ifdef WIDGET_A
176 	i = HUB_WIDGET_ID_MAX + 1;
177 	do {
178 		i--;
179 	} while ((!XBOW_PORT_TYPE_HUB(xbow_p, i)) ||
180 		 (!XBOW_PORT_IS_ENABLED(xbow_p, i)));
181 #else
182 	i = HUB_WIDGET_ID_MIN - 1;
183 	do {
184 		i++;
185 	} while ((!XBOW_PORT_TYPE_HUB(xbow_p, i)) ||
186 		 (!XBOW_PORT_IS_ENABLED(xbow_p, i)));
187 #endif
188 
189 	masterwid = i;
190 	if (nasid != XBOW_PORT_NASID(xbow_p, i))
191 		return 1;
192 
193 	for (i = HUB_WIDGET_ID_MIN; i <= HUB_WIDGET_ID_MAX; i++) {
194 		if (XBOW_PORT_IS_ENABLED(xbow_p, i) &&
195 		    XBOW_PORT_TYPE_IO(xbow_p, i))
196 			probe_one_port(nasid, i, masterwid);
197 	}
198 
199 	return 0;
200 }
201 
202 static void xtalk_probe_node(nasid_t nasid)
203 {
204 	volatile u64		hubreg;
205 	xwidget_part_num_t	partnum;
206 	widgetreg_t		widget_id;
207 
208 	hubreg = REMOTE_HUB_L(nasid, IIO_LLP_CSR);
209 
210 	/* check whether the link is up */
211 	if (!(hubreg & IIO_LLP_CSR_IS_UP))
212 		return;
213 
214 	widget_id = *(volatile widgetreg_t *)
215 		       (RAW_NODE_SWIN_BASE(nasid, 0x0) + WIDGET_ID);
216 	partnum = XWIDGET_PART_NUM(widget_id);
217 
218 	switch (partnum) {
219 	case BRIDGE_WIDGET_PART_NUM:
220 		bridge_platform_create(nasid, 0x8, 0xa);
221 		break;
222 	case XBOW_WIDGET_PART_NUM:
223 	case XXBOW_WIDGET_PART_NUM:
224 		pr_info("xtalk:n%d/0 xbow widget\n", nasid);
225 		xbow_probe(nasid);
226 		break;
227 	default:
228 		pr_info("xtalk:n%d/0 unknown widget (0x%x)\n", nasid, partnum);
229 		break;
230 	}
231 }
232 
233 static int __init xtalk_init(void)
234 {
235 	nasid_t nasid;
236 
237 	for_each_online_node(nasid)
238 		xtalk_probe_node(nasid);
239 
240 	return 0;
241 }
242 arch_initcall(xtalk_init);
243