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 * All Rights Reserved, Copyright (c) FUJITSU LIMITED 2006 23 */ 24 25 #pragma ident "%Z%%M% %I% %E% SMI" 26 27 #include <sys/types.h> 28 #include <sys/sysmacros.h> 29 #include <sys/conf.h> 30 #include <sys/modctl.h> 31 #include <sys/stat.h> 32 #include <sys/async.h> 33 #include <sys/machsystm.h> 34 #include <sys/ksynch.h> 35 #include <sys/ddi.h> 36 #include <sys/sunddi.h> 37 #include <sys/ddifm.h> 38 #include <sys/fm/protocol.h> 39 #include <sys/fm/util.h> 40 #include <sys/kmem.h> 41 #include <sys/fm/io/opl_mc_fm.h> 42 #include <sys/memlist.h> 43 #include <sys/param.h> 44 #include <sys/ontrap.h> 45 #include <vm/page.h> 46 #include <sys/mc-opl.h> 47 48 /* 49 * Function prototypes 50 */ 51 static int mc_open(dev_t *, int, int, cred_t *); 52 static int mc_close(dev_t, int, int, cred_t *); 53 static int mc_ioctl(dev_t, int, intptr_t, int, cred_t *, int *); 54 static int mc_attach(dev_info_t *, ddi_attach_cmd_t); 55 static int mc_detach(dev_info_t *, ddi_detach_cmd_t); 56 57 static int mc_board_add(mc_opl_t *mcp); 58 static int mc_board_del(mc_opl_t *mcp); 59 static int mc_suspend(mc_opl_t *mcp, uint32_t flag); 60 static int mc_resume(mc_opl_t *mcp, uint32_t flag); 61 62 static void insert_mcp(mc_opl_t *mcp); 63 static void delete_mcp(mc_opl_t *mcp); 64 65 static int pa_to_maddr(mc_opl_t *mcp, uint64_t pa, mc_addr_t *maddr); 66 67 static int mc_valid_pa(mc_opl_t *mcp, uint64_t pa); 68 69 int mc_get_mem_unum(int, uint64_t, char *, int, int *); 70 extern int plat_max_boards(void); 71 72 static void mc_get_mlist(mc_opl_t *); 73 74 #pragma weak opl_get_physical_board 75 extern int opl_get_physical_board(int); 76 static int mc_opl_get_physical_board(int); 77 78 /* 79 * Configuration data structures 80 */ 81 static struct cb_ops mc_cb_ops = { 82 mc_open, /* open */ 83 mc_close, /* close */ 84 nulldev, /* strategy */ 85 nulldev, /* print */ 86 nodev, /* dump */ 87 nulldev, /* read */ 88 nulldev, /* write */ 89 mc_ioctl, /* ioctl */ 90 nodev, /* devmap */ 91 nodev, /* mmap */ 92 nodev, /* segmap */ 93 nochpoll, /* poll */ 94 ddi_prop_op, /* cb_prop_op */ 95 0, /* streamtab */ 96 D_MP | D_NEW | D_HOTPLUG, /* Driver compatibility flag */ 97 CB_REV, /* rev */ 98 nodev, /* cb_aread */ 99 nodev /* cb_awrite */ 100 }; 101 102 static struct dev_ops mc_ops = { 103 DEVO_REV, /* rev */ 104 0, /* refcnt */ 105 ddi_getinfo_1to1, /* getinfo */ 106 nulldev, /* identify */ 107 nulldev, /* probe */ 108 mc_attach, /* attach */ 109 mc_detach, /* detach */ 110 nulldev, /* reset */ 111 &mc_cb_ops, /* cb_ops */ 112 (struct bus_ops *)0, /* bus_ops */ 113 nulldev /* power */ 114 }; 115 116 /* 117 * Driver globals 118 */ 119 int mc_patrol_interval_sec = 10; 120 121 int inject_op_delay = 5; 122 123 mc_inst_list_t *mc_instances; 124 static kmutex_t mcmutex; 125 126 void *mc_statep; 127 128 #ifdef DEBUG 129 int oplmc_debug = 1; 130 #endif 131 132 static int mc_debug_show_all; 133 134 extern struct mod_ops mod_driverops; 135 136 static struct modldrv modldrv = { 137 &mod_driverops, /* module type, this one is a driver */ 138 "OPL Memory-controller 1.1", /* module name */ 139 &mc_ops, /* driver ops */ 140 }; 141 142 static struct modlinkage modlinkage = { 143 MODREV_1, /* rev */ 144 (void *)&modldrv, 145 NULL 146 }; 147 148 #pragma weak opl_get_mem_unum 149 extern int (*opl_get_mem_unum)(int, uint64_t, char *, int, int *); 150 151 /* 152 * pseudo-mc node portid format 153 * 154 * [10] = 0 155 * [9] = 1 156 * [8] = LSB_ID[4] = 0 157 * [7:4] = LSB_ID[3:0] 158 * [3:0] = 0 159 * 160 */ 161 162 /* 163 * These are the module initialization routines. 164 */ 165 int 166 _init(void) 167 { 168 int error; 169 170 171 if ((error = ddi_soft_state_init(&mc_statep, 172 sizeof (mc_opl_t), 1)) != 0) 173 return (error); 174 175 mutex_init(&mcmutex, NULL, MUTEX_DRIVER, NULL); 176 if (&opl_get_mem_unum) 177 opl_get_mem_unum = mc_get_mem_unum; 178 179 error = mod_install(&modlinkage); 180 if (error != 0) { 181 if (&opl_get_mem_unum) 182 opl_get_mem_unum = NULL; 183 mutex_destroy(&mcmutex); 184 ddi_soft_state_fini(&mc_statep); 185 } 186 187 return (error); 188 } 189 190 int 191 _fini(void) 192 { 193 int error; 194 195 if ((error = mod_remove(&modlinkage)) != 0) 196 return (error); 197 198 mutex_destroy(&mcmutex); 199 200 if (&opl_get_mem_unum) 201 opl_get_mem_unum = NULL; 202 203 ddi_soft_state_fini(&mc_statep); 204 205 return (0); 206 } 207 208 int 209 _info(struct modinfo *modinfop) 210 { 211 return (mod_info(&modlinkage, modinfop)); 212 } 213 214 static int 215 mc_attach(dev_info_t *devi, ddi_attach_cmd_t cmd) 216 { 217 mc_opl_t *mcp; 218 int instance; 219 220 /* get the instance of this devi */ 221 instance = ddi_get_instance(devi); 222 223 switch (cmd) { 224 case DDI_ATTACH: 225 break; 226 case DDI_RESUME: 227 mcp = ddi_get_soft_state(mc_statep, instance); 228 return (mc_resume(mcp, MC_DRIVER_SUSPENDED)); 229 default: 230 return (DDI_FAILURE); 231 } 232 233 234 if (ddi_soft_state_zalloc(mc_statep, instance) != DDI_SUCCESS) 235 return (DDI_FAILURE); 236 237 if ((mcp = ddi_get_soft_state(mc_statep, instance)) == NULL) { 238 goto bad; 239 } 240 241 /* set informations in mc state */ 242 mcp->mc_dip = devi; 243 244 if (mc_board_add(mcp)) 245 goto bad; 246 247 insert_mcp(mcp); 248 ddi_report_dev(devi); 249 250 return (DDI_SUCCESS); 251 252 bad: 253 ddi_soft_state_free(mc_statep, instance); 254 return (DDI_FAILURE); 255 } 256 257 /* ARGSUSED */ 258 static int 259 mc_detach(dev_info_t *devi, ddi_detach_cmd_t cmd) 260 { 261 int instance; 262 mc_opl_t *mcp; 263 264 /* get the instance of this devi */ 265 instance = ddi_get_instance(devi); 266 if ((mcp = ddi_get_soft_state(mc_statep, instance)) == NULL) { 267 return (DDI_FAILURE); 268 } 269 270 switch (cmd) { 271 case DDI_SUSPEND: 272 return (mc_suspend(mcp, MC_DRIVER_SUSPENDED)); 273 case DDI_DETACH: 274 break; 275 default: 276 return (DDI_FAILURE); 277 } 278 279 mutex_enter(&mcmutex); 280 if (mc_board_del(mcp) != DDI_SUCCESS) { 281 mutex_exit(&mcmutex); 282 return (DDI_FAILURE); 283 } 284 285 delete_mcp(mcp); 286 mutex_exit(&mcmutex); 287 288 /* free up the soft state */ 289 ddi_soft_state_free(mc_statep, instance); 290 291 return (DDI_SUCCESS); 292 } 293 294 /* ARGSUSED */ 295 static int 296 mc_open(dev_t *devp, int flag, int otyp, cred_t *credp) 297 { 298 return (0); 299 } 300 301 /* ARGSUSED */ 302 static int 303 mc_close(dev_t devp, int flag, int otyp, cred_t *credp) 304 { 305 return (0); 306 } 307 308 /* ARGSUSED */ 309 static int 310 mc_ioctl(dev_t dev, int cmd, intptr_t arg, int mode, cred_t *credp, 311 int *rvalp) 312 { 313 return (ENXIO); 314 } 315 316 /* 317 * PA validity check: 318 * This function return 1 if the PA is valid, otherwise 319 * return 0. 320 */ 321 322 /* ARGSUSED */ 323 static int 324 pa_is_valid(mc_opl_t *mcp, uint64_t addr) 325 { 326 /* 327 * Check if the addr is on the board. 328 */ 329 if ((addr < mcp->mc_start_address) || 330 (mcp->mc_start_address + mcp->mc_size <= addr)) 331 return (0); 332 333 if (mcp->mlist == NULL) 334 mc_get_mlist(mcp); 335 336 if (mcp->mlist && address_in_memlist(mcp->mlist, addr, 0)) { 337 return (1); 338 } 339 return (0); 340 } 341 342 /* 343 * mac-pa translation routines. 344 * 345 * Input: mc driver state, (LSB#, Bank#, DIMM address) 346 * Output: physical address 347 * 348 * Valid - return value: 0 349 * Invalid - return value: -1 350 */ 351 static int 352 mcaddr_to_pa(mc_opl_t *mcp, mc_addr_t *maddr, uint64_t *pa) 353 { 354 int i; 355 uint64_t pa_offset = 0; 356 int cs = (maddr->ma_dimm_addr >> CS_SHIFT) & 1; 357 int bank = maddr->ma_bank; 358 mc_addr_t maddr1; 359 int bank0, bank1; 360 361 MC_LOG("mcaddr /LSB%d/B%d/%x\n", maddr->ma_bd, bank, 362 maddr->ma_dimm_addr); 363 364 /* loc validity check */ 365 ASSERT(maddr->ma_bd >= 0 && OPL_BOARD_MAX > maddr->ma_bd); 366 ASSERT(bank >= 0 && OPL_BANK_MAX > bank); 367 368 /* Do translation */ 369 for (i = 0; i < PA_BITS_FOR_MAC; i++) { 370 int pa_bit = 0; 371 int mc_bit = mcp->mc_trans_table[cs][i]; 372 if (mc_bit < MC_ADDRESS_BITS) { 373 pa_bit = (maddr->ma_dimm_addr >> mc_bit) & 1; 374 } else if (mc_bit == MP_NONE) { 375 pa_bit = 0; 376 } else if (mc_bit == MP_BANK_0) { 377 pa_bit = bank & 1; 378 } else if (mc_bit == MP_BANK_1) { 379 pa_bit = (bank >> 1) & 1; 380 } else if (mc_bit == MP_BANK_2) { 381 pa_bit = (bank >> 2) & 1; 382 } 383 pa_offset |= ((uint64_t)pa_bit) << i; 384 } 385 *pa = mcp->mc_start_address + pa_offset; 386 MC_LOG("pa = %lx\n", *pa); 387 388 if (pa_to_maddr(mcp, *pa, &maddr1) == -1) { 389 return (-1); 390 } 391 392 393 if (IS_MIRROR(mcp, maddr->ma_bank)) { 394 bank0 = maddr->ma_bank & ~(1); 395 bank1 = maddr1.ma_bank & ~(1); 396 } else { 397 bank0 = maddr->ma_bank; 398 bank1 = maddr1.ma_bank; 399 } 400 /* 401 * there is no need to check ma_bd because it is generated from 402 * mcp. They are the same. 403 */ 404 if ((bank0 == bank1) && 405 (maddr->ma_dimm_addr == maddr1.ma_dimm_addr)) { 406 return (0); 407 } else { 408 cmn_err(CE_WARN, "Translation error source /LSB%d/B%d/%x, " 409 "PA %lx, target /LSB%d/B%d/%x\n", 410 maddr->ma_bd, bank, maddr->ma_dimm_addr, 411 *pa, maddr1.ma_bd, maddr1.ma_bank, 412 maddr1.ma_dimm_addr); 413 return (-1); 414 } 415 } 416 417 /* 418 * PA to CS (used by pa_to_maddr). 419 */ 420 static int 421 pa_to_cs(mc_opl_t *mcp, uint64_t pa_offset) 422 { 423 int i; 424 int cs = 0; 425 426 for (i = 0; i < PA_BITS_FOR_MAC; i++) { 427 /* MAC address bit<29> is arranged on the same PA bit */ 428 /* on both table. So we may use any table. */ 429 if (mcp->mc_trans_table[0][i] == CS_SHIFT) { 430 cs = (pa_offset >> i) & 1; 431 break; 432 } 433 } 434 return (cs); 435 } 436 437 /* 438 * PA to DIMM (used by pa_to_maddr). 439 */ 440 /* ARGSUSED */ 441 static uint32_t 442 pa_to_dimm(mc_opl_t *mcp, uint64_t pa_offset) 443 { 444 int i; 445 int cs = pa_to_cs(mcp, pa_offset); 446 uint32_t dimm_addr = 0; 447 448 for (i = 0; i < PA_BITS_FOR_MAC; i++) { 449 int pa_bit_value = (pa_offset >> i) & 1; 450 int mc_bit = mcp->mc_trans_table[cs][i]; 451 if (mc_bit < MC_ADDRESS_BITS) { 452 dimm_addr |= pa_bit_value << mc_bit; 453 } 454 } 455 return (dimm_addr); 456 } 457 458 /* 459 * PA to Bank (used by pa_to_maddr). 460 */ 461 static int 462 pa_to_bank(mc_opl_t *mcp, uint64_t pa_offset) 463 { 464 int i; 465 int cs = pa_to_cs(mcp, pa_offset); 466 int bankno = mcp->mc_trans_table[cs][INDEX_OF_BANK_SUPPLEMENT_BIT]; 467 468 469 for (i = 0; i < PA_BITS_FOR_MAC; i++) { 470 int pa_bit_value = (pa_offset >> i) & 1; 471 int mc_bit = mcp->mc_trans_table[cs][i]; 472 switch (mc_bit) { 473 case MP_BANK_0: 474 bankno |= pa_bit_value; 475 break; 476 case MP_BANK_1: 477 bankno |= pa_bit_value << 1; 478 break; 479 case MP_BANK_2: 480 bankno |= pa_bit_value << 2; 481 break; 482 } 483 } 484 485 return (bankno); 486 } 487 488 /* 489 * PA to MAC address translation 490 * 491 * Input: MAC driver state, physicall adress 492 * Output: LSB#, Bank id, mac address 493 * 494 * Valid - return value: 0 495 * Invalid - return value: -1 496 */ 497 498 int 499 pa_to_maddr(mc_opl_t *mcp, uint64_t pa, mc_addr_t *maddr) 500 { 501 uint64_t pa_offset; 502 503 /* PA validity check */ 504 if (!pa_is_valid(mcp, pa)) 505 return (-1); 506 507 508 /* Do translation */ 509 pa_offset = pa - mcp->mc_start_address; 510 511 maddr->ma_bd = mcp->mc_board_num; 512 maddr->ma_bank = pa_to_bank(mcp, pa_offset); 513 maddr->ma_dimm_addr = pa_to_dimm(mcp, pa_offset); 514 MC_LOG("pa %lx -> mcaddr /LSB%d/B%d/%x\n", 515 pa_offset, maddr->ma_bd, maddr->ma_bank, maddr->ma_dimm_addr); 516 return (0); 517 } 518 519 static void 520 mc_ereport_post(mc_aflt_t *mc_aflt) 521 { 522 char buf[FM_MAX_CLASS]; 523 char device_path[MAXPATHLEN]; 524 nv_alloc_t *nva = NULL; 525 nvlist_t *ereport, *detector, *resource; 526 errorq_elem_t *eqep; 527 int nflts; 528 mc_flt_stat_t *flt_stat; 529 int i, n, blen; 530 char *p; 531 uint32_t values[2], synd[2], dslot[2]; 532 533 if (panicstr) { 534 eqep = errorq_reserve(ereport_errorq); 535 if (eqep == NULL) 536 return; 537 ereport = errorq_elem_nvl(ereport_errorq, eqep); 538 nva = errorq_elem_nva(ereport_errorq, eqep); 539 } else { 540 ereport = fm_nvlist_create(nva); 541 } 542 543 /* 544 * Create the scheme "dev" FMRI. 545 */ 546 detector = fm_nvlist_create(nva); 547 resource = fm_nvlist_create(nva); 548 549 nflts = mc_aflt->mflt_nflts; 550 551 ASSERT(nflts >= 1 && nflts <= 2); 552 553 flt_stat = mc_aflt->mflt_stat[0]; 554 (void) ddi_pathname(mc_aflt->mflt_mcp->mc_dip, device_path); 555 (void) fm_fmri_dev_set(detector, FM_DEV_SCHEME_VERSION, NULL, 556 device_path, NULL); 557 558 /* 559 * Encode all the common data into the ereport. 560 */ 561 (void) snprintf(buf, FM_MAX_CLASS, "%s.%s-%s", 562 MC_OPL_ERROR_CLASS, 563 mc_aflt->mflt_is_ptrl ? MC_OPL_PTRL_SUBCLASS : 564 MC_OPL_MI_SUBCLASS, 565 mc_aflt->mflt_erpt_class); 566 567 MC_LOG("mc_ereport_post: ereport %s\n", buf); 568 569 570 fm_ereport_set(ereport, FM_EREPORT_VERSION, buf, 571 fm_ena_generate(mc_aflt->mflt_id, FM_ENA_FMT1), 572 detector, NULL); 573 574 /* 575 * Set payload. 576 */ 577 fm_payload_set(ereport, MC_OPL_BOARD, DATA_TYPE_UINT32, 578 flt_stat->mf_flt_maddr.ma_bd, NULL); 579 580 fm_payload_set(ereport, MC_OPL_PA, DATA_TYPE_UINT64, 581 flt_stat->mf_flt_paddr, NULL); 582 583 if (flt_stat->mf_type == FLT_TYPE_PERMANENT_CE) { 584 fm_payload_set(ereport, MC_OPL_FLT_TYPE, 585 DATA_TYPE_UINT8, ECC_STICKY, NULL); 586 } 587 588 for (i = 0; i < nflts; i++) 589 values[i] = mc_aflt->mflt_stat[i]->mf_flt_maddr.ma_bank; 590 591 fm_payload_set(ereport, MC_OPL_BANK, DATA_TYPE_UINT32_ARRAY, 592 nflts, values, NULL); 593 594 for (i = 0; i < nflts; i++) 595 values[i] = mc_aflt->mflt_stat[i]->mf_cntl; 596 597 fm_payload_set(ereport, MC_OPL_STATUS, DATA_TYPE_UINT32_ARRAY, 598 nflts, values, NULL); 599 600 for (i = 0; i < nflts; i++) 601 values[i] = mc_aflt->mflt_stat[i]->mf_err_add; 602 603 fm_payload_set(ereport, MC_OPL_ERR_ADD, DATA_TYPE_UINT32_ARRAY, 604 nflts, values, NULL); 605 606 for (i = 0; i < nflts; i++) 607 values[i] = mc_aflt->mflt_stat[i]->mf_err_log; 608 609 fm_payload_set(ereport, MC_OPL_ERR_LOG, DATA_TYPE_UINT32_ARRAY, 610 nflts, values, NULL); 611 612 for (i = 0; i < nflts; i++) { 613 flt_stat = mc_aflt->mflt_stat[i]; 614 if (flt_stat->mf_errlog_valid) { 615 synd[i] = flt_stat->mf_synd; 616 dslot[i] = flt_stat->mf_dimm_slot; 617 values[i] = flt_stat->mf_dram_place; 618 } else { 619 synd[i] = 0; 620 dslot[i] = 0; 621 values[i] = 0; 622 } 623 } 624 625 fm_payload_set(ereport, MC_OPL_ERR_SYND, 626 DATA_TYPE_UINT32_ARRAY, nflts, synd, NULL); 627 628 fm_payload_set(ereport, MC_OPL_ERR_DIMMSLOT, 629 DATA_TYPE_UINT32_ARRAY, nflts, dslot, NULL); 630 631 fm_payload_set(ereport, MC_OPL_ERR_DRAM, 632 DATA_TYPE_UINT32_ARRAY, nflts, values, NULL); 633 634 blen = MAXPATHLEN; 635 device_path[0] = 0; 636 p = &device_path[0]; 637 638 for (i = 0; i < nflts; i++) { 639 int bank = flt_stat->mf_flt_maddr.ma_bank; 640 int psb = -1; 641 642 flt_stat = mc_aflt->mflt_stat[i]; 643 psb = mc_opl_get_physical_board( 644 flt_stat->mf_flt_maddr.ma_bd); 645 646 if (psb != -1) { 647 snprintf(p, blen, "/CMU%d/B%d", psb, bank); 648 } else { 649 snprintf(p, blen, "/CMU/B%d", bank); 650 } 651 652 if (flt_stat->mf_errlog_valid) { 653 snprintf(p + strlen(p), blen, "/MEM%d%d%c", 654 bank/2, (bank & 0x1) * 2 + dslot[i] & 1, 655 (dslot[i] & 0x2) ? 'B' : 'A'); 656 } 657 658 n = strlen(&device_path[0]); 659 blen = MAXPATHLEN - n; 660 p = &device_path[n]; 661 if (i < (nflts - 1)) { 662 snprintf(p, blen, " "); 663 n += 1; blen -= 1; p += 1; 664 } 665 } 666 667 /* 668 * UNUM format /LSB#/B#/MEMxyZ 669 * where x is the MAC# = Bank#/2 670 * y is slot info = (Bank# & 0x1)*2 + {0, 1} 0 for DIMM-L, 1 for DIMM-H 671 * DIMM-L is 0 in bit 13, DIMM-H is 1 in bit 13. 672 * Z is A(CS0) or B(CS1) given by bit 14 673 */ 674 (void) fm_fmri_mem_set(resource, FM_MEM_SCHEME_VERSION, 675 NULL, device_path, NULL, 0); 676 677 fm_payload_set(ereport, MC_OPL_RESOURCE, DATA_TYPE_NVLIST, 678 resource, NULL); 679 680 if (panicstr) { 681 errorq_commit(ereport_errorq, eqep, ERRORQ_SYNC); 682 } else { 683 (void) fm_ereport_post(ereport, EVCH_TRYHARD); 684 fm_nvlist_destroy(ereport, FM_NVA_FREE); 685 fm_nvlist_destroy(detector, FM_NVA_FREE); 686 fm_nvlist_destroy(resource, FM_NVA_FREE); 687 } 688 } 689 690 static void 691 mc_err_drain(mc_aflt_t *mc_aflt) 692 { 693 int rv; 694 page_t *pp; 695 uint64_t errors; 696 uint64_t pa = (uint64_t)(-1); 697 698 MC_LOG("mc_err_drain: %s\n", 699 mc_aflt->mflt_erpt_class); 700 /* 701 * we come here only when we have: 702 * In mirror mode: CMPE, MUE, SUE 703 * In normal mode: UE, Permanent CE 704 */ 705 rv = mcaddr_to_pa(mc_aflt->mflt_mcp, 706 &(mc_aflt->mflt_stat[0]->mf_flt_maddr), &pa); 707 if (rv == 0) 708 mc_aflt->mflt_stat[0]->mf_flt_paddr = pa; 709 else 710 mc_aflt->mflt_stat[0]->mf_flt_paddr = (uint64_t)-1; 711 if (rv == 0) { 712 MC_LOG("mc_err_drain:pa = %lx\n", pa); 713 pp = page_numtopp_nolock(pa >> PAGESHIFT); 714 715 if (pp) { 716 /* 717 * Don't keep retiring and make ereports 718 * on bad pages in PTRL case 719 */ 720 MC_LOG("mc_err_drain:pp = %p\n", pp); 721 if (mc_aflt->mflt_is_ptrl) { 722 errors = 0; 723 if (page_retire_check(pa, &errors) == 0) { 724 MC_LOG("Page retired\n"); 725 return; 726 } 727 if (errors & mc_aflt->mflt_pr) { 728 MC_LOG("errors %lx, mflt_pr %x\n", 729 errors, mc_aflt->mflt_pr); 730 return; 731 } 732 } 733 MC_LOG("offline page %p error %x\n", pp, 734 mc_aflt->mflt_pr); 735 (void) page_retire(pa, mc_aflt->mflt_pr); 736 } 737 } 738 mc_ereport_post(mc_aflt); 739 } 740 741 #define DIMM_SIZE 0x80000000 742 743 #define INC_DIMM_ADDR(p, n) \ 744 (p)->ma_dimm_addr += n; \ 745 (p)->ma_dimm_addr &= (DIMM_SIZE - 1) 746 747 /* 748 * The restart address is actually defined in unit of PA[37:6] 749 * the mac patrol will convert that to dimm offset. If the 750 * address is not in the bank, it will continue to search for 751 * the next PA that is within the bank. 752 * 753 * Also the mac patrol scans the dimms based on PA, not 754 * dimm offset. 755 */ 756 757 static int 758 restart_patrol(mc_opl_t *mcp, int bank, mc_addr_info_t *maddr_info) 759 { 760 page_t *pp; 761 uint32_t reg; 762 uint64_t pa; 763 int rv; 764 int loop_count = 0; 765 766 reg = ldphysio(MAC_PTRL_CNTL(mcp, bank)); 767 768 /* already running, so we just return */ 769 if (reg & MAC_CNTL_PTRL_START) 770 return (0); 771 772 if (maddr_info == NULL || (maddr_info->mi_valid == 0)) { 773 MAC_PTRL_START(mcp, bank); 774 return (0); 775 } 776 777 778 rv = mcaddr_to_pa(mcp, &maddr_info->mi_maddr, &pa); 779 if (rv != 0) { 780 MC_LOG("cannot convert mcaddr to pa. use auto restart\n"); 781 MAC_PTRL_START(mcp, bank); 782 return (0); 783 } 784 785 /* 786 * pa is the last address scanned by the mac patrol 787 * we calculate the next restart address as follows: 788 * first we always advance it by 64 byte. Then begin the loop. 789 * loop { 790 * if it is not in phys_install, we advance to next 64 MB boundary 791 * if it is not backed by a page structure, done 792 * if the page is bad, advance to the next page boundary. 793 * else done 794 * if the new address exceeds the board, wrap around. 795 * } <stop if we come back to the same page> 796 */ 797 798 if (pa < mcp->mc_start_address || pa >= (mcp->mc_start_address 799 + mcp->mc_size)) { 800 /* pa is not on this board, just retry */ 801 cmn_err(CE_WARN, "restart_patrol: invalid address %lx " 802 "on board %d\n", pa, mcp->mc_board_num); 803 MAC_PTRL_START(mcp, bank); 804 return (0); 805 } 806 807 MC_LOG("restart_patrol: pa = %lx\n", pa); 808 if (maddr_info->mi_advance) { 809 uint64_t new_pa; 810 811 if (IS_MIRROR(mcp, bank)) 812 new_pa = pa + 64 * 2; 813 else 814 new_pa = pa + 64; 815 816 if (!mc_valid_pa(mcp, new_pa)) { 817 /* Isolation unit size is 64 MB */ 818 #define MC_ISOLATION_BSIZE (64 * 1024 * 1024) 819 MC_LOG("Invalid PA\n"); 820 pa = roundup(new_pa + 1, MC_ISOLATION_BSIZE); 821 } else { 822 pp = page_numtopp_nolock(new_pa >> PAGESHIFT); 823 if (pp != NULL) { 824 uint64_t errors = 0; 825 if (page_retire_check(new_pa, &errors) && 826 (errors == 0)) { 827 MC_LOG("Page has no error\n"); 828 MAC_PTRL_START(mcp, bank); 829 return (0); 830 } 831 /* 832 * skip bad pages 833 * and let the following loop to take care 834 */ 835 pa = roundup(new_pa + 1, PAGESIZE); 836 MC_LOG("Skipping bad page to %lx\n", pa); 837 } else { 838 MC_LOG("Page has no page structure\n"); 839 MAC_PTRL_START(mcp, bank); 840 return (0); 841 } 842 } 843 } 844 845 /* 846 * if we wrap around twice, we just give up and let 847 * mac patrol decide. 848 */ 849 MC_LOG("pa is now %lx\n", pa); 850 while (loop_count <= 1) { 851 if (!mc_valid_pa(mcp, pa)) { 852 MC_LOG("pa is not valid. round up to 64 MB\n"); 853 pa = roundup(pa + 1, 64 * 1024 * 1024); 854 } else { 855 pp = page_numtopp_nolock(pa >> PAGESHIFT); 856 if (pp != NULL) { 857 uint64_t errors = 0; 858 if (page_retire_check(pa, &errors) && 859 (errors == 0)) { 860 MC_LOG("Page has no error\n"); 861 break; 862 } 863 /* skip bad pages */ 864 pa = roundup(pa + 1, PAGESIZE); 865 MC_LOG("Skipping bad page to %lx\n", pa); 866 } else { 867 MC_LOG("Page has no page structure\n"); 868 break; 869 } 870 } 871 if (pa >= (mcp->mc_start_address + mcp->mc_size)) { 872 MC_LOG("Wrap around\n"); 873 pa = mcp->mc_start_address; 874 loop_count++; 875 } 876 } 877 878 /* retstart MAC patrol: PA[37:6] */ 879 MC_LOG("restart at pa = %lx\n", pa); 880 ST_MAC_REG(MAC_RESTART_ADD(mcp, bank), MAC_RESTART_PA(pa)); 881 MAC_PTRL_START_ADD(mcp, bank); 882 883 return (0); 884 } 885 886 /* 887 * Rewriting is used for two purposes. 888 * - to correct the error in memory. 889 * - to determine whether the error is permanent or intermittent. 890 * It's done by writing the address in MAC_BANKm_REWRITE_ADD 891 * and issuing REW_REQ command in MAC_BANKm_PTRL_CNRL. After that, 892 * REW_END (and REW_CE/REW_UE if some error detected) is set when 893 * rewrite operation is done. See 4.7.3 and 4.7.11 in Columbus2 PRM. 894 * 895 * Note that rewrite operation doesn't change RAW_UE to Marked UE. 896 * Therefore, we use it only CE case. 897 */ 898 static uint32_t 899 do_rewrite(mc_opl_t *mcp, int bank, uint32_t dimm_addr) 900 { 901 uint32_t cntl; 902 int count = 0; 903 904 /* first wait to make sure PTRL_STATUS is 0 */ 905 while (count++ < MAX_MC_LOOP_COUNT) { 906 cntl = LD_MAC_REG(MAC_PTRL_CNTL(mcp, bank)); 907 if (!(cntl & MAC_CNTL_PTRL_STATUS)) 908 break; 909 delay(drv_usectohz(10 * 1000)); /* 10 m.s. */ 910 } 911 if (count >= MAX_MC_LOOP_COUNT) 912 goto bad; 913 914 count = 0; 915 916 ST_MAC_REG(MAC_REWRITE_ADD(mcp, bank), dimm_addr); 917 MAC_REW_REQ(mcp, bank); 918 919 do { 920 cntl = LD_MAC_REG(MAC_PTRL_CNTL(mcp, bank)); 921 if (count++ >= MAX_MC_LOOP_COUNT) { 922 goto bad; 923 } else 924 delay(drv_usectohz(10 * 1000)); /* 10 m.s. */ 925 /* 926 * If there are other MEMORY or PCI activities, this 927 * will be BUSY, else it should be set immediately 928 */ 929 } while (!(cntl & MAC_CNTL_REW_END)); 930 931 MAC_CLEAR_ERRS(mcp, bank, MAC_CNTL_REW_ERRS); 932 return (cntl); 933 bad: 934 /* This is bad. Just reset the circuit */ 935 cmn_err(CE_WARN, "mc-opl rewrite timeout on /LSB%d/B%d\n", 936 mcp->mc_board_num, bank); 937 cntl = MAC_CNTL_REW_END; 938 MAC_CMD(mcp, bank, MAC_CNTL_PTRL_RESET); 939 MAC_CLEAR_ERRS(mcp, bank, MAC_CNTL_REW_ERRS); 940 return (cntl); 941 } 942 943 void 944 mc_process_scf_log(mc_opl_t *mcp) 945 { 946 int count = 0; 947 scf_log_t *p; 948 int bank; 949 950 while ((p = mcp->mc_scf_log) != NULL) { 951 bank = p->sl_bank; 952 while ((LD_MAC_REG(MAC_STATIC_ERR_ADD(mcp, p->sl_bank)) 953 & MAC_STATIC_ERR_VLD)) { 954 if (count++ >= (MAX_MC_LOOP_COUNT)) { 955 break; 956 } 957 delay(drv_usectohz(10 * 1000)); /* 10 m.s. */ 958 } 959 960 if (count < MAX_MC_LOOP_COUNT) { 961 ST_MAC_REG(MAC_STATIC_ERR_LOG(mcp, p->sl_bank), 962 p->sl_err_log); 963 964 ST_MAC_REG(MAC_STATIC_ERR_ADD(mcp, p->sl_bank), 965 p->sl_err_add|MAC_STATIC_ERR_VLD); 966 mcp->mc_scf_retry[bank] = 0; 967 } else { 968 /* if we try too many times, just drop the req */ 969 if (mcp->mc_scf_retry[bank]++ <= MAX_SCF_RETRY) { 970 return; 971 } else { 972 cmn_err(CE_WARN, "SCF is not responding. " 973 "Dropping the SCF LOG\n"); 974 } 975 } 976 mcp->mc_scf_log = p->sl_next; 977 mcp->mc_scf_total--; 978 ASSERT(mcp->mc_scf_total >= 0); 979 kmem_free(p, sizeof (scf_log_t)); 980 } 981 } 982 983 void 984 mc_queue_scf_log(mc_opl_t *mcp, mc_flt_stat_t *flt_stat, int bank) 985 { 986 scf_log_t *p; 987 988 if (mcp->mc_scf_total >= MAX_SCF_LOGS) { 989 cmn_err(CE_WARN, 990 "Max# SCF logs excceded on /LSB%d/B%d\n", 991 mcp->mc_board_num, bank); 992 return; 993 } 994 p = kmem_zalloc(sizeof (scf_log_t), KM_SLEEP); 995 p->sl_next = 0; 996 p->sl_err_add = flt_stat->mf_err_add; 997 p->sl_err_log = flt_stat->mf_err_log; 998 p->sl_bank = bank; 999 1000 if (mcp->mc_scf_log == NULL) { 1001 /* 1002 * we rely on mc_scf_log to detect NULL queue. 1003 * mc_scf_log_tail is irrelevant is such case. 1004 */ 1005 mcp->mc_scf_log_tail = mcp->mc_scf_log = p; 1006 } else { 1007 mcp->mc_scf_log_tail->sl_next = p; 1008 mcp->mc_scf_log_tail = p; 1009 } 1010 mcp->mc_scf_total++; 1011 } 1012 1013 /* 1014 * This routine determines what kind of CE happens, intermittent 1015 * or permanent as follows. (See 4.7.3 in Columbus2 PRM.) 1016 * - Do rewrite by issuing REW_REQ command to MAC_PTRL_CNTL register. 1017 * - If CE is still detected on the same address even after doing 1018 * rewrite operation twice, it is determined as permanent error. 1019 * - If error is not detected anymore, it is determined as intermittent 1020 * error. 1021 * - If UE is detected due to rewrite operation, it should be treated 1022 * as UE. 1023 */ 1024 1025 /* ARGSUSED */ 1026 static void 1027 mc_scrub_ce(mc_opl_t *mcp, int bank, mc_flt_stat_t *flt_stat, int ptrl_error) 1028 { 1029 uint32_t cntl; 1030 int i; 1031 1032 flt_stat->mf_type = FLT_TYPE_PERMANENT_CE; 1033 /* 1034 * rewrite request 1st time reads and correct error data 1035 * and write to DIMM. 2nd rewrite request must be issued 1036 * after REW_CE/UE/END is 0. When the 2nd request is completed, 1037 * if REW_CE = 1, then it is permanent CE. 1038 */ 1039 for (i = 0; i < 2; i++) { 1040 cntl = do_rewrite(mcp, bank, flt_stat->mf_err_add); 1041 /* 1042 * If the error becomes UE or CMPE 1043 * we return to the caller immediately. 1044 */ 1045 if (cntl & MAC_CNTL_REW_UE) { 1046 if (ptrl_error) 1047 flt_stat->mf_cntl |= MAC_CNTL_PTRL_UE; 1048 else 1049 flt_stat->mf_cntl |= MAC_CNTL_MI_UE; 1050 flt_stat->mf_type = FLT_TYPE_UE; 1051 return; 1052 } 1053 if (cntl & MAC_CNTL_REW_CMPE) { 1054 if (ptrl_error) 1055 flt_stat->mf_cntl |= MAC_CNTL_PTRL_CMPE; 1056 else 1057 flt_stat->mf_cntl |= MAC_CNTL_MI_CMPE; 1058 flt_stat->mf_type = FLT_TYPE_CMPE; 1059 return; 1060 } 1061 } 1062 if (!(cntl & MAC_CNTL_REW_CE)) { 1063 flt_stat->mf_type = FLT_TYPE_INTERMITTENT_CE; 1064 } 1065 1066 if (flt_stat->mf_type == FLT_TYPE_PERMANENT_CE) { 1067 /* report PERMANENT_CE to SP via SCF */ 1068 if (!(flt_stat->mf_err_log & MAC_ERR_LOG_INVALID)) { 1069 mc_queue_scf_log(mcp, flt_stat, bank); 1070 } 1071 } 1072 } 1073 1074 #define IS_CMPE(cntl, f) ((cntl) & ((f) ? MAC_CNTL_PTRL_CMPE :\ 1075 MAC_CNTL_MI_CMPE)) 1076 #define IS_UE(cntl, f) ((cntl) & ((f) ? MAC_CNTL_PTRL_UE : MAC_CNTL_MI_UE)) 1077 #define IS_CE(cntl, f) ((cntl) & ((f) ? MAC_CNTL_PTRL_CE : MAC_CNTL_MI_CE)) 1078 #define IS_OK(cntl, f) (!((cntl) & ((f) ? MAC_CNTL_PTRL_ERRS : \ 1079 MAC_CNTL_MI_ERRS))) 1080 1081 1082 static int 1083 IS_CE_ONLY(uint32_t cntl, int ptrl_error) 1084 { 1085 if (ptrl_error) { 1086 return ((cntl & MAC_CNTL_PTRL_ERRS) == MAC_CNTL_PTRL_CE); 1087 } else { 1088 return ((cntl & MAC_CNTL_MI_ERRS) == MAC_CNTL_MI_CE); 1089 } 1090 } 1091 1092 void 1093 mc_write_cntl(mc_opl_t *mcp, int bank, uint32_t value) 1094 { 1095 value |= mcp->mc_bank[bank].mcb_ptrl_cntl; 1096 ST_MAC_REG(MAC_PTRL_CNTL(mcp, bank), value); 1097 } 1098 1099 static int 1100 mc_stop(mc_opl_t *mcp, int bank) 1101 { 1102 uint32_t reg; 1103 int count = 0; 1104 1105 reg = LD_MAC_REG(MAC_PTRL_CNTL(mcp, bank)); 1106 1107 if (reg & MAC_CNTL_PTRL_START) 1108 MAC_PTRL_STOP(mcp, bank); 1109 1110 while (count++ <= MAX_MC_LOOP_COUNT) { 1111 reg = LD_MAC_REG(MAC_PTRL_CNTL(mcp, bank)); 1112 if ((reg & MAC_CNTL_PTRL_STATUS) == 0) 1113 return (0); 1114 delay(drv_usectohz(10 * 1000)); /* 10 m.s. */ 1115 } 1116 return (-1); 1117 } 1118 1119 static void 1120 mc_read_ptrl_reg(mc_opl_t *mcp, int bank, mc_flt_stat_t *flt_stat) 1121 { 1122 flt_stat->mf_cntl = LD_MAC_REG(MAC_PTRL_CNTL(mcp, bank)) & 1123 MAC_CNTL_PTRL_ERRS; 1124 flt_stat->mf_err_add = LD_MAC_REG(MAC_PTRL_ERR_ADD(mcp, bank)); 1125 flt_stat->mf_err_log = LD_MAC_REG(MAC_PTRL_ERR_LOG(mcp, bank)); 1126 flt_stat->mf_flt_maddr.ma_bd = mcp->mc_board_num; 1127 flt_stat->mf_flt_maddr.ma_bank = bank; 1128 flt_stat->mf_flt_maddr.ma_dimm_addr = flt_stat->mf_err_add; 1129 } 1130 1131 static void 1132 mc_read_mi_reg(mc_opl_t *mcp, int bank, mc_flt_stat_t *flt_stat) 1133 { 1134 uint32_t status, old_status; 1135 1136 status = LD_MAC_REG(MAC_PTRL_CNTL(mcp, bank)) & 1137 MAC_CNTL_MI_ERRS; 1138 old_status = 0; 1139 1140 /* we keep reading until the status is stable */ 1141 while (old_status != status) { 1142 old_status = status; 1143 flt_stat->mf_err_add = 1144 LD_MAC_REG(MAC_MI_ERR_ADD(mcp, bank)); 1145 flt_stat->mf_err_log = 1146 LD_MAC_REG(MAC_MI_ERR_LOG(mcp, bank)); 1147 status = LD_MAC_REG(MAC_PTRL_CNTL(mcp, bank)) & 1148 MAC_CNTL_MI_ERRS; 1149 if (status == old_status) { 1150 break; 1151 } 1152 } 1153 1154 flt_stat->mf_cntl = status; 1155 flt_stat->mf_flt_maddr.ma_bd = mcp->mc_board_num; 1156 flt_stat->mf_flt_maddr.ma_bank = bank; 1157 flt_stat->mf_flt_maddr.ma_dimm_addr = flt_stat->mf_err_add; 1158 } 1159 1160 1161 /* 1162 * Error philosophy for mirror mode: 1163 * 1164 * PTRL (The error address for both banks are same, since ptrl stops if it 1165 * detects error.) 1166 * - Compaire error Report CMPE. 1167 * 1168 * - UE-UE Report MUE. No rewrite. 1169 * 1170 * - UE-* UE-(CE/OK). Rewrite to scrub UE. Report SUE. 1171 * 1172 * - CE-* CE-(CE/OK). Scrub to determine if CE is permanent. 1173 * If CE is permanent, inform SCF. Once for each 1174 * Dimm. If CE becomes UE or CMPE, go back to above. 1175 * 1176 * 1177 * MI (The error addresses for each bank are the same or different.) 1178 * - Compair error If addresses are the same. Just CMPE. 1179 * If addresses are different (this could happen 1180 * as a result of scrubbing. Report each seperately. 1181 * Only report error info on each side. 1182 * 1183 * - UE-UE Addresses are the same. Report MUE. 1184 * Addresses are different. Report SUE on each bank. 1185 * Rewrite to clear UE. 1186 * 1187 * - UE-* UE-(CE/OK) 1188 * Rewrite to clear UE. Report SUE for the bank. 1189 * 1190 * - CE-* CE-(CE/OK). Scrub to determine if CE is permanent. 1191 * If CE becomes UE or CMPE, go back to above. 1192 * 1193 */ 1194 1195 static int 1196 mc_process_error_mir(mc_opl_t *mcp, mc_aflt_t *mc_aflt, mc_flt_stat_t *flt_stat) 1197 { 1198 int ptrl_error = mc_aflt->mflt_is_ptrl; 1199 int i; 1200 int rv = 0; 1201 1202 MC_LOG("process mirror errors cntl[0] = %x, cntl[1] = %x\n", 1203 flt_stat[0].mf_cntl, flt_stat[1].mf_cntl); 1204 1205 if (ptrl_error) { 1206 if (((flt_stat[0].mf_cntl | flt_stat[1].mf_cntl) 1207 & MAC_CNTL_PTRL_ERRS) == 0) 1208 return (0); 1209 } else { 1210 if (((flt_stat[0].mf_cntl | flt_stat[1].mf_cntl) 1211 & MAC_CNTL_MI_ERRS) == 0) 1212 return (0); 1213 } 1214 1215 /* 1216 * First we take care of the case of CE 1217 * because they can become UE or CMPE 1218 */ 1219 for (i = 0; i < 2; i++) { 1220 if (IS_CE_ONLY(flt_stat[i].mf_cntl, ptrl_error)) { 1221 MC_LOG("CE detected on bank %d\n", 1222 flt_stat[i].mf_flt_maddr.ma_bank); 1223 mc_scrub_ce(mcp, flt_stat[i].mf_flt_maddr.ma_bank, 1224 &flt_stat[i], ptrl_error); 1225 rv = 1; 1226 } 1227 } 1228 1229 /* The above scrubbing can turn CE into UE or CMPE */ 1230 1231 /* 1232 * Now we distinguish two cases: same address or not 1233 * the same address. It might seem more intuitive to 1234 * distinguish PTRL v.s. MI error but it is more 1235 * complicated that way. 1236 */ 1237 1238 if (flt_stat[0].mf_err_add == flt_stat[1].mf_err_add) { 1239 1240 if (IS_CMPE(flt_stat[0].mf_cntl, ptrl_error) || 1241 IS_CMPE(flt_stat[1].mf_cntl, ptrl_error)) { 1242 flt_stat[0].mf_type = FLT_TYPE_CMPE; 1243 flt_stat[1].mf_type = FLT_TYPE_CMPE; 1244 mc_aflt->mflt_erpt_class = MC_OPL_CMPE; 1245 MC_LOG("cmpe error detected\n"); 1246 mc_aflt->mflt_nflts = 2; 1247 mc_aflt->mflt_stat[0] = &flt_stat[0]; 1248 mc_aflt->mflt_stat[1] = &flt_stat[1]; 1249 mc_aflt->mflt_pr = PR_UE; 1250 mc_err_drain(mc_aflt); 1251 return (1); 1252 } 1253 1254 if (IS_UE(flt_stat[0].mf_cntl, ptrl_error) && 1255 IS_UE(flt_stat[1].mf_cntl, ptrl_error)) { 1256 /* Both side are UE's */ 1257 1258 MAC_SET_ERRLOG_INFO(&flt_stat[0]); 1259 MAC_SET_ERRLOG_INFO(&flt_stat[1]); 1260 MC_LOG("MUE detected\n"); 1261 flt_stat[0].mf_type = flt_stat[1].mf_type = 1262 FLT_TYPE_MUE; 1263 mc_aflt->mflt_erpt_class = MC_OPL_MUE; 1264 mc_aflt->mflt_nflts = 2; 1265 mc_aflt->mflt_stat[0] = &flt_stat[0]; 1266 mc_aflt->mflt_stat[1] = &flt_stat[1]; 1267 mc_aflt->mflt_pr = PR_UE; 1268 mc_err_drain(mc_aflt); 1269 return (1); 1270 } 1271 1272 /* Now the only case is UE/CE, UE/OK, or don't care */ 1273 for (i = 0; i < 2; i++) { 1274 if (IS_UE(flt_stat[i].mf_cntl, ptrl_error)) { 1275 /* If we have CE, we would have done REW */ 1276 if (IS_OK(flt_stat[i^1].mf_cntl, ptrl_error)) { 1277 (void) do_rewrite(mcp, 1278 flt_stat[i].mf_flt_maddr.ma_bank, 1279 flt_stat[i].mf_flt_maddr.ma_dimm_addr); 1280 } 1281 flt_stat[i].mf_type = FLT_TYPE_UE; 1282 MAC_SET_ERRLOG_INFO(&flt_stat[i]); 1283 mc_aflt->mflt_erpt_class = MC_OPL_SUE; 1284 mc_aflt->mflt_stat[0] = &flt_stat[i]; 1285 mc_aflt->mflt_nflts = 1; 1286 mc_aflt->mflt_pr = PR_MCE; 1287 mc_err_drain(mc_aflt); 1288 /* Once we hit a UE/CE or UE/OK case, done */ 1289 return (1); 1290 } 1291 } 1292 1293 } else { 1294 /* 1295 * addresses are different. That means errors 1296 * on the 2 banks are not related at all. 1297 */ 1298 for (i = 0; i < 2; i++) { 1299 if (IS_CMPE(flt_stat[i].mf_cntl, ptrl_error)) { 1300 flt_stat[i].mf_type = FLT_TYPE_CMPE; 1301 mc_aflt->mflt_erpt_class = MC_OPL_CMPE; 1302 MC_LOG("cmpe error detected\n"); 1303 mc_aflt->mflt_nflts = 1; 1304 mc_aflt->mflt_stat[0] = &flt_stat[i]; 1305 mc_aflt->mflt_pr = PR_UE; 1306 mc_err_drain(mc_aflt); 1307 /* no more report on this bank */ 1308 flt_stat[i].mf_cntl = 0; 1309 rv = 1; 1310 } 1311 } 1312 1313 for (i = 0; i < 2; i++) { 1314 if (IS_UE(flt_stat[i].mf_cntl, ptrl_error)) { 1315 (void) do_rewrite(mcp, 1316 flt_stat[i].mf_flt_maddr.ma_bank, 1317 flt_stat[i].mf_flt_maddr.ma_dimm_addr); 1318 flt_stat[i].mf_type = FLT_TYPE_UE; 1319 MAC_SET_ERRLOG_INFO(&flt_stat[i]); 1320 mc_aflt->mflt_erpt_class = MC_OPL_SUE; 1321 mc_aflt->mflt_stat[0] = &flt_stat[i]; 1322 mc_aflt->mflt_nflts = 1; 1323 mc_aflt->mflt_pr = PR_MCE; 1324 mc_err_drain(mc_aflt); 1325 rv = 1; 1326 } 1327 } 1328 } 1329 return (rv); 1330 } 1331 1332 static void 1333 mc_error_handler_mir(mc_opl_t *mcp, int bank, mc_addr_info_t *maddr) 1334 { 1335 mc_aflt_t mc_aflt; 1336 mc_flt_stat_t flt_stat[2], mi_flt_stat[2]; 1337 int other_bank; 1338 1339 if (mc_stop(mcp, bank)) { 1340 cmn_err(CE_WARN, "Cannot stop Memory Patrol at /LSB%d/B%d\n", 1341 mcp->mc_board_num, bank); 1342 return; 1343 } 1344 bzero(&mc_aflt, sizeof (mc_aflt_t)); 1345 bzero(&flt_stat, 2 * sizeof (mc_flt_stat_t)); 1346 bzero(&mi_flt_stat, 2 * sizeof (mc_flt_stat_t)); 1347 1348 mc_aflt.mflt_mcp = mcp; 1349 mc_aflt.mflt_id = gethrtime(); 1350 1351 /* Now read all the registers into flt_stat */ 1352 1353 MC_LOG("Reading registers of bank %d\n", bank); 1354 /* patrol registers */ 1355 mc_read_ptrl_reg(mcp, bank, &flt_stat[0]); 1356 1357 ASSERT(maddr); 1358 maddr->mi_maddr = flt_stat[0].mf_flt_maddr; 1359 1360 MC_LOG("ptrl registers cntl %x add %x log %x\n", 1361 flt_stat[0].mf_cntl, 1362 flt_stat[0].mf_err_add, 1363 flt_stat[0].mf_err_log); 1364 1365 /* MI registers */ 1366 mc_read_mi_reg(mcp, bank, &mi_flt_stat[0]); 1367 1368 MC_LOG("MI registers cntl %x add %x log %x\n", 1369 mi_flt_stat[0].mf_cntl, 1370 mi_flt_stat[0].mf_err_add, 1371 mi_flt_stat[0].mf_err_log); 1372 1373 other_bank = bank^1; 1374 1375 MC_LOG("Reading registers of bank %d\n", other_bank); 1376 1377 ASSERT(mcp->mc_bank[other_bank].mcb_status & BANK_INSTALLED); 1378 1379 mc_read_ptrl_reg(mcp, other_bank, &flt_stat[1]); 1380 MC_LOG("ptrl registers cntl %x add %x log %x\n", 1381 flt_stat[1].mf_cntl, 1382 flt_stat[1].mf_err_add, 1383 flt_stat[1].mf_err_log); 1384 1385 /* MI registers */ 1386 mc_read_mi_reg(mcp, other_bank, &mi_flt_stat[1]); 1387 MC_LOG("MI registers cntl %x add %x log %x\n", 1388 mi_flt_stat[1].mf_cntl, 1389 mi_flt_stat[1].mf_err_add, 1390 mi_flt_stat[1].mf_err_log); 1391 1392 /* clear errors once we read all the registers */ 1393 MAC_CLEAR_ERRS(mcp, other_bank, 1394 (MAC_CNTL_PTRL_ERRS|MAC_CNTL_MI_ERRS)); 1395 1396 MAC_CLEAR_ERRS(mcp, bank, (MAC_CNTL_PTRL_ERRS|MAC_CNTL_MI_ERRS)); 1397 1398 /* Process PTRL errors first */ 1399 1400 /* if not error mode, cntl1 is 0 */ 1401 if ((flt_stat[0].mf_err_add & MAC_ERR_ADD_INVALID) || 1402 (flt_stat[0].mf_err_log & MAC_ERR_LOG_INVALID)) 1403 flt_stat[0].mf_cntl = 0; 1404 1405 if ((flt_stat[1].mf_err_add & MAC_ERR_ADD_INVALID) || 1406 (flt_stat[1].mf_err_log & MAC_ERR_LOG_INVALID)) 1407 flt_stat[1].mf_cntl = 0; 1408 1409 mc_aflt.mflt_is_ptrl = 1; 1410 maddr->mi_valid = mc_process_error_mir(mcp, &mc_aflt, &flt_stat[0]); 1411 1412 mc_aflt.mflt_is_ptrl = 0; 1413 mc_process_error_mir(mcp, &mc_aflt, &mi_flt_stat[0]); 1414 } 1415 1416 static int 1417 mc_process_error(mc_opl_t *mcp, int bank, mc_aflt_t *mc_aflt, 1418 mc_flt_stat_t *flt_stat) 1419 { 1420 int ptrl_error = mc_aflt->mflt_is_ptrl; 1421 int rv = 0; 1422 1423 mc_aflt->mflt_erpt_class = NULL; 1424 if (IS_UE(flt_stat->mf_cntl, ptrl_error)) { 1425 MC_LOG("UE deteceted\n"); 1426 flt_stat->mf_type = FLT_TYPE_UE; 1427 mc_aflt->mflt_erpt_class = MC_OPL_UE; 1428 mc_aflt->mflt_pr = PR_UE; 1429 MAC_SET_ERRLOG_INFO(flt_stat); 1430 rv = 1; 1431 } else if (IS_CE(flt_stat->mf_cntl, ptrl_error)) { 1432 MC_LOG("CE deteceted\n"); 1433 MAC_SET_ERRLOG_INFO(flt_stat); 1434 1435 /* Error type can change after scrubing */ 1436 mc_scrub_ce(mcp, bank, flt_stat, ptrl_error); 1437 1438 if (flt_stat->mf_type == FLT_TYPE_PERMANENT_CE) { 1439 mc_aflt->mflt_erpt_class = MC_OPL_CE; 1440 mc_aflt->mflt_pr = PR_MCE; 1441 } else if (flt_stat->mf_type == FLT_TYPE_UE) { 1442 mc_aflt->mflt_erpt_class = MC_OPL_UE; 1443 mc_aflt->mflt_pr = PR_UE; 1444 } 1445 rv = 1; 1446 } 1447 MC_LOG("mc_process_error: fault type %x erpt %s\n", 1448 flt_stat->mf_type, 1449 mc_aflt->mflt_erpt_class); 1450 if (mc_aflt->mflt_erpt_class) { 1451 mc_aflt->mflt_stat[0] = flt_stat; 1452 mc_aflt->mflt_nflts = 1; 1453 mc_err_drain(mc_aflt); 1454 } 1455 return (rv); 1456 } 1457 1458 static void 1459 mc_error_handler(mc_opl_t *mcp, int bank, mc_addr_info_t *maddr) 1460 { 1461 mc_aflt_t mc_aflt; 1462 mc_flt_stat_t flt_stat, mi_flt_stat; 1463 1464 if (mc_stop(mcp, bank)) { 1465 cmn_err(CE_WARN, "Cannot stop Memory Patrol at /LSB%d/B%d\n", 1466 mcp->mc_board_num, bank); 1467 return; 1468 } 1469 1470 bzero(&mc_aflt, sizeof (mc_aflt_t)); 1471 bzero(&flt_stat, sizeof (mc_flt_stat_t)); 1472 bzero(&mi_flt_stat, sizeof (mc_flt_stat_t)); 1473 1474 mc_aflt.mflt_mcp = mcp; 1475 mc_aflt.mflt_id = gethrtime(); 1476 1477 /* patrol registers */ 1478 mc_read_ptrl_reg(mcp, bank, &flt_stat); 1479 1480 ASSERT(maddr); 1481 maddr->mi_maddr = flt_stat.mf_flt_maddr; 1482 1483 MC_LOG("ptrl registers cntl %x add %x log %x\n", 1484 flt_stat.mf_cntl, 1485 flt_stat.mf_err_add, 1486 flt_stat.mf_err_log); 1487 1488 /* MI registers */ 1489 mc_read_mi_reg(mcp, bank, &mi_flt_stat); 1490 1491 MC_LOG("MI registers cntl %x add %x log %x\n", 1492 mi_flt_stat.mf_cntl, 1493 mi_flt_stat.mf_err_add, 1494 mi_flt_stat.mf_err_log); 1495 1496 /* clear errors once we read all the registers */ 1497 MAC_CLEAR_ERRS(mcp, bank, (MAC_CNTL_PTRL_ERRS|MAC_CNTL_MI_ERRS)); 1498 1499 mc_aflt.mflt_is_ptrl = 1; 1500 if ((flt_stat.mf_cntl & MAC_CNTL_PTRL_ERRS) && 1501 ((flt_stat.mf_err_add & MAC_ERR_ADD_INVALID) == 0) && 1502 ((flt_stat.mf_err_log & MAC_ERR_LOG_INVALID) == 0)) { 1503 maddr->mi_valid = mc_process_error(mcp, bank, 1504 &mc_aflt, &flt_stat); 1505 } 1506 mc_aflt.mflt_is_ptrl = 0; 1507 if ((mi_flt_stat.mf_cntl & MAC_CNTL_MI_ERRS) && 1508 ((mi_flt_stat.mf_err_add & MAC_ERR_ADD_INVALID) == 0) && 1509 ((mi_flt_stat.mf_err_log & MAC_ERR_LOG_INVALID) == 0)) { 1510 mc_process_error(mcp, bank, &mc_aflt, &mi_flt_stat); 1511 } 1512 } 1513 1514 /* 1515 * memory patrol error handling algorithm: 1516 * timeout() is used to do periodic polling 1517 * This is the flow chart. 1518 * timeout -> 1519 * mc_check_errors() 1520 * if memory bank is installed, read the status register 1521 * if any error bit is set, 1522 * -> mc_error_handler() 1523 * -> mc_stop() 1524 * -> read all error regsiters 1525 * -> mc_process_error() 1526 * determine error type 1527 * rewrite to clear error or scrub to determine CE type 1528 * inform SCF on permanent CE 1529 * -> mc_err_drain 1530 * page offline processing 1531 * -> mc_ereport_post() 1532 */ 1533 1534 static void 1535 mc_check_errors_func(mc_opl_t *mcp) 1536 { 1537 mc_addr_info_t maddr_info; 1538 int i, error_count = 0; 1539 uint32_t stat, cntl; 1540 1541 /* 1542 * scan errors. 1543 */ 1544 for (i = 0; i < BANKNUM_PER_SB; i++) { 1545 if (mcp->mc_bank[i].mcb_status & BANK_INSTALLED) { 1546 stat = ldphysio(MAC_PTRL_STAT(mcp, i)); 1547 cntl = ldphysio(MAC_PTRL_CNTL(mcp, i)); 1548 if (cntl & MAC_CNTL_PTRL_ADD_MAX) { 1549 mcp->mc_period++; 1550 MC_LOG("mc period %ld on " 1551 "/LSB%d/B%d\n", mcp->mc_period, 1552 mcp->mc_board_num, i); 1553 MAC_CLEAR_MAX(mcp, i); 1554 } 1555 if (mc_debug_show_all) { 1556 MC_LOG("/LSB%d/B%d stat %x cntl %x\n", 1557 mcp->mc_board_num, i, 1558 stat, cntl); 1559 } 1560 if (stat & (MAC_STAT_PTRL_ERRS|MAC_STAT_MI_ERRS)) { 1561 maddr_info.mi_valid = 0; 1562 maddr_info.mi_advance = 1; 1563 if (IS_MIRROR(mcp, i)) 1564 mc_error_handler_mir(mcp, i, 1565 &maddr_info); 1566 else 1567 mc_error_handler(mcp, i, &maddr_info); 1568 1569 error_count++; 1570 restart_patrol(mcp, i, &maddr_info); 1571 } else { 1572 restart_patrol(mcp, i, NULL); 1573 } 1574 } 1575 } 1576 mc_process_scf_log(mcp); 1577 if (error_count > 0) 1578 mcp->mc_last_error += error_count; 1579 else 1580 mcp->mc_last_error = 0; 1581 } 1582 1583 /* this is just a wrapper for the above func */ 1584 1585 static void 1586 mc_check_errors(void *arg) 1587 { 1588 mc_opl_t *mcp = (mc_opl_t *)arg; 1589 clock_t interval; 1590 1591 /* 1592 * scan errors. 1593 */ 1594 mutex_enter(&mcp->mc_lock); 1595 mcp->mc_tid = 0; 1596 if ((mcp->mc_status & MC_POLL_RUNNING) && 1597 !(mcp->mc_status & MC_SOFT_SUSPENDED)) { 1598 mc_check_errors_func(mcp); 1599 1600 if (mcp->mc_last_error > 0) { 1601 interval = (mcp->mc_interval_hz) >> mcp->mc_last_error; 1602 if (interval < 1) 1603 interval = 1; 1604 } else 1605 interval = mcp->mc_interval_hz; 1606 1607 mcp->mc_tid = timeout(mc_check_errors, mcp, 1608 interval); 1609 } 1610 mutex_exit(&mcp->mc_lock); 1611 } 1612 1613 static void 1614 get_ptrl_start_address(mc_opl_t *mcp, int bank, mc_addr_t *maddr) 1615 { 1616 maddr->ma_bd = mcp->mc_board_num; 1617 maddr->ma_bank = bank; 1618 maddr->ma_dimm_addr = 0; 1619 } 1620 1621 typedef struct mc_mem_range { 1622 uint64_t addr; 1623 uint64_t size; 1624 } mc_mem_range_t; 1625 1626 static int 1627 get_base_address(mc_opl_t *mcp) 1628 { 1629 mc_mem_range_t *mem_range; 1630 int len; 1631 1632 if (ddi_getlongprop(DDI_DEV_T_ANY, mcp->mc_dip, DDI_PROP_DONTPASS, 1633 "sb-mem-ranges", (caddr_t)&mem_range, &len) != DDI_SUCCESS) { 1634 return (DDI_FAILURE); 1635 } 1636 1637 mcp->mc_start_address = mem_range->addr; 1638 mcp->mc_size = mem_range->size; 1639 1640 kmem_free(mem_range, len); 1641 return (DDI_SUCCESS); 1642 } 1643 1644 struct mc_addr_spec { 1645 uint32_t bank; 1646 uint32_t phys_hi; 1647 uint32_t phys_lo; 1648 }; 1649 1650 #define REGS_PA(m, i) ((((uint64_t)m[i].phys_hi)<<32) | m[i].phys_lo) 1651 1652 static char *mc_tbl_name[] = { 1653 "cs0-mc-pa-trans-table", 1654 "cs1-mc-pa-trans-table" 1655 }; 1656 1657 static int 1658 mc_valid_pa(mc_opl_t *mcp, uint64_t pa) 1659 { 1660 struct memlist *ml; 1661 1662 if (mcp->mlist == NULL) 1663 mc_get_mlist(mcp); 1664 1665 for (ml = mcp->mlist; ml; ml = ml->next) { 1666 if (ml->address <= pa && pa < (ml->address + ml->size)) 1667 return (1); 1668 } 1669 return (0); 1670 } 1671 1672 static void 1673 mc_memlist_delete(struct memlist *mlist) 1674 { 1675 struct memlist *ml; 1676 1677 for (ml = mlist; ml; ml = mlist) { 1678 mlist = ml->next; 1679 kmem_free(ml, sizeof (struct memlist)); 1680 } 1681 } 1682 1683 static struct memlist * 1684 mc_memlist_dup(struct memlist *mlist) 1685 { 1686 struct memlist *hl = NULL, *tl, **mlp; 1687 1688 if (mlist == NULL) 1689 return (NULL); 1690 1691 mlp = &hl; 1692 tl = *mlp; 1693 for (; mlist; mlist = mlist->next) { 1694 *mlp = kmem_alloc(sizeof (struct memlist), KM_SLEEP); 1695 (*mlp)->address = mlist->address; 1696 (*mlp)->size = mlist->size; 1697 (*mlp)->prev = tl; 1698 tl = *mlp; 1699 mlp = &((*mlp)->next); 1700 } 1701 *mlp = NULL; 1702 1703 return (hl); 1704 } 1705 1706 1707 static struct memlist * 1708 mc_memlist_del_span(struct memlist *mlist, uint64_t base, uint64_t len) 1709 { 1710 uint64_t end; 1711 struct memlist *ml, *tl, *nlp; 1712 1713 if (mlist == NULL) 1714 return (NULL); 1715 1716 end = base + len; 1717 if ((end <= mlist->address) || (base == end)) 1718 return (mlist); 1719 1720 for (tl = ml = mlist; ml; tl = ml, ml = nlp) { 1721 uint64_t mend; 1722 1723 nlp = ml->next; 1724 1725 if (end <= ml->address) 1726 break; 1727 1728 mend = ml->address + ml->size; 1729 if (base < mend) { 1730 if (base <= ml->address) { 1731 ml->address = end; 1732 if (end >= mend) 1733 ml->size = 0ull; 1734 else 1735 ml->size = mend - ml->address; 1736 } else { 1737 ml->size = base - ml->address; 1738 if (end < mend) { 1739 struct memlist *nl; 1740 /* 1741 * splitting an memlist entry. 1742 */ 1743 nl = kmem_alloc(sizeof (struct memlist), 1744 KM_SLEEP); 1745 nl->address = end; 1746 nl->size = mend - nl->address; 1747 if ((nl->next = nlp) != NULL) 1748 nlp->prev = nl; 1749 nl->prev = ml; 1750 ml->next = nl; 1751 nlp = nl; 1752 } 1753 } 1754 if (ml->size == 0ull) { 1755 if (ml == mlist) { 1756 if ((mlist = nlp) != NULL) 1757 nlp->prev = NULL; 1758 kmem_free(ml, sizeof (struct memlist)); 1759 if (mlist == NULL) 1760 break; 1761 ml = nlp; 1762 } else { 1763 if ((tl->next = nlp) != NULL) 1764 nlp->prev = tl; 1765 kmem_free(ml, sizeof (struct memlist)); 1766 ml = tl; 1767 } 1768 } 1769 } 1770 } 1771 1772 return (mlist); 1773 } 1774 1775 static void 1776 mc_get_mlist(mc_opl_t *mcp) 1777 { 1778 struct memlist *mlist; 1779 1780 memlist_read_lock(); 1781 mlist = mc_memlist_dup(phys_install); 1782 memlist_read_unlock(); 1783 1784 if (mlist) { 1785 mlist = mc_memlist_del_span(mlist, 0ull, mcp->mc_start_address); 1786 } 1787 1788 if (mlist) { 1789 uint64_t startpa, endpa; 1790 1791 startpa = mcp->mc_start_address + mcp->mc_size; 1792 endpa = ptob(physmax + 1); 1793 if (endpa > startpa) { 1794 mlist = mc_memlist_del_span(mlist, 1795 startpa, endpa - startpa); 1796 } 1797 } 1798 1799 if (mlist) { 1800 mcp->mlist = mlist; 1801 } 1802 } 1803 1804 int 1805 mc_board_add(mc_opl_t *mcp) 1806 { 1807 struct mc_addr_spec *macaddr; 1808 int len, i, bk, cc; 1809 mc_addr_info_t maddr; 1810 uint32_t mirr; 1811 1812 mutex_init(&mcp->mc_lock, NULL, MUTEX_DRIVER, NULL); 1813 1814 /* 1815 * Get configurations from "pseudo-mc" node which includes: 1816 * board# : LSB number 1817 * mac-addr : physical base address of MAC registers 1818 * csX-mac-pa-trans-table: translation table from DIMM address 1819 * to physical address or vice versa. 1820 */ 1821 mcp->mc_board_num = (int)ddi_getprop(DDI_DEV_T_ANY, mcp->mc_dip, 1822 DDI_PROP_DONTPASS, "board#", -1); 1823 1824 /* 1825 * Get start address in this CAB. It can be gotten from 1826 * "sb-mem-ranges" property. 1827 */ 1828 1829 if (get_base_address(mcp) == DDI_FAILURE) { 1830 mutex_destroy(&mcp->mc_lock); 1831 return (DDI_FAILURE); 1832 } 1833 /* get mac-pa trans tables */ 1834 for (i = 0; i < MC_TT_CS; i++) { 1835 len = MC_TT_ENTRIES; 1836 cc = ddi_getlongprop_buf(DDI_DEV_T_ANY, mcp->mc_dip, 1837 DDI_PROP_DONTPASS, mc_tbl_name[i], 1838 (caddr_t)mcp->mc_trans_table[i], &len); 1839 1840 if (cc != DDI_SUCCESS) { 1841 bzero(mcp->mc_trans_table[i], MC_TT_ENTRIES); 1842 } 1843 } 1844 mcp->mlist = NULL; 1845 1846 mc_get_mlist(mcp); 1847 1848 /* initialize bank informations */ 1849 cc = ddi_getlongprop(DDI_DEV_T_ANY, mcp->mc_dip, DDI_PROP_DONTPASS, 1850 "mc-addr", (caddr_t)&macaddr, &len); 1851 if (cc != DDI_SUCCESS) { 1852 cmn_err(CE_WARN, "Cannot get mc-addr. err=%d\n", cc); 1853 mutex_destroy(&mcp->mc_lock); 1854 return (DDI_FAILURE); 1855 } 1856 1857 for (i = 0; i < len / sizeof (struct mc_addr_spec); i++) { 1858 struct mc_bank *bankp; 1859 uint32_t reg; 1860 1861 /* 1862 * setup bank 1863 */ 1864 bk = macaddr[i].bank; 1865 bankp = &(mcp->mc_bank[bk]); 1866 bankp->mcb_status = BANK_INSTALLED; 1867 bankp->mcb_reg_base = REGS_PA(macaddr, i); 1868 1869 reg = LD_MAC_REG(MAC_PTRL_CNTL(mcp, bk)); 1870 bankp->mcb_ptrl_cntl = (reg & MAC_CNTL_PTRL_PRESERVE_BITS); 1871 1872 /* 1873 * check if mirror mode 1874 */ 1875 mirr = LD_MAC_REG(MAC_MIRR(mcp, bk)); 1876 1877 if (mirr & MAC_MIRR_MIRROR_MODE) { 1878 MC_LOG("Mirror -> /LSB%d/B%d\n", 1879 mcp->mc_board_num, bk); 1880 bankp->mcb_status |= BANK_MIRROR_MODE; 1881 /* 1882 * The following bit is only used for 1883 * error injection. We should clear it 1884 */ 1885 if (mirr & MAC_MIRR_BANK_EXCLUSIVE) 1886 ST_MAC_REG(MAC_MIRR(mcp, bk), 1887 0); 1888 } 1889 1890 /* 1891 * restart if not mirror mode or the other bank 1892 * of the mirror is not running 1893 */ 1894 if (!(mirr & MAC_MIRR_MIRROR_MODE) || 1895 !(mcp->mc_bank[bk^1].mcb_status & 1896 BANK_PTRL_RUNNING)) { 1897 MC_LOG("Starting up /LSB%d/B%d\n", 1898 mcp->mc_board_num, bk); 1899 get_ptrl_start_address(mcp, bk, &maddr.mi_maddr); 1900 maddr.mi_valid = 1; 1901 maddr.mi_advance = 0; 1902 restart_patrol(mcp, bk, &maddr); 1903 } else { 1904 MC_LOG("Not starting up /LSB%d/B%d\n", 1905 mcp->mc_board_num, bk); 1906 } 1907 bankp->mcb_status |= BANK_PTRL_RUNNING; 1908 } 1909 kmem_free(macaddr, len); 1910 1911 /* 1912 * set interval in HZ. 1913 */ 1914 for (i = 0; i < BANKNUM_PER_SB; i++) { 1915 mcp->mc_scf_retry[i] = 0; 1916 } 1917 mcp->mc_last_error = 0; 1918 mcp->mc_period = 0; 1919 1920 mcp->mc_interval_hz = drv_usectohz(mc_patrol_interval_sec * 1000000); 1921 /* restart memory patrol checking */ 1922 mcp->mc_status |= MC_POLL_RUNNING; 1923 mcp->mc_tid = timeout(mc_check_errors, mcp, mcp->mc_interval_hz); 1924 1925 return (DDI_SUCCESS); 1926 } 1927 1928 int 1929 mc_board_del(mc_opl_t *mcp) 1930 { 1931 int i; 1932 scf_log_t *p; 1933 timeout_id_t tid = 0; 1934 1935 /* 1936 * cleanup mac state 1937 */ 1938 mutex_enter(&mcp->mc_lock); 1939 for (i = 0; i < BANKNUM_PER_SB; i++) { 1940 if (mcp->mc_bank[i].mcb_status & BANK_INSTALLED) { 1941 if (mc_stop(mcp, i)) { 1942 mutex_exit(&mcp->mc_lock); 1943 return (-1); 1944 } 1945 mcp->mc_bank[i].mcb_status &= ~BANK_INSTALLED; 1946 } 1947 } 1948 1949 /* stop memory patrol checking */ 1950 if (mcp->mc_status & MC_POLL_RUNNING) { 1951 mcp->mc_status &= ~MC_POLL_RUNNING; 1952 tid = mcp->mc_tid; 1953 mcp->mc_tid = 0; 1954 } 1955 1956 /* just throw away all the scf logs */ 1957 while ((p = mcp->mc_scf_log) != NULL) { 1958 mcp->mc_scf_log = p->sl_next; 1959 mcp->mc_scf_total--; 1960 kmem_free(p, sizeof (scf_log_t)); 1961 } 1962 1963 if (mcp->mlist) 1964 mc_memlist_delete(mcp->mlist); 1965 1966 mutex_exit(&mcp->mc_lock); 1967 if (tid) 1968 (void) untimeout(tid); 1969 1970 mutex_destroy(&mcp->mc_lock); 1971 return (DDI_SUCCESS); 1972 } 1973 1974 int 1975 mc_suspend(mc_opl_t *mcp, uint32_t flag) 1976 { 1977 timeout_id_t tid = 0; 1978 int i; 1979 /* stop memory patrol checking */ 1980 mutex_enter(&mcp->mc_lock); 1981 if (mcp->mc_status & MC_POLL_RUNNING) { 1982 for (i = 0; i < BANKNUM_PER_SB; i++) { 1983 if (mcp->mc_bank[i].mcb_status & BANK_INSTALLED) { 1984 if (mc_stop(mcp, i)) { 1985 mutex_exit(&mcp->mc_lock); 1986 return (-1); 1987 } 1988 } 1989 } 1990 mcp->mc_status &= ~MC_POLL_RUNNING; 1991 tid = mcp->mc_tid; 1992 } 1993 mcp->mc_status |= flag; 1994 mcp->mc_tid = 0; 1995 mutex_exit(&mcp->mc_lock); 1996 if (tid) 1997 (void) untimeout(tid); 1998 1999 return (DDI_SUCCESS); 2000 } 2001 2002 /* caller must clear the SUSPEND bits or this will do nothing */ 2003 2004 int 2005 mc_resume(mc_opl_t *mcp, uint32_t flag) 2006 { 2007 int i; 2008 uint64_t basepa; 2009 2010 mutex_enter(&mcp->mc_lock); 2011 basepa = mcp->mc_start_address; 2012 if (get_base_address(mcp) == DDI_FAILURE) { 2013 mutex_exit(&mcp->mc_lock); 2014 return (DDI_FAILURE); 2015 } 2016 2017 if (basepa != mcp->mc_start_address) { 2018 if (mcp->mlist) 2019 mc_memlist_delete(mcp->mlist); 2020 mcp->mlist = NULL; 2021 mc_get_mlist(mcp); 2022 } 2023 2024 mcp->mc_status &= ~flag; 2025 mcp->mc_list->mc_start_address = mcp->mc_start_address; 2026 2027 if (mcp->mc_status & (MC_SOFT_SUSPENDED | MC_DRIVER_SUSPENDED)) { 2028 mutex_exit(&mcp->mc_lock); 2029 return (DDI_SUCCESS); 2030 } 2031 2032 if (!(mcp->mc_status & MC_POLL_RUNNING)) { 2033 /* restart memory patrol checking */ 2034 mcp->mc_status |= MC_POLL_RUNNING; 2035 for (i = 0; i < BANKNUM_PER_SB; i++) { 2036 if (mcp->mc_bank[i].mcb_status & BANK_INSTALLED) { 2037 restart_patrol(mcp, i, NULL); 2038 } 2039 } 2040 /* check error asap */ 2041 mcp->mc_tid = timeout(mc_check_errors, mcp, 1); 2042 } 2043 mutex_exit(&mcp->mc_lock); 2044 2045 return (DDI_SUCCESS); 2046 } 2047 2048 static mc_opl_t * 2049 mc_pa_to_mcp(uint64_t pa) 2050 { 2051 mc_inst_list_t *p; 2052 ASSERT(MUTEX_HELD(&mcmutex)); 2053 for (p = mc_instances; p; p = p->next) { 2054 /* if mac patrol is suspended, we cannot rely on it */ 2055 if (!(p->mc_opl->mc_status & MC_POLL_RUNNING) || 2056 (p->mc_opl->mc_status & MC_SOFT_SUSPENDED)) 2057 continue; 2058 if ((p->mc_start_address <= pa) && 2059 (pa < (p->mc_start_address + p->mc_size))) { 2060 return (p->mc_opl); 2061 } 2062 } 2063 return (NULL); 2064 } 2065 2066 /* 2067 * Get Physical Board number from Logical one. 2068 */ 2069 static int 2070 mc_opl_get_physical_board(int sb) 2071 { 2072 if (&opl_get_physical_board) { 2073 return (opl_get_physical_board(sb)); 2074 } 2075 2076 cmn_err(CE_NOTE, "!opl_get_physical_board() not loaded\n"); 2077 return (-1); 2078 } 2079 2080 /* ARGSUSED */ 2081 int 2082 mc_get_mem_unum(int synd_code, uint64_t flt_addr, char *buf, int buflen, 2083 int *lenp) 2084 { 2085 mc_opl_t *mcp; 2086 int bank; 2087 int sb; 2088 2089 mutex_enter(&mcmutex); 2090 2091 if (((mcp = mc_pa_to_mcp(flt_addr)) == NULL) || 2092 (!pa_is_valid(mcp, flt_addr))) { 2093 mutex_exit(&mcmutex); 2094 if (snprintf(buf, buflen, "UNKNOWN") >= buflen) { 2095 return (ENOSPC); 2096 } else { 2097 if (lenp) 2098 *lenp = strlen(buf); 2099 } 2100 return (0); 2101 } 2102 2103 bank = pa_to_bank(mcp, flt_addr - mcp->mc_start_address); 2104 sb = mc_opl_get_physical_board(mcp->mc_board_num); 2105 2106 if (sb == -1) { 2107 mutex_exit(&mcmutex); 2108 return (ENXIO); 2109 } 2110 2111 if (snprintf(buf, buflen, "/CMU%d/B%d", sb, bank) >= buflen) { 2112 mutex_exit(&mcmutex); 2113 return (ENOSPC); 2114 } else { 2115 if (lenp) 2116 *lenp = strlen(buf); 2117 } 2118 mutex_exit(&mcmutex); 2119 return (0); 2120 } 2121 2122 int 2123 opl_mc_suspend() 2124 { 2125 mc_opl_t *mcp; 2126 mc_inst_list_t *p; 2127 2128 mutex_enter(&mcmutex); 2129 2130 for (p = mc_instances; p; p = p->next) { 2131 mcp = p->mc_opl; 2132 (void) mc_suspend(mcp, MC_SOFT_SUSPENDED); 2133 } 2134 2135 mutex_exit(&mcmutex); 2136 return (0); 2137 } 2138 2139 int 2140 opl_mc_resume() 2141 { 2142 mc_opl_t *mcp; 2143 mc_inst_list_t *p; 2144 2145 mutex_enter(&mcmutex); 2146 2147 for (p = mc_instances; p; p = p->next) { 2148 mcp = p->mc_opl; 2149 (void) mc_resume(mcp, MC_SOFT_SUSPENDED); 2150 } 2151 2152 mutex_exit(&mcmutex); 2153 return (0); 2154 } 2155 2156 static void 2157 insert_mcp(mc_opl_t *mcp) 2158 { 2159 mc_inst_list_t *p; 2160 2161 p = kmem_zalloc(sizeof (mc_inst_list_t), KM_SLEEP); 2162 p->mc_opl = mcp; 2163 p->mc_board_num = mcp->mc_board_num; 2164 p->mc_start_address = mcp->mc_start_address; 2165 p->mc_size = mcp->mc_size; 2166 mcp->mc_list = p; 2167 2168 mutex_enter(&mcmutex); 2169 2170 p->next = mc_instances; 2171 mc_instances = p; 2172 2173 mutex_exit(&mcmutex); 2174 } 2175 2176 static void 2177 delete_mcp(mc_opl_t *mcp) 2178 { 2179 mc_inst_list_t *prev, *current; 2180 mc_inst_list_t *p; 2181 2182 p = mcp->mc_list; 2183 2184 if (mc_instances == p) { 2185 mc_instances = p->next; 2186 kmem_free(p, sizeof (mc_inst_list_t)); 2187 return; 2188 } 2189 prev = mc_instances; 2190 for (current = mc_instances; current != NULL; current = current->next) { 2191 if (current == p) { 2192 prev->next = p->next; 2193 kmem_free(p, sizeof (mc_inst_list_t)); 2194 return; 2195 } 2196 prev = current; 2197 } 2198 } 2199 2200 /* Error injection interface */ 2201 2202 /* ARGSUSED */ 2203 int 2204 mc_inject_error(int error_type, uint64_t pa, uint32_t flags) 2205 { 2206 mc_opl_t *mcp; 2207 int bank; 2208 uint32_t dimm_addr; 2209 uint32_t cntl; 2210 mc_addr_info_t maddr; 2211 uint32_t data, stat; 2212 int both_sides = 0; 2213 uint64_t pa0; 2214 on_trap_data_t otd; 2215 extern void cpu_flush_ecache(void); 2216 2217 MC_LOG("HW mc_inject_error(%x, %lx, %x)\n", error_type, pa, flags); 2218 2219 mutex_enter(&mcmutex); 2220 2221 if ((mcp = mc_pa_to_mcp(pa)) == NULL) { 2222 mutex_exit(&mcmutex); 2223 MC_LOG("mc_inject_error: invalid pa\n"); 2224 return (ENOTSUP); 2225 } 2226 2227 mutex_enter(&mcp->mc_lock); 2228 mutex_exit(&mcmutex); 2229 2230 if (mcp->mc_status & (MC_SOFT_SUSPENDED | MC_DRIVER_SUSPENDED)) { 2231 mutex_exit(&mcp->mc_lock); 2232 MC_LOG("mc-opl has been suspended. No error injection.\n"); 2233 return (EBUSY); 2234 } 2235 2236 /* convert pa to offset within the board */ 2237 MC_LOG("pa %lx, offset %lx\n", pa, pa - mcp->mc_start_address); 2238 2239 if (!pa_is_valid(mcp, pa)) { 2240 mutex_exit(&mcp->mc_lock); 2241 return (EINVAL); 2242 } 2243 2244 pa0 = pa - mcp->mc_start_address; 2245 2246 bank = pa_to_bank(mcp, pa0); 2247 2248 if (flags & MC_INJECT_FLAG_OTHER) 2249 bank = bank ^ 1; 2250 2251 if (MC_INJECT_MIRROR(error_type) && !IS_MIRROR(mcp, bank)) { 2252 mutex_exit(&mcp->mc_lock); 2253 MC_LOG("Not mirror mode\n"); 2254 return (EINVAL); 2255 } 2256 2257 dimm_addr = pa_to_dimm(mcp, pa0); 2258 2259 MC_LOG("injecting error to /LSB%d/B%d/D%x\n", 2260 mcp->mc_board_num, bank, dimm_addr); 2261 2262 2263 switch (error_type) { 2264 case MC_INJECT_INTERMITTENT_MCE: 2265 case MC_INJECT_PERMANENT_MCE: 2266 case MC_INJECT_MUE: 2267 both_sides = 1; 2268 } 2269 2270 if (flags & MC_INJECT_FLAG_RESET) 2271 ST_MAC_REG(MAC_EG_CNTL(mcp, bank), 0); 2272 2273 ST_MAC_REG(MAC_EG_ADD(mcp, bank), dimm_addr & MAC_EG_ADD_MASK); 2274 2275 if (both_sides) { 2276 ST_MAC_REG(MAC_EG_CNTL(mcp, bank^1), 0); 2277 ST_MAC_REG(MAC_EG_ADD(mcp, bank^1), 2278 dimm_addr & MAC_EG_ADD_MASK); 2279 } 2280 2281 switch (error_type) { 2282 case MC_INJECT_UE: 2283 case MC_INJECT_SUE: 2284 case MC_INJECT_MUE: 2285 if (flags & MC_INJECT_FLAG_PATH) { 2286 cntl = MAC_EG_ADD_FIX 2287 |MAC_EG_FORCE_READ00|MAC_EG_FORCE_READ16 2288 |MAC_EG_DERR_ONCE; 2289 } else { 2290 cntl = MAC_EG_ADD_FIX|MAC_EG_FORCE_DERR00 2291 |MAC_EG_FORCE_DERR16|MAC_EG_DERR_ONCE; 2292 } 2293 flags |= MC_INJECT_FLAG_ST; 2294 break; 2295 case MC_INJECT_INTERMITTENT_CE: 2296 case MC_INJECT_INTERMITTENT_MCE: 2297 if (flags & MC_INJECT_FLAG_PATH) { 2298 cntl = MAC_EG_ADD_FIX 2299 |MAC_EG_FORCE_READ00 2300 |MAC_EG_DERR_ONCE; 2301 } else { 2302 cntl = MAC_EG_ADD_FIX 2303 |MAC_EG_FORCE_DERR16 2304 |MAC_EG_DERR_ONCE; 2305 } 2306 flags |= MC_INJECT_FLAG_ST; 2307 break; 2308 case MC_INJECT_PERMANENT_CE: 2309 case MC_INJECT_PERMANENT_MCE: 2310 if (flags & MC_INJECT_FLAG_PATH) { 2311 cntl = MAC_EG_ADD_FIX 2312 |MAC_EG_FORCE_READ00 2313 |MAC_EG_DERR_ALWAYS; 2314 } else { 2315 cntl = MAC_EG_ADD_FIX 2316 |MAC_EG_FORCE_DERR16 2317 |MAC_EG_DERR_ALWAYS; 2318 } 2319 flags |= MC_INJECT_FLAG_ST; 2320 break; 2321 case MC_INJECT_CMPE: 2322 data = 0xabcdefab; 2323 stphys(pa, data); 2324 cpu_flush_ecache(); 2325 MC_LOG("CMPE: writing data %x to %lx\n", data, pa); 2326 ST_MAC_REG(MAC_MIRR(mcp, bank), MAC_MIRR_BANK_EXCLUSIVE); 2327 stphys(pa, data ^ 0xffffffff); 2328 cpu_flush_ecache(); 2329 ST_MAC_REG(MAC_MIRR(mcp, bank), 0); 2330 MC_LOG("CMPE: write new data %xto %lx\n", data, pa); 2331 cntl = 0; 2332 break; 2333 case MC_INJECT_NOP: 2334 cntl = 0; 2335 break; 2336 default: 2337 MC_LOG("mc_inject_error: invalid option\n"); 2338 cntl = 0; 2339 } 2340 2341 if (cntl) { 2342 ST_MAC_REG(MAC_EG_CNTL(mcp, bank), cntl & MAC_EG_SETUP_MASK); 2343 ST_MAC_REG(MAC_EG_CNTL(mcp, bank), cntl); 2344 2345 if (both_sides) { 2346 ST_MAC_REG(MAC_EG_CNTL(mcp, bank^1), cntl & 2347 MAC_EG_SETUP_MASK); 2348 ST_MAC_REG(MAC_EG_CNTL(mcp, bank^1), cntl); 2349 } 2350 } 2351 2352 /* 2353 * For all injection cases except compare error, we 2354 * must write to the PA to trigger the error. 2355 */ 2356 2357 if (flags & MC_INJECT_FLAG_ST) { 2358 data = 0xf0e0d0c0; 2359 MC_LOG("Writing %x to %lx\n", data, pa); 2360 stphys(pa, data); 2361 cpu_flush_ecache(); 2362 } 2363 2364 delay(inject_op_delay * drv_usectohz(1000 * 1000)); 2365 2366 2367 if (flags & MC_INJECT_FLAG_LD) { 2368 if (flags & MC_INJECT_FLAG_NO_TRAP) { 2369 if (on_trap(&otd, OT_DATA_EC)) { 2370 no_trap(); 2371 MC_LOG("Trap occurred\n"); 2372 } else { 2373 MC_LOG("On-trap Reading from %lx\n", pa); 2374 data = ldphys(pa); 2375 no_trap(); 2376 MC_LOG("data = %x\n", data); 2377 } 2378 } else { 2379 MC_LOG("Reading from %lx\n", pa); 2380 data = ldphys(pa); 2381 MC_LOG("data = %x\n", data); 2382 } 2383 } 2384 2385 if (flags & MC_INJECT_FLAG_RESTART) { 2386 delay(inject_op_delay * drv_usectohz(1000 * 1000)); 2387 2388 MC_LOG("Restart patrol\n"); 2389 if (mc_stop(mcp, bank)) { 2390 cmn_err(CE_WARN, "Cannot stop Memory Patrol at " 2391 "/LSB%d/B%d\n", mcp->mc_board_num, bank); 2392 mutex_exit(&mcp->mc_lock); 2393 return (EIO); 2394 } 2395 maddr.mi_maddr.ma_bd = mcp->mc_board_num; 2396 maddr.mi_maddr.ma_bank = bank; 2397 maddr.mi_maddr.ma_dimm_addr = dimm_addr; 2398 maddr.mi_valid = 1; 2399 maddr.mi_advance = 0; 2400 restart_patrol(mcp, bank, &maddr); 2401 } 2402 2403 if (flags & MC_INJECT_FLAG_POLL) { 2404 delay(inject_op_delay * drv_usectohz(1000 * 1000)); 2405 2406 MC_LOG("Poll patrol error\n"); 2407 stat = LD_MAC_REG(MAC_PTRL_STAT(mcp, bank)); 2408 cntl = LD_MAC_REG(MAC_PTRL_CNTL(mcp, bank)); 2409 if (stat & (MAC_STAT_PTRL_ERRS|MAC_STAT_MI_ERRS)) { 2410 maddr.mi_valid = 0; 2411 maddr.mi_advance = 1; 2412 if (IS_MIRROR(mcp, bank)) 2413 mc_error_handler_mir(mcp, bank, 2414 &maddr); 2415 else 2416 mc_error_handler(mcp, bank, &maddr); 2417 2418 restart_patrol(mcp, bank, &maddr); 2419 } else 2420 restart_patrol(mcp, bank, NULL); 2421 } 2422 2423 mutex_exit(&mcp->mc_lock); 2424 return (0); 2425 } 2426 2427 void 2428 mc_stphysio(uint64_t pa, uint32_t data) 2429 { 2430 MC_LOG("0x%x -> pa(%lx)\n", data, pa); 2431 stphysio(pa, data); 2432 } 2433 2434 uint32_t 2435 mc_ldphysio(uint64_t pa) 2436 { 2437 uint32_t rv; 2438 2439 rv = ldphysio(pa); 2440 MC_LOG("pa(%lx) = 0x%x\n", pa, rv); 2441 return (rv); 2442 } 2443