1 /* 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 
32 #include "ti_properties.h"
33 
34 #include <string.h>
35 #include <vector>
36 
37 #include "jni.h"
38 #include "nativehelper/scoped_local_ref.h"
39 #include "nativehelper/scoped_utf_chars.h"
40 
41 #include "art_jvmti.h"
42 #include "runtime.h"
43 #include "thread-current-inl.h"
44 #include "ti_phase.h"
45 #include "well_known_classes.h"
46 
47 namespace openjdkjvmti {
48 
49 // Hardcoded properties. Tests ensure that these are consistent with libcore's view, as seen
50 // in System.java and AndroidHardcodedSystemProperties.java.
51 static constexpr const char* kProperties[][2] = {
52     // Recommended by the spec.
53     { "java.vm.vendor", "The Android Project" },
54     { "java.vm.version", "2.1.0" },  // This is Runtime::GetVersion().
55     { "java.vm.name", "Dalvik" },
56     // Android does not provide java.vm.info.
57     //
58     // These are other values provided by AndroidHardcodedSystemProperties.
59     { "java.class.version", "50.0" },
60     { "java.version", "0" },
61     { "java.compiler", "" },
62     { "java.ext.dirs", "" },
63 
64     { "java.specification.name", "Dalvik Core Library" },
65     { "java.specification.vendor", "The Android Project" },
66     { "java.specification.version", "0.9" },
67 
68     { "java.vendor", "The Android Project" },
69     { "java.vendor.url", "http://www.android.com/" },
70     { "java.vm.name", "Dalvik" },
71     { "java.vm.specification.name", "Dalvik Virtual Machine Specification" },
72     { "java.vm.specification.vendor", "The Android Project" },
73     { "java.vm.specification.version", "0.9" },
74     { "java.vm.vendor", "The Android Project" },
75 
76     { "java.vm.vendor.url", "http://www.android.com/" },
77 
78     { "java.net.preferIPv6Addresses", "false" },
79 
80     { "file.encoding", "UTF-8" },
81 
82     { "file.separator", "/" },
83     { "line.separator", "\n" },
84     { "path.separator", ":" },
85 
86     { "os.name", "Linux" },
87 };
88 static constexpr size_t kPropertiesSize = arraysize(kProperties);
89 static constexpr const char* kPropertyLibraryPath = "java.library.path";
90 static constexpr const char* kPropertyClassPath = "java.class.path";
91 
GetSystemProperties(jvmtiEnv * env,jint * count_ptr,char *** property_ptr)92 jvmtiError PropertiesUtil::GetSystemProperties(jvmtiEnv* env,
93                                                jint* count_ptr,
94                                                char*** property_ptr) {
95   if (count_ptr == nullptr || property_ptr == nullptr) {
96     return ERR(NULL_POINTER);
97   }
98   jvmtiError array_alloc_result;
99   JvmtiUniquePtr<char*[]> array_data_ptr = AllocJvmtiUniquePtr<char*[]>(env,
100                                                                         kPropertiesSize + 2,
101                                                                         &array_alloc_result);
102   if (array_data_ptr == nullptr) {
103     return array_alloc_result;
104   }
105 
106   std::vector<JvmtiUniquePtr<char[]>> property_copies;
107 
108   {
109     jvmtiError libpath_result;
110     JvmtiUniquePtr<char[]> libpath_data = CopyString(env, kPropertyLibraryPath, &libpath_result);
111     if (libpath_data == nullptr) {
112       return libpath_result;
113     }
114     array_data_ptr.get()[0] = libpath_data.get();
115     property_copies.push_back(std::move(libpath_data));
116   }
117 
118   {
119     jvmtiError classpath_result;
120     JvmtiUniquePtr<char[]> classpath_data = CopyString(env, kPropertyClassPath, &classpath_result);
121     if (classpath_data == nullptr) {
122       return classpath_result;
123     }
124     array_data_ptr.get()[1] = classpath_data.get();
125     property_copies.push_back(std::move(classpath_data));
126   }
127 
128   for (size_t i = 0; i != kPropertiesSize; ++i) {
129     jvmtiError data_result;
130     JvmtiUniquePtr<char[]> data = CopyString(env, kProperties[i][0], &data_result);
131     if (data == nullptr) {
132       return data_result;
133     }
134     array_data_ptr.get()[i + 2] = data.get();
135     property_copies.push_back(std::move(data));
136   }
137 
138   // Everything is OK, release the data.
139   *count_ptr = kPropertiesSize + 2;
140   *property_ptr = array_data_ptr.release();
141   for (auto& uptr : property_copies) {
142     uptr.release();
143   }
144 
145   return ERR(NONE);
146 }
147 
Copy(jvmtiEnv * env,const char * in,char ** out)148 static jvmtiError Copy(jvmtiEnv* env, const char* in, char** out) {
149   jvmtiError result;
150   JvmtiUniquePtr<char[]> data = CopyString(env, in, &result);
151   *out = data.release();
152   return result;
153 }
154 
155 // See dalvik_system_VMRuntime.cpp.
DefaultToDot(const std::string & class_path)156 static const char* DefaultToDot(const std::string& class_path) {
157   return class_path.empty() ? "." : class_path.c_str();
158 }
159 
160 // Handle kPropertyLibraryPath.
GetLibraryPath(jvmtiEnv * env,char ** value_ptr)161 static jvmtiError GetLibraryPath(jvmtiEnv* env, char** value_ptr) {
162   const std::vector<std::string>& runtime_props = art::Runtime::Current()->GetProperties();
163   for (const std::string& prop_assignment : runtime_props) {
164     size_t assign_pos = prop_assignment.find('=');
165     if (assign_pos != std::string::npos && assign_pos > 0) {
166       if (prop_assignment.substr(0, assign_pos) == kPropertyLibraryPath) {
167         return Copy(env, prop_assignment.substr(assign_pos + 1).c_str(), value_ptr);
168       }
169     }
170   }
171   if (!PhaseUtil::IsLivePhase()) {
172     return ERR(NOT_AVAILABLE);
173   }
174   // We expect this call to be rare. So don't optimize.
175   DCHECK(art::Thread::Current() != nullptr);
176   JNIEnv* jni_env = art::Thread::Current()->GetJniEnv();
177   jmethodID get_prop = jni_env->GetStaticMethodID(art::WellKnownClasses::java_lang_System,
178                                                   "getProperty",
179                                                   "(Ljava/lang/String;)Ljava/lang/String;");
180   CHECK(get_prop != nullptr);
181 
182   ScopedLocalRef<jobject> input_str(jni_env, jni_env->NewStringUTF(kPropertyLibraryPath));
183   if (input_str.get() == nullptr) {
184     jni_env->ExceptionClear();
185     return ERR(OUT_OF_MEMORY);
186   }
187 
188   ScopedLocalRef<jobject> prop_res(
189       jni_env, jni_env->CallStaticObjectMethod(art::WellKnownClasses::java_lang_System,
190                                                get_prop,
191                                                input_str.get()));
192   if (jni_env->ExceptionCheck() == JNI_TRUE) {
193     jni_env->ExceptionClear();
194     return ERR(INTERNAL);
195   }
196   if (prop_res.get() == nullptr) {
197     *value_ptr = nullptr;
198     return ERR(NONE);
199   }
200 
201   ScopedUtfChars chars(jni_env, reinterpret_cast<jstring>(prop_res.get()));
202   return Copy(env, chars.c_str(), value_ptr);
203 }
204 
GetSystemProperty(jvmtiEnv * env,const char * property,char ** value_ptr)205 jvmtiError PropertiesUtil::GetSystemProperty(jvmtiEnv* env,
206                                              const char* property,
207                                              char** value_ptr) {
208   if (property == nullptr || value_ptr == nullptr) {
209     return ERR(NULL_POINTER);
210   }
211 
212   if (strcmp(property, kPropertyLibraryPath) == 0) {
213     return GetLibraryPath(env, value_ptr);
214   }
215 
216   if (strcmp(property, kPropertyClassPath) == 0) {
217     return Copy(env, DefaultToDot(art::Runtime::Current()->GetClassPathString()), value_ptr);
218   }
219 
220   for (size_t i = 0; i != kPropertiesSize; ++i) {
221     if (strcmp(property, kProperties[i][0]) == 0) {
222       return Copy(env, kProperties[i][1], value_ptr);
223     }
224   }
225 
226   return ERR(NOT_AVAILABLE);
227 }
228 
SetSystemProperty(jvmtiEnv * env,const char * property,const char * value)229 jvmtiError PropertiesUtil::SetSystemProperty([[maybe_unused]] jvmtiEnv* env,
230                                              [[maybe_unused]] const char* property,
231                                              [[maybe_unused]] const char* value) {
232   // We do not allow manipulation of any property here.
233   return ERR(NOT_AVAILABLE);
234 }
235 
236 }  // namespace openjdkjvmti
237