| /******************************************************************************* |
| * Copyright (c) 2005 The Regents of the University of California. |
| * This material was produced under U.S. Government contract W-7405-ENG-36 |
| * for Los Alamos National Laboratory, which is operated by the University |
| * of California for the U.S. Department of Energy. The U.S. Government has |
| * rights to use, reproduce, and distribute this software. NEITHER THE |
| * GOVERNMENT NOR THE UNIVERSITY MAKES ANY WARRANTY, EXPRESS OR IMPLIED, OR |
| * ASSUMES ANY LIABILITY FOR THE USE OF THIS SOFTWARE. If software is modified |
| * to produce derivative works, such modified software should be clearly marked, |
| * so as not to confuse it with the version available from LANL. |
| * |
| * Additionally, this program and the accompanying materials |
| * are made available under the terms of the Eclipse Public License v1.0 |
| * which accompanies this distribution, and is available at |
| * http://www.eclipse.org/legal/epl-v10.html |
| * |
| * LA-CC 04-115 |
| *******************************************************************************/ |
| package org.eclipse.ptp.debug.core; |
| |
| import java.util.ArrayList; |
| import java.util.HashMap; |
| import java.util.List; |
| |
| import org.eclipse.cdt.core.IBinaryParser.IBinaryObject; |
| import org.eclipse.core.resources.IMarker; |
| import org.eclipse.core.resources.IResource; |
| import org.eclipse.core.runtime.CoreException; |
| import org.eclipse.core.runtime.IPath; |
| import org.eclipse.core.runtime.IProgressMonitor; |
| import org.eclipse.core.runtime.IStatus; |
| import org.eclipse.core.runtime.Path; |
| import org.eclipse.core.runtime.Status; |
| import org.eclipse.core.runtime.jobs.Job; |
| import org.eclipse.debug.core.DebugException; |
| import org.eclipse.debug.core.DebugPlugin; |
| import org.eclipse.debug.core.model.IBreakpoint; |
| import org.eclipse.ptp.core.PreferenceConstants; |
| import org.eclipse.ptp.core.elements.IPJob; |
| import org.eclipse.ptp.core.util.BitList; |
| import org.eclipse.ptp.debug.core.cdi.IPCDISession; |
| import org.eclipse.ptp.debug.core.cdi.model.IPCDITarget; |
| import org.eclipse.ptp.debug.core.cdi.model.IPCDITargetConfiguration; |
| import org.eclipse.ptp.debug.core.events.IPDebugEvent; |
| import org.eclipse.ptp.debug.core.events.IPDebugInfo; |
| import org.eclipse.ptp.debug.core.events.PDebugEvent; |
| import org.eclipse.ptp.debug.core.events.PDebugInfo; |
| import org.eclipse.ptp.debug.core.events.PDebugRegisterInfo; |
| import org.eclipse.ptp.debug.core.launch.IPLaunch; |
| import org.eclipse.ptp.debug.core.model.IPBreakpoint; |
| import org.eclipse.ptp.debug.core.model.IPDebugTarget; |
| import org.eclipse.ptp.debug.core.model.IPLineBreakpoint; |
| import org.eclipse.ptp.debug.internal.core.breakpoint.PLineBreakpoint; |
| import org.eclipse.ptp.debug.internal.core.model.PDebugTarget; |
| import org.eclipse.ptp.debug.internal.core.model.PSession; |
| |
| /** |
| * @author clement chu |
| * |
| */ |
| public class PCDIDebugModel { |
| private final String SESSION_KEY = "session_key"; |
| private DebugJobStorage jobStorage = new DebugJobStorage("Job"); |
| private DebugJobStorage sessionStorage = new DebugJobStorage("Session"); |
| |
| public void shutdown() { |
| for (IPSession pSession : sessionStorage.getJobValueCollection().toArray(new IPSession[0])) { |
| if (pSession != null) { |
| IPJob job = pSession.getJob(); |
| if(!job.isTerminated()) { |
| // FIXME: what should happen here? |
| //job.removeAllProcesses(); |
| pSession.getPCDISession().shutdown(); |
| try { |
| PDebugUtils.println("-----PCDIDebugModel - waiting debugger to stop"); |
| Thread.sleep(2000); |
| } catch (InterruptedException e) { |
| e.printStackTrace(); |
| } |
| } |
| } |
| } |
| jobStorage.closeDebugJobStorage(); |
| sessionStorage.closeDebugJobStorage(); |
| } |
| public void shutdownSession(IPJob job) { |
| if (job != null) { |
| IPSession pSession = (IPSession)sessionStorage.removeValue(job.getID(), SESSION_KEY); |
| if (pSession != null) { |
| pSession.getPCDISession().shutdown(); |
| } |
| } |
| } |
| public IPCDISession getPCDISession(String job_id) { |
| IPSession pSession = (IPSession)sessionStorage.getValue(job_id, SESSION_KEY); |
| if (pSession != null) { |
| return pSession.getPCDISession(); |
| } |
| return null; |
| } |
| public IPSession createDebuggerSession(IAbstractDebugger debugger, IPLaunch launch, IBinaryObject exe, IProgressMonitor monitor) throws CoreException { |
| IPSession pSession = new PSession(debugger.createDebuggerSession(launch, exe, monitor)); |
| if (!monitor.isCanceled()) { |
| IPJob job = launch.getPJob(); |
| sessionStorage.addValue(job.getID(), SESSION_KEY, pSession); |
| |
| newJob(job, pSession.getPCDISession().createBitList()); |
| fireSessionEvent(job, pSession.getPCDISession()); |
| pSession.getPCDISession().start(monitor); |
| } |
| return pSession; |
| } |
| public static String getPluginIdentifier() { |
| return PTPDebugCorePlugin.getUniqueIdentifier(); |
| } |
| public void fireSessionEvent(IPJob job, IPCDISession session) { |
| IPDebugInfo info = new PDebugInfo(job, null, null, null); |
| PTPDebugCorePlugin.getDefault().fireDebugEvent(new PDebugEvent(session, IPDebugEvent.CREATE, IPDebugEvent.DEBUGGER, info)); |
| } |
| public void fireRegisterEvent(IPJob job, BitList tasks, boolean refresh) { |
| if (!tasks.isEmpty()) { |
| IPCDISession session = getPCDISession(job.getID()); |
| if (session != null) { |
| IPDebugInfo info = new PDebugRegisterInfo(job, tasks, tasks, null, refresh); |
| PTPDebugCorePlugin.getDefault().fireDebugEvent(new PDebugEvent(session, IPDebugEvent.CREATE, IPDebugEvent.REGISTER, info)); |
| } |
| } |
| } |
| public void fireUnregisterEvent(IPJob job, BitList tasks, boolean refresh) { |
| if (!tasks.isEmpty()) { |
| IPCDISession session = getPCDISession(job.getID()); |
| if (session != null) { |
| IPDebugInfo info = new PDebugRegisterInfo(job, tasks, null, tasks, refresh); |
| PTPDebugCorePlugin.getDefault().fireDebugEvent(new PDebugEvent(session, IPDebugEvent.TERMINATE, IPDebugEvent.REGISTER, info)); |
| } |
| } |
| } |
| /************************************************** |
| * Register / Unregister |
| **************************************************/ |
| public void removeDebugTarget(final IPLaunch launch, final BitList tasks, final boolean refresh) { |
| Job aJob = new Job("Removing the debug targets...") { |
| protected IStatus run(IProgressMonitor monitor) { |
| int[] taskArray = tasks.toArray(); |
| for (int i=0; i<taskArray.length; i++) { |
| IPDebugTarget debugTarget = launch.getDebugTarget(taskArray[i]); |
| if (debugTarget != null) { |
| launch.removeDebugTarget(debugTarget); |
| debugTarget.terminated(); |
| } |
| } |
| fireUnregisterEvent(launch.getPJob(), tasks, refresh); |
| return Status.OK_STATUS; |
| } |
| }; |
| aJob.setSystem(true); |
| aJob.setPriority(Job.INTERACTIVE); |
| aJob.schedule(); |
| } |
| public void addNewDebugTargets(final IPLaunch launch, final BitList tasks, final IPCDITarget[] cdiTargets, final IBinaryObject file, final boolean resumeTarget, final boolean refresh) { |
| Job aJob = new Job("Creating new debug targets...") { |
| protected IStatus run(IProgressMonitor monitor) { |
| boolean allowTerminate = true; |
| boolean allowDisconnect = false; |
| for (int i=0; i<cdiTargets.length; i++) { |
| IPDebugTarget target = new PDebugTarget(launch, cdiTargets[i], null, file, allowTerminate, allowDisconnect); |
| IPCDITargetConfiguration config = cdiTargets[i].getConfiguration(); |
| if (config.supportsResume() && resumeTarget) { |
| try { |
| target.resume(); |
| } catch (DebugException e) { |
| PTPDebugCorePlugin.log(e); |
| } |
| } |
| launch.addDebugTarget(target); |
| } |
| fireRegisterEvent(launch.getPJob(), tasks, refresh); |
| return Status.OK_STATUS; |
| } |
| }; |
| aJob.setSystem(true); |
| aJob.setPriority(Job.INTERACTIVE); |
| aJob.schedule(); |
| } |
| /************************************************** |
| * Breakpoint |
| **************************************************/ |
| public IPLineBreakpoint[] lineBreakpointsExists(String sourceHandle, IResource resource, int lineNumber) throws CoreException { |
| IBreakpoint[] breakpoints = getPBreakpoints(); |
| List<IBreakpoint> foundBreakpoints = new ArrayList<IBreakpoint>(0); |
| for (int i = 0; i < breakpoints.length; i++) { |
| if (!(breakpoints[i] instanceof IPLineBreakpoint)) |
| continue; |
| IPLineBreakpoint breakpoint = (IPLineBreakpoint) breakpoints[i]; |
| if (sameSourceHandle(sourceHandle, breakpoint.getSourceHandle())) { |
| if (breakpoint.getMarker().getResource().equals(resource)) { |
| if (breakpoint.getLineNumber() == lineNumber) { |
| foundBreakpoints.add(breakpoint); |
| } |
| } |
| } |
| } |
| return (IPLineBreakpoint[]) foundBreakpoints.toArray(new IPLineBreakpoint[foundBreakpoints.size()]); |
| } |
| // remove global breapoint or the breakpoint same as job id given |
| public IPLineBreakpoint lineBreakpointExists(IPLineBreakpoint[] breakpoints, String job_id) throws CoreException { |
| for (int i = 0; i < breakpoints.length; i++) { |
| String bpt_job_id = breakpoints[i].getJobId(); |
| if (bpt_job_id.equals(IPBreakpoint.GLOBAL) || bpt_job_id.equals(job_id)) |
| return breakpoints[i]; |
| } |
| return null; |
| } |
| public boolean sameSourceHandle(String handle1, String handle2) { |
| if (handle1 == null || handle2 == null) |
| return false; |
| IPath path1 = new Path(handle1); |
| IPath path2 = new Path(handle2); |
| if (path1.isValidPath(handle1) && path2.isValidPath(handle2)) |
| return path1.equals(path2); |
| return handle1.equals(handle2); |
| } |
| public IBreakpoint createLineBreakpoint(String sourceHandle, IResource resource, int lineNumber, boolean enabled, int ignoreCount, String condition, boolean register, String set_id, String job_id, String jobName) throws CoreException { |
| HashMap<String, Object> attributes = new HashMap<String, Object>(10); |
| attributes.put(IBreakpoint.ID, PTPDebugCorePlugin.getUniqueIdentifier()); |
| attributes.put(IMarker.LINE_NUMBER, new Integer(lineNumber)); |
| attributes.put(IBreakpoint.ENABLED, new Boolean(enabled)); |
| attributes.put(IPBreakpoint.SOURCE_HANDLE, sourceHandle); |
| attributes.put(IPBreakpoint.IGNORE_COUNT, new Integer(ignoreCount)); |
| attributes.put(IPBreakpoint.CONDITION, condition); |
| attributes.put(IPBreakpoint.SET_ID, set_id); |
| attributes.put(IPBreakpoint.CUR_SET_ID, set_id); |
| attributes.put(IPBreakpoint.JOB_ID, job_id); |
| attributes.put(IPBreakpoint.JOB_NAME, jobName); |
| return new PLineBreakpoint(resource, attributes, register); |
| } |
| public IBreakpoint[] getPBreakpoints() { |
| return DebugPlugin.getDefault().getBreakpointManager().getBreakpoints(getPluginIdentifier()); |
| } |
| public IBreakpoint[] findPBreakpoints(String job_id, String set_id) throws CoreException { |
| List<IBreakpoint> bptList = new ArrayList<IBreakpoint>(); |
| IBreakpoint[] breakpoints = getPBreakpoints(); |
| for (int i = 0; i < breakpoints.length; i++) { |
| if (!(breakpoints[i] instanceof IPLineBreakpoint)) |
| continue; |
| IPLineBreakpoint breakpoint = (IPLineBreakpoint) breakpoints[i]; |
| if (breakpoint.getJobId().equals(job_id) && breakpoint.getSetId().equals(set_id)) { |
| bptList.add(breakpoint); |
| } |
| } |
| return (IBreakpoint[]) bptList.toArray(new IBreakpoint[bptList.size()]); |
| } |
| public IPBreakpoint[] findPBreakpoints(String job_id, boolean includeGlobal) throws CoreException { |
| List<IBreakpoint> bptList = new ArrayList<IBreakpoint>(); |
| IBreakpoint[] breakpoints = getPBreakpoints(); |
| for (int i = 0; i < breakpoints.length; i++) { |
| if (!(breakpoints[i] instanceof IPBreakpoint)) |
| continue; |
| IPBreakpoint breakpoint = (IPBreakpoint) breakpoints[i]; |
| String bp_job_id = breakpoint.getJobId(); |
| if (bp_job_id.equals(job_id) || (includeGlobal && bp_job_id.equals(IPBreakpoint.GLOBAL))) { |
| bptList.add(breakpoint); |
| } |
| } |
| return (IPBreakpoint[]) bptList.toArray(new IPBreakpoint[bptList.size()]); |
| } |
| public void deletePBreakpoint(final String job_id, final String set_id) { |
| Job aJob = new Job("Deleting parallel breakpoint...") { |
| public IStatus run(IProgressMonitor monitor) { |
| try { |
| IBreakpoint[] breakpoints = findPBreakpoints(job_id, set_id); |
| if (breakpoints.length > 0) { |
| DebugPlugin.getDefault().getBreakpointManager().removeBreakpoints(breakpoints, true); |
| } |
| } catch (CoreException e) { |
| return e.getStatus(); |
| } |
| return Status.OK_STATUS; |
| } |
| }; |
| aJob.setSystem(true); |
| aJob.setPriority(Job.INTERACTIVE); |
| aJob.schedule(); |
| } |
| public void deletePBreakpoint(final String job_id) { |
| Job aJob = new Job("Deleting parallel breakpoint...") { |
| public IStatus run(IProgressMonitor monitor) { |
| try { |
| IBreakpoint[] breakpoints = findPBreakpoints(job_id, false); |
| if (breakpoints.length > 0) { |
| DebugPlugin.getDefault().getBreakpointManager().removeBreakpoints(breakpoints, true); |
| } |
| } catch (CoreException e) { |
| return e.getStatus(); |
| } |
| return Status.OK_STATUS; |
| } |
| }; |
| aJob.setSystem(true); |
| aJob.setPriority(Job.INTERACTIVE); |
| aJob.schedule(); |
| } |
| /** Updating the set id of each breakpoint |
| * @param set_id set id |
| * @param monitor |
| * @throws CoreException |
| */ |
| public void updatePBreakpoints(String set_id, IProgressMonitor monitor) throws CoreException { |
| try { |
| IBreakpoint[] breakpoints = getPBreakpoints(); |
| monitor.beginTask("Updating parallel breakpoint...", breakpoints.length); |
| for (int i = 0; i < breakpoints.length; i++) { |
| if (!(breakpoints[i] instanceof IPBreakpoint)) |
| continue; |
| ((IPBreakpoint) breakpoints[i]).setCurSetId(set_id); |
| monitor.worked(1); |
| } |
| } finally { |
| monitor.done(); |
| } |
| } |
| |
| /************************************************** |
| * Debug Job |
| **************************************************/ |
| public void newJob(IPJob job, BitList rootTasks) { |
| createSet(job.getID(), PreferenceConstants.SET_ROOT_ID, rootTasks); |
| } |
| public void createSet(String job_id, String set_id, BitList tasks) { |
| jobStorage.addValue(job_id, set_id, tasks); |
| } |
| public void addTasks(String job_id, String set_id, BitList tasks) { |
| ((BitList)jobStorage.getValue(job_id, set_id)).or(tasks); // add tasks |
| } |
| public void removeTasks(String job_id, String set_id, BitList tasks) { |
| ((BitList)jobStorage.getValue(job_id, set_id)).andNot(tasks); // remove tasks |
| } |
| public void deleteSet(String job_id, String set_id) { |
| jobStorage.removeValue(job_id, set_id); |
| } |
| public BitList getTasks(String job_id, String set_id) { |
| BitList tasks = ((BitList)jobStorage.getValue(job_id, set_id)); |
| if (tasks != null) |
| return tasks.copy(); |
| return null; |
| } |
| public void deleteJob(IPJob job) { |
| jobStorage.removeJobStorage(job.getID()); |
| } |
| } |