summaryrefslogtreecommitdiff
path: root/java/java-impl/src/com/intellij/compilerOutputIndex/impl/quickInheritance/QuickMethodsIndex.java
blob: df11a049bc0d6f91c848f906462dbce564873bf6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
package com.intellij.compilerOutputIndex.impl.quickInheritance;

import com.intellij.compilerOutputIndex.api.descriptor.HashSetDataExternalizer;
import com.intellij.compilerOutputIndex.api.fs.AsmUtil;
import com.intellij.compilerOutputIndex.api.indexer.CompilerOutputBaseIndex;
import com.intellij.compilerOutputIndex.api.indexer.CompilerOutputIndexer;
import com.intellij.openapi.project.Project;
import com.intellij.openapi.util.Ref;
import com.intellij.util.indexing.DataIndexer;
import com.intellij.util.indexing.ID;
import com.intellij.util.indexing.StorageException;
import com.intellij.util.indexing.ValueContainer;
import com.intellij.util.io.EnumeratorStringDescriptor;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.jetbrains.asm4.ClassReader;
import org.jetbrains.asm4.ClassVisitor;
import org.jetbrains.asm4.MethodVisitor;
import org.jetbrains.asm4.Opcodes;

import java.util.*;

/**
 * @author Dmitry Batkovich
 */
public class QuickMethodsIndex extends CompilerOutputBaseIndex<String, Set<String>> {

  public static QuickMethodsIndex getInstance(final Project project) {
    return CompilerOutputIndexer.getInstance(project).getIndex(QuickMethodsIndex.class);
  }

  public QuickMethodsIndex() {
    super(new EnumeratorStringDescriptor(), new HashSetDataExternalizer<String>(new EnumeratorStringDescriptor()));
  }

  @Override
  protected ID<String, Set<String>> getIndexId() {
    return generateIndexId(QuickMethodsIndex.class);
  }

  @Override
  protected int getVersion() {
    return 0;
  }

  protected Set<String> getMethodsNames(final String classQName) {
    final Ref<Set<String>> methodsRef = Ref.create();
    try {
      myIndex.getData(classQName).forEach(new ValueContainer.ContainerAction<Set<String>>() {
        @Override
        public boolean perform(final int id, final Set<String> value) {
          methodsRef.set(value);
          return true;
        }
      });
      final Set<String> methods = methodsRef.get();
      return methods == null ? Collections.<String>emptySet() : methods;
    }
    catch (StorageException e) {
      throw new RuntimeException(e);
    }
  }

  @Override
  protected DataIndexer<String, Set<String>, ClassReader> getIndexer() {
    return new DataIndexer<String, Set<String>, ClassReader>() {
      @NotNull
      @Override
      public Map<String, Set<String>> map(final ClassReader inputData) {
        final Map<String, Set<String>> map = new HashMap<String, Set<String>>();
        inputData.accept(new ClassVisitor(Opcodes.ASM4) {

          private String myClassName;
          private final HashSet<String> myMethodNames = new HashSet<String>();

          @Override
          public void visit(final int i, final int i2, final String name, final String s2, final String s3, final String[] strings) {
            myClassName = AsmUtil.getQualifiedClassName(name);
          }

          @Override
          public void visitEnd() {
            map.put(myClassName, myMethodNames);
          }

          @Nullable
          @Override
          public MethodVisitor visitMethod(final int access,
                                           final String name,
                                           final String desc,
                                           final String sign,
                                           final String[] exceptions) {
            if ((access & Opcodes.ACC_STATIC) == 0) {
              myMethodNames.add(name);
            }
            return null;
          }
        }, Opcodes.ASM4);
        return map;
      }
    };
  }
}