aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/misc/sgi-xp/xpc_sn2.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/misc/sgi-xp/xpc_sn2.c')
-rw-r--r--drivers/misc/sgi-xp/xpc_sn2.c111
1 files changed, 111 insertions, 0 deletions
diff --git a/drivers/misc/sgi-xp/xpc_sn2.c b/drivers/misc/sgi-xp/xpc_sn2.c
new file mode 100644
index 000000000000..5a37348715c7
--- /dev/null
+++ b/drivers/misc/sgi-xp/xpc_sn2.c
@@ -0,0 +1,111 @@
1/*
2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
4 * for more details.
5 *
6 * Copyright (c) 2008 Silicon Graphics, Inc. All Rights Reserved.
7 */
8
9/*
10 * Cross Partition Communication (XPC) sn2-based functions.
11 *
12 * Architecture specific implementation of common functions.
13 *
14 */
15
16#include <linux/kernel.h>
17#include <asm/uncached.h>
18#include <asm/sn/sn_sal.h>
19#include "xpc.h"
20
21struct xpc_vars *xpc_vars;
22struct xpc_vars_part *xpc_vars_part;
23
24static enum xp_retval
25xpc_rsvd_page_init_sn2(struct xpc_rsvd_page *rp)
26{
27 AMO_t *amos_page;
28 u64 nasid_array = 0;
29 int i;
30 int ret;
31
32 xpc_vars = XPC_RP_VARS(rp);
33
34 rp->sn.vars_pa = __pa(xpc_vars);
35
36 xpc_vars_part = XPC_RP_VARS_PART(rp);
37
38 /*
39 * Before clearing xpc_vars, see if a page of AMOs had been previously
40 * allocated. If not we'll need to allocate one and set permissions
41 * so that cross-partition AMOs are allowed.
42 *
43 * The allocated AMO page needs MCA reporting to remain disabled after
44 * XPC has unloaded. To make this work, we keep a copy of the pointer
45 * to this page (i.e., amos_page) in the struct xpc_vars structure,
46 * which is pointed to by the reserved page, and re-use that saved copy
47 * on subsequent loads of XPC. This AMO page is never freed, and its
48 * memory protections are never restricted.
49 */
50 amos_page = xpc_vars->amos_page;
51 if (amos_page == NULL) {
52 amos_page = (AMO_t *)TO_AMO(uncached_alloc_page(0, 1));
53 if (amos_page == NULL) {
54 dev_err(xpc_part, "can't allocate page of AMOs\n");
55 return xpNoMemory;
56 }
57
58 /*
59 * Open up AMO-R/W to cpu. This is done for Shub 1.1 systems
60 * when xpc_allow_IPI_ops() is called via xpc_hb_init().
61 */
62 if (!enable_shub_wars_1_1()) {
63 ret = sn_change_memprotect(ia64_tpa((u64)amos_page),
64 PAGE_SIZE,
65 SN_MEMPROT_ACCESS_CLASS_1,
66 &nasid_array);
67 if (ret != 0) {
68 dev_err(xpc_part, "can't change memory "
69 "protections\n");
70 uncached_free_page(__IA64_UNCACHED_OFFSET |
71 TO_PHYS((u64)amos_page), 1);
72 return xpSalError;
73 }
74 }
75 }
76
77 /* clear xpc_vars */
78 memset(xpc_vars, 0, sizeof(struct xpc_vars));
79
80 xpc_vars->version = XPC_V_VERSION;
81 xpc_vars->act_nasid = cpuid_to_nasid(0);
82 xpc_vars->act_phys_cpuid = cpu_physical_id(0);
83 xpc_vars->vars_part_pa = __pa(xpc_vars_part);
84 xpc_vars->amos_page_pa = ia64_tpa((u64)amos_page);
85 xpc_vars->amos_page = amos_page; /* save for next load of XPC */
86
87 /* clear xpc_vars_part */
88 memset((u64 *)xpc_vars_part, 0, sizeof(struct xpc_vars_part) *
89 xp_max_npartitions);
90
91 /* initialize the activate IRQ related AMO variables */
92 for (i = 0; i < xp_nasid_mask_words; i++)
93 (void)xpc_IPI_init(XPC_ACTIVATE_IRQ_AMOS + i);
94
95 /* initialize the engaged remote partitions related AMO variables */
96 (void)xpc_IPI_init(XPC_ENGAGED_PARTITIONS_AMO);
97 (void)xpc_IPI_init(XPC_DISENGAGE_REQUEST_AMO);
98
99 return xpSuccess;
100}
101
102void
103xpc_init_sn2(void)
104{
105 xpc_rsvd_page_init = xpc_rsvd_page_init_sn2;
106}
107
108void
109xpc_exit_sn2(void)
110{
111}