Skip to main content
aboutsummaryrefslogblamecommitdiffstats
blob: 8efdabd4572752113c2a36409fb888492595bf06 (plain) (tree)
1
2
3
4
5
6
7
8
9
                                                                                
                                                       
                                                                   
                                                                        
                                                           

                                            
                
                                                                       
                                         
                                                                                 
                                                     
 
                                         
                                                
                                                 
                                                      
                                                    
                                                   
                                                           
                                             
                                             
                                                             
                                                          
                                                                   
                                                        
                                               
                                             
                                                  
                                               
                                                    
                                           
                                          
                                                             
                                                                  
                                                             
                                                       
                                                       
                                                     
                                                         
                                                  

                                                            
                                                        
                                                               
                                                                         
                                               

   
                           
   
                                                                                               
 
                                                                                           

                                                                      


                                                                 
                                         


                                                              
                                         

         
                                                               
                                         
         
                                                       
                                         
         
                                                                
                                         

                                                                         
                                         

                                                                  
                                         
         
                                                        
                                         

                                                       
                                         
         
                                                        
                                         
         
                                                       
                                         

                                                     
                                         

                                                         
                                         

                                                           
                                         
         


                                                                                

     
                                                           
                                                       
                                               
        

                                                                 

                             

                                                                  
                                                                                                     
                                                                     

                                                        
                                                                                              
            
                                                
                                          
             
         
        

                                                                          

                                                  
                                         
                                                         
                                                  
                                                              
                        


                                              
                                                                              


                                                       
                                          

                 

                                                         
                 

                                                                                                            




                                                
 



                                         
 



                                     
                                                  




                                                                           
         
        
                                                   




















                                                                                                                               
                                             
                                                




                                                                                   
                
                                                        
         
 
                                                      

                                                                           


                                          
                                           
                                                             
                 
         
        
 
                                             
                                                                     

                                                                         


                                                             
                                                                       
                                                                                                                      




                                                                              

                                                                                                                                  
                                 


                              

         
 
                                          
                                     
                                     



                                                             

         
 
                                 
                                           

         
 
                                          
                                                  

         
                                         
                                                                                          
                                                               
                                          

                                                                        

                                           
                        

         
                                  


                                                            
                                                     
                                             
                                                                                                   


                                                                                         
                                                                                                        


                                                                                  
                                              
                             
                                                                 

                                                          
                                                  

                     

                             

         
 
                                       
                         
                                                                                                                 
                    

     
                                                                      
                                                    
                                                  

                                              
                                             
                            

                               
                                                                                                  
                                                              
                  

                                    



                                                                                                      
                                         
                                             
                                 


                                                                                                    
                                                                                                        


                                                                      
                         
                 
         



                                                                                                
                                                     
                                                                                                                


                                                                              

                                 




                       
                                                                             
                                                                                           

                                                               
                                 
                                              
                                                                                                                        
                                                                                                                
                                                                 
                                                              

                                                          

                                                  


                 
 
 

                               
     
 

                                                         

                                     
                                                                          
     
 


                                                                                     

                                                 


                                                                                 
                             
                                                                                         
                                                                                





                                                                                        

                                                        

                                                                                  
                                                                                    







                                       
    
 
                                        
                                                 

     
 
                                                 
                                                          

     
 

                                                              

                                                





                                      
                                                                                           
                                                                                                  
                                                       
 
                   

                               
                                                   
                                                                


                                                      
                                                                  
                                                                                      
                                                                          
                                                                                       
                     
                                                                                    
                                    
                     
             
                                                
                                                         
                    
                      

                               


                     



                                



                                                                                         

                               
                                                   
                                                                


                                                      
                                                                
                                                                                      
                                                                      

                                                                                       
                                                                

                                
                                              


                             
                               


                     





                                                        

                                                            






                                            
                               
                                                                  

     
                             
                                                                

     
                                 
                                                                    

     
                                   

                                                           
     



                                           


                                  




                                                                                                      
         



                                                                     

































                                                                                                     
 
/*******************************************************************************
 * Copyright (c) 2004, 2008 IBM Corporation 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:
 *     Andrew Niefer (IBM Corporation) - initial API and implementation
 *     Markus Schorn (Wind River Systems)
 *******************************************************************************/
