/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 2007 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ #pragma ident "%Z%%M% %I% %E% SMI" #include <sys/cpuvar.h> #include <sys/kdi_impl.h> #include <sys/reboot.h> #include <sys/errno.h> #include <sys/atomic.h> #include <sys/kmem.h> kdi_debugvec_t *kdi_dvec; struct modctl *kdi_dmods; static kdi_dtrace_state_t kdi_dtrace_state = KDI_DTSTATE_IDLE; void kdi_dvec_vmready(void) { kdi_dvec->dv_kctl_vmready(); kdi_dvec->dv_vmready(); } void kdi_dvec_memavail(void) { /* * The driver will allocate more memory (if requested), and will call * dv_memavail itself. */ kdi_dvec->dv_kctl_memavail(); } #if defined(__x86) void kdi_dvec_handle_fault(greg_t trapno, greg_t pc, greg_t sp, int cpuid) { kdi_dvec->dv_handle_fault(trapno, pc, sp, cpuid); } #endif #if defined(__sparc) /* * Called on the CPU being initialized */ void kdi_dvec_cpu_init(struct cpu *cp) { kdi_dvec->dv_kctl_cpu_init(); kdi_dvec->dv_cpu_init(cp); } void kdi_dvec_cpr_restart(void) { kdi_dvec->dv_kctl_cpu_init(); kdi_dvec->dv_cpr_restart(); } #endif /* __sparc */ void kdi_dvec_modavail(void) { kdi_dvec->dv_kctl_modavail(); } void kdi_dvec_thravail(void) { kdi_dvec->dv_kctl_thravail(); } void kdi_dvec_mod_loaded(struct modctl *modp) { kdi_dvec->dv_mod_loaded(modp); } void kdi_dvec_mod_unloading(struct modctl *modp) { kdi_dvec->dv_mod_unloading(modp); } kdi_dtrace_state_t kdi_dtrace_get_state(void) { return (kdi_dtrace_state); } int kdi_dtrace_set(kdi_dtrace_set_t transition) { kdi_dtrace_state_t new, cur; do { cur = kdi_dtrace_state; switch (transition) { case KDI_DTSET_DTRACE_ACTIVATE: if (cur == KDI_DTSTATE_KMDB_BPT_ACTIVE) return (EBUSY); if (cur == KDI_DTSTATE_DTRACE_ACTIVE) return (0); new = KDI_DTSTATE_DTRACE_ACTIVE; break; case KDI_DTSET_DTRACE_DEACTIVATE: if (cur == KDI_DTSTATE_KMDB_BPT_ACTIVE) return (EBUSY); if (cur == KDI_DTSTATE_IDLE) return (0); new = KDI_DTSTATE_IDLE; break; case KDI_DTSET_KMDB_BPT_ACTIVATE: if (cur == KDI_DTSTATE_DTRACE_ACTIVE) return (EBUSY); if (cur == KDI_DTSTATE_KMDB_BPT_ACTIVE) return (0); new = KDI_DTSTATE_KMDB_BPT_ACTIVE; break; case KDI_DTSET_KMDB_BPT_DEACTIVATE: if (cur == KDI_DTSTATE_DTRACE_ACTIVE) return (EBUSY); if (cur == KDI_DTSTATE_IDLE) return (0); new = KDI_DTSTATE_IDLE; break; default: return (EINVAL); } } while (cas32((uint_t *)&kdi_dtrace_state, cur, new) != cur); return (0); }