view usr/src/cmd/mdb/common/mdb/mdb_frame.c @ 14167:7ac6fb740bcf

3946 ::gcore (fix sparc build)
author Christopher Siden <chris.siden@delphix.com>
date Tue, 27 Aug 2013 10:51:34 -0800
parents 153d1be61d60
children
line wrap: on
line source

/*
 * 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 2006 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */

#pragma ident	"%Z%%M%	%I%	%E% SMI"

/*
 * Utility routines to manage debugger frames and commands.  A debugger frame
 * is used by each invocation of mdb_run() (the main parsing loop) to manage
 * its state.  Refer to the comments in mdb.c for more information on frames.
 * Each frame has a list of commands (that is, a dcmd, argument list, and
 * optional address list) that represent a pipeline after it has been parsed.
 */

#include <mdb/mdb_debug.h>
#include <mdb/mdb_frame.h>
#include <mdb/mdb_modapi.h>
#include <mdb/mdb_err.h>
#include <mdb/mdb_lex.h>
#include <mdb/mdb_io.h>
#include <mdb/mdb.h>

mdb_cmd_t *
mdb_cmd_create(mdb_idcmd_t *idcp, mdb_argvec_t *argv)
{
	mdb_cmd_t *cp = mdb_zalloc(sizeof (mdb_cmd_t), UM_NOSLEEP);

	if (cp == NULL) {
		warn("failed to allocate memory for command");
		longjmp(mdb.m_frame->f_pcb, MDB_ERR_NOMEM);
	}

	mdb_list_append(&mdb.m_frame->f_cmds, cp);
	mdb_argvec_copy(&cp->c_argv, argv);
	mdb_argvec_zero(argv);
	cp->c_dcmd = idcp;

	return (cp);
}

void
mdb_cmd_destroy(mdb_cmd_t *cp)
{
	mdb_addrvec_destroy(&cp->c_addrv);
	mdb_argvec_destroy(&cp->c_argv);
	mdb_vcb_purge(cp->c_vcbs);
	mdb_free(cp, sizeof (mdb_cmd_t));
}

void
mdb_cmd_reset(mdb_cmd_t *cp)
{
	mdb_addrvec_destroy(&cp->c_addrv);
	mdb_vcb_purge(cp->c_vcbs);
	cp->c_vcbs = NULL;
}

void
mdb_frame_reset(mdb_frame_t *fp)
{
	mdb_cmd_t *cp;

	while ((cp = mdb_list_next(&fp->f_cmds)) != NULL) {
		mdb_list_delete(&fp->f_cmds, cp);
		mdb_cmd_destroy(cp);
	}
	fp->f_cp = NULL;
	fp->f_pcmd = NULL;

	while (mdb_iob_stack_size(&fp->f_ostk) != 0) {
		mdb_iob_destroy(mdb.m_out);
		mdb.m_out = mdb_iob_stack_pop(&fp->f_ostk);
	}

	mdb_wcb_purge(&fp->f_wcbs);
	mdb_recycle(&fp->f_mblks);
}

void
mdb_frame_push(mdb_frame_t *fp)
{
	mdb_intr_disable();

	if (mdb.m_fmark == NULL)
		mdb.m_fmark = fp;

	mdb_lex_state_save(mdb.m_frame->f_lstate);

	bzero(fp, sizeof (mdb_frame_t));
	mdb_lex_state_create(fp);
	mdb_list_append(&mdb.m_flist, fp);

	fp->f_flags = mdb.m_flags & MDB_FL_VOLATILE;
	fp->f_pcmd = mdb.m_frame->f_pcmd;
	fp->f_id = mdb.m_fid++;
	mdb.m_frame->f_dot = mdb_nv_get_value(mdb.m_dot);

	mdb.m_frame = fp;
	mdb.m_depth++;

	mdb_dprintf(MDB_DBG_DSTK, "push frame <%u> mark=%p in=%s out=%s\n",
	    fp->f_id, (void *)mdb.m_fmark,
	    mdb_iob_name(mdb.m_in), mdb_iob_name(mdb.m_out));

	mdb_intr_enable();
}

void
mdb_frame_pop(mdb_frame_t *fp, int err)
{
	mdb_intr_disable();

	ASSERT(mdb_iob_stack_size(&fp->f_istk) == 0);
	ASSERT(mdb_iob_stack_size(&fp->f_ostk) == 0);
	ASSERT(mdb_list_next(&fp->f_cmds) == NULL);
	ASSERT(fp->f_mblks == NULL);
	ASSERT(fp->f_wcbs == NULL);

	mdb_dprintf(MDB_DBG_DSTK, "pop frame <%u> status=%s\n",
	    fp->f_id, mdb_err2str(err));

	if (mdb.m_frame == fp) {
		mdb.m_flags &= ~MDB_FL_VOLATILE;
		mdb.m_flags |= fp->f_flags;
		mdb_frame_switch(mdb_list_prev(fp));
	}

	if (mdb.m_fmark == fp)
		mdb.m_fmark = NULL;

	mdb_lex_state_destroy(fp);

	mdb_list_delete(&mdb.m_flist, fp);
	ASSERT(mdb.m_depth != 0);
	mdb.m_depth--;

	mdb_intr_enable();
}

void
mdb_frame_switch(mdb_frame_t *frame)
{
	mdb_lex_state_save(mdb.m_frame->f_lstate);
	mdb.m_frame->f_dot = mdb_nv_get_value(mdb.m_dot);
	mdb.m_frame = frame;
	mdb_lex_state_restore(mdb.m_frame->f_lstate);
	mdb_dprintf(MDB_DBG_DSTK, "switch to frame <%u>\n", mdb.m_frame->f_id);

	mdb_nv_set_value(mdb.m_dot, frame->f_dot);
}

void
mdb_frame_set_pipe(mdb_frame_t *frame)
{
	frame->pipe = TRUE;
}

void
mdb_frame_clear_pipe(mdb_frame_t *frame)
{
	frame->pipe = FALSE;
}

mdb_frame_t *
mdb_frame_pipe(void)
{
	mdb_frame_t *frame = mdb_list_prev(mdb.m_frame);

	while (frame && frame->pipe == FALSE)
		frame = mdb_list_prev(frame);

	return (frame);
}