package org.eclipse.cdt.internal.core.dom.parser.cpp;

import org.eclipse.cdt.core.dom.ILinkage;
import org.eclipse.cdt.core.dom.ast.ASTTypeUtil;
import org.eclipse.cdt.core.dom.ast.DOMException;
import org.eclipse.cdt.core.dom.ast.IASTDeclSpecifier;
import org.eclipse.cdt.core.dom.ast.IASTDeclaration;
import org.eclipse.cdt.core.dom.ast.IASTDeclarator;
import org.eclipse.cdt.core.dom.ast.IASTFunctionDefinition;
import org.eclipse.cdt.core.dom.ast.IASTName;
import org.eclipse.cdt.core.dom.ast.IASTNode;
import org.eclipse.cdt.core.dom.ast.IASTParameterDeclaration;
import org.eclipse.cdt.core.dom.ast.IASTSimpleDeclaration;
import org.eclipse.cdt.core.dom.ast.IASTStandardFunctionDeclarator;
import org.eclipse.cdt.core.dom.ast.IASTTranslationUnit;
import org.eclipse.cdt.core.dom.ast.IASTTypeId;
import org.eclipse.cdt.core.dom.ast.IBinding;
import org.eclipse.cdt.core.dom.ast.IFunctionType;
import org.eclipse.cdt.core.dom.ast.IParameter;
import org.eclipse.cdt.core.dom.ast.IProblemBinding;
import org.eclipse.cdt.core.dom.ast.IScope;
import org.eclipse.cdt.core.dom.ast.IType;
import org.eclipse.cdt.core.dom.ast.cpp.ICPPASTDeclSpecifier;
import org.eclipse.cdt.core.dom.ast.cpp.ICPPASTFunctionDeclarator;
import org.eclipse.cdt.core.dom.ast.cpp.ICPPASTQualifiedName;
import org.eclipse.cdt.core.dom.ast.cpp.ICPPBlockScope;
import org.eclipse.cdt.core.dom.ast.cpp.ICPPClassScope;
import org.eclipse.cdt.core.dom.ast.cpp.ICPPFunction;
import org.eclipse.cdt.core.dom.ast.cpp.ICPPFunctionType;
import org.eclipse.cdt.core.parser.util.ArrayUtil;
import org.eclipse.cdt.internal.core.dom.Linkage;
import org.eclipse.cdt.internal.core.dom.parser.ASTInternal;
import org.eclipse.cdt.internal.core.dom.parser.ASTNode;
import org.eclipse.cdt.internal.core.dom.parser.ProblemBinding;
import org.eclipse.cdt.internal.core.dom.parser.cpp.semantics.CPPVisitor;
import org.eclipse.core.runtime.PlatformObject;

/**
 * Binding for c++ function
 */
public class CPPFunction extends PlatformObject implements ICPPFunction, ICPPInternalFunction {

    public static class CPPFunctionProblem extends ProblemBinding implements ICPPFunction {
        public CPPFunctionProblem(IASTNode node, int id, char[] arg) {
            super(node, id, arg);
        }

        public IParameter[] getParameters() throws DOMException {
            throw new DOMException(this);
        }

        public IScope getFunctionScope() throws DOMException {
            throw new DOMException(this);
        }

        public ICPPFunctionType getType() throws DOMException {
            throw new DOMException(this);
        }
        public boolean isStatic() throws DOMException {
            throw new DOMException(this);
        }
        public String[] getQualifiedName() throws DOMException {
            throw new DOMException(this);
        }
        public char[][] getQualifiedNameCharArray() throws DOMException {
            throw new DOMException(this);
        }
        public boolean isGloballyQualified() throws DOMException {
            throw new DOMException(this);
        }
        public boolean isMutable() throws DOMException {
            throw new DOMException(this);
        }
        public boolean isInline() throws DOMException {
            throw new DOMException(this);
        }
        public boolean isExternC() throws DOMException {
            throw new DOMException(this);
        }
        public boolean isExtern() throws DOMException {
            throw new DOMException(this);
        }
        public boolean isAuto() throws DOMException {
            throw new DOMException(this);
        }
        public boolean isRegister() throws DOMException {
            throw new DOMException(this);
        }
        public boolean takesVarArgs() throws DOMException {
            throw new DOMException(this);
        }
		public IType[] getExceptionSpecification() throws DOMException {
            throw new DOMException(this);
		}
    }
    
