Package uk.co.cwspencer.ideagdb.debug

Source Code of uk.co.cwspencer.ideagdb.debug.GdbExecutionStackFrame

package uk.co.cwspencer.ideagdb.debug;

import com.intellij.icons.AllIcons;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.openapi.vfs.LocalFileSystem;
import com.intellij.openapi.vfs.VirtualFile;
import com.intellij.ui.SimpleColoredComponent;
import com.intellij.ui.SimpleTextAttributes;
import com.intellij.xdebugger.XDebuggerBundle;
import com.intellij.xdebugger.XDebuggerUtil;
import com.intellij.xdebugger.XSourcePosition;
import com.intellij.xdebugger.evaluation.XDebuggerEvaluator;
import com.intellij.xdebugger.frame.XCompositeNode;
import com.intellij.xdebugger.frame.XStackFrame;
import com.intellij.xdebugger.frame.XValueChildrenList;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import uk.co.cwspencer.gdb.Gdb;
import uk.co.cwspencer.gdb.messages.*;

import java.io.File;

/**
* Class for providing information about a stack frame.
*/
public class GdbExecutionStackFrame extends XStackFrame {
    private static final Logger m_log =
            Logger.getInstance("#uk.co.cwspencer.ideagdb.debug.GdbExecutionStackFrame");

    // The GDB instance
    private Gdb m_gdb;

    // The thread the frame is in
    private int m_thread;

    // The GDB stack frame
    private GdbStackFrame m_frame;
    private int m_frameNo;

    // The expression evaluator
    private GdbEvaluator m_evaluator;

    /**
     * Constructor.
     *
     * @param gdb    Handle to the GDB instance.
     * @param thread The thread the frame is in.
     * @param frame  The GDB stack frame to wrap.
     */
    public GdbExecutionStackFrame(Gdb gdb, int thread, GdbStackFrame frame) {
        m_gdb = gdb;
        m_thread = thread;
        m_frame = frame;

        // The top frame doesn't have a level set
        m_frameNo = m_frame.level == null ? 0 : m_frame.level;
    }

    /**
     * Returns an object that can be used to determine if two stack frames after a debugger step
     * are the same.
     *
     * @return The equality object.
     */
    @Nullable
    @Override
    public Object getEqualityObject() {
        // TODO: This would be better if we could actually determine if two frames represent the
        // same position, but it doesn't really matter, and this is good enough to stop the debugger
        // from collapsing variable trees when we step the debugger
        return GdbExecutionStackFrame.class;
    }

    /**
     * Returns an expression evaluator in the context of this stack frame.
     */
    @Nullable
    @Override
    public XDebuggerEvaluator getEvaluator() {
        if (m_evaluator == null) {
            m_evaluator = new GdbEvaluator(m_gdb, m_thread, m_frameNo);
        }
        return m_evaluator;
    }

    /**
     * Gets the source position of the stack frame, if available.
     *
     * @return The source position, or null if it is not available.
     */
    @Nullable
    @Override
    public XSourcePosition getSourcePosition() {
        if (m_frame.fileAbsolute == null || m_frame.line == null) {
            return null;
        }

        String path = m_frame.fileAbsolute.replace(File.separatorChar, '/');
        VirtualFile file = LocalFileSystem.getInstance().findFileByPath(path);
        if (file == null) {
            return null;
        }

        return XDebuggerUtil.getInstance().createPosition(file, m_frame.line - 1);
    }

    /**
     * Controls the presentation of the frame in the stack trace.
     *
     * @param component The stack frame visual component.
     */
    public void customizePresentation(SimpleColoredComponent component) {
        if (m_frame.address == null) {
            component.append(XDebuggerBundle.message("invalid.frame"),
                    SimpleTextAttributes.ERROR_ATTRIBUTES);
            return;
        }

        // Format the frame information
        XSourcePosition sourcePosition = getSourcePosition();
        if (m_frame.function != null) {
            // Strip any arguments from the function name
            String function = m_frame.function;
            int parenIndex = function.indexOf('(');
            if (parenIndex != -1) {
                function = function.substring(0, parenIndex);
            }

            if (sourcePosition != null) {
                component.append(function + "():" + (sourcePosition.getLine() + 1),
                        SimpleTextAttributes.REGULAR_ATTRIBUTES);
                component.append(" (" + sourcePosition.getFile().getName() + ")",
                        SimpleTextAttributes.GRAY_ITALIC_ATTRIBUTES);
            } else {
                component.append(function + "()", SimpleTextAttributes.GRAY_ATTRIBUTES);
                component.append(" (", SimpleTextAttributes.GRAY_ITALIC_ATTRIBUTES);
                if (m_frame.module != null) {
                    component.append(m_frame.module + ":",
                            SimpleTextAttributes.GRAY_ITALIC_ATTRIBUTES);
                }
                component.append("0x" + Long.toHexString(m_frame.address) + ")",
                        SimpleTextAttributes.GRAY_ITALIC_ATTRIBUTES);
            }
        } else if (sourcePosition != null) {
            component.append(
                    sourcePosition.getFile().getName() + ":" + (sourcePosition.getLine() + 1),
                    SimpleTextAttributes.REGULAR_ATTRIBUTES);
        } else {
            String addressStr = "0x" + Long.toHexString(m_frame.address);
            component.append(addressStr, SimpleTextAttributes.GRAY_ITALIC_ATTRIBUTES);
            component.appendFixedTextFragmentWidth(addressStr.length());
        }
        component.setIcon(AllIcons.Debugger.StackFrame);
    }

    /**
     * Gets the variables available on this frame. This passes the request and returns immediately;
     * the data is supplied to node asynchronously.
     *
     * @param node The node into which the variables are inserted.
     */
    @Override
    public void computeChildren(@NotNull final XCompositeNode node) {
        // TODO: This can be called multiple times if the user changes the value of a variable. We
        // shouldn't really call -stack-list-variables more than once in this case (i.e., only call
        // -var-update after the first call)
        m_gdb.getVariablesForFrame(m_thread, m_frameNo, new Gdb.GdbEventCallback() {
            @Override
            public void onGdbCommandCompleted(GdbEvent event) {
                onGdbVariablesReady(event, node);
            }
        });
    }

    /**
     * Callback function for when GDB has responded to our stack variables request.
     *
     * @param event The event.
     * @param node  The node passed to computeChildren().
     */
    private void onGdbVariablesReady(GdbEvent event, final XCompositeNode node) {
        if (event instanceof GdbErrorEvent) {
            node.setErrorMessage(((GdbErrorEvent) event).message);
            return;
        }
        if (!(event instanceof GdbVariableObjects)) {
            node.setErrorMessage("Unexpected data received from GDB");
            m_log.warn("Unexpected event " + event + " received from variable objects request");
            return;
        }

        // Inspect the data
        GdbVariableObjects variables = (GdbVariableObjects) event;
        if (variables.objects == null || variables.objects.isEmpty()) {
            // No data
            node.addChildren(XValueChildrenList.EMPTY, true);
        }

        // Build a XValueChildrenList
        XValueChildrenList children = new XValueChildrenList(variables.objects.size());
        for (GdbVariableObject variable : variables.objects) {
            children.add(variable.expression, new GdbValue(m_gdb, variable));
        }
        node.addChildren(children, true);
    }
}
TOP

Related Classes of uk.co.cwspencer.ideagdb.debug.GdbExecutionStackFrame

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.