blob: cb0853bdb5a299c6343cb1ad1ade5f477bb8bb65 [file] [log] [blame]
Alex Lighta7e38d82017-01-19 14:57:28 -08001/* Copyright (C) 2017 The Android Open Source Project
2 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
3 *
4 * This file implements interfaces from the file jvmti.h. This implementation
5 * is licensed under the same terms as the file jvmti.h. The
6 * copyright and license information for the file jvmti.h follows.
7 *
8 * Copyright (c) 2003, 2011, Oracle and/or its affiliates. All rights reserved.
9 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
10 *
11 * This code is free software; you can redistribute it and/or modify it
12 * under the terms of the GNU General Public License version 2 only, as
13 * published by the Free Software Foundation. Oracle designates this
14 * particular file as subject to the "Classpath" exception as provided
15 * by Oracle in the LICENSE file that accompanied this code.
16 *
17 * This code is distributed in the hope that it will be useful, but WITHOUT
18 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
19 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
20 * version 2 for more details (a copy is included in the LICENSE file that
21 * accompanied this code).
22 *
23 * You should have received a copy of the GNU General Public License version
24 * 2 along with this work; if not, write to the Free Software Foundation,
25 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
26 *
27 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
28 * or visit www.oracle.com if you need additional information or have any
29 * questions.
30 */
31
Andreas Gampe06c42a52017-07-26 14:17:14 -070032#ifndef ART_OPENJDKJVMTI_TI_CLASS_DEFINITION_H_
33#define ART_OPENJDKJVMTI_TI_CLASS_DEFINITION_H_
Alex Lighta7e38d82017-01-19 14:57:28 -080034
Alex Lightca97ada2018-02-02 09:25:31 -080035#include <stddef.h>
36#include <sys/mman.h>
37#include <sys/types.h>
38
Alex Lighta7e38d82017-01-19 14:57:28 -080039#include "art_jvmti.h"
40
Vladimir Markoe1993c72017-06-14 17:01:38 +010041#include "base/array_ref.h"
David Sehr79e26072018-04-06 17:58:50 -070042#include "base/mem_map.h"
Alex Lightd55b8442019-10-15 15:46:07 -070043#include "events.h"
Vladimir Markoe1993c72017-06-14 17:01:38 +010044
Alex Lighta7e38d82017-01-19 14:57:28 -080045namespace openjdkjvmti {
46
47// A struct that stores data needed for redefining/transforming classes. This structure should only
48// even be accessed from a single thread and must not survive past the completion of the
49// redefinition/retransformation function that created it.
Alex Lightb7354d52017-03-30 15:17:01 -070050class ArtClassDefinition {
Alex Lighta7e38d82017-01-19 14:57:28 -080051 public:
Jack Nudelman139a5b92021-09-30 20:04:13 +000052 // If we support doing a on-demand dex-dequickening using signal handlers.
Roland Levillaindc744e52021-10-11 11:25:29 +000053 static constexpr bool kEnableOnDemandDexDequicken = true;
Jack Nudelman139a5b92021-09-30 20:04:13 +000054
Alex Light51271782017-03-24 17:28:30 -070055 ArtClassDefinition()
Alex Lightb7354d52017-03-30 15:17:01 -070056 : klass_(nullptr),
57 loader_(nullptr),
58 name_(),
59 protection_domain_(nullptr),
Vladimir Markoc34bebf2018-08-16 16:12:49 +010060 dex_data_mmap_(),
61 temp_mmap_(),
Alex Light64e4c142018-01-30 13:46:37 -080062 dex_data_memory_(),
Alex Lightca97ada2018-02-02 09:25:31 -080063 initial_dex_file_unquickened_(nullptr),
Alex Light64e4c142018-01-30 13:46:37 -080064 dex_data_(),
Alex Lightca97ada2018-02-02 09:25:31 -080065 current_dex_memory_(),
Alex Light64e4c142018-01-30 13:46:37 -080066 current_dex_file_(),
67 redefined_(false),
68 from_class_ext_(false),
Alex Lightd55b8442019-10-15 15:46:07 -070069 initialized_(false),
70 structural_transform_update_(false) {}
Alex Lightb7354d52017-03-30 15:17:01 -070071
Alex Light64e4c142018-01-30 13:46:37 -080072 void InitFirstLoad(const char* descriptor,
73 art::Handle<art::mirror::ClassLoader> klass_loader,
74 const art::DexFile& dex_file);
75 jvmtiError Init(art::Thread* self, jclass klass);
76 jvmtiError Init(art::Thread* self, const jvmtiClassDefinition& def);
Alex Light51271782017-03-24 17:28:30 -070077
Alex Lighta7e38d82017-01-19 14:57:28 -080078 ArtClassDefinition(ArtClassDefinition&& o) = default;
Alex Lightb7354d52017-03-30 15:17:01 -070079 ArtClassDefinition& operator=(ArtClassDefinition&& o) = default;
Alex Lighta7e38d82017-01-19 14:57:28 -080080
Alex Lightd55b8442019-10-15 15:46:07 -070081 void SetNewDexData(jint new_dex_len, unsigned char* new_dex_data, ArtJvmtiEvent event) {
Alex Lightb7354d52017-03-30 15:17:01 -070082 DCHECK(IsInitialized());
Alex Lighta7e38d82017-01-19 14:57:28 -080083 if (new_dex_data == nullptr) {
84 return;
Alex Light64e4c142018-01-30 13:46:37 -080085 } else {
86 art::ArrayRef<const unsigned char> new_data(new_dex_data, new_dex_len);
87 if (new_data != dex_data_) {
88 dex_data_memory_.resize(new_dex_len);
89 memcpy(dex_data_memory_.data(), new_dex_data, new_dex_len);
90 dex_data_ = art::ArrayRef<const unsigned char>(dex_data_memory_);
Alex Lightd55b8442019-10-15 15:46:07 -070091 if (event == ArtJvmtiEvent::kStructuralDexFileLoadHook) {
92 structural_transform_update_ = true;
93 }
Alex Light64e4c142018-01-30 13:46:37 -080094 }
Alex Lighta7e38d82017-01-19 14:57:28 -080095 }
96 }
97
Alex Lightd55b8442019-10-15 15:46:07 -070098 bool HasStructuralChanges() const {
99 return structural_transform_update_;
100 }
101
Vladimir Markoe1993c72017-06-14 17:01:38 +0100102 art::ArrayRef<const unsigned char> GetNewOriginalDexFile() const {
Alex Lightb7354d52017-03-30 15:17:01 -0700103 DCHECK(IsInitialized());
104 if (redefined_) {
Alex Light64e4c142018-01-30 13:46:37 -0800105 return current_dex_file_;
Alex Light40528472017-03-28 09:07:36 -0700106 } else {
Vladimir Markoe1993c72017-06-14 17:01:38 +0100107 return art::ArrayRef<const unsigned char>();
Alex Light40528472017-03-28 09:07:36 -0700108 }
109 }
110
Alex Lightca97ada2018-02-02 09:25:31 -0800111 bool ContainsAddress(uintptr_t ptr) const {
Vladimir Markoc34bebf2018-08-16 16:12:49 +0100112 return dex_data_mmap_.IsValid() &&
113 reinterpret_cast<uintptr_t>(dex_data_mmap_.Begin()) <= ptr &&
114 reinterpret_cast<uintptr_t>(dex_data_mmap_.End()) > ptr;
Alex Lightca97ada2018-02-02 09:25:31 -0800115 }
116
Alex Light64e4c142018-01-30 13:46:37 -0800117 bool IsModified() const REQUIRES_SHARED(art::Locks::mutator_lock_);
Alex Lighta7e38d82017-01-19 14:57:28 -0800118
Alex Lightb7354d52017-03-30 15:17:01 -0700119 bool IsInitialized() const {
Alex Light64e4c142018-01-30 13:46:37 -0800120 return initialized_;
Alex Lightb7354d52017-03-30 15:17:01 -0700121 }
122
123 jclass GetClass() const {
124 DCHECK(IsInitialized());
125 return klass_;
126 }
127
128 jobject GetLoader() const {
129 DCHECK(IsInitialized());
130 return loader_;
131 }
132
133 const std::string& GetName() const {
134 DCHECK(IsInitialized());
135 return name_;
136 }
137
Jack Nudelman139a5b92021-09-30 20:04:13 +0000138 bool IsLazyDefinition() const {
139 DCHECK(IsInitialized());
140 return dex_data_mmap_.IsValid() &&
141 dex_data_.data() == dex_data_mmap_.Begin() &&
142 dex_data_mmap_.GetProtect() == PROT_NONE;
143 }
144
Alex Lightb7354d52017-03-30 15:17:01 -0700145 jobject GetProtectionDomain() const {
146 DCHECK(IsInitialized());
147 return protection_domain_;
148 }
149
Vladimir Markoe1993c72017-06-14 17:01:38 +0100150 art::ArrayRef<const unsigned char> GetDexData() const {
Alex Lightb7354d52017-03-30 15:17:01 -0700151 DCHECK(IsInitialized());
Alex Light64e4c142018-01-30 13:46:37 -0800152 return dex_data_;
Alex Lightb7354d52017-03-30 15:17:01 -0700153 }
154
Alex Lightca97ada2018-02-02 09:25:31 -0800155 void InitializeMemory() const;
156
Alex Lighta7e38d82017-01-19 14:57:28 -0800157 private:
Alex Light64e4c142018-01-30 13:46:37 -0800158 jvmtiError InitCommon(art::Thread* self, jclass klass);
159
160 template<typename GetOriginalDexFile>
161 void InitWithDex(GetOriginalDexFile get_original, const art::DexFile* quick_dex)
162 REQUIRES_SHARED(art::Locks::mutator_lock_);
Alex Lightb7354d52017-03-30 15:17:01 -0700163
164 jclass klass_;
165 jobject loader_;
166 std::string name_;
167 jobject protection_domain_;
Alex Light64e4c142018-01-30 13:46:37 -0800168
Jack Nudelman139a5b92021-09-30 20:04:13 +0000169 // Mmap that will be filled with the original-dex-file lazily if it needs to be de-quickened or
170 // de-compact-dex'd
Vladimir Markoc34bebf2018-08-16 16:12:49 +0100171 mutable art::MemMap dex_data_mmap_;
Alex Lightca97ada2018-02-02 09:25:31 -0800172 // This is a temporary mmap we will use to be able to fill the dex file data atomically.
Vladimir Markoc34bebf2018-08-16 16:12:49 +0100173 mutable art::MemMap temp_mmap_;
Alex Lightca97ada2018-02-02 09:25:31 -0800174
Alex Light64e4c142018-01-30 13:46:37 -0800175 // A unique_ptr to the current dex_data if it needs to be cleaned up.
176 std::vector<unsigned char> dex_data_memory_;
177
Alex Lightca97ada2018-02-02 09:25:31 -0800178 const art::DexFile* initial_dex_file_unquickened_;
179
Alex Light64e4c142018-01-30 13:46:37 -0800180 // A ref to the current dex data. This is either dex_data_memory_, or current_dex_file_. This is
181 // what the dex file will be turned into.
182 art::ArrayRef<const unsigned char> dex_data_;
183
184 // This is only used if we failed to create a mmap to store the dequickened data
185 std::vector<unsigned char> current_dex_memory_;
186
187 // This is a dequickened version of what is loaded right now. It is either current_dex_memory_ (if
188 // no other redefinition has ever happened to this) or the current dex_file_ directly (if this
189 // class has been redefined, thus it cannot have any quickened stuff).
190 art::ArrayRef<const unsigned char> current_dex_file_;
191
Alex Lightb7354d52017-03-30 15:17:01 -0700192 bool redefined_;
193
Alex Light64e4c142018-01-30 13:46:37 -0800194 // If we got the initial dex_data_ from a class_ext
195 bool from_class_ext_;
196
197 bool initialized_;
198
Alex Lightd55b8442019-10-15 15:46:07 -0700199 // Set if we had a new dex from the given transform type.
200 bool structural_transform_update_;
201
Alex Lightb7354d52017-03-30 15:17:01 -0700202 DISALLOW_COPY_AND_ASSIGN(ArtClassDefinition);
Alex Lighta7e38d82017-01-19 14:57:28 -0800203};
204
205} // namespace openjdkjvmti
206
Andreas Gampe06c42a52017-07-26 14:17:14 -0700207#endif // ART_OPENJDKJVMTI_TI_CLASS_DEFINITION_H_