| /* | 
 |  * This file is subject to the terms and conditions of the GNU General Public | 
 |  * License.  See the file "COPYING" in the main directory of this archive | 
 |  * for more details. | 
 |  * | 
 |  * Copyright (c) 2008 Silicon Graphics, Inc.  All Rights Reserved. | 
 |  */ | 
 |  | 
 | /* | 
 |  * Cross Partition (XP) uv-based functions. | 
 |  * | 
 |  *      Architecture specific implementation of common functions. | 
 |  * | 
 |  */ | 
 |  | 
 | #include <linux/device.h> | 
 | #include <asm/uv/uv_hub.h> | 
 | #if defined CONFIG_X86_64 | 
 | #include <asm/uv/bios.h> | 
 | #elif defined CONFIG_IA64_GENERIC || defined CONFIG_IA64_SGI_UV | 
 | #include <asm/sn/sn_sal.h> | 
 | #endif | 
 | #include "../sgi-gru/grukservices.h" | 
 | #include "xp.h" | 
 |  | 
 | /* | 
 |  * Convert a virtual memory address to a physical memory address. | 
 |  */ | 
 | static unsigned long | 
 | xp_pa_uv(void *addr) | 
 | { | 
 | 	return uv_gpa(addr); | 
 | } | 
 |  | 
 | static enum xp_retval | 
 | xp_remote_memcpy_uv(unsigned long dst_gpa, const unsigned long src_gpa, | 
 | 		    size_t len) | 
 | { | 
 | 	int ret; | 
 |  | 
 | 	ret = gru_copy_gpa(dst_gpa, src_gpa, len); | 
 | 	if (ret == 0) | 
 | 		return xpSuccess; | 
 |  | 
 | 	dev_err(xp, "gru_copy_gpa() failed, dst_gpa=0x%016lx src_gpa=0x%016lx " | 
 | 		"len=%ld\n", dst_gpa, src_gpa, len); | 
 | 	return xpGruCopyError; | 
 | } | 
 |  | 
 | static int | 
 | xp_cpu_to_nasid_uv(int cpuid) | 
 | { | 
 | 	/* ??? Is this same as sn2 nasid in mach/part bitmaps set up by SAL? */ | 
 | 	return UV_PNODE_TO_NASID(uv_cpu_to_pnode(cpuid)); | 
 | } | 
 |  | 
 | static enum xp_retval | 
 | xp_expand_memprotect_uv(unsigned long phys_addr, unsigned long size) | 
 | { | 
 | 	int ret; | 
 |  | 
 | #if defined CONFIG_X86_64 | 
 | 	ret = uv_bios_change_memprotect(phys_addr, size, UV_MEMPROT_ALLOW_RW); | 
 | 	if (ret != BIOS_STATUS_SUCCESS) { | 
 | 		dev_err(xp, "uv_bios_change_memprotect(,, " | 
 | 			"UV_MEMPROT_ALLOW_RW) failed, ret=%d\n", ret); | 
 | 		return xpBiosError; | 
 | 	} | 
 |  | 
 | #elif defined CONFIG_IA64_GENERIC || defined CONFIG_IA64_SGI_UV | 
 | 	u64 nasid_array; | 
 |  | 
 | 	ret = sn_change_memprotect(phys_addr, size, SN_MEMPROT_ACCESS_CLASS_1, | 
 | 				   &nasid_array); | 
 | 	if (ret != 0) { | 
 | 		dev_err(xp, "sn_change_memprotect(,, " | 
 | 			"SN_MEMPROT_ACCESS_CLASS_1,) failed ret=%d\n", ret); | 
 | 		return xpSalError; | 
 | 	} | 
 | #else | 
 | 	#error not a supported configuration | 
 | #endif | 
 | 	return xpSuccess; | 
 | } | 
 |  | 
 | static enum xp_retval | 
 | xp_restrict_memprotect_uv(unsigned long phys_addr, unsigned long size) | 
 | { | 
 | 	int ret; | 
 |  | 
 | #if defined CONFIG_X86_64 | 
 | 	ret = uv_bios_change_memprotect(phys_addr, size, | 
 | 					UV_MEMPROT_RESTRICT_ACCESS); | 
 | 	if (ret != BIOS_STATUS_SUCCESS) { | 
 | 		dev_err(xp, "uv_bios_change_memprotect(,, " | 
 | 			"UV_MEMPROT_RESTRICT_ACCESS) failed, ret=%d\n", ret); | 
 | 		return xpBiosError; | 
 | 	} | 
 |  | 
 | #elif defined CONFIG_IA64_GENERIC || defined CONFIG_IA64_SGI_UV | 
 | 	u64 nasid_array; | 
 |  | 
 | 	ret = sn_change_memprotect(phys_addr, size, SN_MEMPROT_ACCESS_CLASS_0, | 
 | 				   &nasid_array); | 
 | 	if (ret != 0) { | 
 | 		dev_err(xp, "sn_change_memprotect(,, " | 
 | 			"SN_MEMPROT_ACCESS_CLASS_0,) failed ret=%d\n", ret); | 
 | 		return xpSalError; | 
 | 	} | 
 | #else | 
 | 	#error not a supported configuration | 
 | #endif | 
 | 	return xpSuccess; | 
 | } | 
 |  | 
 | enum xp_retval | 
 | xp_init_uv(void) | 
 | { | 
 | 	BUG_ON(!is_uv()); | 
 |  | 
 | 	xp_max_npartitions = XP_MAX_NPARTITIONS_UV; | 
 | 	xp_partition_id = sn_partition_id; | 
 | 	xp_region_size = sn_region_size; | 
 |  | 
 | 	xp_pa = xp_pa_uv; | 
 | 	xp_remote_memcpy = xp_remote_memcpy_uv; | 
 | 	xp_cpu_to_nasid = xp_cpu_to_nasid_uv; | 
 | 	xp_expand_memprotect = xp_expand_memprotect_uv; | 
 | 	xp_restrict_memprotect = xp_restrict_memprotect_uv; | 
 |  | 
 | 	return xpSuccess; | 
 | } | 
 |  | 
 | void | 
 | xp_exit_uv(void) | 
 | { | 
 | 	BUG_ON(!is_uv()); | 
 | } |