Package processing.mode.experimental

Source Code of processing.mode.experimental.LineBreakpoint

/*
* Copyright (C) 2012 Martin Leopold <m@martinleopold.com>
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
*/
package processing.mode.experimental;

import com.sun.jdi.AbsentInformationException;
import com.sun.jdi.Location;
import com.sun.jdi.ReferenceType;
import com.sun.jdi.request.BreakpointRequest;
import java.util.List;
import java.util.logging.Level;
import java.util.logging.Logger;

import static processing.mode.experimental.ExperimentalMode.log;
import static processing.mode.experimental.ExperimentalMode.logE;
import static processing.mode.experimental.ExperimentalMode.log2;

/**
* Model/Controller of a line breakpoint. Can be set before or while debugging.
* Adds a highlight using the debuggers view ({@link DebugEditor}).
*
* @author Martin Leopold <m@martinleopold.com>
*/
public class LineBreakpoint implements ClassLoadListener {

    protected Debugger dbg; // the debugger
    protected LineID line; // the line this breakpoint is set on
    protected BreakpointRequest bpr; // the request on the VM's event request manager
    protected ReferenceType theClass; // the class containing this breakpoint, null when not yet loaded

    /**
     * Create a {@link LineBreakpoint}. If in a debug session, will try to
     * immediately set the breakpoint. If not in a debug session or the
     * corresponding class is not yet loaded the breakpoint will activate on
     * class load.
     *
     * @param line the line id to create the breakpoint on
     * @param dbg the {@link Debugger}
     */
    public LineBreakpoint(LineID line, Debugger dbg) {
        this.line = line;
        line.startTracking(dbg.editor().getTab(line.fileName()).getDocument());
        this.dbg = dbg;
        theClass = dbg.getClass(className()); // try to get the class immediately, may return null if not yet loaded
        set(); // activate the breakpoint (show highlight, attach if debugger is running)
        Logger.getLogger(LineBreakpoint.class.getName()).log(Level.INFO, "LBP Created " +toString() + " class: " + className(), new Object[]{});
    }

    /**
     * Create a {@link LineBreakpoint} on a line in the current tab.
     *
     * @param lineIdx the line index of the current tab to create the breakpoint
     * on
     * @param dbg the {@link Debugger}
     */
    // TODO: remove and replace by {@link #LineBreakpoint(LineID line, Debugger dbg)}
    public LineBreakpoint(int lineIdx, Debugger dbg) {
        this(dbg.editor().getLineIDInCurrentTab(lineIdx), dbg);
    }

    /**
     * Get the line id this breakpoint is on.
     *
     * @return the line id
     */
    public LineID lineID() {
        return line;
    }

    /**
     * Test if this breakpoint is on a certain line.
     *
     * @param testLine the line id to test
     * @return true if this breakpoint is on the given line
     */
    public boolean isOnLine(LineID testLine) {
        return line.equals(testLine);
    }

    /**
     * Attach this breakpoint to the VM. Creates and enables a
     * {@link BreakpointRequest}. VM needs to be paused.
     */
    protected void attach() {
        if (!dbg.isPaused()) {
            Logger.getLogger(LineBreakpoint.class.getName()).log(Level.WARNING, "can't attach breakpoint, debugger not paused");
            return;
        }

        if (theClass == null) {
            Logger.getLogger(LineBreakpoint.class.getName()).log(Level.WARNING, "can't attach breakpoint, class not loaded: {0}", className());
            return;
        }

        // find line in java space
        LineID javaLine = dbg.sketchToJavaLine(line);
        if (javaLine == null) {
            Logger.getLogger(LineBreakpoint.class.getName()).log(Level.WARNING, "couldn't find line {0} in the java code", line);
            return;
        }
        try {
            Logger.getLogger(LineBreakpoint.class.getName()).log(Level.WARNING, "BPs of class: {0} , line " + (javaLine.lineIdx() + 1), new Object[]{theClass});
            List<Location> locations = theClass.locationsOfLine(javaLine.lineIdx() + 1);
            if (locations.isEmpty()) {
                Logger.getLogger(LineBreakpoint.class.getName()).log(Level.WARNING, "no location found for line {0} -> {1}", new Object[]{line, javaLine});
                return;
            }
            // use first found location
            bpr = dbg.vm().eventRequestManager().createBreakpointRequest(locations.get(0));
            bpr.enable();
            Logger.getLogger(LineBreakpoint.class.getName()).log(Level.INFO, "attached breakpoint to {0} -> {1}", new Object[]{line, javaLine});
        } catch (AbsentInformationException ex) {
            Logger.getLogger(Debugger.class.getName()).log(Level.SEVERE, null, ex);
        }
    }

    /**
     * Detach this breakpoint from the VM. Deletes the
     * {@link BreakpointRequest}.
     */
    protected void detach() {
        if (bpr != null) {
            dbg.vm().eventRequestManager().deleteEventRequest(bpr);
            bpr = null;
        }
    }

    /**
     * Set this breakpoint. Adds the line highlight. If Debugger is paused also
     * attaches the breakpoint by calling {@link #attach()}.
     */
    protected void set() {
        dbg.addClassLoadListener(this); // class may not yet be loaded
        dbg.editor().addBreakpointedLine(line);
        if (theClass != null && dbg.isPaused()) { // class is loaded
            // immediately activate the breakpoint
            attach();
        }
        if (dbg.editor().isInCurrentTab(line)) {
            dbg.editor().getSketch().setModified(true);
        }
    }

    /**
     * Remove this breakpoint. Clears the highlight and detaches the breakpoint
     * if the debugger is paused.
     */
    public void remove() {
        dbg.removeClassLoadListener(this);
        //System.out.println("removing " + line.lineIdx());
        dbg.editor().removeBreakpointedLine(line.lineIdx());
        if (dbg.isPaused()) {
            // immediately remove the breakpoint
            detach();
        }
        line.stopTracking();
        if (dbg.editor().isInCurrentTab(line)) {
            dbg.editor().getSketch().setModified(true);
        }
    }

//    public void enable() {
//    }
//
//    public void disable() {
//    }
    @Override
    public String toString() {
        return line.toString();
    }

    /**
     * Get the name of the class this breakpoint belongs to. Needed for fetching
     * the right location to create a breakpoint request.
     *
     * @return the class name
     */
    protected String className() {
        if (line.fileName().endsWith(".pde")) {
            // standard tab
            ReferenceType mainClass = dbg.getMainClass();
            //System.out.println(dbg.getMainClass().name());
            if (mainClass == null) {
                return null;
            }
            return dbg.getMainClass().name();
        }

        if (line.fileName().endsWith(".java")) {
            // pure java tab
            return line.fileName().substring(0, line.fileName().lastIndexOf(".java"));
        }

        return null;
    }

    /**
     * Event handler called when a class is loaded in the debugger. Causes the
     * breakpoint to be attached, if its class was loaded.
     *
     * @param theClass the class that was just loaded.
     */
    @Override
    public void classLoaded(ReferenceType theClass) {
        // check if our class is being loaded
        log("Class Loaded: " + theClass.name());
        if (theClass.name().equals(className())) {
            this.theClass = theClass;
            attach();
        }
        for (ReferenceType ct : theClass.nestedTypes()) {
          log("Nested " + ct.name());
        }
    }
}
TOP

Related Classes of processing.mode.experimental.LineBreakpoint

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.