#include <sys/param.h>
#include <sys/systm.h>
#include <sys/types.h>
#include <sys/mutex.h>
#include <machine/db_machdep.h>
struct db_mutex ddb_mp_mutex = DB_MUTEX_INITIALIZER;
volatile int ddb_state = DDB_STATE_NOT_RUNNING;
volatile cpuid_t ddb_active_cpu;
extern volatile int db_switch_cpu;
extern volatile long db_switch_to_cpu;
int
db_enter_ddb(void)
{
int i;
db_mtx_enter(&ddb_mp_mutex);
if (ddb_state == DDB_STATE_NOT_RUNNING) {
ddb_active_cpu = cpu_number();
ddb_state = DDB_STATE_RUNNING;
curcpu()->ci_ddb_paused = CI_DDB_INDDB;
db_mtx_leave(&ddb_mp_mutex);
for (i = 0; i < MAXCPUS; i++) {
if (cpu_info[i] != NULL && i != cpu_number() &&
cpu_info[i]->ci_ddb_paused != CI_DDB_STOPPED) {
cpu_info[i]->ci_ddb_paused = CI_DDB_SHOULDSTOP;
i386_send_ipi(cpu_info[i], I386_IPI_DDB);
}
}
return (1);
}
if (ddb_active_cpu == cpu_number() && ddb_state == DDB_STATE_EXITING) {
for (i = 0; i < MAXCPUS; i++) {
if (cpu_info[i] != NULL) {
cpu_info[i]->ci_ddb_paused = CI_DDB_RUNNING;
}
}
db_mtx_leave(&ddb_mp_mutex);
return (0);
}
if (ddb_active_cpu == cpu_number() && db_switch_cpu) {
curcpu()->ci_ddb_paused = CI_DDB_SHOULDSTOP;
db_switch_cpu = 0;
ddb_active_cpu = db_switch_to_cpu;
cpu_info[db_switch_to_cpu]->ci_ddb_paused = CI_DDB_ENTERDDB;
}
while (ddb_active_cpu != cpu_number() &&
curcpu()->ci_ddb_paused != CI_DDB_RUNNING) {
if (curcpu()->ci_ddb_paused == CI_DDB_SHOULDSTOP)
curcpu()->ci_ddb_paused = CI_DDB_STOPPED;
db_mtx_leave(&ddb_mp_mutex);
while (ddb_active_cpu != cpu_number() &&
curcpu()->ci_ddb_paused != CI_DDB_RUNNING)
;
db_mtx_enter(&ddb_mp_mutex);
}
if (ddb_active_cpu == cpu_number() && ddb_state == DDB_STATE_RUNNING) {
curcpu()->ci_ddb_paused = CI_DDB_INDDB;
db_mtx_leave(&ddb_mp_mutex);
return (1);
} else {
db_mtx_leave(&ddb_mp_mutex);
return (0);
}
}
void
db_startcpu(int cpu)
{
if (cpu != cpu_number() && cpu_info[cpu] != NULL) {
db_mtx_enter(&ddb_mp_mutex);
cpu_info[cpu]->ci_ddb_paused = CI_DDB_RUNNING;
db_mtx_leave(&ddb_mp_mutex);
}
}
void
db_stopcpu(int cpu)
{
db_mtx_enter(&ddb_mp_mutex);
if (cpu != cpu_number() && cpu_info[cpu] != NULL &&
cpu_info[cpu]->ci_ddb_paused != CI_DDB_STOPPED) {
cpu_info[cpu]->ci_ddb_paused = CI_DDB_SHOULDSTOP;
db_mtx_leave(&ddb_mp_mutex);
i386_send_ipi(cpu_info[cpu], I386_IPI_DDB);
} else {
db_mtx_leave(&ddb_mp_mutex);
}
}
void
i386_ipi_db(struct cpu_info *ci)
{
db_enter();
}