1e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver/*
2e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * Copyright 2012, Google Inc.
3e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * All rights reserved.
4e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver *
5e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * Redistribution and use in source and binary forms, with or without
6e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * modification, are permitted provided that the following conditions are
7e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * met:
8e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver *
9e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver *     * Redistributions of source code must retain the above copyright
10e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * notice, this list of conditions and the following disclaimer.
11e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver *     * Redistributions in binary form must reproduce the above
12e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * copyright notice, this list of conditions and the following disclaimer
13e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * in the documentation and/or other materials provided with the
14e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * distribution.
15e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver *     * Neither the name of Google Inc. nor the names of its
16e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * contributors may be used to endorse or promote products derived from
17e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * this software without specific prior written permission.
18e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver *
19e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver */
31e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver
32e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruverpackage org.jf.dexlib2.immutable;
33e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver
34e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruverimport com.google.common.collect.ImmutableList;
3522c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruverimport com.google.common.collect.ImmutableSet;
3622c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruverimport com.google.common.collect.ImmutableSortedSet;
3722c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruverimport com.google.common.collect.Ordering;
38f81150ad43efac4d590f6b7ea1425896b7ffb011Ben Gruverimport org.jf.dexlib2.base.reference.BaseMethodReference;
39f4662586da3c902c3a0ed21a174b814e248c7fb8Ben Gruverimport org.jf.dexlib2.iface.Annotation;
40f4662586da3c902c3a0ed21a174b814e248c7fb8Ben Gruverimport org.jf.dexlib2.iface.Method;
41f4662586da3c902c3a0ed21a174b814e248c7fb8Ben Gruverimport org.jf.dexlib2.iface.MethodImplementation;
42f4662586da3c902c3a0ed21a174b814e248c7fb8Ben Gruverimport org.jf.dexlib2.iface.MethodParameter;
4322c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruverimport org.jf.util.ImmutableConverter;
442d7e1111358e2b8cc951a46dc8b0217a7fa0deadBen Gruverimport org.jf.util.ImmutableUtils;
45e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver
46e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruverimport javax.annotation.Nonnull;
47e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruverimport javax.annotation.Nullable;
4822c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruverimport java.util.Set;
49e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver
50f81150ad43efac4d590f6b7ea1425896b7ffb011Ben Gruverpublic class ImmutableMethod extends BaseMethodReference implements Method {
5122c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver    @Nonnull protected final String definingClass;
5222c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver    @Nonnull protected final String name;
5322c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver    @Nonnull protected final ImmutableList<? extends ImmutableMethodParameter> parameters;
5422c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver    @Nonnull protected final String returnType;
5522c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver    protected final int accessFlags;
5622c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver    @Nonnull protected final ImmutableSet<? extends ImmutableAnnotation> annotations;
5722c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver    @Nullable protected final ImmutableMethodImplementation methodImplementation;
58e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver
5922c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver    public ImmutableMethod(@Nonnull String definingClass,
60a8e05220c14778d93c97911044ff5124aadbd77cBen Gruver                           @Nonnull String name,
615fa302678ce3a8e08fa8d2e8dbc5424781e751a6Ben Gruver                           @Nullable Iterable<? extends MethodParameter> parameters,
62e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                           @Nonnull String returnType,
63e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                           int accessFlags,
6422c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver                           @Nullable Set<? extends Annotation> annotations,
65e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                           @Nullable MethodImplementation methodImplementation) {
6622c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver        this.definingClass = definingClass;
67e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        this.name = name;
68e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        this.parameters = ImmutableMethodParameter.immutableListOf(parameters);
69e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        this.returnType = returnType;
70e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        this.accessFlags = accessFlags;
7122c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver        this.annotations = ImmutableAnnotation.immutableSetOf(annotations);
72e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        this.methodImplementation = ImmutableMethodImplementation.of(methodImplementation);
73e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver    }
74e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver
7522c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver    public ImmutableMethod(@Nonnull String definingClass,
76a8e05220c14778d93c97911044ff5124aadbd77cBen Gruver                           @Nonnull String name,
77e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                           @Nullable ImmutableList<? extends ImmutableMethodParameter> parameters,
78e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                           @Nonnull String returnType,
79e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                           int accessFlags,
8022c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver                           @Nullable ImmutableSet<? extends ImmutableAnnotation> annotations,
81e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                           @Nullable ImmutableMethodImplementation methodImplementation) {
8222c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver        this.definingClass = definingClass;
83e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        this.name = name;
842d7e1111358e2b8cc951a46dc8b0217a7fa0deadBen Gruver        this.parameters = ImmutableUtils.nullToEmptyList(parameters);
85e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        this.returnType = returnType;
86e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        this.accessFlags = accessFlags;
8722c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver        this.annotations = ImmutableUtils.nullToEmptySet(annotations);
88e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        this.methodImplementation = methodImplementation;
89e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver    }
90e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver
91e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver    public static ImmutableMethod of(Method method) {
92e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        if (method instanceof ImmutableMethod) {
93e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver            return (ImmutableMethod)method;
94e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        }
95e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver        return new ImmutableMethod(
9622c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver                method.getDefiningClass(),
97e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                method.getName(),
98e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                method.getParameters(),
99e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                method.getReturnType(),
100e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                method.getAccessFlags(),
101e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                method.getAnnotations(),
102e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                method.getImplementation());
103e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver    }
104e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver
1057b89cbdf6b7636fed95d1773c2f3ce1c4d9ef0b1Ben Gruver    @Override @Nonnull public String getDefiningClass() { return definingClass; }
1067b89cbdf6b7636fed95d1773c2f3ce1c4d9ef0b1Ben Gruver    @Override @Nonnull public String getName() { return name; }
1077b89cbdf6b7636fed95d1773c2f3ce1c4d9ef0b1Ben Gruver    @Override @Nonnull public ImmutableList<? extends CharSequence> getParameterTypes() { return parameters; }
1087b89cbdf6b7636fed95d1773c2f3ce1c4d9ef0b1Ben Gruver    @Override @Nonnull public ImmutableList<? extends ImmutableMethodParameter> getParameters() { return parameters; }
1097b89cbdf6b7636fed95d1773c2f3ce1c4d9ef0b1Ben Gruver    @Override @Nonnull public String getReturnType() { return returnType; }
1107b89cbdf6b7636fed95d1773c2f3ce1c4d9ef0b1Ben Gruver    @Override public int getAccessFlags() { return accessFlags; }
1117b89cbdf6b7636fed95d1773c2f3ce1c4d9ef0b1Ben Gruver    @Override @Nonnull public ImmutableSet<? extends ImmutableAnnotation> getAnnotations() { return annotations; }
1127b89cbdf6b7636fed95d1773c2f3ce1c4d9ef0b1Ben Gruver    @Override @Nullable public ImmutableMethodImplementation getImplementation() { return methodImplementation; }
113e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver
114e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver    @Nonnull
11522c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver    public static ImmutableSortedSet<ImmutableMethod> immutableSetOf(@Nullable Iterable<? extends Method> list) {
11622c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver        return CONVERTER.toSortedSet(Ordering.natural(), list);
117e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver    }
118e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver
11922c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver    private static final ImmutableConverter<ImmutableMethod, Method> CONVERTER =
12022c3185bb7c8618437eabe6c597549e0989ec4e6Ben Gruver            new ImmutableConverter<ImmutableMethod, Method>() {
121e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                @Override
122ddf20219422e40a1e60268d8049093602d7bacf8Ben Gruver                protected boolean isImmutable(@Nonnull Method item) {
123e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                    return item instanceof ImmutableMethod;
124e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                }
125e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver
126ddf20219422e40a1e60268d8049093602d7bacf8Ben Gruver                @Nonnull
127e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                @Override
128ddf20219422e40a1e60268d8049093602d7bacf8Ben Gruver                protected ImmutableMethod makeImmutable(@Nonnull Method item) {
129e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                    return ImmutableMethod.of(item);
130e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver                }
131e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver            };
132e2f00f0eba79723388f2152db7b68c64872d7eb3Ben Gruver}
133