001    /*
002     * Copyright 2010-2013 JetBrains s.r.o.
003     *
004     * Licensed under the Apache License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.apache.org/licenses/LICENSE-2.0
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    
017    package org.jetbrains.jet.lang.resolve.calls;
018    
019    import org.jetbrains.annotations.NotNull;
020    import org.jetbrains.jet.lang.descriptors.CallableDescriptor;
021    import org.jetbrains.jet.lang.descriptors.CallableMemberDescriptor;
022    import org.jetbrains.jet.lang.descriptors.Visibilities;
023    import org.jetbrains.jet.lang.resolve.calls.context.BasicCallResolutionContext;
024    import org.jetbrains.jet.lang.resolve.calls.model.ResolvedCall;
025    import org.jetbrains.jet.lang.resolve.scopes.JetScope;
026    
027    import static org.jetbrains.jet.lang.resolve.BindingContext.NEED_SYNTHETIC_ACCESSOR;
028    
029    public class NeedSyntheticCallResolverExtension implements CallResolverExtension {
030    
031        @Override
032        public  <F extends CallableDescriptor> void run(
033                @NotNull ResolvedCall<F> resolvedCall,
034                @NotNull BasicCallResolutionContext context
035        ) {
036            CallableDescriptor targetDescriptor = resolvedCall.getResultingDescriptor();
037            if (needSyntheticAccessor(context.scope, targetDescriptor)) {
038                context.trace.record(NEED_SYNTHETIC_ACCESSOR, (CallableMemberDescriptor) targetDescriptor.getOriginal(), Boolean.TRUE);
039            }
040        }
041    
042        //Necessary synthetic accessors in outer classes generated via old logic: CodegenContext.getAccessor
043        //Generation of accessors in nested classes (to invoke from outer,
044        //      e.g.: from class to classobject) controlled via NEED_SYNTHETIC_ACCESSOR slice
045        private boolean needSyntheticAccessor(JetScope invokationScope, CallableDescriptor targetDescriptor) {
046            return targetDescriptor instanceof CallableMemberDescriptor &&
047                   targetDescriptor.getVisibility() == Visibilities.PRIVATE &&
048                   targetDescriptor.getContainingDeclaration() != invokationScope.getContainingDeclaration().getContainingDeclaration();
049        }
050    }