	protected ICPPASTFunctionDeclarator[] declarations;
	protected ICPPASTFunctionDeclarator definition;
	protected ICPPFunctionType type = null;
	
	private static final int FULLY_RESOLVED         = 1;
	private static final int RESOLUTION_IN_PROGRESS = 1 << 1;
	private int bits = 0;
	
	public CPPFunction(ICPPASTFunctionDeclarator declarator) {
	    if (declarator != null) {
			IASTNode parent = CPPVisitor.findOutermostDeclarator(declarator).getParent();
			if (parent instanceof IASTFunctionDefinition)
				definition = declarator;
			else
				declarations = new ICPPASTFunctionDeclarator[] { declarator };
	    
		    IASTName name= getASTName();
		    name.setBinding(this);
	    }
	}
	
	private void resolveAllDeclarations() {
	    if ((bits & (FULLY_RESOLVED | RESOLUTION_IN_PROGRESS)) == 0) {
	        bits |= RESOLUTION_IN_PROGRESS;
		    IASTTranslationUnit tu = null;
	        if (definition != null) {
	            tu = definition.getTranslationUnit();
	        } else if (declarations != null) {
	            tu = declarations[0].getTranslationUnit();
	        } else {
	            //implicit binding
	            IScope scope = getScope();
                try {
                    IASTNode node = ASTInternal.getPhysicalNodeOfScope(scope);
                    if (node != null) {
                    	tu = node.getTranslationUnit();
                    }
                } catch (DOMException e) {
                }
	        }
	        if (tu != null) {
	            CPPVisitor.getDeclarations(tu, this);
	        }
	        declarations = (ICPPASTFunctionDeclarator[]) ArrayUtil.trim(ICPPASTFunctionDeclarator.class,
	        		declarations);
	        bits |= FULLY_RESOLVED;
	        bits &= ~RESOLUTION_IN_PROGRESS;
	    }
	}
	
 
    public IASTNode[] getDeclarations() {
        return declarations;
    }


    public IASTNode getDefinition() {
        return definition;
    }
    
	public void addDefinition(IASTNode node) {
		ICPPASTFunctionDeclarator dtor = extractFunctionDtor(node);
		if (dtor != null) {
			updateParameterBindings(dtor);
			definition = dtor;
		}
	}
	
	public void addDeclaration(IASTNode node) {
		ICPPASTFunctionDeclarator dtor = extractFunctionDtor(node);
		if (dtor != null) {
			updateParameterBindings(dtor);

			if (declarations == null) {
				declarations = new ICPPASTFunctionDeclarator[] { dtor };
				return;
			}

			// Keep the lowest offset declaration in [0]
			if (declarations.length > 0 && ((ASTNode)node).getOffset() < ((ASTNode)declarations[0]).getOffset()) {
				declarations = (ICPPASTFunctionDeclarator[]) ArrayUtil.prepend(ICPPASTFunctionDeclarator.class,
						declarations, dtor);
			} else {
				declarations = (ICPPASTFunctionDeclarator[]) ArrayUtil.append(ICPPASTFunctionDeclarator.class,
						declarations, dtor);
			}
		}
	}
	
	private ICPPASTFunctionDeclarator extractFunctionDtor(IASTNode node) {
		if (node instanceof IASTName)
			node = node.getParent();
		if (node instanceof IASTDeclarator == false)
			return null;
		node= CPPVisitor.findTypeRelevantDeclarator((IASTDeclarator) node);
		if (node instanceof ICPPASTFunctionDeclarator == false)
			return null;
		
		return (ICPPASTFunctionDeclarator) node;
	}

	public void removeDeclaration(IASTNode node) {
		ICPPASTFunctionDeclarator dtor = extractFunctionDtor(node);
		if (definition == dtor) {
			definition = null;
			return;
		}
		if (declarations != null) {
			ArrayUtil.remove(declarations, dtor);
		}
	}
	

