blob: a1901f08f3e3fa5b7ab493240a1290cdf4219a6a [file] [log] [blame]
Jeff Haoc3acfc52016-08-29 14:18:26 -07001/*
2 * Copyright (C) 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Andreas Gampe8cf9cb32017-07-19 09:28:38 -070017#include <sstream>
Jeff Haoc3acfc52016-08-29 14:18:26 -070018#include <string>
19#include <vector>
Jeff Haoc3acfc52016-08-29 14:18:26 -070020
21#include <sys/types.h>
22#include <unistd.h>
23
Jeff Hao042e8982016-10-19 11:17:11 -070024#include "base/unix_file/fd_file.h"
Jeff Haoc3acfc52016-08-29 14:18:26 -070025#include "common_runtime_test.h"
David Sehr013fd802018-01-11 22:55:24 -080026#include "dex/art_dex_file_loader.h"
David Sehre1123402018-02-01 02:46:18 -080027#include "dex/base64_test_util.h"
David Sehr9e734c72018-01-04 17:56:19 -080028#include "dex/code_item_accessors-inl.h"
29#include "dex/dex_file-inl.h"
30#include "dex/dex_file_loader.h"
Mathieu Chartier75175552018-01-25 11:23:01 -080031#include "dexlayout.h"
David Sehr97c381e2017-02-01 15:09:58 -080032#include "exec_utils.h"
Mathieu Chartierd00e02b2017-05-24 12:04:13 -070033#include "jit/profile_compilation_info.h"
Jeff Haoc3acfc52016-08-29 14:18:26 -070034#include "utils.h"
35
36namespace art {
37
Jeff Hao042e8982016-10-19 11:17:11 -070038static const char kDexFileLayoutInputDex[] =
39 "ZGV4CjAzNQD1KW3+B8NAB0f2A/ZVIBJ0aHrGIqcpVTAUAgAAcAAAAHhWNBIAAAAAAAAAAIwBAAAH"
40 "AAAAcAAAAAQAAACMAAAAAQAAAJwAAAAAAAAAAAAAAAMAAACoAAAAAgAAAMAAAAAUAQAAAAEAADAB"
41 "AAA4AQAAQAEAAEgBAABNAQAAUgEAAGYBAAADAAAABAAAAAUAAAAGAAAABgAAAAMAAAAAAAAAAAAA"
42 "AAAAAAABAAAAAAAAAAIAAAAAAAAAAAAAAAAAAAACAAAAAAAAAAEAAAAAAAAAdQEAAAAAAAABAAAA"
43 "AAAAAAIAAAAAAAAAAgAAAAAAAAB/AQAAAAAAAAEAAQABAAAAaQEAAAQAAABwEAIAAAAOAAEAAQAB"
44 "AAAAbwEAAAQAAABwEAIAAAAOAAY8aW5pdD4ABkEuamF2YQAGQi5qYXZhAANMQTsAA0xCOwASTGph"
45 "dmEvbGFuZy9PYmplY3Q7AAFWAAQABw48AAQABw48AAAAAQAAgIAEgAIAAAEAAYCABJgCAAAACwAA"
46 "AAAAAAABAAAAAAAAAAEAAAAHAAAAcAAAAAIAAAAEAAAAjAAAAAMAAAABAAAAnAAAAAUAAAADAAAA"
47 "qAAAAAYAAAACAAAAwAAAAAEgAAACAAAAAAEAAAIgAAAHAAAAMAEAAAMgAAACAAAAaQEAAAAgAAAC"
48 "AAAAdQEAAAAQAAABAAAAjAEAAA==";
49
Jeff Haoe17f5892017-02-23 16:14:04 -080050// Dex file with catch handler unreferenced by try blocks.
51// Constructed by building a dex file with try/catch blocks and hex editing.
52static const char kUnreferencedCatchHandlerInputDex[] =
53 "ZGV4CjAzNQD+exd52Y0f9nY5x5GmInXq5nXrO6Kl2RV4AwAAcAAAAHhWNBIAAAAAAAAAANgCAAAS"
54 "AAAAcAAAAAgAAAC4AAAAAwAAANgAAAABAAAA/AAAAAQAAAAEAQAAAQAAACQBAAA0AgAARAEAANYB"
55 "AADeAQAA5gEAAO4BAAAAAgAADwIAACYCAAA9AgAAUQIAAGUCAAB5AgAAfwIAAIUCAACIAgAAjAIA"
56 "AKECAACnAgAArAIAAAQAAAAFAAAABgAAAAcAAAAIAAAACQAAAAwAAAAOAAAADAAAAAYAAAAAAAAA"
57 "DQAAAAYAAADIAQAADQAAAAYAAADQAQAABQABABAAAAAAAAAAAAAAAAAAAgAPAAAAAQABABEAAAAD"
58 "AAAAAAAAAAAAAAABAAAAAwAAAAAAAAADAAAAAAAAAMgCAAAAAAAAAQABAAEAAAC1AgAABAAAAHAQ"
59 "AwAAAA4AAwABAAIAAgC6AgAAIQAAAGIAAAAaAQoAbiACABAAYgAAABoBCwBuIAIAEAAOAA0AYgAA"
60 "ABoBAQBuIAIAEAAo8A0AYgAAABoBAgBuIAIAEAAo7gAAAAAAAAcAAQAHAAAABwABAAIBAg8BAhgA"
61 "AQAAAAQAAAABAAAABwAGPGluaXQ+AAZDYXRjaDEABkNhdGNoMgAQSGFuZGxlclRlc3QuamF2YQAN"
62 "TEhhbmRsZXJUZXN0OwAVTGphdmEvaW8vUHJpbnRTdHJlYW07ABVMamF2YS9sYW5nL0V4Y2VwdGlv"
63 "bjsAEkxqYXZhL2xhbmcvT2JqZWN0OwASTGphdmEvbGFuZy9TdHJpbmc7ABJMamF2YS9sYW5nL1N5"
64 "c3RlbTsABFRyeTEABFRyeTIAAVYAAlZMABNbTGphdmEvbGFuZy9TdHJpbmc7AARtYWluAANvdXQA"
65 "B3ByaW50bG4AAQAHDgAEAQAHDn17AncdHoseAAAAAgAAgYAExAIBCdwCAAANAAAAAAAAAAEAAAAA"
66 "AAAAAQAAABIAAABwAAAAAgAAAAgAAAC4AAAAAwAAAAMAAADYAAAABAAAAAEAAAD8AAAABQAAAAQA"
67 "AAAEAQAABgAAAAEAAAAkAQAAASAAAAIAAABEAQAAARAAAAIAAADIAQAAAiAAABIAAADWAQAAAyAA"
68 "AAIAAAC1AgAAACAAAAEAAADIAgAAABAAAAEAAADYAgAA";
69
Jeff Haoac462712017-03-02 10:59:43 -080070// Dex file with 0-size (catch all only) catch handler unreferenced by try blocks.
71// Constructed by building a dex file with try/catch blocks and hex editing.
72static const char kUnreferenced0SizeCatchHandlerInputDex[] =
73 "ZGV4CjAzNQCEbEEvMstSNpQpjPdfMEfUBS48cis2QRJoAwAAcAAAAHhWNBIAAAAAAAAAAMgCAAAR"
74 "AAAAcAAAAAcAAAC0AAAAAwAAANAAAAABAAAA9AAAAAQAAAD8AAAAAQAAABwBAAAsAgAAPAEAAOoB"
75 "AADyAQAABAIAABMCAAAqAgAAPgIAAFICAABmAgAAaQIAAG0CAACCAgAAhgIAAIoCAACQAgAAlQIA"
76 "AJ4CAACiAgAAAgAAAAMAAAAEAAAABQAAAAYAAAAHAAAACQAAAAcAAAAFAAAAAAAAAAgAAAAFAAAA"
77 "3AEAAAgAAAAFAAAA5AEAAAQAAQANAAAAAAAAAAAAAAAAAAIADAAAAAEAAQAOAAAAAgAAAAAAAAAA"
78 "AAAAAQAAAAIAAAAAAAAAAQAAAAAAAAC5AgAAAAAAAAEAAQABAAAApgIAAAQAAABwEAMAAAAOAAQA"
79 "AQACAAIAqwIAAC8AAABiAAAAGgEPAG4gAgAQAGIAAAAaAQoAbiACABAAYgAAABoBEABuIAIAEABi"
80 "AAAAGgELAG4gAgAQAA4ADQBiAQAAGgIKAG4gAgAhACcADQBiAQAAGgILAG4gAgAhACcAAAAAAAAA"
81 "BwABAA4AAAAHAAEAAgAdACYAAAABAAAAAwAAAAEAAAAGAAY8aW5pdD4AEEhhbmRsZXJUZXN0Lmph"
82 "dmEADUxIYW5kbGVyVGVzdDsAFUxqYXZhL2lvL1ByaW50U3RyZWFtOwASTGphdmEvbGFuZy9PYmpl"
83 "Y3Q7ABJMamF2YS9sYW5nL1N0cmluZzsAEkxqYXZhL2xhbmcvU3lzdGVtOwABVgACVkwAE1tMamF2"
84 "YS9sYW5nL1N0cmluZzsAAmYxAAJmMgAEbWFpbgADb3V0AAdwcmludGxuAAJ0MQACdDIAAQAHDgAE"
85 "AQAHDnl7eXkCeB2bAAAAAgAAgYAEvAIBCdQCAA0AAAAAAAAAAQAAAAAAAAABAAAAEQAAAHAAAAAC"
86 "AAAABwAAALQAAAADAAAAAwAAANAAAAAEAAAAAQAAAPQAAAAFAAAABAAAAPwAAAAGAAAAAQAAABwB"
87 "AAABIAAAAgAAADwBAAABEAAAAgAAANwBAAACIAAAEQAAAOoBAAADIAAAAgAAAKYCAAAAIAAAAQAA"
88 "ALkCAAAAEAAAAQAAAMgCAAA=";
89
Jeff Haoa64a64c2017-03-06 15:05:38 -080090// Dex file with an unreferenced catch handler at end of code item.
91// Constructed by building a dex file with try/catch blocks and hex editing.
92static const char kUnreferencedEndingCatchHandlerInputDex[] =
93 "ZGV4CjAzNQCEflufI6xGTDDRmLpbfYi6ujPrDLIwvYcEBAAAcAAAAHhWNBIAAAAAAAAAAGQDAAAT"
94 "AAAAcAAAAAgAAAC8AAAAAwAAANwAAAABAAAAAAEAAAUAAAAIAQAAAQAAADABAAC0AgAAUAEAAE4C"
95 "AABWAgAAXgIAAGYCAAB4AgAAhwIAAJ4CAAC1AgAAyQIAAN0CAADxAgAA9wIAAP0CAAAAAwAABAMA"
96 "ABkDAAAcAwAAIgMAACcDAAAEAAAABQAAAAYAAAAHAAAACAAAAAkAAAAMAAAADgAAAAwAAAAGAAAA"
97 "AAAAAA0AAAAGAAAAQAIAAA0AAAAGAAAASAIAAAUAAQARAAAAAAAAAAAAAAAAAAAADwAAAAAAAgAQ"
98 "AAAAAQABABIAAAADAAAAAAAAAAAAAAABAAAAAwAAAAAAAAADAAAAAAAAAFADAAAAAAAAAQABAAEA"
99 "AAAwAwAABAAAAHAQBAAAAA4AAgAAAAIAAgA1AwAAIQAAAGIAAAAaAQoAbiADABAAYgAAABoBCwBu"
100 "IAMAEAAOAA0AYgAAABoBAQBuIAMAEAAo8A0AYgAAABoBAgBuIAMAEAAo7gAAAAAAAAcAAQAHAAAA"
101 "BwABAAIBAg8BAhgAAwABAAIAAgBCAwAAIQAAAGIAAAAaAQoAbiADABAAYgAAABoBCwBuIAMAEAAO"
102 "AA0AYgAAABoBAQBuIAMAEAAo8A0AYgAAABoBAgBuIAMAEAAo7gAAAAAAAAcAAQAHAAAABwABAAIB"
103 "Ag8BAhgAAQAAAAQAAAABAAAABwAGPGluaXQ+AAZDYXRjaDEABkNhdGNoMgAQSGFuZGxlclRlc3Qu"
104 "amF2YQANTEhhbmRsZXJUZXN0OwAVTGphdmEvaW8vUHJpbnRTdHJlYW07ABVMamF2YS9sYW5nL0V4"
105 "Y2VwdGlvbjsAEkxqYXZhL2xhbmcvT2JqZWN0OwASTGphdmEvbGFuZy9TdHJpbmc7ABJMamF2YS9s"
106 "YW5nL1N5c3RlbTsABFRyeTEABFRyeTIAAVYAAlZMABNbTGphdmEvbGFuZy9TdHJpbmc7AAFhAARt"
107 "YWluAANvdXQAB3ByaW50bG4AAQAHDgAEAAcOfHsCeB0eih4AEQEABw59ewJ3HR6LHgAAAAMAAIGA"
108 "BNACAQnoAgEJ1AMAAA0AAAAAAAAAAQAAAAAAAAABAAAAEwAAAHAAAAACAAAACAAAALwAAAADAAAA"
109 "AwAAANwAAAAEAAAAAQAAAAABAAAFAAAABQAAAAgBAAAGAAAAAQAAADABAAABIAAAAwAAAFABAAAB"
110 "EAAAAgAAAEACAAACIAAAEwAAAE4CAAADIAAAAwAAADADAAAAIAAAAQAAAFADAAAAEAAAAQAAAGQD"
111 "AAA=";
112
Mathieu Chartier6b0dbef2017-02-21 12:45:30 -0800113// Dex file with multiple code items that have the same debug_info_off_. Constructed by a modified
114// dexlayout on XandY.
115static const char kDexFileDuplicateOffset[] =
116 "ZGV4CjAzNwAQfXfPCB8qCxo7MqdFhmHZQwCv8+udHD8MBAAAcAAAAHhWNBIAAAAAAAAAAFQDAAAT"
117 "AAAAcAAAAAgAAAC8AAAAAQAAANwAAAABAAAA6AAAAAUAAADwAAAAAwAAABgBAACUAgAAeAEAABQC"
118 "AAAeAgAAJgIAACsCAAAyAgAANwIAAFsCAAB7AgAAngIAALICAAC1AgAAvQIAAMUCAADIAgAA1QIA"
119 "AOkCAADvAgAA9QIAAPwCAAACAAAAAwAAAAQAAAAFAAAABgAAAAcAAAAIAAAACQAAAAkAAAAHAAAA"
120 "AAAAAAIAAQASAAAAAAAAAAEAAAABAAAAAQAAAAIAAAAAAAAAAgAAAAEAAAAGAAAAAQAAAAAAAAAA"
121 "AAAABgAAAAAAAAAKAAAAAAAAACsDAAAAAAAAAQAAAAAAAAAGAAAAAAAAAAsAAAD0AQAANQMAAAAA"
122 "AAACAAAAAAAAAAAAAAAAAAAACwAAAAQCAAA/AwAAAAAAAAIAAAAUAwAAGgMAAAEAAAAjAwAAAQAB"
123 "AAEAAAAFAAAABAAAAHAQBAAAAA4AAQABAAEAAAAFAAAABAAAAHAQBAAAAA4AAQAAAAEAAAAFAAAA"
124 "CAAAACIAAQBwEAEAAABpAAAADgABAAEAAQAAAAUAAAAEAAAAcBAAAAAADgB4AQAAAAAAAAAAAAAA"
125 "AAAAhAEAAAAAAAAAAAAAAAAAAAg8Y2xpbml0PgAGPGluaXQ+AANMWDsABUxZJFo7AANMWTsAIkxk"
126 "YWx2aWsvYW5ub3RhdGlvbi9FbmNsb3NpbmdDbGFzczsAHkxkYWx2aWsvYW5ub3RhdGlvbi9Jbm5l"
127 "ckNsYXNzOwAhTGRhbHZpay9hbm5vdGF0aW9uL01lbWJlckNsYXNzZXM7ABJMamF2YS9sYW5nL09i"
128 "amVjdDsAAVYABlguamF2YQAGWS5qYXZhAAFaAAthY2Nlc3NGbGFncwASZW1pdHRlcjogamFjay00"
129 "LjI1AARuYW1lAAR0aGlzAAV2YWx1ZQABegARAAcOABMABw4AEgAHDnYAEQAHDgACAwERGAICBAIN"
130 "BAgPFwwCBQERHAEYAQAAAQAAgIAEjAMAAAEAAYCABKQDAQACAAAIAoiABLwDAYCABNwDAAAADwAA"
131 "AAAAAAABAAAAAAAAAAEAAAATAAAAcAAAAAIAAAAIAAAAvAAAAAMAAAABAAAA3AAAAAQAAAABAAAA"
132 "6AAAAAUAAAAFAAAA8AAAAAYAAAADAAAAGAEAAAMQAAACAAAAeAEAAAEgAAAEAAAAjAEAAAYgAAAC"
133 "AAAA9AEAAAIgAAATAAAAFAIAAAMgAAAEAAAA/wIAAAQgAAADAAAAFAMAAAAgAAADAAAAKwMAAAAQ"
134 "AAABAAAAVAMAAA==";
135
Jeff Haof914f8e2017-02-21 16:14:23 -0800136// Dex file with null value for annotations_off in the annotation_set_ref_list.
137// Constructed by building a dex file with annotations and hex editing.
138static const char kNullSetRefListElementInputDex[] =
139 "ZGV4CjAzNQB1iA+7ZwgkF+7E6ZesYFc2lRAR3qnRAanwAwAAcAAAAHhWNBIAAAAAAAAAACADAAAS"
140 "AAAAcAAAAAgAAAC4AAAAAwAAANgAAAABAAAA/AAAAAQAAAAEAQAAAgAAACQBAACMAgAAZAEAAOgB"
141 "AADwAQAAAAIAAAMCAAAQAgAAIAIAADQCAABIAgAAawIAAI0CAAC1AgAAyAIAANECAADUAgAA2QIA"
142 "ANwCAADjAgAA6QIAAAMAAAAEAAAABQAAAAYAAAAHAAAACAAAAAkAAAAMAAAAAgAAAAMAAAAAAAAA"
143 "DAAAAAcAAAAAAAAADQAAAAcAAADgAQAABgAGAAsAAAAAAAEAAAAAAAAAAgAOAAAAAQAAABAAAAAC"
144 "AAEAAAAAAAAAAAAAAAAAAgAAAAAAAAABAAAAsAEAAAgDAAAAAAAAAQAAAAEmAAACAAAA2AEAAAoA"
145 "AADIAQAAFgMAAAAAAAACAAAAAAAAAHwBAAABAAAA/AIAAAAAAAABAAAAAgMAAAEAAQABAAAA8AIA"
146 "AAQAAABwEAMAAAAOAAIAAgAAAAAA9QIAAAEAAAAOAAAAAAAAAAAAAAAAAAAAAQAAAAEAAABkAQAA"
147 "cAEAAAAAAAAAAAAAAAAAAAEAAAAEAAAAAgAAAAMAAwAGPGluaXQ+AA5Bbm5vQ2xhc3MuamF2YQAB"
148 "TAALTEFubm9DbGFzczsADkxNeUFubm90YXRpb247ABJMamF2YS9sYW5nL09iamVjdDsAEkxqYXZh"
149 "L2xhbmcvU3RyaW5nOwAhTGphdmEvbGFuZy9hbm5vdGF0aW9uL0Fubm90YXRpb247ACBMamF2YS9s"
150 "YW5nL2Fubm90YXRpb24vUmV0ZW50aW9uOwAmTGphdmEvbGFuZy9hbm5vdGF0aW9uL1JldGVudGlv"
151 "blBvbGljeTsAEU15QW5ub3RhdGlvbi5qYXZhAAdSVU5USU1FAAFWAANWTEwAAWEABWFOYW1lAARu"
152 "YW1lAAV2YWx1ZQABAAcOAAICAAAHDgABBQERGwABAQEQFw8AAAIAAICABIQDAQmcAwAAAAECgQgA"
153 "AAARAAAAAAAAAAEAAAAAAAAAAQAAABIAAABwAAAAAgAAAAgAAAC4AAAAAwAAAAMAAADYAAAABAAA"
154 "AAEAAAD8AAAABQAAAAQAAAAEAQAABgAAAAIAAAAkAQAAAhAAAAEAAABkAQAAAxAAAAMAAABwAQAA"
155 "ASAAAAIAAACEAQAABiAAAAIAAACwAQAAARAAAAIAAADYAQAAAiAAABIAAADoAQAAAyAAAAIAAADw"
156 "AgAABCAAAAIAAAD8AgAAACAAAAIAAAAIAwAAABAAAAEAAAAgAwAA";
157
Jeff Haoe17f5892017-02-23 16:14:04 -0800158// Dex file with shared empty class data item for multiple class defs.
159// Constructing by building a dex file with multiple classes and hex editing.
160static const char kMultiClassDataInputDex[] =
161 "ZGV4CjAzNQALJgF9TtnLq748xVe/+wyxETrT9lTEiW6YAQAAcAAAAHhWNBIAAAAAAAAAADQBAAAI"
162 "AAAAcAAAAAQAAACQAAAAAAAAAAAAAAACAAAAoAAAAAAAAAAAAAAAAgAAALAAAACoAAAA8AAAAPAA"
163 "AAD4AAAAAAEAAAMBAAAIAQAADQEAACEBAAAkAQAAAgAAAAMAAAAEAAAABQAAAAEAAAAGAAAAAgAA"
164 "AAcAAAABAAAAAQYAAAMAAAAAAAAAAAAAAAAAAAAnAQAAAAAAAAIAAAABBgAAAwAAAAAAAAABAAAA"
165 "AAAAACcBAAAAAAAABkEuamF2YQAGQi5qYXZhAAFJAANMQTsAA0xCOwASTGphdmEvbGFuZy9PYmpl"
166 "Y3Q7AAFhAAFiAAAAAAABAAAAARkAAAAIAAAAAAAAAAEAAAAAAAAAAQAAAAgAAABwAAAAAgAAAAQA"
167 "AACQAAAABAAAAAIAAACgAAAABgAAAAIAAACwAAAAAiAAAAgAAADwAAAAACAAAAIAAAAnAQAAABAA"
168 "AAEAAAA0AQAA";
169
170// Dex file with code info followed by non 4-byte aligned section.
171// Constructed a dex file with code info followed by string data and hex edited.
172static const char kUnalignedCodeInfoInputDex[] =
173 "ZGV4CjAzNQDXJzXNb4iWn2SLhmLydW/8h1K9moERIw7UAQAAcAAAAHhWNBIAAAAAAAAAAEwBAAAG"
174 "AAAAcAAAAAMAAACIAAAAAQAAAJQAAAAAAAAAAAAAAAMAAACgAAAAAQAAALgAAAD8AAAA2AAAAAIB"
175 "AAAKAQAAEgEAABcBAAArAQAALgEAAAIAAAADAAAABAAAAAQAAAACAAAAAAAAAAAAAAAAAAAAAAAA"
176 "AAUAAAABAAAAAAAAAAAAAAABAAAAAQAAAAAAAAABAAAAAAAAADsBAAAAAAAAAQABAAEAAAAxAQAA"
177 "BAAAAHAQAgAAAA4AAQABAAAAAAA2AQAAAQAAAA4ABjxpbml0PgAGQS5qYXZhAANMQTsAEkxqYXZh"
178 "L2xhbmcvT2JqZWN0OwABVgABYQABAAcOAAMABw4AAAABAQCBgATYAQEB8AEAAAALAAAAAAAAAAEA"
179 "AAAAAAAAAQAAAAYAAABwAAAAAgAAAAMAAACIAAAAAwAAAAEAAACUAAAABQAAAAMAAACgAAAABgAA"
180 "AAEAAAC4AAAAASAAAAIAAADYAAAAAiAAAAYAAAACAQAAAyAAAAIAAAAxAQAAACAAAAEAAAA7AQAA"
181 "ABAAAAEAAABMAQAA";
Jeff Hao44652a32017-02-22 14:20:41 -0800182
Jeff Hao32a60ab2017-03-01 16:44:50 -0800183// Dex file with class data section preceding code items.
184// Constructed by passing dex file through dexmerger tool and hex editing.
185static const char kClassDataBeforeCodeInputDex[] =
186 "ZGV4CjAzNQCZKmCu3XXn4zvxCh5VH0gZNNobEAcsc49EAgAAcAAAAHhWNBIAAAAAAAAAAAQBAAAJ"
187 "AAAAcAAAAAQAAACUAAAAAgAAAKQAAAAAAAAAAAAAAAUAAAC8AAAAAQAAAOQAAABAAQAABAEAAPgB"
188 "AAAAAgAACAIAAAsCAAAQAgAAJAIAACcCAAAqAgAALQIAAAIAAAADAAAABAAAAAUAAAACAAAAAAAA"
189 "AAAAAAAFAAAAAwAAAAAAAAABAAEAAAAAAAEAAAAGAAAAAQAAAAcAAAABAAAACAAAAAIAAQAAAAAA"
190 "AQAAAAEAAAACAAAAAAAAAAEAAAAAAAAAjAEAAAAAAAALAAAAAAAAAAEAAAAAAAAAAQAAAAkAAABw"
191 "AAAAAgAAAAQAAACUAAAAAwAAAAIAAACkAAAABQAAAAUAAAC8AAAABgAAAAEAAADkAAAAABAAAAEA"
192 "AAAEAQAAACAAAAEAAACMAQAAASAAAAQAAACkAQAAAiAAAAkAAAD4AQAAAyAAAAQAAAAwAgAAAAAB"
193 "AwCBgASkAwEBvAMBAdADAQHkAwAAAQABAAEAAAAwAgAABAAAAHAQBAAAAA4AAgABAAAAAAA1AgAA"
194 "AgAAABIQDwACAAEAAAAAADoCAAACAAAAEiAPAAIAAQAAAAAAPwIAAAIAAAASMA8ABjxpbml0PgAG"
195 "QS5qYXZhAAFJAANMQTsAEkxqYXZhL2xhbmcvT2JqZWN0OwABVgABYQABYgABYwABAAcOAAMABw4A"
196 "BgAHDgAJAAcOAA==";
197
Jeff Haod9be7682017-04-05 09:29:43 -0700198// Dex file with local info containing a null type descriptor.
199// Constructed a dex file with debug info sequence containing DBG_RESTART_LOCAL without any
200// DBG_START_LOCAL to give it a declared type.
201static const char kUnknownTypeDebugInfoInputDex[] =
202 "ZGV4CjAzNQBtKqZfzjHLNSNwW2A6Bz9FuCEX0sL+FF38AQAAcAAAAHhWNBIAAAAAAAAAAHQBAAAI"
203 "AAAAcAAAAAQAAACQAAAAAgAAAKAAAAAAAAAAAAAAAAMAAAC4AAAAAQAAANAAAAAMAQAA8AAAABwB"
204 "AAAkAQAALAEAAC8BAAA0AQAASAEAAEsBAABOAQAAAgAAAAMAAAAEAAAABQAAAAIAAAAAAAAAAAAA"
205 "AAUAAAADAAAAAAAAAAEAAQAAAAAAAQAAAAYAAAACAAEAAAAAAAEAAAABAAAAAgAAAAAAAAABAAAA"
206 "AAAAAGMBAAAAAAAAAQABAAEAAABUAQAABAAAAHAQAgAAAA4AAgABAAAAAABZAQAAAgAAABIQDwAG"
207 "PGluaXQ+AAZBLmphdmEAAUkAA0xBOwASTGphdmEvbGFuZy9PYmplY3Q7AAFWAAFhAAR0aGlzAAEA"
208 "Bw4AAwAHDh4GAAYAAAAAAQEAgYAE8AEBAYgCAAAACwAAAAAAAAABAAAAAAAAAAEAAAAIAAAAcAAA"
209 "AAIAAAAEAAAAkAAAAAMAAAACAAAAoAAAAAUAAAADAAAAuAAAAAYAAAABAAAA0AAAAAEgAAACAAAA"
210 "8AAAAAIgAAAIAAAAHAEAAAMgAAACAAAAVAEAAAAgAAABAAAAYwEAAAAQAAABAAAAdAEAAA==";
211
Jeff Haod212d5b2017-04-26 12:09:06 -0700212// Dex file with multiple class data items pointing to the same code item.
213// Constructed by hex editing.
214static const char kDuplicateCodeItemInputDex[] =
215 "ZGV4CjAzNQCwKtVglQOmLWuHwldN5jkBOInC7mTMhJMAAgAAcAAAAHhWNBIAAAAAAAAAAHgBAAAH"
216 "AAAAcAAAAAMAAACMAAAAAQAAAJgAAAAAAAAAAAAAAAQAAACkAAAAAQAAAMQAAAAcAQAA5AAAACQB"
217 "AAAsAQAANAEAADkBAABNAQAAUAEAAFMBAAACAAAAAwAAAAQAAAAEAAAAAgAAAAAAAAAAAAAAAAAA"
218 "AAAAAAAFAAAAAAAAAAYAAAABAAAAAAAAAAAAAAABAAAAAQAAAAAAAAABAAAAAAAAAGUBAAAAAAAA"
219 "AQABAAEAAABWAQAABAAAAHAQAwAAAA4AAQABAAAAAABbAQAAAQAAAA4AAAABAAEAAAAAAGABAAAB"
220 "AAAADgAAAAY8aW5pdD4ABkEuamF2YQADTEE7ABJMamF2YS9sYW5nL09iamVjdDsAAVYAAWEAAWIA"
221 "AQAHDgADAAcOAAUABw4AAAABAgCBgATkAQEA/AEBAPwBAAsAAAAAAAAAAQAAAAAAAAABAAAABwAA"
222 "AHAAAAACAAAAAwAAAIwAAAADAAAAAQAAAJgAAAAFAAAABAAAAKQAAAAGAAAAAQAAAMQAAAABIAAA"
223 "AwAAAOQAAAACIAAABwAAACQBAAADIAAAAwAAAFYBAAAAIAAAAQAAAGUBAAAAEAAAAQAAAHgBAAA=";
224
Mathieu Chartiera27af082018-02-03 16:12:23 -0800225// Returns the default compact dex option for dexlayout based on kDefaultCompactDexLevel.
226static std::vector<std::string> DefaultCompactDexOption() {
227 return (kDefaultCompactDexLevel == CompactDexLevel::kCompactDexLevelFast) ?
228 std::vector<std::string>{"-x", "fast"} : std::vector<std::string>{"-x", "none"};
229}
230
Mathieu Chartier6b0dbef2017-02-21 12:45:30 -0800231static void WriteBase64ToFile(const char* base64, File* file) {
Jeff Hao042e8982016-10-19 11:17:11 -0700232 // Decode base64.
233 CHECK(base64 != nullptr);
234 size_t length;
235 std::unique_ptr<uint8_t[]> bytes(DecodeBase64(base64, &length));
Mathieu Chartier6b0dbef2017-02-21 12:45:30 -0800236 CHECK(bytes != nullptr);
Jeff Hao042e8982016-10-19 11:17:11 -0700237 if (!file->WriteFully(bytes.get(), length)) {
238 PLOG(FATAL) << "Failed to write base64 as file";
239 }
Mathieu Chartier6b0dbef2017-02-21 12:45:30 -0800240}
241
242static void WriteFileBase64(const char* base64, const char* location) {
243 // Write to provided file.
244 std::unique_ptr<File> file(OS::CreateEmptyFile(location));
245 CHECK(file != nullptr);
246 WriteBase64ToFile(base64, file.get());
Jeff Hao042e8982016-10-19 11:17:11 -0700247 if (file->FlushCloseOrErase() != 0) {
248 PLOG(FATAL) << "Could not flush and close test file.";
249 }
250}
251
Jeff Haoc3acfc52016-08-29 14:18:26 -0700252class DexLayoutTest : public CommonRuntimeTest {
253 protected:
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800254 std::string GetDexLayoutPath() {
255 return GetTestAndroidRoot() + "/bin/dexlayoutd";
Jeff Haoc3acfc52016-08-29 14:18:26 -0700256 }
257
Jeff Haoa8621002016-10-04 18:13:44 +0000258 // Runs FullPlainOutput test.
259 bool FullPlainOutputExec(std::string* error_msg) {
Jeff Haoc3acfc52016-08-29 14:18:26 -0700260 // TODO: dexdump2 -> dexdump ?
261 ScratchFile dexdump_output;
Andreas Gampeca620d72016-11-08 08:09:33 -0800262 const std::string& dexdump_filename = dexdump_output.GetFilename();
Jeff Haoc3acfc52016-08-29 14:18:26 -0700263 std::string dexdump = GetTestAndroidRoot() + "/bin/dexdump2";
264 EXPECT_TRUE(OS::FileExists(dexdump.c_str())) << dexdump << " should be a valid file path";
Jeff Haoc3acfc52016-08-29 14:18:26 -0700265
266 ScratchFile dexlayout_output;
Andreas Gampeca620d72016-11-08 08:09:33 -0800267 const std::string& dexlayout_filename = dexlayout_output.GetFilename();
Jeff Haoc3acfc52016-08-29 14:18:26 -0700268
Jeff Haoa8621002016-10-04 18:13:44 +0000269 for (const std::string &dex_file : GetLibCoreDexFileNames()) {
270 std::vector<std::string> dexdump_exec_argv =
271 { dexdump, "-d", "-f", "-h", "-l", "plain", "-o", dexdump_filename, dex_file };
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800272 std::vector<std::string> dexlayout_args =
273 { "-d", "-f", "-h", "-l", "plain", "-o", dexlayout_filename, dex_file };
Jeff Haoa8621002016-10-04 18:13:44 +0000274 if (!::art::Exec(dexdump_exec_argv, error_msg)) {
275 return false;
276 }
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800277 if (!DexLayoutExec(dexlayout_args, error_msg)) {
Jeff Haoa8621002016-10-04 18:13:44 +0000278 return false;
279 }
280 std::vector<std::string> diff_exec_argv =
281 { "/usr/bin/diff", dexdump_filename, dexlayout_filename };
282 if (!::art::Exec(diff_exec_argv, error_msg)) {
283 return false;
284 }
Jeff Haoc3acfc52016-08-29 14:18:26 -0700285 }
286 return true;
287 }
288
Jeff Haoa8621002016-10-04 18:13:44 +0000289 // Runs DexFileOutput test.
290 bool DexFileOutputExec(std::string* error_msg) {
291 ScratchFile tmp_file;
Andreas Gampeca620d72016-11-08 08:09:33 -0800292 const std::string& tmp_name = tmp_file.GetFilename();
293 size_t tmp_last_slash = tmp_name.rfind('/');
Jeff Haoa8621002016-10-04 18:13:44 +0000294 std::string tmp_dir = tmp_name.substr(0, tmp_last_slash + 1);
Jeff Haoa8621002016-10-04 18:13:44 +0000295
296 for (const std::string &dex_file : GetLibCoreDexFileNames()) {
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800297 std::vector<std::string> dexlayout_args =
298 { "-w", tmp_dir, "-o", tmp_name, dex_file };
Mathieu Chartiera27af082018-02-03 16:12:23 -0800299 if (!DexLayoutExec(dexlayout_args, error_msg, /*pass_default_cdex_option*/ false)) {
Jeff Haoa8621002016-10-04 18:13:44 +0000300 return false;
301 }
Andreas Gampe5555dd12017-08-24 13:50:21 -0700302 size_t dex_file_last_slash = dex_file.rfind('/');
Jeff Haoa8621002016-10-04 18:13:44 +0000303 std::string dex_file_name = dex_file.substr(dex_file_last_slash + 1);
304 std::vector<std::string> unzip_exec_argv =
305 { "/usr/bin/unzip", dex_file, "classes.dex", "-d", tmp_dir};
306 if (!::art::Exec(unzip_exec_argv, error_msg)) {
307 return false;
308 }
309 std::vector<std::string> diff_exec_argv =
310 { "/usr/bin/diff", tmp_dir + "classes.dex" , tmp_dir + dex_file_name };
311 if (!::art::Exec(diff_exec_argv, error_msg)) {
312 return false;
313 }
Orion Hodson98273672018-02-06 11:50:27 +0000314 if (!UnlinkFile(tmp_dir + "classes.dex")) {
Jeff Haoa8621002016-10-04 18:13:44 +0000315 return false;
316 }
Orion Hodson98273672018-02-06 11:50:27 +0000317 if (!UnlinkFile(tmp_dir + dex_file_name)) {
Jeff Haoa8621002016-10-04 18:13:44 +0000318 return false;
319 }
320 }
Jeff Hao042e8982016-10-19 11:17:11 -0700321 return true;
322 }
Jeff Haoa8621002016-10-04 18:13:44 +0000323
Mathieu Chartieraf7c9022017-10-27 09:42:46 -0700324 template <typename Mutator>
325 bool MutateDexFile(File* output_dex, const std::string& input_jar, const Mutator& mutator) {
326 std::vector<std::unique_ptr<const DexFile>> dex_files;
327 std::string error_msg;
David Sehr013fd802018-01-11 22:55:24 -0800328 const ArtDexFileLoader dex_file_loader;
329 CHECK(dex_file_loader.Open(input_jar.c_str(),
330 input_jar.c_str(),
331 /*verify*/ true,
332 /*verify_checksum*/ true,
333 &error_msg,
334 &dex_files)) << error_msg;
Mathieu Chartieraf7c9022017-10-27 09:42:46 -0700335 EXPECT_EQ(dex_files.size(), 1u) << "Only one input dex is supported";
336 for (const std::unique_ptr<const DexFile>& dex : dex_files) {
337 CHECK(dex->EnableWrite()) << "Failed to enable write";
338 mutator(const_cast<DexFile*>(dex.get()));
339 if (!output_dex->WriteFully(dex->Begin(), dex->Size())) {
340 return false;
341 }
342 }
343 if (output_dex->Flush() != 0) {
344 PLOG(FATAL) << "Could not flush the output file.";
345 }
346 return true;
347 }
348
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700349 // Create a profile with some subset of methods and classes.
350 void CreateProfile(const std::string& input_dex,
351 const std::string& out_profile,
352 const std::string& dex_location) {
353 std::vector<std::unique_ptr<const DexFile>> dex_files;
354 std::string error_msg;
David Sehr013fd802018-01-11 22:55:24 -0800355 const ArtDexFileLoader dex_file_loader;
356 bool result = dex_file_loader.Open(input_dex.c_str(),
357 input_dex,
358 /*verify*/ true,
359 /*verify_checksum*/ false,
360 &error_msg,
361 &dex_files);
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700362
363 ASSERT_TRUE(result) << error_msg;
364 ASSERT_GE(dex_files.size(), 1u);
365
366 size_t profile_methods = 0;
367 size_t profile_classes = 0;
368 ProfileCompilationInfo pfi;
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700369 std::set<DexCacheResolvedClasses> classes;
370 for (const std::unique_ptr<const DexFile>& dex_file : dex_files) {
371 for (uint32_t i = 0; i < dex_file->NumMethodIds(); i += 2) {
Mathieu Chartierbbe3a5e2017-06-13 16:36:17 -0700372 uint8_t flags = 0u;
373
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700374 if ((i & 3) != 0) {
Mathieu Chartierbbe3a5e2017-06-13 16:36:17 -0700375 flags |= ProfileCompilationInfo::MethodHotness::kFlagHot;
Mathieu Chartierea650f32017-05-24 12:04:13 -0700376 ++profile_methods;
377 } else if ((i & 2) != 0) {
Mathieu Chartierbbe3a5e2017-06-13 16:36:17 -0700378 flags |= ProfileCompilationInfo::MethodHotness::kFlagStartup;
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700379 ++profile_methods;
380 }
Mathieu Chartierbbe3a5e2017-06-13 16:36:17 -0700381 pfi.AddMethodIndex(static_cast<ProfileCompilationInfo::MethodHotness::Flag>(flags),
382 dex_location,
383 dex_file->GetLocationChecksum(),
384 /*dex_method_idx*/i,
385 dex_file->NumMethodIds());
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700386 }
387 DexCacheResolvedClasses cur_classes(dex_location,
388 dex_location,
Mathieu Chartierea650f32017-05-24 12:04:13 -0700389 dex_file->GetLocationChecksum(),
390 dex_file->NumMethodIds());
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700391 // Add every even class too.
392 for (uint32_t i = 0; i < dex_file->NumClassDefs(); i += 1) {
Mathieu Chartierea650f32017-05-24 12:04:13 -0700393 if ((i & 2) == 0) {
394 cur_classes.AddClass(dex_file->GetClassDef(i).class_idx_);
395 ++profile_classes;
396 }
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700397 }
Mathieu Chartierea650f32017-05-24 12:04:13 -0700398 classes.insert(cur_classes);
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700399 }
Mathieu Chartierbbe3a5e2017-06-13 16:36:17 -0700400 pfi.AddClasses(classes);
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700401 // Write to provided file.
402 std::unique_ptr<File> file(OS::CreateEmptyFile(out_profile.c_str()));
403 ASSERT_TRUE(file != nullptr);
404 pfi.Save(file->Fd());
405 if (file->FlushCloseOrErase() != 0) {
406 PLOG(FATAL) << "Could not flush and close test file.";
407 }
408 EXPECT_GE(profile_methods, 0u);
409 EXPECT_GE(profile_classes, 0u);
410 }
411
Jeff Haod9be7682017-04-05 09:29:43 -0700412 // Runs DexFileLayout test.
Jeff Hao042e8982016-10-19 11:17:11 -0700413 bool DexFileLayoutExec(std::string* error_msg) {
414 ScratchFile tmp_file;
Andreas Gampe641a4732017-08-24 13:21:35 -0700415 const std::string& tmp_name = tmp_file.GetFilename();
Andreas Gampe5555dd12017-08-24 13:50:21 -0700416 size_t tmp_last_slash = tmp_name.rfind('/');
Jeff Hao042e8982016-10-19 11:17:11 -0700417 std::string tmp_dir = tmp_name.substr(0, tmp_last_slash + 1);
418
419 // Write inputs and expected outputs.
420 std::string dex_file = tmp_dir + "classes.dex";
421 WriteFileBase64(kDexFileLayoutInputDex, dex_file.c_str());
422 std::string profile_file = tmp_dir + "primary.prof";
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700423 CreateProfile(dex_file, profile_file, dex_file);
424 // WriteFileBase64(kDexFileLayoutInputProfile, profile_file.c_str());
Jeff Hao042e8982016-10-19 11:17:11 -0700425 std::string output_dex = tmp_dir + "classes.dex.new";
426
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800427 std::vector<std::string> dexlayout_args =
428 { "-v", "-w", tmp_dir, "-o", tmp_name, "-p", profile_file, dex_file };
429 if (!DexLayoutExec(dexlayout_args, error_msg)) {
Jeff Hao042e8982016-10-19 11:17:11 -0700430 return false;
431 }
Mathieu Chartierfa0aa092017-03-27 15:43:54 -0700432
433 // -v makes sure that the layout did not corrupt the dex file.
Orion Hodson98273672018-02-06 11:50:27 +0000434 if (!UnlinkFile(dex_file) || !UnlinkFile(profile_file) || !UnlinkFile(output_dex)) {
Jeff Hao042e8982016-10-19 11:17:11 -0700435 return false;
436 }
Jeff Haoa8621002016-10-04 18:13:44 +0000437 return true;
438 }
Jeff Hao44652a32017-02-22 14:20:41 -0800439
Bharadwaj Kalandhabhatta02ef6402017-05-17 17:13:39 -0700440 // Runs DexFileLayout test twice (second time is run on output of first time)
441 // for behavior consistency.
442 bool DexFileLayoutFixedPointExec(std::string* error_msg) {
443 ScratchFile tmp_file;
Andreas Gampe641a4732017-08-24 13:21:35 -0700444 const std::string& tmp_name = tmp_file.GetFilename();
Andreas Gampe5555dd12017-08-24 13:50:21 -0700445 size_t tmp_last_slash = tmp_name.rfind('/');
Bharadwaj Kalandhabhatta02ef6402017-05-17 17:13:39 -0700446 std::string tmp_dir = tmp_name.substr(0, tmp_last_slash + 1);
447
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700448 // Unzip the test dex file to the classes.dex destination. It is required to unzip since
449 // opening from jar recalculates the dex location checksum.
Bharadwaj Kalandhabhatta02ef6402017-05-17 17:13:39 -0700450 std::string dex_file = tmp_dir + "classes.dex";
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700451
452 std::vector<std::string> unzip_args = {
453 "/usr/bin/unzip",
454 GetTestDexFileName("ManyMethods"),
455 "classes.dex",
456 "-d",
457 tmp_dir,
458 };
459 if (!art::Exec(unzip_args, error_msg)) {
460 LOG(ERROR) << "Failed to unzip dex";
461 return false;
462 }
463
Bharadwaj Kalandhabhatta02ef6402017-05-17 17:13:39 -0700464 std::string profile_file = tmp_dir + "primary.prof";
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700465 CreateProfile(dex_file, profile_file, dex_file);
Bharadwaj Kalandhabhatta02ef6402017-05-17 17:13:39 -0700466 std::string output_dex = tmp_dir + "classes.dex.new";
467 std::string second_output_dex = tmp_dir + "classes.dex.new.new";
468
Bharadwaj Kalandhabhatta02ef6402017-05-17 17:13:39 -0700469 // -v makes sure that the layout did not corrupt the dex file.
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800470 std::vector<std::string> dexlayout_args =
471 { "-i", "-v", "-w", tmp_dir, "-o", tmp_name, "-p", profile_file, dex_file };
Mathieu Chartiera27af082018-02-03 16:12:23 -0800472 if (!DexLayoutExec(dexlayout_args, error_msg, /*pass_default_cdex_option*/ false)) {
Bharadwaj Kalandhabhatta02ef6402017-05-17 17:13:39 -0700473 return false;
474 }
475
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700476 // Recreate the profile with the new dex location. This is required so that the profile dex
477 // location matches.
478 CreateProfile(dex_file, profile_file, output_dex);
479
Bharadwaj Kalandhabhatta02ef6402017-05-17 17:13:39 -0700480 // -v makes sure that the layout did not corrupt the dex file.
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700481 // -i since the checksum won't match from the first layout.
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800482 std::vector<std::string> second_dexlayout_args =
483 { "-i", "-v", "-w", tmp_dir, "-o", tmp_name, "-p", profile_file, output_dex };
Mathieu Chartiera27af082018-02-03 16:12:23 -0800484 if (!DexLayoutExec(second_dexlayout_args, error_msg, /*pass_default_cdex_option*/ false)) {
Bharadwaj Kalandhabhatta02ef6402017-05-17 17:13:39 -0700485 return false;
486 }
487
488 bool diff_result = true;
489 std::vector<std::string> diff_exec_argv =
490 { "/usr/bin/diff", output_dex, second_output_dex };
491 if (!::art::Exec(diff_exec_argv, error_msg)) {
492 diff_result = false;
493 }
494
Orion Hodson98273672018-02-06 11:50:27 +0000495 std::vector<std::string> test_files = { dex_file, profile_file, output_dex, second_output_dex };
496 for (auto test_file : test_files) {
497 if (!UnlinkFile(test_file)) {
498 return false;
499 }
Bharadwaj Kalandhabhatta02ef6402017-05-17 17:13:39 -0700500 }
501
502 return diff_result;
503 }
504
Jeff Haoac462712017-03-02 10:59:43 -0800505 // Runs UnreferencedCatchHandlerTest & Unreferenced0SizeCatchHandlerTest.
506 bool UnreferencedCatchHandlerExec(std::string* error_msg, const char* filename) {
Jeff Hao44652a32017-02-22 14:20:41 -0800507 ScratchFile tmp_file;
Andreas Gampe641a4732017-08-24 13:21:35 -0700508 const std::string& tmp_name = tmp_file.GetFilename();
Andreas Gampe5555dd12017-08-24 13:50:21 -0700509 size_t tmp_last_slash = tmp_name.rfind('/');
Jeff Hao44652a32017-02-22 14:20:41 -0800510 std::string tmp_dir = tmp_name.substr(0, tmp_last_slash + 1);
511
512 // Write inputs and expected outputs.
513 std::string input_dex = tmp_dir + "classes.dex";
Jeff Haoac462712017-03-02 10:59:43 -0800514 WriteFileBase64(filename, input_dex.c_str());
Jeff Hao44652a32017-02-22 14:20:41 -0800515 std::string output_dex = tmp_dir + "classes.dex.new";
516
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800517 std::vector<std::string> dexlayout_args = { "-w", tmp_dir, "-o", "/dev/null", input_dex };
Mathieu Chartiera27af082018-02-03 16:12:23 -0800518 if (!DexLayoutExec(dexlayout_args, error_msg, /*pass_default_cdex_option*/ false)) {
Jeff Hao44652a32017-02-22 14:20:41 -0800519 return false;
520 }
521
522 // Diff input and output. They should be the same.
523 std::vector<std::string> diff_exec_argv = { "/usr/bin/diff", input_dex, output_dex };
524 if (!::art::Exec(diff_exec_argv, error_msg)) {
525 return false;
526 }
527
Orion Hodson98273672018-02-06 11:50:27 +0000528 std::vector<std::string> dex_files = { input_dex, output_dex };
529 for (auto dex_file : dex_files) {
530 if (!UnlinkFile(dex_file)) {
531 return false;
532 }
Jeff Hao44652a32017-02-22 14:20:41 -0800533 }
534 return true;
535 }
Jeff Haod9be7682017-04-05 09:29:43 -0700536
537 bool DexLayoutExec(ScratchFile* dex_file,
538 const char* dex_filename,
539 ScratchFile* profile_file,
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800540 const std::vector<std::string>& dexlayout_args) {
Mathieu Chartieraf7c9022017-10-27 09:42:46 -0700541 if (dex_filename != nullptr) {
542 WriteBase64ToFile(dex_filename, dex_file->GetFile());
543 EXPECT_EQ(dex_file->GetFile()->Flush(), 0);
544 }
Jeff Haod9be7682017-04-05 09:29:43 -0700545 if (profile_file != nullptr) {
Mathieu Chartierd00e02b2017-05-24 12:04:13 -0700546 CreateProfile(dex_file->GetFilename(), profile_file->GetFilename(), dex_file->GetFilename());
Jeff Haod9be7682017-04-05 09:29:43 -0700547 }
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800548
Jeff Haod9be7682017-04-05 09:29:43 -0700549 std::string error_msg;
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800550 const bool result = DexLayoutExec(dexlayout_args, &error_msg);
Jeff Haod9be7682017-04-05 09:29:43 -0700551 if (!result) {
552 LOG(ERROR) << "Error: " << error_msg;
553 return false;
554 }
555 return true;
556 }
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800557
Mathieu Chartiera27af082018-02-03 16:12:23 -0800558 bool DexLayoutExec(const std::vector<std::string>& dexlayout_args,
559 std::string* error_msg,
560 bool pass_default_cdex_option = true) {
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800561 std::vector<std::string> argv;
562
563 std::string dexlayout = GetDexLayoutPath();
564 CHECK(OS::FileExists(dexlayout.c_str())) << dexlayout << " should be a valid file path";
565 argv.push_back(dexlayout);
Mathieu Chartiera27af082018-02-03 16:12:23 -0800566 if (pass_default_cdex_option) {
567 std::vector<std::string> cdex_level = DefaultCompactDexOption();
568 argv.insert(argv.end(), cdex_level.begin(), cdex_level.end());
569 }
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800570
571 argv.insert(argv.end(), dexlayout_args.begin(), dexlayout_args.end());
572
573 return ::art::Exec(argv, error_msg);
574 }
Orion Hodson98273672018-02-06 11:50:27 +0000575
576 bool UnlinkFile(const std::string& file_path) {
577 return unix_file::FdFile(file_path, 0, false).Unlink();
578 }
Jeff Haoc3acfc52016-08-29 14:18:26 -0700579};
580
581
582TEST_F(DexLayoutTest, FullPlainOutput) {
Jeff Hao0f7eaeb2016-08-31 17:56:13 -0700583 // Disable test on target.
584 TEST_DISABLED_FOR_TARGET();
Jeff Haoc3acfc52016-08-29 14:18:26 -0700585 std::string error_msg;
Jeff Haoa8621002016-10-04 18:13:44 +0000586 ASSERT_TRUE(FullPlainOutputExec(&error_msg)) << error_msg;
587}
588
589TEST_F(DexLayoutTest, DexFileOutput) {
590 // Disable test on target.
591 TEST_DISABLED_FOR_TARGET();
592 std::string error_msg;
593 ASSERT_TRUE(DexFileOutputExec(&error_msg)) << error_msg;
Jeff Haoc3acfc52016-08-29 14:18:26 -0700594}
595
Jeff Hao042e8982016-10-19 11:17:11 -0700596TEST_F(DexLayoutTest, DexFileLayout) {
597 // Disable test on target.
598 TEST_DISABLED_FOR_TARGET();
599 std::string error_msg;
600 ASSERT_TRUE(DexFileLayoutExec(&error_msg)) << error_msg;
601}
602
Bharadwaj Kalandhabhatta02ef6402017-05-17 17:13:39 -0700603TEST_F(DexLayoutTest, DexFileLayoutFixedPoint) {
604 // Disable test on target.
605 TEST_DISABLED_FOR_TARGET();
606 std::string error_msg;
607 ASSERT_TRUE(DexFileLayoutFixedPointExec(&error_msg)) << error_msg;
608}
609
Jeff Haoe17f5892017-02-23 16:14:04 -0800610TEST_F(DexLayoutTest, UnreferencedCatchHandler) {
611 // Disable test on target.
612 TEST_DISABLED_FOR_TARGET();
613 std::string error_msg;
Jeff Haoac462712017-03-02 10:59:43 -0800614 ASSERT_TRUE(UnreferencedCatchHandlerExec(&error_msg,
615 kUnreferencedCatchHandlerInputDex)) << error_msg;
Jeff Haoe17f5892017-02-23 16:14:04 -0800616}
Jeff Haoac462712017-03-02 10:59:43 -0800617
618TEST_F(DexLayoutTest, Unreferenced0SizeCatchHandler) {
619 // Disable test on target.
620 TEST_DISABLED_FOR_TARGET();
621 std::string error_msg;
622 ASSERT_TRUE(UnreferencedCatchHandlerExec(&error_msg,
623 kUnreferenced0SizeCatchHandlerInputDex)) << error_msg;
624}
625
Jeff Haoa64a64c2017-03-06 15:05:38 -0800626TEST_F(DexLayoutTest, UnreferencedEndingCatchHandler) {
627 // Disable test on target.
628 TEST_DISABLED_FOR_TARGET();
629 std::string error_msg;
630 ASSERT_TRUE(UnreferencedCatchHandlerExec(&error_msg,
631 kUnreferencedEndingCatchHandlerInputDex)) << error_msg;
632}
633
Mathieu Chartier6b0dbef2017-02-21 12:45:30 -0800634TEST_F(DexLayoutTest, DuplicateOffset) {
Jeff Haod9be7682017-04-05 09:29:43 -0700635 ScratchFile temp_dex;
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800636 std::vector<std::string> dexlayout_args =
637 { "-a", "-i", "-o", "/dev/null", temp_dex.GetFilename() };
Jeff Haod9be7682017-04-05 09:29:43 -0700638 ASSERT_TRUE(DexLayoutExec(&temp_dex,
639 kDexFileDuplicateOffset,
640 nullptr /* profile_file */,
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800641 dexlayout_args));
Mathieu Chartier6b0dbef2017-02-21 12:45:30 -0800642}
643
Jeff Haof914f8e2017-02-21 16:14:23 -0800644TEST_F(DexLayoutTest, NullSetRefListElement) {
Jeff Haod9be7682017-04-05 09:29:43 -0700645 ScratchFile temp_dex;
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800646 std::vector<std::string> dexlayout_args = { "-o", "/dev/null", temp_dex.GetFilename() };
Jeff Haod9be7682017-04-05 09:29:43 -0700647 ASSERT_TRUE(DexLayoutExec(&temp_dex,
648 kNullSetRefListElementInputDex,
649 nullptr /* profile_file */,
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800650 dexlayout_args));
Jeff Haof914f8e2017-02-21 16:14:23 -0800651}
652
Jeff Haoe17f5892017-02-23 16:14:04 -0800653TEST_F(DexLayoutTest, MultiClassData) {
Jeff Haod9be7682017-04-05 09:29:43 -0700654 ScratchFile temp_dex;
655 ScratchFile temp_profile;
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800656 std::vector<std::string> dexlayout_args =
657 { "-p", temp_profile.GetFilename(), "-o", "/dev/null", temp_dex.GetFilename() };
Jeff Haod9be7682017-04-05 09:29:43 -0700658 ASSERT_TRUE(DexLayoutExec(&temp_dex,
659 kMultiClassDataInputDex,
660 &temp_profile,
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800661 dexlayout_args));
Jeff Haoe17f5892017-02-23 16:14:04 -0800662}
663
664TEST_F(DexLayoutTest, UnalignedCodeInfo) {
Jeff Haod9be7682017-04-05 09:29:43 -0700665 ScratchFile temp_dex;
666 ScratchFile temp_profile;
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800667 std::vector<std::string> dexlayout_args =
668 { "-p", temp_profile.GetFilename(), "-o", "/dev/null", temp_dex.GetFilename() };
Jeff Haod9be7682017-04-05 09:29:43 -0700669 ASSERT_TRUE(DexLayoutExec(&temp_dex,
670 kUnalignedCodeInfoInputDex,
671 &temp_profile,
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800672 dexlayout_args));
Jeff Hao44652a32017-02-22 14:20:41 -0800673}
674
Jeff Hao32a60ab2017-03-01 16:44:50 -0800675TEST_F(DexLayoutTest, ClassDataBeforeCode) {
Jeff Haod9be7682017-04-05 09:29:43 -0700676 ScratchFile temp_dex;
677 ScratchFile temp_profile;
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800678 std::vector<std::string> dexlayout_args =
679 { "-p", temp_profile.GetFilename(), "-o", "/dev/null", temp_dex.GetFilename() };
Jeff Haod9be7682017-04-05 09:29:43 -0700680 ASSERT_TRUE(DexLayoutExec(&temp_dex,
681 kClassDataBeforeCodeInputDex,
682 &temp_profile,
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800683 dexlayout_args));
Jeff Haod9be7682017-04-05 09:29:43 -0700684}
685
686TEST_F(DexLayoutTest, UnknownTypeDebugInfo) {
687 ScratchFile temp_dex;
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800688 std::vector<std::string> dexlayout_args = { "-o", "/dev/null", temp_dex.GetFilename() };
Jeff Haod9be7682017-04-05 09:29:43 -0700689 ASSERT_TRUE(DexLayoutExec(&temp_dex,
690 kUnknownTypeDebugInfoInputDex,
691 nullptr /* profile_file */,
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800692 dexlayout_args));
Jeff Hao32a60ab2017-03-01 16:44:50 -0800693}
694
Jeff Haod212d5b2017-04-26 12:09:06 -0700695TEST_F(DexLayoutTest, DuplicateCodeItem) {
696 ScratchFile temp_dex;
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800697 std::vector<std::string> dexlayout_args = { "-o", "/dev/null", temp_dex.GetFilename() };
Jeff Haod212d5b2017-04-26 12:09:06 -0700698 ASSERT_TRUE(DexLayoutExec(&temp_dex,
699 kDuplicateCodeItemInputDex,
700 nullptr /* profile_file */,
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800701 dexlayout_args));
Jeff Haod212d5b2017-04-26 12:09:06 -0700702}
703
Mathieu Chartieraf7c9022017-10-27 09:42:46 -0700704// Test that instructions that go past the end of the code items don't cause crashes.
705TEST_F(DexLayoutTest, CodeItemOverrun) {
706 ScratchFile temp_dex;
707 MutateDexFile(temp_dex.GetFile(), GetTestDexFileName("ManyMethods"), [] (DexFile* dex) {
708 bool mutated_successfully = false;
709 // Change the dex instructions to make an opcode that spans past the end of the code item.
710 for (size_t i = 0; i < dex->NumClassDefs(); ++i) {
711 const DexFile::ClassDef& def = dex->GetClassDef(i);
712 const uint8_t* data = dex->GetClassData(def);
713 if (data == nullptr) {
714 continue;
715 }
716 ClassDataItemIterator it(*dex, data);
717 it.SkipAllFields();
Mathieu Chartierb7c273c2017-11-10 18:07:56 -0800718 while (it.HasNextMethod()) {
Mathieu Chartieraf7c9022017-10-27 09:42:46 -0700719 DexFile::CodeItem* item = const_cast<DexFile::CodeItem*>(it.GetMethodCodeItem());
720 if (item != nullptr) {
Mathieu Chartier698ebbc2018-01-05 11:00:42 -0800721 CodeItemInstructionAccessor instructions(*dex, item);
Mathieu Chartieraf7c9022017-10-27 09:42:46 -0700722 if (instructions.begin() != instructions.end()) {
723 DexInstructionIterator last_instruction = instructions.begin();
724 for (auto dex_it = instructions.begin(); dex_it != instructions.end(); ++dex_it) {
725 last_instruction = dex_it;
726 }
727 if (last_instruction->SizeInCodeUnits() == 1) {
728 // Set the opcode to something that will go past the end of the code item.
Mathieu Chartier2b2bef22017-10-26 17:10:19 -0700729 const_cast<Instruction&>(last_instruction.Inst()).SetOpcode(
Mathieu Chartieraf7c9022017-10-27 09:42:46 -0700730 Instruction::CONST_STRING_JUMBO);
731 mutated_successfully = true;
Mathieu Chartier176190c2017-10-31 09:58:07 -0700732 // Test that the safe iterator doesn't go past the end.
733 SafeDexInstructionIterator it2(instructions.begin(), instructions.end());
734 while (!it2.IsErrorState()) {
735 ++it2;
736 }
737 EXPECT_TRUE(it2 == last_instruction);
738 EXPECT_TRUE(it2 < instructions.end());
Mathieu Chartieraf7c9022017-10-27 09:42:46 -0700739 }
740 }
741 }
742 it.Next();
743 }
744 }
745 CHECK(mutated_successfully)
746 << "Failed to find candidate code item with only one code unit in last instruction.";
747 });
Mathieu Chartiera27af082018-02-03 16:12:23 -0800748
749 std::string error_msg;
750
751 ScratchFile tmp_file;
752 const std::string& tmp_name = tmp_file.GetFilename();
753 size_t tmp_last_slash = tmp_name.rfind('/');
754 std::string tmp_dir = tmp_name.substr(0, tmp_last_slash + 1);
755 ScratchFile profile_file;
756
757 std::vector<std::string> dexlayout_args =
758 { "-i",
759 "-v",
760 "-w", tmp_dir,
761 "-o", tmp_name,
762 "-p", profile_file.GetFilename(),
763 temp_dex.GetFilename()
764 };
765 // -v makes sure that the layout did not corrupt the dex file.
Mathieu Chartieraf7c9022017-10-27 09:42:46 -0700766 ASSERT_TRUE(DexLayoutExec(&temp_dex,
767 /*dex_filename*/ nullptr,
Mathieu Chartiera27af082018-02-03 16:12:23 -0800768 &profile_file,
Andreas Gampe693bfbf2017-11-10 12:23:31 -0800769 dexlayout_args));
Orion Hodson98273672018-02-06 11:50:27 +0000770 ASSERT_TRUE(UnlinkFile(temp_dex.GetFilename() + ".new"));
Mathieu Chartieraf7c9022017-10-27 09:42:46 -0700771}
772
Mathieu Chartier2f36d2f2017-11-20 15:45:25 -0800773// Test that link data is written out (or at least the header is updated).
774TEST_F(DexLayoutTest, LinkData) {
Mathieu Chartierfe9185a2017-11-21 19:01:10 -0800775 TEST_DISABLED_FOR_TARGET();
Mathieu Chartier2f36d2f2017-11-20 15:45:25 -0800776 ScratchFile temp_dex;
777 size_t file_size = 0;
778 MutateDexFile(temp_dex.GetFile(), GetTestDexFileName("ManyMethods"), [&] (DexFile* dex) {
779 DexFile::Header& header = const_cast<DexFile::Header&>(dex->GetHeader());
780 header.link_off_ = header.file_size_;
781 header.link_size_ = 16 * KB;
782 header.file_size_ += header.link_size_;
783 file_size = header.file_size_;
784 });
785 TEMP_FAILURE_RETRY(temp_dex.GetFile()->SetLength(file_size));
786
787 std::string error_msg;
788
789 ScratchFile tmp_file;
790 const std::string& tmp_name = tmp_file.GetFilename();
791 size_t tmp_last_slash = tmp_name.rfind('/');
792 std::string tmp_dir = tmp_name.substr(0, tmp_last_slash + 1);
793 ScratchFile profile_file;
794
795 std::vector<std::string> dexlayout_args =
796 { "-i",
797 "-v",
798 "-w", tmp_dir,
799 "-o", tmp_name,
800 "-p", profile_file.GetFilename(),
801 temp_dex.GetFilename()
802 };
803 // -v makes sure that the layout did not corrupt the dex file.
804 ASSERT_TRUE(DexLayoutExec(&temp_dex,
805 /*dex_filename*/ nullptr,
806 &profile_file,
807 dexlayout_args));
Orion Hodson98273672018-02-06 11:50:27 +0000808 ASSERT_TRUE(UnlinkFile(temp_dex.GetFilename() + ".new"));
Mathieu Chartier2f36d2f2017-11-20 15:45:25 -0800809}
810
Mathieu Chartier75175552018-01-25 11:23:01 -0800811TEST_F(DexLayoutTest, ClassFilter) {
812 std::vector<std::unique_ptr<const DexFile>> dex_files;
813 std::string error_msg;
814 const ArtDexFileLoader dex_file_loader;
815 const std::string input_jar = GetTestDexFileName("ManyMethods");
816 CHECK(dex_file_loader.Open(input_jar.c_str(),
817 input_jar.c_str(),
818 /*verify*/ true,
819 /*verify_checksum*/ true,
820 &error_msg,
821 &dex_files)) << error_msg;
822 ASSERT_EQ(dex_files.size(), 1u);
823 for (const std::unique_ptr<const DexFile>& dex_file : dex_files) {
824 EXPECT_GT(dex_file->NumClassDefs(), 1u);
825 for (uint32_t i = 0; i < dex_file->NumClassDefs(); ++i) {
826 const DexFile::ClassDef& class_def = dex_file->GetClassDef(i);
827 LOG(INFO) << dex_file->GetClassDescriptor(class_def);
828 }
829 Options options;
830 // Filter out all the classes other than the one below based on class descriptor.
831 options.class_filter_.insert("LManyMethods$Strings;");
832 DexLayout dexlayout(options,
833 /*info*/ nullptr,
834 /*out_file*/ nullptr,
835 /*header*/ nullptr);
836 std::unique_ptr<DexContainer> out;
837 dexlayout.ProcessDexFile(dex_file->GetLocation().c_str(),
838 dex_file.get(),
839 /*dex_file_index*/ 0,
840 &out);
841 std::unique_ptr<const DexFile> output_dex_file(
842 dex_file_loader.OpenWithDataSection(
843 out->GetMainSection()->Begin(),
844 out->GetMainSection()->Size(),
845 out->GetDataSection()->Begin(),
846 out->GetDataSection()->Size(),
847 dex_file->GetLocation().c_str(),
848 /* checksum */ 0,
849 /*oat_dex_file*/ nullptr,
850 /* verify */ true,
851 /*verify_checksum*/ false,
852 &error_msg));
853 ASSERT_TRUE(output_dex_file != nullptr);
854
Mathieu Chartier807b21b2018-01-29 05:18:24 -0800855 ASSERT_EQ(output_dex_file->NumClassDefs(), options.class_filter_.size());
Mathieu Chartier75175552018-01-25 11:23:01 -0800856 for (uint32_t i = 0; i < output_dex_file->NumClassDefs(); ++i) {
Mathieu Chartier807b21b2018-01-29 05:18:24 -0800857 // Check that every class in the output dex file is in the filter.
Mathieu Chartier75175552018-01-25 11:23:01 -0800858 const DexFile::ClassDef& class_def = output_dex_file->GetClassDef(i);
859 ASSERT_TRUE(options.class_filter_.find(output_dex_file->GetClassDescriptor(class_def)) !=
860 options.class_filter_.end());
861 }
862 }
863}
864
Jeff Haoc3acfc52016-08-29 14:18:26 -0700865} // namespace art