Bug 10071 - add support for debug hovers to DSF
authordadubrow
Thu, 05 Nov 2009 15:17:54 -0600
changeset 110 9262ca4bdfff
parent 106 397355a0f517
child 111 c2563c416525
Bug 10071 - add support for debug hovers to DSF
cdt/cdt_6_0_x/org.eclipse.cdt.debug.ui/META-INF/MANIFEST.MF
cdt/cdt_6_0_x/org.eclipse.cdt.debug.ui/src/org/eclipse/cdt/debug/internal/ui/editors/DebugTextHover.java
cdt/cdt_6_0_x/org.eclipse.cdt.debug.ui/src/org/eclipse/cdt/debug/ui/editors/AbstractDebugTextHover.java
cdt/cdt_6_0_x/org.eclipse.cdt.dsf.ui/src/org/eclipse/cdt/dsf/debug/ui/DebugTextHover.java
--- a/cdt/cdt_6_0_x/org.eclipse.cdt.debug.ui/META-INF/MANIFEST.MF	Tue Nov 03 15:25:26 2009 -0600
+++ b/cdt/cdt_6_0_x/org.eclipse.cdt.debug.ui/META-INF/MANIFEST.MF	Thu Nov 05 15:17:54 2009 -0600
@@ -23,6 +23,7 @@
  org.eclipse.cdt.debug.ui.breakpointactions,
  org.eclipse.cdt.debug.ui.breakpoints,
  org.eclipse.cdt.debug.ui.disassembly,
+ org.eclipse.cdt.debug.ui.editors,
  org.eclipse.cdt.debug.ui.importexecutable,
  org.eclipse.cdt.debug.ui.sourcelookup
 Require-Bundle: org.eclipse.ui.ide;bundle-version="[3.2.0,4.0.0)",
--- a/cdt/cdt_6_0_x/org.eclipse.cdt.debug.ui/src/org/eclipse/cdt/debug/internal/ui/editors/DebugTextHover.java	Tue Nov 03 15:25:26 2009 -0600
+++ b/cdt/cdt_6_0_x/org.eclipse.cdt.debug.ui/src/org/eclipse/cdt/debug/internal/ui/editors/DebugTextHover.java	Thu Nov 05 15:17:54 2009 -0600
@@ -7,281 +7,22 @@
  *
  * Contributors:
  *     QNX Software Systems - Initial API and implementation
+ *     Nokia - Refactored into CDI specific implementation of AbstractDebugTextHover
+
  *******************************************************************************/
 package org.eclipse.cdt.debug.internal.ui.editors;
 
-import java.util.regex.Matcher;
-import java.util.regex.Pattern;
-
 import org.eclipse.cdt.debug.core.model.ICStackFrame;
-import org.eclipse.cdt.debug.internal.ui.CDebugUIUtils;
-import org.eclipse.cdt.debug.ui.CDebugUIPlugin;
-import org.eclipse.cdt.ui.text.c.hover.ICEditorTextHover;
+import org.eclipse.cdt.debug.ui.editors.AbstractDebugTextHover;
 import org.eclipse.core.runtime.IAdaptable;
 import org.eclipse.debug.core.DebugException;
-import org.eclipse.debug.ui.IDebugUIConstants;
-import org.eclipse.jface.text.BadLocationException;
-import org.eclipse.jface.text.IDocument;
-import org.eclipse.jface.text.IInformationControlCreator;
-import org.eclipse.jface.text.IRegion;
-import org.eclipse.jface.text.ITextHoverExtension;
-import org.eclipse.jface.text.ITextViewer;
-import org.eclipse.jface.viewers.ISelection;
 import org.eclipse.jface.viewers.IStructuredSelection;