	public IParameter[] getParameters() {
	    IASTStandardFunctionDeclarator dtor = getPreferredDtor();
		IASTParameterDeclaration[] params = dtor.getParameters();
		int size = params.length;
		IParameter[] result = new IParameter[ size ];
		if (size > 0) {
			for (int i = 0; i < size; i++) {
				IASTParameterDeclaration p = params[i];
				final IASTName name = CPPVisitor.findInnermostDeclarator(p.getDeclarator()).getName();
				final IBinding binding= name.resolveBinding();
				if (binding instanceof IParameter) {
					result[i]= (IParameter) binding;
				}
				else {
					result[i] = new CPPParameter.CPPParameterProblem(p, IProblemBinding.SEMANTIC_INVALID_TYPE,
							name.toCharArray());
				}
			}
		}
		return result;
	}


	public IScope getFunctionScope() {
	    resolveAllDeclarations();
	    if (definition != null) {
			return definition.getFunctionScope();
	    } 
	        
	    return declarations[0].getFunctionScope();
	}


	public String getName() {
	    return getASTName().toString();
	}


	public char[] getNameCharArray() {
		return getASTName().getSimpleID();
	}
	
	protected IASTName getASTName() {
		IASTDeclarator dtor = (definition != null) ? definition : declarations[0];
		dtor= CPPVisitor.findInnermostDeclarator(dtor);
	    IASTName name= dtor.getName();
	    if (name instanceof ICPPASTQualifiedName) {
	        IASTName[] ns = ((ICPPASTQualifiedName)name).getNames();
	        name = ns[ ns.length - 1 ];
	    }
	    return name;
	}

	public IScope getScope() {
	    IASTName n = getASTName();
	    IScope scope = CPPVisitor.getContainingScope(n);
	    if (scope instanceof ICPPClassScope) {
	    	ICPPASTDeclSpecifier declSpec = null;
		    if (definition != null) {
		    	IASTNode node = CPPVisitor.findOutermostDeclarator(definition).getParent();
		        IASTFunctionDefinition def = (IASTFunctionDefinition) node;
			    declSpec = (ICPPASTDeclSpecifier) def.getDeclSpecifier();    
		    } else {
		    	IASTNode node = CPPVisitor.findOutermostDeclarator(declarations[0]).getParent();
		        IASTSimpleDeclaration decl = (IASTSimpleDeclaration)node; 
		        declSpec = (ICPPASTDeclSpecifier) decl.getDeclSpecifier();
		    }
		    if (declSpec.isFriend()) {
		        try {
	                while (scope instanceof ICPPClassScope) {
		                scope = scope.getParent();
	                }
		        } catch (DOMException e) {
	            }
		    }
	    }
		return scope;
	}


    public ICPPFunctionType getType() {
        if (type == null)
            type = (ICPPFunctionType) CPPVisitor.createType((definition != null) ? definition : declarations[0]);
        return type;
    }

    public IBinding resolveParameter(IASTParameterDeclaration param) {
        IASTDeclarator dtor = param.getDeclarator();
        while (dtor.getNestedDeclarator() != null)
            dtor = dtor.getNestedDeclarator();
    	IASTName name = dtor.getName();
    	IBinding binding = name.getBinding();
    	if (binding != null)
    		return binding;
		
    	IASTStandardFunctionDeclarator fdtor = (IASTStandardFunctionDeclarator) param.getParent();
    	IASTParameterDeclaration[] ps = fdtor.getParameters();
    	int i = 0;
    	for (; i < ps.length; i++) {
    		if (param == ps[i])
    			break;
    	}
    	
    	//create a new binding and set it for the corresponding parameter in all known defns and decls
    	binding = new CPPParameter(name);
    	IASTParameterDeclaration temp = null;
    	if (definition != null) {
    		IASTParameterDeclaration[] paramDecls = definition.getParameters();
    		if (paramDecls.length > i) { // This will be less than i if we have a void parameter
	    		temp = paramDecls[i];
	    		IASTName n = CPPVisitor.findInnermostDeclarator(temp.getDeclarator()).getName();
	    		if (n != name) {
	    		    n.setBinding(binding);
	    		    ((CPPParameter)binding).addDeclaration(n);
	    		}
    		}
    	}
    	if (declarations != null) {
    		for (int j = 0; j < declarations.length && declarations[j] != null; j++) {
    			IASTParameterDeclaration[] paramDecls = declarations[j].getParameters();
    			if (paramDecls.length > i) {
	    			temp = paramDecls[i];
	        		IASTName n = CPPVisitor.findInnermostDeclarator(temp.getDeclarator()).getName();
	        		if (n != name) {
	        		    n.setBinding(binding);
	        		    ((CPPParameter)binding).addDeclaration(n);
	        		}
    			}
    		}
    	}
    	return binding;
    }
    
