1 /*
2  * Copyright (c) 2010, 2015, Oracle and/or its affiliates. All rights reserved.
3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4  *
5  * This code is free software; you can redistribute it and/or modify it
6  * under the terms of the GNU General Public License version 2 only, as
7  * published by the Free Software Foundation.
8  *
9  * This code is distributed in the hope that it will be useful, but WITHOUT
10  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
12  * version 2 for more details (a copy is included in the LICENSE file that
13  * accompanied this code).
14  *
15  * You should have received a copy of the GNU General Public License version
16  * 2 along with this work; if not, write to the Free Software Foundation,
17  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18  *
19  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20  * or visit www.oracle.com if you need additional information or have any
21  * questions.
22  */
23 package jdk.vm.ci.code;
24 
25 import java.util.Arrays;
26 import java.util.Collections;
27 import java.util.IdentityHashMap;
28 import java.util.Set;
29 
30 import jdk.vm.ci.meta.JavaKind;
31 import jdk.vm.ci.meta.JavaValue;
32 import jdk.vm.ci.meta.ResolvedJavaField;
33 import jdk.vm.ci.meta.ResolvedJavaType;
34 
35 /**
36  * An instance of this class represents an object whose allocation was removed by escape analysis.
37  * The information stored in the {@link VirtualObject} is used during deoptimization to recreate the
38  * object.
39  */
40 public final class VirtualObject implements JavaValue {
41 
42     private final ResolvedJavaType type;
43     private JavaValue[] values;
44     private JavaKind[] slotKinds;
45     private final int id;
46 
47     /**
48      * Creates a new {@link VirtualObject} for the given type, with the given fields. If
49      * {@code type} is an instance class then {@code values} provides the values for the fields
50      * returned by {@link ResolvedJavaType#getInstanceFields(boolean) getInstanceFields(true)}. If
51      * {@code type} is an array then the length of the values array determines the reallocated array
52      * length.
53      *
54      * @param type the type of the object whose allocation was removed during compilation. This can
55      *            be either an instance of an array type.
56      * @param id a unique id that identifies the object within the debug information for one
57      *            position in the compiled code.
58      * @return a new {@link VirtualObject} instance.
59      */
get(ResolvedJavaType type, int id)60     public static VirtualObject get(ResolvedJavaType type, int id) {
61         return new VirtualObject(type, id);
62     }
63 
VirtualObject(ResolvedJavaType type, int id)64     private VirtualObject(ResolvedJavaType type, int id) {
65         this.type = type;
66         this.id = id;
67     }
68 
appendValue(StringBuilder buf, JavaValue value, Set<VirtualObject> visited)69     private static StringBuilder appendValue(StringBuilder buf, JavaValue value, Set<VirtualObject> visited) {
70         if (value instanceof VirtualObject) {
71             VirtualObject vo = (VirtualObject) value;
72             buf.append("vobject:").append(vo.type.toJavaName(false)).append(':').append(vo.id);
73             if (!visited.contains(vo)) {
74                 visited.add(vo);
75                 buf.append('{');
76                 if (vo.values == null) {
77                     buf.append("<uninitialized>");
78                 } else {
79                     if (vo.type.isArray()) {
80                         for (int i = 0; i < vo.values.length; i++) {
81                             if (i != 0) {
82                                 buf.append(',');
83                             }
84                             buf.append(i).append('=');
85                             appendValue(buf, vo.values[i], visited);
86                         }
87                     } else {
88                         ResolvedJavaField[] fields = vo.type.getInstanceFields(true);
89                         assert fields.length == vo.values.length : vo.type + ", fields=" + Arrays.toString(fields) + ", values=" + Arrays.toString(vo.values);
90                         for (int i = 0; i < vo.values.length; i++) {
91                             if (i != 0) {
92                                 buf.append(',');
93                             }
94                             buf.append(fields[i].getName()).append('=');
95                             appendValue(buf, vo.values[i], visited);
96                         }
97                     }
98                 }
99                 buf.append('}');
100             }
101         } else {
102             buf.append(value);
103         }
104         return buf;
105     }
106 
107     @Override
toString()108     public String toString() {
109         Set<VirtualObject> visited = Collections.newSetFromMap(new IdentityHashMap<VirtualObject, Boolean>());
110         return appendValue(new StringBuilder(), this, visited).toString();
111     }
112 
113     /**
114      * Returns the type of the object whose allocation was removed during compilation. This can be
115      * either an instance of an array type.
116      */
getType()117     public ResolvedJavaType getType() {
118         return type;
119     }
120 
121     /**
122      * Returns the array containing all the values to be stored into the object when it is
123      * recreated. This field is intentional exposed as a mutable array that a compiler may modify
124      * (e.g. during register allocation).
125      */
126     @SuppressFBWarnings(value = "EI_EXPOSE_REP", justification = "`values` is intentional mutable")//
getValues()127     public JavaValue[] getValues() {
128         return values;
129     }
130 
131     /**
132      * Returns the kind of the value at {@code index}.
133      */
getSlotKind(int index)134     public JavaKind getSlotKind(int index) {
135         return slotKinds[index];
136     }
137 
138     /**
139      * Returns the unique id that identifies the object within the debug information for one
140      * position in the compiled code.
141      */
getId()142     public int getId() {
143         return id;
144     }
145 
146     /**
147      * Overwrites the current set of values with a new one.
148      *
149      * @param values an array containing all the values to be stored into the object when it is
150      *            recreated.
151      * @param slotKinds an array containing the Java kinds of the values. This must have the same
152      *            length as {@code values}. This array is now owned by this object and must not be
153      *            mutated by the caller.
154      */
155     @SuppressFBWarnings(value = "EI_EXPOSE_REP2", justification = "caller transfers ownership of `slotKinds`")
setValues(JavaValue[] values, JavaKind[] slotKinds)156     public void setValues(JavaValue[] values, JavaKind[] slotKinds) {
157         assert values.length == slotKinds.length;
158         this.values = values;
159         this.slotKinds = slotKinds;
160     }
161 
162     @Override
hashCode()163     public int hashCode() {
164         return 42 + type.hashCode();
165     }
166 
167     @Override
equals(Object o)168     public boolean equals(Object o) {
169         if (o == this) {
170             return true;
171         }
172         if (o instanceof VirtualObject) {
173             VirtualObject l = (VirtualObject) o;
174             if (!l.type.equals(type) || l.values.length != values.length) {
175                 return false;
176             }
177             for (int i = 0; i < values.length; i++) {
178                 /*
179                  * Virtual objects can form cycles. Calling equals() could therefore lead to
180                  * infinite recursion.
181                  */
182                 if (!same(values[i], l.values[i])) {
183                     return false;
184                 }
185             }
186             return true;
187         }
188         return false;
189     }
190 
same(Object o1, Object o2)191     private static boolean same(Object o1, Object o2) {
192         return o1 == o2;
193     }
194 }
195