-import org.eclipse.ui.IEditorPart;
-import org.eclipse.ui.IPartListener;
-import org.eclipse.ui.ISelectionListener;
-import org.eclipse.ui.IWorkbenchPage;
-import org.eclipse.ui.IWorkbenchPart;
-
-/* The class LanguageOperators protects some language specific
- * operator information used by the DebugTextHover class.
- */
-
-class LanguageOperators {
-	public String getAssignmentOperator() {
-		return "="; //$NON-NLS-1$
-	}
-
-	public String getGreaterThanEqualToOperator() {
-		return ">="; //$NON-NLS-1$
-	}
-
-	public String getEqualToOperator() {
-		return "=="; //$NON-NLS-1$
-	}
-
-	public String getNotEqualToOperator() {
-		return "!="; //$NON-NLS-1$
-	}
-
-	public String getLessThenEqualToOperator() {
-		return "<="; //$NON-NLS-1$
-	}
-
-	public String getValueChangeOperatorsRegex() {
-		return "(\\+\\+)|(\\-\\-)|(\\+\\=)|" //$NON-NLS-1$
-				+ "(\\-\\=)|(\\*\\=)|(/\\=)|(\\&\\=)" //$NON-NLS-1$
-				+ "(\\%\\=)|(\\^\\=)|(\\|\\=)|(\\<\\<\\=)|(\\>\\>\\=)"; //$NON-NLS-1$
-	}
-
-	public String getEqualToOperatorsRegex() {
-		return "\\=\\=|\\<\\=|\\>\\=|!\\="; //$NON-NLS-1$
-	}
-
-	public String getIdentifierRegex() {
-		return "[_A-Za-z][_A-Za-z0-9]*"; //$NON-NLS-1$
-	}
-}
 
 /**
  * The text hovering support for C/C++ debugger.
  */
 