    protected void updateParameterBindings(ICPPASTFunctionDeclarator fdtor) {
    	ICPPASTFunctionDeclarator orig = definition != null ? definition : declarations[0];
    	IASTParameterDeclaration[] ops = orig.getParameters();
    	IASTParameterDeclaration[] nps = fdtor.getParameters();
    	CPPParameter temp = null;
    	for (int i = 0; i < ops.length; i++) {
    		temp = (CPPParameter) CPPVisitor.findInnermostDeclarator(ops[i].getDeclarator()).getName().getBinding();
    		if (temp != null && nps.length > i) {		//length could be different, ie 0 or 1 with void
    		    IASTDeclarator dtor = nps[i].getDeclarator();
    		    while (dtor.getNestedDeclarator() != null)
    		        dtor = dtor.getNestedDeclarator();
    		    IASTName name = dtor.getName();
    			name.setBinding(temp);
    			temp.addDeclaration(name);
    		}
    	}
    }


    public boolean isStatic() {
        return isStatic(true);
    }

    public boolean isStatic(boolean resolveAll) {
        if (resolveAll && (bits & FULLY_RESOLVED) == 0) {
            resolveAllDeclarations();
        }
		return hasStorageClass(this, IASTDeclSpecifier.sc_static);
    }

//	static public boolean isStatic
//        //2 state bits, most significant = whether or not we've figure this out yet
//        //least significant = whether or not we are static
//        int state = (bits & IS_STATIC) >> 2;
//        if (state > 1) return (state % 2 != 0);
//        
//        IASTDeclSpecifier declSpec = null;
//        IASTFunctionDeclarator dtor = (IASTFunctionDeclarator) getDefinition();
//        if (dtor != null) {
//	        declSpec = ((IASTFunctionDefinition)dtor.getParent()).getDeclSpecifier();
//	        if (declSpec.getStorageClass() == IASTDeclSpecifier.sc_static) {
//	            bits |= 3 << 2;
//	            return true;
//	        }
//        }
//        
//        IASTFunctionDeclarator[] dtors = (IASTFunctionDeclarator[]) getDeclarations();
//        if (dtors != null) {
//	        for (int i = 0; i < dtors.length; i++) {
//	            IASTNode parent = dtors[i].getParent();
//	            declSpec = ((IASTSimpleDeclaration)parent).getDeclSpecifier();
//	            if (declSpec.getStorageClass() == IASTDeclSpecifier.sc_static) {
//	                bits |= 3 << 2;
//	                return true;
//	            }
//	        }
//        }
//        bits |= 2 << 2;
//        return false;
//    }
    

    public String[] getQualifiedName() {
        return CPPVisitor.getQualifiedName(this);
    }


    public char[][] getQualifiedNameCharArray() {
        return CPPVisitor.getQualifiedNameCharArray(this);
    }


    public boolean isGloballyQualified() throws DOMException {
        IScope scope = getScope();
        while (scope != null) {
            if (scope instanceof ICPPBlockScope)
                return false;
            scope = scope.getParent();
        }
        return true;
    }

	static public boolean hasStorageClass(ICPPInternalFunction function, int storage) {
	    ICPPASTFunctionDeclarator dtor = (ICPPASTFunctionDeclarator) function.getDefinition();
	    IASTNode[] ds = function.getDeclarations();

        int i = -1;
        do {
            if (dtor != null) {
                IASTNode parent = dtor.getParent();
	            while (!(parent instanceof IASTDeclaration))
	                parent = parent.getParent();
	            
	            IASTDeclSpecifier declSpec = null;
	            if (parent instanceof IASTSimpleDeclaration) {
	                declSpec = ((IASTSimpleDeclaration)parent).getDeclSpecifier();
	            } else if (parent instanceof IASTFunctionDefinition) {
	                declSpec = ((IASTFunctionDefinition)parent).getDeclSpecifier();
	            }
	            if (declSpec != null && declSpec.getStorageClass() == storage) {
	            	return true;
	            }
            }
            if (ds != null && ++i < ds.length) {
            	dtor = (ICPPASTFunctionDeclarator) ds[i];
            } else {
                break;
            }
        } while (dtor != null);
        return false;
	}
	
