xref: /illumos-gate/usr/src/uts/sun4v/io/px/px_tools_4v.c (revision 1a220b56b93ff1dc80855691548503117af4cc10)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 #pragma ident	"%Z%%M%	%I%	%E% SMI"
27 
28 #include <sys/sysmacros.h>
29 #include <sys/machsystm.h>
30 #include <sys/cpuvar.h>
31 #include <sys/ddi_implfuncs.h>
32 #include <sys/hypervisor_api.h>
33 #include <sys/hsvc.h>
34 #include <px_obj.h>
35 #include <sys/pci_tools.h>
36 #include <px_tools_var.h>
37 #include "px_lib4v.h"
38 #include <px_tools_ext.h>
39 
40 /*
41  * Delay needed to have a safe environment envelop any error which could
42  * surface.  The larger the number of bridges and switches, the larger the
43  * number needed here.
44  *
45  * Note: this is a workaround until a better solution is found.  While this
46  * number is high, given enough bridges and switches in the device path, this
47  * workaround can break.  Also, other PIL 15 interrupts besides the ones we are
48  * enveloping could delay processing of the interrupt we are trying to protect.
49  */
50 int pxtool_cfg_delay_usec = 2500;
51 int pxtool_iomem_delay_usec = 25000;
52 
53 /* Currently there is no way of getting this info from hypervisor. */
54 #define	INTERRUPT_MAPPING_ENTRIES	64
55 
56 /* Number of inos per root complex. */
57 int pxtool_num_inos = INTERRUPT_MAPPING_ENTRIES;
58 
59 /* Verify hypervisor version for DIAG functions ra2pa and hpriv. */
60 #define	PXTOOL_HYP_VER_UNINIT	0
61 #define	PXTOOL_HYP_VER_BAD	1
62 #define	PXTOOL_HYP_VER_OK	2
63 
64 static int pxtool_hyp_version = PXTOOL_HYP_VER_UNINIT;
65 
66 /* Swap endianness. */
67 static uint64_t
68 pxtool_swap_endian(uint64_t data, int size)
69 {
70 	typedef union {
71 		uint64_t data64;
72 		uint8_t data8[8];
73 	} data_split_t;
74 
75 	data_split_t orig_data;
76 	data_split_t returned_data;
77 	int i;
78 
79 	orig_data.data64 = data;
80 	returned_data.data64 = 0;
81 
82 	for (i = 0; i < size; i++) {
83 		returned_data.data8[7 - i] = orig_data.data8[8 - size + i];
84 	}
85 
86 	return (returned_data.data64);
87 }
88 
89 static void
90 pxtool_validate_diag_hyp_svc(dev_info_t *dip, int *diag_svc_status_p)
91 {
92 	uint64_t pxtool_diag_maj_ver;
93 	uint64_t pxtool_diag_min_ver;
94 	int ret;
95 
96 	if (*diag_svc_status_p == PXTOOL_HYP_VER_UNINIT) {
97 
98 		*diag_svc_status_p = PXTOOL_HYP_VER_BAD;
99 
100 		/*
101 		 * Verify that hypervisor DIAG API has been
102 		 * negotiated (by unix).
103 		 */
104 		if ((ret = hsvc_version(HSVC_GROUP_DIAG,
105 		    &pxtool_diag_maj_ver, &pxtool_diag_min_ver)) != 0) {
106 			DBG(DBG_TOOLS, dip,
107 			    "diag hypervisor svc not negotiated: "
108 			    "grp:0x%lx, errno:%d\n", HSVC_GROUP_DIAG, ret);
109 
110 		} else if (pxtool_diag_maj_ver == 1) {
111 			/*
112 			 * Major version 1 is OK.
113 			 *
114 			 * Code maintainers: if the version changes, check for
115 			 * API changes in hv_ra2pa() and hv_hpriv() before
116 			 * accepting the new version.
117 			 */
118 			*diag_svc_status_p = PXTOOL_HYP_VER_OK;
119 
120 		} else {
121 			DBG(DBG_TOOLS, dip,
122 			    "diag hypervisor svc: bad major number: "
123 			    "grp:0x%lx, maj:0x%lx, min:0x%lx\n",
124 			    HSVC_GROUP_DIAG, pxtool_diag_maj_ver,
125 			    pxtool_diag_min_ver);
126 		}
127 	}
128 }
129 
130 static int
131 pxtool_phys_access(px_t *px_p, uintptr_t dev_addr,
132     uint64_t *data_p, boolean_t is_big_endian, boolean_t is_write)
133 {
134 	uint64_t rfunc, pfunc;
135 	uint64_t rdata_addr, pdata_addr;
136 	uint64_t to_addr, from_addr;
137 	uint64_t local_data;
138 	int rval;
139 	dev_info_t *dip = px_p->px_dip;
140 
141 	DBG(DBG_TOOLS, dip,
142 	    "pxtool_phys_access: dev_addr:0x%" PRIx64 "\n", dev_addr);
143 	DBG(DBG_TOOLS, dip, "    data_addr:0x%" PRIx64 ", is_write:%s\n",
144 	    data_p, (is_write ? "yes" : "no"));
145 
146 	if (pxtool_hyp_version != PXTOOL_HYP_VER_OK) {
147 		pxtool_validate_diag_hyp_svc(dip, &pxtool_hyp_version);
148 		if (pxtool_hyp_version != PXTOOL_HYP_VER_OK) {
149 			DBG(DBG_TOOLS, dip, "Couldn't validate diag hyp svc\n");
150 			return (EPERM);
151 		}
152 	}
153 
154 	if ((rfunc = va_to_pa((void *)px_phys_acc_4v))  == (uint64_t)-1) {
155 		DBG(DBG_TOOLS, dip, "Error getting real addr for function\n");
156 		return (EIO);
157 	}
158 
159 	if ((pfunc = hv_ra2pa(rfunc)) == -1) {
160 		DBG(DBG_TOOLS, dip, "Error getting phys addr for function\n");
161 		return (EIO);
162 	}
163 
164 	if ((rdata_addr = va_to_pa((void *)&local_data))  == (uint64_t)-1) {
165 		DBG(DBG_TOOLS, dip, "Error getting real addr for data_p\n");
166 		return (EIO);
167 	}
168 
169 	if ((pdata_addr = hv_ra2pa(rdata_addr)) == -1) {
170 		DBG(DBG_TOOLS, dip, "Error getting phys addr for data ptr\n");
171 		return (EIO);
172 	}
173 
174 	if (is_write) {
175 		to_addr = dev_addr;
176 		from_addr = pdata_addr;
177 
178 		if (is_big_endian)
179 			local_data = *data_p;
180 		else
181 			local_data =
182 			    pxtool_swap_endian(*data_p, sizeof (uint64_t));
183 	} else {
184 		to_addr = pdata_addr;
185 		from_addr = dev_addr;
186 	}
187 
188 	rval = hv_hpriv((void *)pfunc, from_addr, to_addr, NULL);
189 	switch (rval) {
190 	case H_ENOACCESS:	/* Returned by non-debug hypervisor. */
191 		rval = ENOTSUP;
192 		break;
193 	case H_EOK:
194 		rval = SUCCESS;
195 		break;
196 	default:
197 		rval = EIO;
198 		break;
199 	}
200 
201 	if ((rval == SUCCESS) && (!is_write)) {
202 		if (is_big_endian)
203 			*data_p = local_data;
204 		else
205 			*data_p =
206 			    pxtool_swap_endian(local_data, sizeof (uint64_t));
207 	}
208 
209 	return (rval);
210 }
211 
212 /*
213  * This function is for PCI config space access.
214  * It assumes that offset, bdf, acc_attr are valid in prg_p.
215  * This function modifies prg_p status and data.
216  *
217  * prg_p->phys_addr isn't used.
218  */
219 
220 /*ARGSUSED*/
221 int
222 pxtool_pcicfg_access(px_t *px_p, pcitool_reg_t *prg_p,
223     uint64_t *data_p, boolean_t is_write)
224 {
225 	pci_cfg_data_t data;
226 	on_trap_data_t otd;
227 	dev_info_t *dip = px_p->px_dip;
228 	px_pec_t *pec_p = px_p->px_pec_p;
229 	pci_device_t bdf = PX_GET_BDF(prg_p);
230 	size_t size = PCITOOL_ACC_ATTR_SIZE(prg_p->acc_attr);
231 	int rval = 0;
232 
233 	/* Alignment checking. */
234 	if (!IS_P2ALIGNED(prg_p->offset, size)) {
235 		DBG(DBG_TOOLS, dip, "not aligned.\n");
236 		prg_p->status = PCITOOL_NOT_ALIGNED;
237 		return (EINVAL);
238 	}
239 
240 	mutex_enter(&pec_p->pec_pokefault_mutex);
241 	pec_p->pec_ontrap_data = &otd;
242 
243 	if (is_write) {
244 
245 		if (PCITOOL_ACC_IS_BIG_ENDIAN(prg_p->acc_attr))
246 			data.qw = pxtool_swap_endian(*data_p, size);
247 		else
248 			data.qw = *data_p;
249 
250 		switch (size) {
251 			case sizeof (uint8_t):
252 				data.b = (uint8_t)data.qw;
253 				break;
254 			case sizeof (uint16_t):
255 				data.w = (uint16_t)data.qw;
256 				break;
257 			case sizeof (uint32_t):
258 				data.dw = (uint32_t)data.qw;
259 				break;
260 			case sizeof (uint64_t):
261 				break;
262 		}
263 
264 		DBG(DBG_TOOLS, dip, "put: bdf:0x%x, off:0x%" PRIx64 ", size:"
265 		    "0x%" PRIx64 ", data:0x%" PRIx64 "\n",
266 		    bdf, prg_p->offset, size, data.qw);
267 
268 		pec_p->pec_safeacc_type = DDI_FM_ERR_POKE;
269 
270 		if (!on_trap(&otd, OT_DATA_ACCESS)) {
271 			otd.ot_trampoline = (uintptr_t)&poke_fault;
272 			rval = hvio_config_put(px_p->px_dev_hdl, bdf,
273 			    prg_p->offset, size, data);
274 		} else
275 			rval = H_EIO;
276 
277 		if (otd.ot_trap & OT_DATA_ACCESS)
278 			rval = H_EIO;
279 
280 	} else {
281 
282 		data.qw = 0;
283 
284 		pec_p->pec_safeacc_type = DDI_FM_ERR_PEEK;
285 
286 		if (!on_trap(&otd, OT_DATA_ACCESS)) {
287 			otd.ot_trampoline = (uintptr_t)&peek_fault;
288 			rval = hvio_config_get(px_p->px_dev_hdl, bdf,
289 			    prg_p->offset, size, &data);
290 		} else
291 			rval = H_EIO;
292 
293 		DBG(DBG_TOOLS, dip, "get: bdf:0x%x, off:0x%" PRIx64 ", size:"
294 		    "0x%" PRIx64 ", data:0x%" PRIx64 "\n",
295 		    bdf, prg_p->offset, size, data.qw);
296 
297 		switch (size) {
298 			case sizeof (uint8_t):
299 				*data_p = data.b;
300 				break;
301 			case sizeof (uint16_t):
302 				*data_p = data.w;
303 				break;
304 			case sizeof (uint32_t):
305 				*data_p = data.dw;
306 				break;
307 			case sizeof (uint64_t):
308 				*data_p = data.qw;
309 				break;
310 			default:
311 				DBG(DBG_TOOLS, dip,
312 				    "bad size:0x%" PRIx64 "\n", size);
313 				break;
314 		}
315 		if (PCITOOL_ACC_IS_BIG_ENDIAN(prg_p->acc_attr))
316 			*data_p = pxtool_swap_endian(*data_p, size);
317 	}
318 
319 	/*
320 	 * Workaround: delay taking down safe access env.
321 	 * For more info, see comments where pxtool_cfg_delay_usec is declared.
322 	 */
323 	if (pxtool_cfg_delay_usec > 0)
324 		drv_usecwait(pxtool_cfg_delay_usec);
325 
326 	no_trap();
327 	pec_p->pec_ontrap_data = NULL;
328 	pec_p->pec_safeacc_type = DDI_FM_ERR_UNEXPECTED;
329 	mutex_exit(&pec_p->pec_pokefault_mutex);
330 
331 	if (rval != SUCCESS) {
332 		prg_p->status = PCITOOL_INVALID_ADDRESS;
333 		rval = EINVAL;
334 	} else
335 		prg_p->status = PCITOOL_SUCCESS;
336 
337 	return (rval);
338 }
339 
340 
341 /*
342  * This function is for PCI IO space and memory space access.
343  * It assumes that offset, bdf, acc_attr are current in prg_p.
344  * It assumes that prg_p->phys_addr is the final phys addr (including offset).
345  * This function modifies prg_p status and data.
346  */
347 int
348 pxtool_pciiomem_access(px_t *px_p, pcitool_reg_t *prg_p,
349     uint64_t *data_p, boolean_t is_write)
350 {
351 	on_trap_data_t otd;
352 	uint32_t io_stat = 0;
353 	dev_info_t *dip = px_p->px_dip;
354 	px_pec_t *pec_p = px_p->px_pec_p;
355 	size_t size = PCITOOL_ACC_ATTR_SIZE(prg_p->acc_attr);
356 	int rval = 0;
357 
358 	/* Alignment checking. */
359 	if (!IS_P2ALIGNED(prg_p->offset, size)) {
360 		DBG(DBG_TOOLS, dip, "not aligned.\n");
361 		prg_p->status = PCITOOL_NOT_ALIGNED;
362 		return (EINVAL);
363 	}
364 
365 	mutex_enter(&pec_p->pec_pokefault_mutex);
366 	pec_p->pec_ontrap_data = &otd;
367 
368 	if (is_write) {
369 		pci_device_t bdf = PX_GET_BDF(prg_p);
370 
371 		if (PCITOOL_ACC_IS_BIG_ENDIAN(prg_p->acc_attr))
372 			*data_p = pxtool_swap_endian(*data_p, size);
373 
374 		pec_p->pec_safeacc_type = DDI_FM_ERR_POKE;
375 
376 		if (!on_trap(&otd, OT_DATA_ACCESS)) {
377 			otd.ot_trampoline = (uintptr_t)&poke_fault;
378 			rval = hvio_poke(px_p->px_dev_hdl, prg_p->phys_addr,
379 			    size, *data_p, bdf, &io_stat);
380 		} else
381 			rval = H_EIO;
382 
383 		if (otd.ot_trap & OT_DATA_ACCESS)
384 			rval = H_EIO;
385 
386 		DBG(DBG_TOOLS, dip, "iomem:phys_addr:0x%" PRIx64 ", bdf:0x%x, "
387 		    "rval:%d, io_stat:%d\n", prg_p->phys_addr, bdf,
388 		    rval, io_stat);
389 	} else {
390 
391 		*data_p = 0;
392 
393 		pec_p->pec_safeacc_type = DDI_FM_ERR_PEEK;
394 
395 		if (!on_trap(&otd, OT_DATA_ACCESS)) {
396 			otd.ot_trampoline = (uintptr_t)&peek_fault;
397 			rval = hvio_peek(px_p->px_dev_hdl, prg_p->phys_addr,
398 			    size, &io_stat, data_p);
399 		} else
400 			rval = H_EIO;
401 
402 		DBG(DBG_TOOLS, dip, "iomem:phys_addr:0x%" PRIx64 ", "
403 		    "size:0x%" PRIx64 ", hdl:0x%" PRIx64 ", "
404 		    "rval:%d, io_stat:%d\n", prg_p->phys_addr,
405 		    size, px_p->px_dev_hdl, rval, io_stat);
406 		DBG(DBG_TOOLS, dip, "read data:0x%" PRIx64 "\n", *data_p);
407 
408 		if (PCITOOL_ACC_IS_BIG_ENDIAN(prg_p->acc_attr))
409 			*data_p = pxtool_swap_endian(*data_p, size);
410 	}
411 
412 	/*
413 	 * Workaround: delay taking down safe access env.
414 	 * For more info, see comment where pxtool_iomem_delay_usec is declared.
415 	 */
416 	if (pxtool_iomem_delay_usec > 0)
417 		delay(drv_usectohz(pxtool_iomem_delay_usec));
418 
419 	no_trap();
420 	pec_p->pec_ontrap_data = NULL;
421 	pec_p->pec_safeacc_type = DDI_FM_ERR_UNEXPECTED;
422 	mutex_exit(&pec_p->pec_pokefault_mutex);
423 
424 	if (rval != SUCCESS) {
425 		prg_p->status = PCITOOL_INVALID_ADDRESS;
426 		rval = EINVAL;
427 	} else if (io_stat != SUCCESS) {
428 		prg_p->status = PCITOOL_IO_ERROR;
429 		rval = EIO;
430 	} else
431 		prg_p->status = PCITOOL_SUCCESS;
432 
433 	return (rval);
434 }
435 
436 
437 /*ARGSUSED*/
438 int
439 pxtool_dev_reg_ops_platchk(dev_info_t *dip, pcitool_reg_t *prg_p)
440 {
441 	return (SUCCESS);
442 }
443 
444 
445 /*
446  * Perform register accesses on the nexus device itself.
447  */
448 int
449 pxtool_bus_reg_ops(dev_info_t *dip, void *arg, int cmd, int mode)
450 {
451 
452 	pcitool_reg_t		prg;
453 	size_t			size;
454 	px_t			*px_p = DIP_TO_STATE(dip);
455 	boolean_t		is_write = B_FALSE;
456 	uint32_t		rval = 0;
457 
458 	if (cmd == PCITOOL_NEXUS_SET_REG)
459 		is_write = B_TRUE;
460 
461 	DBG(DBG_TOOLS, dip, "pxtool_bus_reg_ops set/get reg\n");
462 
463 	/* Read data from userland. */
464 	if (ddi_copyin(arg, &prg, sizeof (pcitool_reg_t),
465 	    mode) != DDI_SUCCESS) {
466 		DBG(DBG_TOOLS, dip, "Error reading arguments\n");
467 		return (EFAULT);
468 	}
469 
470 	size = PCITOOL_ACC_ATTR_SIZE(prg.acc_attr);
471 
472 	DBG(DBG_TOOLS, dip, "raw bus:0x%x, dev:0x%x, func:0x%x\n",
473 	    prg.bus_no, prg.dev_no, prg.func_no);
474 	DBG(DBG_TOOLS, dip, "barnum:0x%x, offset:0x%" PRIx64 ", acc:0x%x\n",
475 	    prg.barnum, prg.offset, prg.acc_attr);
476 	DBG(DBG_TOOLS, dip, "data:0x%" PRIx64 ", phys_addr:0x%" PRIx64 "\n",
477 	    prg.data, prg.phys_addr);
478 
479 	/*
480 	 * If bank num == ff, base phys addr passed in from userland.
481 	 *
482 	 * Normal bank specification is invalid, as there is no OBP property to
483 	 * back it up.
484 	 */
485 	if (prg.barnum != PCITOOL_BASE) {
486 		prg.status = PCITOOL_OUT_OF_RANGE;
487 		rval = EINVAL;
488 		goto done;
489 	}
490 
491 	/* Allow only size of 8-bytes. */
492 	if (size != sizeof (uint64_t)) {
493 		prg.status = PCITOOL_INVALID_SIZE;
494 		rval = EINVAL;
495 		goto done;
496 	}
497 
498 	/* Alignment checking. */
499 	if (!IS_P2ALIGNED(prg.offset, size)) {
500 		DBG(DBG_TOOLS, dip, "not aligned.\n");
501 		prg.status = PCITOOL_NOT_ALIGNED;
502 		rval = EINVAL;
503 		goto done;
504 	}
505 
506 	prg.phys_addr += prg.offset;
507 
508 	/*  XXX do some kind of checking here? */
509 
510 	/* Access device.  prg.status is modified. */
511 	rval = pxtool_phys_access(px_p, prg.phys_addr, &prg.data,
512 	    PCITOOL_ACC_IS_BIG_ENDIAN(prg.acc_attr), is_write);
513 done:
514 	if (ddi_copyout(&prg, arg, sizeof (pcitool_reg_t),
515 	    mode) != DDI_SUCCESS) {
516 		DBG(DBG_TOOLS, dip, "Copyout failed.\n");
517 		return (EFAULT);
518 	}
519 
520 	return (rval);
521 }
522