-public class DebugTextHover implements ICEditorTextHover, ITextHoverExtension,
-		ISelectionListener, IPartListener {
-
-	static final private int MAX_HOVER_INFO_SIZE = 100;
-
-	protected ISelection fSelection = null;
-
-	protected IEditorPart fEditor;
-
-	/**
-	 * Constructor for DebugTextHover.
-	 */
-	public DebugTextHover() {
-		super();
-	}
-
-	/*
-	 * (non-Javadoc)
-	 * 
-	 * @see org.eclipse.jface.text.ITextHover#getHoverInfo(org.eclipse.jface.text.ITextViewer,
-	 *      org.eclipse.jface.text.IRegion)
-	 */
-	public String getHoverInfo(ITextViewer textViewer, IRegion hoverRegion) {
-		ICStackFrame frame = getFrame();
-		if (frame != null && frame.canEvaluate()) {
-			try {
-				IDocument document = textViewer.getDocument();
-				if (document == null)
-					return null;
-				String expression = document.get(hoverRegion.getOffset(),
-						hoverRegion.getLength());
-				if (expression == null)
-					return null;
-				expression = expression.trim();
-				if (expression.length() == 0)
-					return null;
-				LanguageOperators operatorsObj = new LanguageOperators();
-
-				Pattern pattern = Pattern.compile(operatorsObj
-						.getValueChangeOperatorsRegex());
-				Matcher matcher = pattern.matcher(expression);
-
-				boolean match_found = matcher.find();
-				// Get matching string
-				// If the expression has some operators which can change the
-				// value of a variable, that expresssion should not be
-				// evaluated.
-				if (match_found) {
-					return null;
-				}
-				pattern = Pattern.compile(operatorsObj
-						.getEqualToOperatorsRegex());
-				String[] tokens = pattern.split(expression);
-				for (int i = 0; i < tokens.length; i++) {
-					//If the expression contains assignment operator that
-					// can change the value of a variable, the expression
-					// should not be evaluated.
-					if (tokens[i].indexOf(operatorsObj
-							.getAssignmentOperator()) != -1)
-						return null;
-				}
-				//Supressing function calls from evaluation.
-				String functionCallRegex = operatorsObj.getIdentifierRegex() + "\\s*\\("; //$NON-NLS-1$
-				pattern = Pattern.compile(functionCallRegex);
-				matcher = pattern.matcher(expression);
-				match_found = matcher.find();
-				if (match_found) {
-					return null;
-				}
-				StringBuffer buffer = new StringBuffer();
-				String result = evaluateExpression(frame, expression);
-				if (result == null)
-					return null;
-				try {
-					appendVariable(buffer, makeHTMLSafe(expression), makeHTMLSafe(result.trim()));
-				} catch (DebugException x) {
-					CDebugUIPlugin.log(x);
-				}
-				if (buffer.length() > 0) {
-					return buffer.toString();
-				}
-			} catch (BadLocationException x) {
-				CDebugUIPlugin.log(x);
-			}
-		}
-		return null;
-	}
-
-	/*
-	 * (non-Javadoc)
-	 * 
-	 * @see org.eclipse.jface.text.ITextHover#getHoverRegion(org.eclipse.jface.text.ITextViewer, int)
-	 */
-	public IRegion getHoverRegion(ITextViewer viewer, int offset) {
-		/*
-		 * Point selectedRange = viewer.getSelectedRange(); if ( selectedRange.x >= 0 && selectedRange.y > 0 && offset >= selectedRange.x && offset <=
-		 * selectedRange.x + selectedRange.y ) return new Region( selectedRange.x, selectedRange.y );
-		 */
-		if (viewer != null)
-			return CDebugUIUtils.findWord(viewer.getDocument(), offset);
-		return null;
-	}
-
-	private String evaluateExpression(ICStackFrame frame, String expression) {
-		String result = null;
-		try {
-			result = frame.evaluateExpressionToString(expression);
-		} catch (DebugException e) {
-			// ignore
-		}
-		return result;
-	}
-
-	/**
-	 * Append HTML for the given variable to the given buffer
-	 */
-	private static void appendVariable(StringBuffer buffer, String expression, String value) throws DebugException {
-		if (value.length() > MAX_HOVER_INFO_SIZE)
-			value = value.substring(0, MAX_HOVER_INFO_SIZE) + " ..."; //$NON-NLS-1$
-		buffer.append("<p>"); //$NON-NLS-1$
-		buffer.append("<pre>").append(expression).append("</pre>"); //$NON-NLS-1$ //$NON-NLS-2$
-		buffer.append(" = "); //$NON-NLS-1$
-		buffer.append("<b><pre>").append(value).append("</pre></b>"); //$NON-NLS-1$ //$NON-NLS-2$
-		buffer.append("</p>"); //$NON-NLS-1$
-	}
-
-	/*
-	 * (non-Javadoc)
-	 * 
-	 * @see org.eclipse.cdt.ui.text.c.hover.ICEditorTextHover#setEditor(org.eclipse.ui.IEditorPart)
-	 */
-	public void setEditor(IEditorPart editor) {
-		if (editor != null) {
-			fEditor = editor;
-			final IWorkbenchPage page = editor.getSite().getPage();
-			page.addSelectionListener(IDebugUIConstants.ID_DEBUG_VIEW, this);
-			page.addPartListener(this);
-			// initialize selection
-			Runnable r = new Runnable() {
-
-				public void run() {
-					fSelection = page
-							.getSelection(IDebugUIConstants.ID_DEBUG_VIEW);
-				}
-			};
-			CDebugUIPlugin.getStandardDisplay().asyncExec(r);
-		}
-	}
-
-	/*
-	 * (non-Javadoc)
-	 * 
-	 * @see org.eclipse.ui.ISelectionListener#selectionChanged(org.eclipse.ui.IWorkbenchPart, org.eclipse.jface.viewers.ISelection)
-	 */
-	public void selectionChanged(IWorkbenchPart part, ISelection selection) {
-		fSelection = selection;
-	}
-
-	/*
-	 * (non-Javadoc)
-	 * 
-	 * @see org.eclipse.ui.IPartListener#partActivated(org.eclipse.ui.IWorkbenchPart)
-	 */
-	public void partActivated(IWorkbenchPart part) {
-	}
-
-	/*
-	 * (non-Javadoc)
-	 * 
-	 * @see org.eclipse.ui.IPartListener#partBroughtToTop(org.eclipse.ui.IWorkbenchPart)
-	 */
-	public void partBroughtToTop(IWorkbenchPart part) {
-	}
-
-	/*
-	 * (non-Javadoc)
-	 * 
-	 * @see org.eclipse.ui.IPartListener#partClosed(org.eclipse.ui.IWorkbenchPart)
-	 */
-	public void partClosed(IWorkbenchPart part) {
-		if (part.equals(fEditor)) {
-			IWorkbenchPage page = fEditor.getSite().getPage();
-			page.removeSelectionListener(IDebugUIConstants.ID_DEBUG_VIEW, this);
-			page.removePartListener(this);
-			fSelection = null;
-			fEditor = null;
-		}
-	}
-
-	/*
-	 * (non-Javadoc)
-	 * 
-	 * @see org.eclipse.ui.IPartListener#partDeactivated(org.eclipse.ui.IWorkbenchPart)
-	 */
-	public void partDeactivated(IWorkbenchPart part) {
-	}
-
-	/*
-	 * (non-Javadoc)
-	 * 
-	 * @see org.eclipse.ui.IPartListener#partOpened(org.eclipse.ui.IWorkbenchPart)
-	 */
-	public void partOpened(IWorkbenchPart part) {
-	}
+public class DebugTextHover extends AbstractDebugTextHover {
 
 	/**
 	 * Returns the evaluation stack frame, or <code>null</code> if none.
@@ -302,37 +43,24 @@
 		return null;
 	}
 
-	/*
-	 * (non-Javadoc)
-	 * 
-	 * @see org.eclipse.jface.text.ITextHoverExtension#getHoverControlCreator()
-	 */
-	public IInformationControlCreator getHoverControlCreator() {
-		return null;
+	@Override
+	protected boolean canEvaluate() {
+		ICStackFrame frame = getFrame();
+		if (frame != null)
+			return frame.canEvaluate();
+		
+		return false;
 	}
 
-	/**
-	 * Replace any characters in the given String that would confuse an HTML parser with their escape sequences.
-	 */
-	private static String makeHTMLSafe(String string) {
-		StringBuffer buffer = new StringBuffer(string.length());
-		for (int i = 0; i != string.length(); i++) {
-			char ch = string.charAt(i);
-			switch (ch) {
-			case '&':
-				buffer.append("&amp;"); //$NON-NLS-1$
-				break;
-			case '<':
-				buffer.append("&lt;"); //$NON-NLS-1$
-				break;
-			case '>':
-				buffer.append("&gt;"); //$NON-NLS-1$
-				break;
-			default:
-				buffer.append(ch);
-				break;
-			}
+	@Override
+	protected String evaluateExpression(String expression) {
+		ICStackFrame frame = getFrame();
+		String result = null;
+		try {
+			result = frame.evaluateExpressionToString(expression);
+		} catch (DebugException e) {
+			// ignore
 		}
-		return buffer.toString();
+		return result;
 	}
 }
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/cdt/cdt_6_0_x/org.eclipse.cdt.debug.ui/src/org/eclipse/cdt/debug/ui/editors/AbstractDebugTextHover.java	Thu Nov 05 15:17:54 2009 -0600
@@ -0,0 +1,253 @@
+/*******************************************************************************
+ * Copyright (c) 2000, 2007 QNX Software Systems and others.
+ * All rights reserved. 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
+ *
+ * Contributors:
+ *     QNX Software Systems - Initial API and implementation
+ *     Nokia - Refactored from DebugTextHover to remove CDI dependency
+ *     
+ *******************************************************************************/
+package org.eclipse.cdt.debug.ui.editors;
+
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
+
+import org.eclipse.cdt.debug.internal.ui.CDebugUIUtils;
+import org.eclipse.cdt.debug.ui.CDebugUIPlugin;
+import org.eclipse.cdt.ui.text.c.hover.ICEditorTextHover;
+import org.eclipse.debug.core.DebugException;
+import org.eclipse.debug.ui.IDebugUIConstants;
+import org.eclipse.jface.text.BadLocationException;
+import org.eclipse.jface.text.IDocument;
+import org.eclipse.jface.text.IInformationControlCreator;
+import org.eclipse.jface.text.IRegion;
+import org.eclipse.jface.text.ITextHoverExtension;
+import org.eclipse.jface.text.ITextViewer;
+import org.eclipse.jface.viewers.ISelection;
+import org.eclipse.ui.IEditorPart;
+import org.eclipse.ui.IPartListener;
+import org.eclipse.ui.ISelectionListener;
+import org.eclipse.ui.IWorkbenchPage;
+import org.eclipse.ui.IWorkbenchPart;
+
+/* The class LanguageOperators protects some language specific
+ * operator information used by the DebugTextHover class.
+ */
+
+class LanguageOperators {
+	public String getAssignmentOperator() {
+		return "="; //$NON-NLS-1$
+	}
+
+	public String getGreaterThanEqualToOperator() {
+		return ">="; //$NON-NLS-1$
+	}
+
+	public String getEqualToOperator() {
+		return "=="; //$NON-NLS-1$
+	}
+
+	public String getNotEqualToOperator() {
+		return "!="; //$NON-NLS-1$
+	}
+
+	public String getLessThenEqualToOperator() {
+		return "<="; //$NON-NLS-1$
+	}
+
+	public String getValueChangeOperatorsRegex() {
+		return "(\\+\\+)|(\\-\\-)|(\\+\\=)|" //$NON-NLS-1$
+				+ "(\\-\\=)|(\\*\\=)|(/\\=)|(\\&\\=)" //$NON-NLS-1$
+				+ "(\\%\\=)|(\\^\\=)|(\\|\\=)|(\\<\\<\\=)|(\\>\\>\\=)"; //$NON-NLS-1$
+	}
+
+	public String getEqualToOperatorsRegex() {
+		return "\\=\\=|\\<\\=|\\>\\=|!\\="; //$NON-NLS-1$
+	}
+
+	public String getIdentifierRegex() {
+		return "[_A-Za-z][_A-Za-z0-9]*"; //$NON-NLS-1$
+	}
+}
+
+/**
+ * The text hovering support for C/C++ debugger.
+ */
+
+public abstract class AbstractDebugTextHover implements ICEditorTextHover, ITextHoverExtension,
+		ISelectionListener, IPartListener {
+
+	static final private int MAX_HOVER_INFO_SIZE = 100;
+
+	protected ISelection fSelection = null;
+
+	protected IEditorPart fEditor;
+
+	protected abstract boolean canEvaluate();
+	
+	protected abstract String evaluateExpression(String expression);
+	
+	public String getHoverInfo(ITextViewer textViewer, IRegion hoverRegion) {
+		if (canEvaluate()) {
+			try {
+				IDocument document = textViewer.getDocument();
+				if (document == null)
+					return null;
+				String expression = document.get(hoverRegion.getOffset(),
+						hoverRegion.getLength());
+				if (expression == null)
+					return null;
+				expression = expression.trim();
+				if (expression.length() == 0)
+					return null;
+				LanguageOperators operatorsObj = new LanguageOperators();
+
+				Pattern pattern = Pattern.compile(operatorsObj
+						.getValueChangeOperatorsRegex());
+				Matcher matcher = pattern.matcher(expression);
+
+				boolean match_found = matcher.find();
+				// Get matching string
+				// If the expression has some operators which can change the
+				// value of a variable, that expresssion should not be
+				// evaluated.
+				if (match_found) {
+					return null;
+				}
+				pattern = Pattern.compile(operatorsObj
+						.getEqualToOperatorsRegex());
+				String[] tokens = pattern.split(expression);
+				for (int i = 0; i < tokens.length; i++) {
+					//If the expression contains assignment operator that
+					// can change the value of a variable, the expression
+					// should not be evaluated.
+					if (tokens[i].indexOf(operatorsObj
+							.getAssignmentOperator()) != -1)
+						return null;
+				}
+				//Supressing function calls from evaluation.
+				String functionCallRegex = operatorsObj.getIdentifierRegex() + "\\s*\\("; //$NON-NLS-1$
+				pattern = Pattern.compile(functionCallRegex);
+				matcher = pattern.matcher(expression);
+				match_found = matcher.find();
+				if (match_found) {
+					return null;
+				}
+				StringBuffer buffer = new StringBuffer();
+				String result = evaluateExpression(expression);
+				if (result == null)
+					return null;
+				try {
+					appendVariable(buffer, makeHTMLSafe(expression), makeHTMLSafe(result.trim()));
+				} catch (DebugException x) {
+					CDebugUIPlugin.log(x);
+				}
+				if (buffer.length() > 0) {
+					return buffer.toString();
+				}
+			} catch (BadLocationException x) {
+				CDebugUIPlugin.log(x);
+			}
+		}
+		return null;
+	}
+
+	public IRegion getHoverRegion(ITextViewer viewer, int offset) {
+		/*
+		 * Point selectedRange = viewer.getSelectedRange(); if ( selectedRange.x >= 0 && selectedRange.y > 0 && offset >= selectedRange.x && offset <=
+		 * selectedRange.x + selectedRange.y ) return new Region( selectedRange.x, selectedRange.y );
+		 */
+		if (viewer != null)
+			return CDebugUIUtils.findWord(viewer.getDocument(), offset);
+		return null;
+	}
+
+	/**
+	 * Append HTML for the given variable to the given buffer
+	 */
+	private static void appendVariable(StringBuffer buffer, String expression, String value) throws DebugException {
+		if (value.length() > MAX_HOVER_INFO_SIZE)
+			value = value.substring(0, MAX_HOVER_INFO_SIZE) + " ..."; //$NON-NLS-1$
+		buffer.append("<p>"); //$NON-NLS-1$
+		buffer.append("<pre>").append(expression).append("</pre>"); //$NON-NLS-1$ //$NON-NLS-2$
+		buffer.append(" = "); //$NON-NLS-1$
+		buffer.append("<b><pre>").append(value).append("</pre></b>"); //$NON-NLS-1$ //$NON-NLS-2$
+		buffer.append("</p>"); //$NON-NLS-1$
+	}
+
+	public void setEditor(IEditorPart editor) {
+		if (editor != null) {
+			fEditor = editor;
+			final IWorkbenchPage page = editor.getSite().getPage();
+			page.addSelectionListener(IDebugUIConstants.ID_DEBUG_VIEW, this);
+			page.addPartListener(this);
+			// initialize selection
+			Runnable r = new Runnable() {
+
+				public void run() {
+					fSelection = page
+							.getSelection(IDebugUIConstants.ID_DEBUG_VIEW);
+				}
+			};
+			CDebugUIPlugin.getStandardDisplay().asyncExec(r);
+		}
+	}
+
+	public void selectionChanged(IWorkbenchPart part, ISelection selection) {
+		fSelection = selection;
+	}
+
+	public void partActivated(IWorkbenchPart part) {
+	}
+
+	public void partBroughtToTop(IWorkbenchPart part) {
+	}
+
+	public void partClosed(IWorkbenchPart part) {
+		if (part.equals(fEditor)) {
+			IWorkbenchPage page = fEditor.getSite().getPage();
+			page.removeSelectionListener(IDebugUIConstants.ID_DEBUG_VIEW, this);
+			page.removePartListener(this);
+			fSelection = null;
+			fEditor = null;
+		}
+	}
+
+	public void partDeactivated(IWorkbenchPart part) {
+	}
+
+	public void partOpened(IWorkbenchPart part) {
+	}
+
+	public IInformationControlCreator getHoverControlCreator() {
+		return null;
+	}
+
+	/**
+	 * Replace any characters in the given String that would confuse an HTML parser with their escape sequences.
+	 */
+	private static String makeHTMLSafe(String string) {
+		StringBuffer buffer = new StringBuffer(string.length());
+		for (int i = 0; i != string.length(); i++) {
+			char ch = string.charAt(i);
+			switch (ch) {
+			case '&':
+				buffer.append("&amp;"); //$NON-NLS-1$
+				break;
+			case '<':
+				buffer.append("&lt;"); //$NON-NLS-1$
+				break;
+			case '>':
+				buffer.append("&gt;"); //$NON-NLS-1$
+				break;
+			default:
+				buffer.append(ch);
+				break;
+			}
+		}
+		return buffer.toString();
+	}
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/cdt/cdt_6_0_x/org.eclipse.cdt.dsf.ui/src/org/eclipse/cdt/dsf/debug/ui/DebugTextHover.java	Thu Nov 05 15:17:54 2009 -0600
@@ -0,0 +1,108 @@
+/**
+* Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
+* All rights reserved.
+* This component and the accompanying materials are made available
+* under the terms of the License "Eclipse Public License v1.0"
+* which accompanies this distribution, and is available
+* at the URL "http://www.eclipse.org/legal/epl-v10.html".
+*
+* Initial Contributors:
+* Nokia Corporation - initial contribution.
+*
+* Contributors:
+*
+* Description: 
+*
+*/
+
+package org.eclipse.cdt.dsf.debug.ui;
+
+import org.eclipse.cdt.debug.ui.editors.AbstractDebugTextHover;
+import org.eclipse.cdt.dsf.concurrent.DataRequestMonitor;
+import org.eclipse.cdt.dsf.concurrent.Query;
+import org.eclipse.cdt.dsf.debug.service.IExpressions;
+import org.eclipse.cdt.dsf.debug.service.IFormattedValues;
+import org.eclipse.cdt.dsf.debug.service.IExpressions.IExpressionDMContext;
+import org.eclipse.cdt.dsf.debug.service.IFormattedValues.FormattedValueDMContext;
+import org.eclipse.cdt.dsf.debug.service.IFormattedValues.FormattedValueDMData;
+import org.eclipse.cdt.dsf.debug.service.IStack.IFrameDMContext;
+import org.eclipse.cdt.dsf.internal.ui.DsfUIPlugin;
+import org.eclipse.cdt.dsf.service.DsfServicesTracker;
+import org.eclipse.cdt.dsf.service.DsfSession;
+import org.eclipse.core.runtime.IAdaptable;
+import org.eclipse.jface.viewers.IStructuredSelection;
+
+/**
+ * An implementation of AbstractDebugTextHover using DSF services
+ */
+public class DebugTextHover extends AbstractDebugTextHover {
+
+	public class GetExpressionValueQuery extends Query<FormattedValueDMData> {
+    	private final IFrameDMContext frame;
+    	private final String expression;
+		private DsfServicesTracker dsfServicesTracker;
+
+        public GetExpressionValueQuery(IFrameDMContext frame, String expression, DsfServicesTracker dsfServicesTracker) {
+            this.frame = frame;
+			this.expression = expression;
+			this.dsfServicesTracker = dsfServicesTracker;
+        }
+
+        @Override
+        protected void execute(final DataRequestMonitor<FormattedValueDMData> rm) {
+            DsfSession session = DsfSession.getSession(frame.getSessionId());
+			IExpressions expressions = dsfServicesTracker.getService(IExpressions.class);
+    		IExpressionDMContext expressionDMC = expressions.createExpression(frame, expression);
+    		FormattedValueDMContext formattedValueContext = expressions.getFormattedValueContext(expressionDMC, IFormattedValues.NATURAL_FORMAT);
+        	expressions.getFormattedExpressionValue(formattedValueContext,
+        			new DataRequestMonitor<FormattedValueDMData>(session.getExecutor(), rm) {
+                @Override
+                protected void handleSuccess() {
+                    rm.setData(getData());
+                    rm.done();
+                }
+                
+                @Override
+        		protected void handleFailure() {
+        			rm.done();
+        		}
+            });
+        }
+    }
+
+	protected IFrameDMContext getFrame() {
+		if (fSelection instanceof IStructuredSelection) {
+			IStructuredSelection selection = (IStructuredSelection) fSelection;
+			if (selection.size() == 1) {
+				Object element = selection.getFirstElement();
+				if (element instanceof IAdaptable) {
+					return (IFrameDMContext) ((IAdaptable) element).getAdapter(IFrameDMContext.class);
+				}
+			}
+		}
+		return null;
+	}
+
+	@Override
+	protected boolean canEvaluate() {
+		return getFrame() != null;
+	}
+
+	@Override
+	protected String evaluateExpression(String expression) {
+		IFrameDMContext frame = getFrame();
+		String sessionId = frame.getSessionId();
+		DsfServicesTracker dsfServicesTracker = new DsfServicesTracker(DsfUIPlugin.getBundleContext(), sessionId);
+		GetExpressionValueQuery query = new GetExpressionValueQuery(frame, expression, dsfServicesTracker);
+		DsfSession session = DsfSession.getSession(sessionId);
+        session.getExecutor().execute(query);
+        try {
+        	FormattedValueDMData data = query.get();
+        	if (data != null)
+        		return data.getFormattedValue();
+        } catch (Exception e) {
+        }
+        return null;
+	}
+
+}