    public boolean isMutable() {
        return false;
    }

    public boolean isInline() throws DOMException {
	    ICPPASTFunctionDeclarator dtor = (ICPPASTFunctionDeclarator) getDefinition();
        ICPPASTFunctionDeclarator[] ds = (ICPPASTFunctionDeclarator[]) getDeclarations();
        int i = -1;
        do {
            if (dtor != null) {
                IASTNode parent = dtor.getParent();
	            while (!(parent instanceof IASTDeclaration))
	                parent = parent.getParent();
	            
	            IASTDeclSpecifier declSpec = null;
	            if (parent instanceof IASTSimpleDeclaration)
	                declSpec = ((IASTSimpleDeclaration)parent).getDeclSpecifier();
	            else if (parent instanceof IASTFunctionDefinition)
	                declSpec = ((IASTFunctionDefinition)parent).getDeclSpecifier();
	            
	            if (declSpec != null && declSpec.isInline())
                    return true;
            }
            if (ds != null && ++i < ds.length)
                dtor = ds[i];
            else
                break;
        } while (dtor != null);
        return false;
    }

    public boolean isExternC() throws DOMException {
	    if (CPPVisitor.isExternC(getDefinition())) {
	    	return true;
	    }
        IASTNode[] ds= getDeclarations();
        if (ds != null) {
        	for (IASTNode element : ds) {
        		if (CPPVisitor.isExternC(element)) {
        			return true;
        		}
			}
        }
        return false;
    }

    public boolean isExtern() {
        return hasStorageClass(this, IASTDeclSpecifier.sc_extern);
    }

    public boolean isAuto() {
        return hasStorageClass(this, IASTDeclSpecifier.sc_auto);
    }

    public boolean isRegister() {
        return hasStorageClass(this, IASTDeclSpecifier.sc_register);
    }

    public boolean takesVarArgs() {
        ICPPASTFunctionDeclarator dtor= getPreferredDtor();
        return dtor != null ? dtor.takesVarArgs() : false;
    }
    
	public ILinkage getLinkage() {
		return Linkage.CPP_LINKAGE;
	}
	
	@Override
	public String toString() {
		StringBuilder result = new StringBuilder();
		result.append(getName());
		IFunctionType t = getType();
		result.append(t != null ? ASTTypeUtil.getParameterTypeString(t) : "()"); //$NON-NLS-1$
		return result.toString();
	}
	
	public IBinding getOwner() throws DOMException {
		return CPPVisitor.findNameOwner(getASTName(), false);
	}

	public IType[] getExceptionSpecification() throws DOMException {
		ICPPASTFunctionDeclarator declarator = getPreferredDtor();
		if (declarator != null) {
			IASTTypeId[] astTypeIds= declarator.getExceptionSpecification();
			if (astTypeIds.equals(ICPPASTFunctionDeclarator.NO_EXCEPTION_SPECIFICATION)) 
				return null;
			
			if (astTypeIds.equals(IASTTypeId.EMPTY_TYPEID_ARRAY)) 
				return IType.EMPTY_TYPE_ARRAY;
			
			IType[] typeIds = new IType[astTypeIds.length];
			for (int i=0; i<astTypeIds.length; ++i) {
				typeIds[i] = CPPVisitor.createType(astTypeIds[i]);
			}
			return typeIds;
		}
		return null;
	}
	
	protected ICPPASTFunctionDeclarator getPreferredDtor() {
        ICPPASTFunctionDeclarator dtor = (ICPPASTFunctionDeclarator) getDefinition();
        if (dtor != null) 
        	return dtor;

        ICPPASTFunctionDeclarator[] dtors = (ICPPASTFunctionDeclarator[]) getDeclarations();
        if (dtors != null) {
        	for (ICPPASTFunctionDeclarator declarator : dtors) {
        		if (declarator != null) 
        			return declarator;
        	}
        }
        return null;
	}
}

Back to the top