| /******************************************************************************* |
| * Copyright (c) 2000, 2003 IBM Corporation and others. |
| * All rights reserved. This program and the accompanying materials |
| * are made available under the terms of the Common Public License v1.0 |
| * which accompanies this distribution, and is available at |
| * http://www.eclipse.org/legal/cpl-v10.html |
| * |
| * Contributors: |
| * IBM Corporation - initial API and implementation |
| *******************************************************************************/ |
| package org.eclipse.jdt.internal.debug.core.model; |
| |
| |
| import java.text.MessageFormat; |
| import java.util.ArrayList; |
| import java.util.Collections; |
| import java.util.Iterator; |
| import java.util.List; |
| |
| import org.eclipse.debug.core.DebugException; |
| import org.eclipse.jdt.debug.core.IJavaClassType; |
| import org.eclipse.jdt.debug.core.IJavaInterfaceType; |
| import org.eclipse.jdt.debug.core.IJavaObject; |
| import org.eclipse.jdt.debug.core.IJavaThread; |
| import org.eclipse.jdt.debug.core.IJavaValue; |
| |
| import com.sun.jdi.ClassType; |
| import com.sun.jdi.InterfaceType; |
| import com.sun.jdi.Method; |
| import com.sun.jdi.ObjectReference; |
| import com.sun.jdi.Value; |
| |
| /** |
| * The class of an object in a debug target. |
| */ |
| public class JDIClassType extends JDIReferenceType implements IJavaClassType { |
| |
| /** |
| * Cosntructs a new class type on the given target referencing |
| * the specified class type. |
| */ |
| public JDIClassType(JDIDebugTarget target, ClassType type) { |
| super(target, type); |
| } |
| |
| /** |
| * @see IJavaClassType#newInstance(String, IJavaValue[], IJavaThread) |
| */ |
| public IJavaObject newInstance(String signature, IJavaValue[] args, IJavaThread thread) throws DebugException { |
| if (getUnderlyingType() instanceof ClassType) { |
| ClassType clazz = (ClassType)getUnderlyingType(); |
| JDIThread javaThread = (JDIThread)thread; |
| List arguments = convertArguments(args); |
| Method method = null; |
| try { |
| List methods = clazz.methodsByName("<init>", signature); //$NON-NLS-1$ |
| if (methods.isEmpty()) { |
| getDebugTarget().requestFailed(MessageFormat.format(JDIDebugModelMessages.getString("JDIClassType.Type_does_not_implement_cosntructor"), new String[]{signature}), null); //$NON-NLS-1$ |
| } else { |
| method = (Method)methods.get(0); |
| } |
| } catch (RuntimeException e) { |
| getDebugTarget().targetRequestFailed(MessageFormat.format(JDIDebugModelMessages.getString("JDIClassType.exception_while_performing_method_lookup_for_constructor"), new String[] {e.toString(), signature}), e); //$NON-NLS-1$ |
| } |
| ObjectReference result = javaThread.newInstance(clazz, method, arguments); |
| return (IJavaObject)JDIValue.createValue(getDebugTarget(), result); |
| } |
| getDebugTarget().requestFailed(JDIDebugModelMessages.getString("JDIClassType.Type_is_not_a_class_type"), null); //$NON-NLS-1$ |
| // execution will not fall through to here, |
| // as #requestFailed will throw an exception |
| return null; |
| } |
| |
| /** |
| * @see IJavaType#sendMessage(String, String, IJavaValue[], IJavaThread) |
| */ |
| public IJavaValue sendMessage(String selector, String signature, IJavaValue[] args, IJavaThread thread) throws DebugException { |
| if (getUnderlyingType() instanceof ClassType) { |
| ClassType clazz = (ClassType)getUnderlyingType(); |
| JDIThread javaThread = (JDIThread)thread; |
| List arguments = convertArguments(args); |
| Method method = null; |
| try { |
| List methods = clazz.methodsByName(selector, signature); |
| if (methods.isEmpty()) { |
| getDebugTarget().requestFailed(MessageFormat.format(JDIDebugModelMessages.getString("JDIClassType.Type_does_not_implement_selector"), new String[] {selector, signature}), null); //$NON-NLS-1$ |
| } else { |
| method = (Method)methods.get(0); |
| } |
| } catch (RuntimeException e) { |
| getDebugTarget().targetRequestFailed(MessageFormat.format(JDIDebugModelMessages.getString("JDIClassType.exception_while_performing_method_lookup_for_selector"), new String[] {e.toString(), selector, signature}), e); //$NON-NLS-1$ |
| } |
| Value result = javaThread.invokeMethod(clazz, null, method, arguments, false); |
| return JDIValue.createValue(getDebugTarget(), result); |
| } |
| getDebugTarget().requestFailed(JDIDebugModelMessages.getString("JDIClassType.Type_is_not_a_class_type"), null); //$NON-NLS-1$ |
| // execution will not fall through to here, |
| // as #requestFailed will throw an exception |
| return null; |
| } |
| |
| /** |
| * Utility method to convert argument array to an |
| * argument list. |
| * |
| * @param args array of arguments, as <code>IJavaValue</code>s, |
| * possibly <code>null</code> or empty |
| * @return a list of underlying <code>Value</code>s |
| */ |
| protected List convertArguments(IJavaValue[] args) { |
| List arguments = null; |
| if (args == null) { |
| arguments = Collections.EMPTY_LIST; |
| } else { |
| arguments= new ArrayList(args.length); |
| for (int i = 0; i < args.length; i++) { |
| arguments.add(((JDIValue)args[i]).getUnderlyingValue()); |
| } |
| } |
| return arguments; |
| } |
| |
| /** |
| * @see IJavaClassType#getSuperclass() |
| */ |
| public IJavaClassType getSuperclass() throws DebugException { |
| try { |
| ClassType superclazz = ((ClassType)getUnderlyingType()).superclass(); |
| if (superclazz != null) { |
| return (IJavaClassType)JDIType.createType(getDebugTarget(), superclazz); |
| } |
| } catch (RuntimeException e) { |
| getDebugTarget().targetRequestFailed(MessageFormat.format(JDIDebugModelMessages.getString("JDIClassType.exception_while_retrieving_superclass"), new String[] {e.toString()}), e); //$NON-NLS-1$ |
| } |
| // it is possible to return null |
| return null; |
| } |
| |
| /* (non-Javadoc) |
| * @see org.eclipse.jdt.debug.core.IJavaClassType#getAllInterfaces() |
| */ |
| public IJavaInterfaceType[] getAllInterfaces() throws DebugException { |
| try { |
| List interfaceList = ((ClassType)getUnderlyingType()).allInterfaces(); |
| List javaInterfaceTypeList = new ArrayList(interfaceList.size()); |
| Iterator iterator = interfaceList.iterator(); |
| while (iterator.hasNext()) { |
| InterfaceType interfaceType = (InterfaceType) iterator.next(); |
| if (interfaceType != null) { |
| javaInterfaceTypeList.add(JDIType.createType(getDebugTarget(), interfaceType)); |
| } |
| } |
| IJavaInterfaceType[] javaInterfaceTypeArray = new IJavaInterfaceType[javaInterfaceTypeList.size()]; |
| javaInterfaceTypeArray = (IJavaInterfaceType[]) javaInterfaceTypeList.toArray(javaInterfaceTypeArray); |
| return javaInterfaceTypeArray; |
| } catch (RuntimeException re) { |
| getDebugTarget().targetRequestFailed(MessageFormat.format(JDIDebugModelMessages.getString("JDIClassType.exception_while_retrieving_superclass"), new String[] {re.toString()}), re); //$NON-NLS-1$ |
| } |
| return new IJavaInterfaceType[0]; |
| } |
| |
| /* (non-Javadoc) |
| * @see org.eclipse.jdt.debug.core.IJavaClassType#getInterfaces() |
| */ |
| public IJavaInterfaceType[] getInterfaces() throws DebugException { |
| try { |
| List interfaceList = ((ClassType)getUnderlyingType()).interfaces(); |
| List javaInterfaceTypeList = new ArrayList(interfaceList.size()); |
| Iterator iterator = interfaceList.iterator(); |
| while (iterator.hasNext()) { |
| InterfaceType interfaceType = (InterfaceType) iterator.next(); |
| if (interfaceType != null) { |
| javaInterfaceTypeList.add(JDIType.createType(getDebugTarget(), interfaceType)); |
| } |
| } |
| IJavaInterfaceType[] javaInterfaceTypeArray = new IJavaInterfaceType[javaInterfaceTypeList.size()]; |
| javaInterfaceTypeArray = (IJavaInterfaceType[]) javaInterfaceTypeList.toArray(javaInterfaceTypeArray); |
| return javaInterfaceTypeArray; |
| } catch (RuntimeException re) { |
| getDebugTarget().targetRequestFailed(MessageFormat.format(JDIDebugModelMessages.getString("JDIClassType.exception_while_retrieving_superclass"), new String[] {re.toString()}), re); //$NON-NLS-1$ |
| } |
| return new IJavaInterfaceType[0]; |
| } |
| |
| } |
| |