aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/misc/sgi-xp/xpc_partition.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/misc/sgi-xp/xpc_partition.c')
-rw-r--r--drivers/misc/sgi-xp/xpc_partition.c149
1 files changed, 44 insertions, 105 deletions
diff --git a/drivers/misc/sgi-xp/xpc_partition.c b/drivers/misc/sgi-xp/xpc_partition.c
index 6c82f2050974..1db84cb49143 100644
--- a/drivers/misc/sgi-xp/xpc_partition.c
+++ b/drivers/misc/sgi-xp/xpc_partition.c
@@ -20,7 +20,6 @@
20#include <linux/cache.h> 20#include <linux/cache.h>
21#include <linux/mmzone.h> 21#include <linux/mmzone.h>
22#include <linux/nodemask.h> 22#include <linux/nodemask.h>
23#include <asm/uncached.h>
24#include <asm/sn/intr.h> 23#include <asm/sn/intr.h>
25#include <asm/sn/sn_sal.h> 24#include <asm/sn/sn_sal.h>
26#include <asm/sn/nodepda.h> 25#include <asm/sn/nodepda.h>
@@ -44,11 +43,10 @@ u64 xpc_prot_vec[MAX_NUMNODES];
44struct xpc_rsvd_page *xpc_rsvd_page; 43struct xpc_rsvd_page *xpc_rsvd_page;
45static u64 *xpc_part_nasids; 44static u64 *xpc_part_nasids;
46static u64 *xpc_mach_nasids; 45static u64 *xpc_mach_nasids;
47struct xpc_vars *xpc_vars;
48struct xpc_vars_part *xpc_vars_part;
49 46
50static int xp_nasid_mask_bytes; /* actual size in bytes of nasid mask */ 47/* >>> next two variables should be 'xpc_' if they remain here */
51static int xp_nasid_mask_words; /* actual size in words of nasid mask */ 48static int xp_sizeof_nasid_mask; /* actual size in bytes of nasid mask */
49int xp_nasid_mask_words; /* actual size in words of nasid mask */
52 50
53struct xpc_partition *xpc_partitions; 51struct xpc_partition *xpc_partitions;
54 52
@@ -150,12 +148,10 @@ xpc_get_rsvd_page_pa(int nasid)
150 * communications. 148 * communications.
151 */ 149 */
152struct xpc_rsvd_page * 150struct xpc_rsvd_page *
153xpc_rsvd_page_init(void) 151xpc_setup_rsvd_page(void)
154{ 152{
155 struct xpc_rsvd_page *rp; 153 struct xpc_rsvd_page *rp;
156 AMO_t *amos_page; 154 u64 rp_pa;
157 u64 rp_pa, nasid_array = 0;
158 int i, ret;
159 155
160 /* get the local reserved page's address */ 156 /* get the local reserved page's address */
161 157
@@ -168,110 +164,44 @@ xpc_rsvd_page_init(void)
168 } 164 }
169 rp = (struct xpc_rsvd_page *)__va(rp_pa); 165 rp = (struct xpc_rsvd_page *)__va(rp_pa);
170 166
171 if (rp->partid != sn_partition_id) { 167 if (rp->SAL_version < 3) {
172 dev_err(xpc_part, "the reserved page's partid of %d should be " 168 /* SAL_versions < 3 had a SAL_partid defined as a u8 */
173 "%d\n", rp->partid, sn_partition_id); 169 rp->SAL_partid &= 0xff;
170 }
171 BUG_ON(rp->SAL_partid != sn_partition_id);
172
173 if (rp->SAL_partid < 0 || rp->SAL_partid >= xp_max_npartitions) {
174 dev_err(xpc_part, "the reserved page's partid of %d is outside "
175 "supported range (< 0 || >= %d)\n", rp->SAL_partid,
176 xp_max_npartitions);
174 return NULL; 177 return NULL;
175 } 178 }
176 179
177 rp->version = XPC_RP_VERSION; 180 rp->version = XPC_RP_VERSION;
181 rp->max_npartitions = xp_max_npartitions;
178 182
179 /* establish the actual sizes of the nasid masks */ 183 /* establish the actual sizes of the nasid masks */
180 if (rp->SAL_version == 1) { 184 if (rp->SAL_version == 1) {
181 /* SAL_version 1 didn't set the nasids_size field */ 185 /* SAL_version 1 didn't set the nasids_size field */
182 rp->nasids_size = 128; 186 rp->SAL_nasids_size = 128;
183 } 187 }
184 xp_nasid_mask_bytes = rp->nasids_size; 188 xp_sizeof_nasid_mask = rp->SAL_nasids_size;
185 xp_nasid_mask_words = xp_nasid_mask_bytes / 8; 189 xp_nasid_mask_words = DIV_ROUND_UP(xp_sizeof_nasid_mask,
190 BYTES_PER_WORD);
186 191
187 /* setup the pointers to the various items in the reserved page */ 192 /* setup the pointers to the various items in the reserved page */
188 xpc_part_nasids = XPC_RP_PART_NASIDS(rp); 193 xpc_part_nasids = XPC_RP_PART_NASIDS(rp);
189 xpc_mach_nasids = XPC_RP_MACH_NASIDS(rp); 194 xpc_mach_nasids = XPC_RP_MACH_NASIDS(rp);
190 xpc_vars = XPC_RP_VARS(rp);
191 xpc_vars_part = XPC_RP_VARS_PART(rp);
192 195
193 /* 196 if (xpc_rsvd_page_init(rp) != xpSuccess)
194 * Before clearing xpc_vars, see if a page of AMOs had been previously 197 return NULL;
195 * allocated. If not we'll need to allocate one and set permissions
196 * so that cross-partition AMOs are allowed.
197 *
198 * The allocated AMO page needs MCA reporting to remain disabled after
199 * XPC has unloaded. To make this work, we keep a copy of the pointer
200 * to this page (i.e., amos_page) in the struct xpc_vars structure,
201 * which is pointed to by the reserved page, and re-use that saved copy
202 * on subsequent loads of XPC. This AMO page is never freed, and its
203 * memory protections are never restricted.
204 */
205 amos_page = xpc_vars->amos_page;
206 if (amos_page == NULL) {
207 amos_page = (AMO_t *)TO_AMO(uncached_alloc_page(0, 1));
208 if (amos_page == NULL) {
209 dev_err(xpc_part, "can't allocate page of AMOs\n");
210 return NULL;
211 }
212
213 /*
214 * Open up AMO-R/W to cpu. This is done for Shub 1.1 systems
215 * when xpc_allow_IPI_ops() is called via xpc_hb_init().
216 */
217 if (!enable_shub_wars_1_1()) {
218 ret = sn_change_memprotect(ia64_tpa((u64)amos_page),
219 PAGE_SIZE,
220 SN_MEMPROT_ACCESS_CLASS_1,
221 &nasid_array);
222 if (ret != 0) {
223 dev_err(xpc_part, "can't change memory "
224 "protections\n");
225 uncached_free_page(__IA64_UNCACHED_OFFSET |
226 TO_PHYS((u64)amos_page), 1);
227 return NULL;
228 }
229 }
230 } else if (!IS_AMO_ADDRESS((u64)amos_page)) {
231 /*
232 * EFI's XPBOOT can also set amos_page in the reserved page,
233 * but it happens to leave it as an uncached physical address
234 * and we need it to be an uncached virtual, so we'll have to
235 * convert it.
236 */
237 if (!IS_AMO_PHYS_ADDRESS((u64)amos_page)) {
238 dev_err(xpc_part, "previously used amos_page address "
239 "is bad = 0x%p\n", (void *)amos_page);
240 return NULL;
241 }
242 amos_page = (AMO_t *)TO_AMO((u64)amos_page);
243 }
244
245 /* clear xpc_vars */
246 memset(xpc_vars, 0, sizeof(struct xpc_vars));
247
248 xpc_vars->version = XPC_V_VERSION;
249 xpc_vars->act_nasid = cpuid_to_nasid(0);
250 xpc_vars->act_phys_cpuid = cpu_physical_id(0);
251 xpc_vars->vars_part_pa = __pa(xpc_vars_part);
252 xpc_vars->amos_page_pa = ia64_tpa((u64)amos_page);
253 xpc_vars->amos_page = amos_page; /* save for next load of XPC */
254
255 /* clear xpc_vars_part */
256 memset((u64 *)xpc_vars_part, 0, sizeof(struct xpc_vars_part) *
257 xp_max_npartitions);
258
259 /* initialize the activate IRQ related AMO variables */
260 for (i = 0; i < xp_nasid_mask_words; i++)
261 (void)xpc_IPI_init(XPC_ACTIVATE_IRQ_AMOS + i);
262
263 /* initialize the engaged remote partitions related AMO variables */
264 (void)xpc_IPI_init(XPC_ENGAGED_PARTITIONS_AMO);
265 (void)xpc_IPI_init(XPC_DISENGAGE_REQUEST_AMO);
266
267 /* timestamp of when reserved page was setup by XPC */
268 rp->stamp = CURRENT_TIME;
269 198
270 /* 199 /*
200 * Set timestamp of when reserved page was setup by XPC.
271 * This signifies to the remote partition that our reserved 201 * This signifies to the remote partition that our reserved
272 * page is initialized. 202 * page is initialized.
273 */ 203 */
274 rp->vars_pa = __pa(xpc_vars); 204 rp->stamp = CURRENT_TIME;
275 205
276 return rp; 206 return rp;
277} 207}
@@ -465,7 +395,7 @@ xpc_get_remote_rp(int nasid, u64 *discovered_nasids,
465 395
466 /* pull over the reserved page header and part_nasids mask */ 396 /* pull over the reserved page header and part_nasids mask */
467 ret = xp_remote_memcpy(remote_rp, (void *)*remote_rp_pa, 397 ret = xp_remote_memcpy(remote_rp, (void *)*remote_rp_pa,
468 XPC_RP_HEADER_SIZE + xp_nasid_mask_bytes); 398 XPC_RP_HEADER_SIZE + xp_sizeof_nasid_mask);
469 if (ret != xpSuccess) 399 if (ret != xpSuccess)
470 return ret; 400 return ret;
471 401
@@ -476,19 +406,28 @@ xpc_get_remote_rp(int nasid, u64 *discovered_nasids,
476 discovered_nasids[i] |= remote_part_nasids[i]; 406 discovered_nasids[i] |= remote_part_nasids[i];
477 } 407 }
478 408
479 /* check that the partid is for another partition */ 409 /* check that the partid is valid and is for another partition */
480 410
481 if (remote_rp->partid < 0 || remote_rp->partid >= xp_max_npartitions) 411 if (remote_rp->SAL_partid < 0 ||
412 remote_rp->SAL_partid >= xp_max_npartitions) {
482 return xpInvalidPartid; 413 return xpInvalidPartid;
414 }
483 415
484 if (remote_rp->partid == sn_partition_id) 416 if (remote_rp->SAL_partid == sn_partition_id)
485 return xpLocalPartid; 417 return xpLocalPartid;
486 418
419 /* see if the rest of the reserved page has been set up by XPC */
420 if (timespec_equal(&remote_rp->stamp, &ZERO_STAMP))
421 return xpRsvdPageNotSet;
422
487 if (XPC_VERSION_MAJOR(remote_rp->version) != 423 if (XPC_VERSION_MAJOR(remote_rp->version) !=
488 XPC_VERSION_MAJOR(XPC_RP_VERSION)) { 424 XPC_VERSION_MAJOR(XPC_RP_VERSION)) {
489 return xpBadVersion; 425 return xpBadVersion;
490 } 426 }
491 427
428 if (remote_rp->max_npartitions <= sn_partition_id)
429 return xpInvalidPartid;
430
492 return xpSuccess; 431 return xpSuccess;
493} 432}
494 433
@@ -592,7 +531,7 @@ xpc_identify_act_IRQ_req(int nasid)
592 int remote_rp_version; 531 int remote_rp_version;
593 int reactivate = 0; 532 int reactivate = 0;
594 int stamp_diff; 533 int stamp_diff;
595 struct timespec remote_rp_stamp = { 0, 0 }; 534 struct timespec remote_rp_stamp = { 0, 0 }; /*>>> ZERO_STAMP */
596 short partid; 535 short partid;
597 struct xpc_partition *part; 536 struct xpc_partition *part;
598 enum xp_retval ret; 537 enum xp_retval ret;
@@ -608,12 +547,12 @@ xpc_identify_act_IRQ_req(int nasid)
608 return; 547 return;
609 } 548 }
610 549
611 remote_vars_pa = remote_rp->vars_pa; 550 remote_vars_pa = remote_rp->sn.vars_pa;
612 remote_rp_version = remote_rp->version; 551 remote_rp_version = remote_rp->version;
613 if (XPC_SUPPORTS_RP_STAMP(remote_rp_version)) 552 if (XPC_SUPPORTS_RP_STAMP(remote_rp_version))
614 remote_rp_stamp = remote_rp->stamp; 553 remote_rp_stamp = remote_rp->stamp;
615 554
616 partid = remote_rp->partid; 555 partid = remote_rp->SAL_partid;
617 part = &xpc_partitions[partid]; 556 part = &xpc_partitions[partid];
618 557
619 /* pull over the cross partition variables */ 558 /* pull over the cross partition variables */
@@ -977,7 +916,7 @@ xpc_discovery(void)
977 enum xp_retval ret; 916 enum xp_retval ret;
978 917
979 remote_rp = xpc_kmalloc_cacheline_aligned(XPC_RP_HEADER_SIZE + 918 remote_rp = xpc_kmalloc_cacheline_aligned(XPC_RP_HEADER_SIZE +
980 xp_nasid_mask_bytes, 919 xp_sizeof_nasid_mask,
981 GFP_KERNEL, &remote_rp_base); 920 GFP_KERNEL, &remote_rp_base);
982 if (remote_rp == NULL) 921 if (remote_rp == NULL)
983 return; 922 return;
@@ -1063,9 +1002,9 @@ xpc_discovery(void)
1063 continue; 1002 continue;
1064 } 1003 }
1065 1004
1066 remote_vars_pa = remote_rp->vars_pa; 1005 remote_vars_pa = remote_rp->sn.vars_pa;
1067 1006
1068 partid = remote_rp->partid; 1007 partid = remote_rp->SAL_partid;
1069 part = &xpc_partitions[partid]; 1008 part = &xpc_partitions[partid];
1070 1009
1071 /* pull over the cross partition variables */ 1010 /* pull over the cross partition variables */
@@ -1155,5 +1094,5 @@ xpc_initiate_partid_to_nasids(short partid, void *nasid_mask)
1155 part_nasid_pa = (u64)XPC_RP_PART_NASIDS(part->remote_rp_pa); 1094 part_nasid_pa = (u64)XPC_RP_PART_NASIDS(part->remote_rp_pa);
1156 1095
1157 return xp_remote_memcpy(nasid_mask, (void *)part_nasid_pa, 1096 return xp_remote_memcpy(nasid_mask, (void *)part_nasid_pa,
1158 xp_nasid_mask_bytes); 1097 xp_sizeof_nasid_mask);
1159} 1098}