1*795d594fSAndroid Build Coastguard Worker /* Copyright (C) 2018 The Android Open Source Project
2*795d594fSAndroid Build Coastguard Worker * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
3*795d594fSAndroid Build Coastguard Worker *
4*795d594fSAndroid Build Coastguard Worker * This file implements interfaces from the file jvmti.h. This implementation
5*795d594fSAndroid Build Coastguard Worker * is licensed under the same terms as the file jvmti.h. The
6*795d594fSAndroid Build Coastguard Worker * copyright and license information for the file jvmti.h follows.
7*795d594fSAndroid Build Coastguard Worker *
8*795d594fSAndroid Build Coastguard Worker * Copyright (c) 2003, 2011, Oracle and/or its affiliates. All rights reserved.
9*795d594fSAndroid Build Coastguard Worker * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
10*795d594fSAndroid Build Coastguard Worker *
11*795d594fSAndroid Build Coastguard Worker * This code is free software; you can redistribute it and/or modify it
12*795d594fSAndroid Build Coastguard Worker * under the terms of the GNU General Public License version 2 only, as
13*795d594fSAndroid Build Coastguard Worker * published by the Free Software Foundation. Oracle designates this
14*795d594fSAndroid Build Coastguard Worker * particular file as subject to the "Classpath" exception as provided
15*795d594fSAndroid Build Coastguard Worker * by Oracle in the LICENSE file that accompanied this code.
16*795d594fSAndroid Build Coastguard Worker *
17*795d594fSAndroid Build Coastguard Worker * This code is distributed in the hope that it will be useful, but WITHOUT
18*795d594fSAndroid Build Coastguard Worker * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
19*795d594fSAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
20*795d594fSAndroid Build Coastguard Worker * version 2 for more details (a copy is included in the LICENSE file that
21*795d594fSAndroid Build Coastguard Worker * accompanied this code).
22*795d594fSAndroid Build Coastguard Worker *
23*795d594fSAndroid Build Coastguard Worker * You should have received a copy of the GNU General Public License version
24*795d594fSAndroid Build Coastguard Worker * 2 along with this work; if not, write to the Free Software Foundation,
25*795d594fSAndroid Build Coastguard Worker * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
26*795d594fSAndroid Build Coastguard Worker *
27*795d594fSAndroid Build Coastguard Worker * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
28*795d594fSAndroid Build Coastguard Worker * or visit www.oracle.com if you need additional information or have any
29*795d594fSAndroid Build Coastguard Worker * questions.
30*795d594fSAndroid Build Coastguard Worker */
31*795d594fSAndroid Build Coastguard Worker
32*795d594fSAndroid Build Coastguard Worker #include "ti_logging.h"
33*795d594fSAndroid Build Coastguard Worker
34*795d594fSAndroid Build Coastguard Worker #include "art_jvmti.h"
35*795d594fSAndroid Build Coastguard Worker
36*795d594fSAndroid Build Coastguard Worker #include "base/logging.h"
37*795d594fSAndroid Build Coastguard Worker #include "base/mutex.h"
38*795d594fSAndroid Build Coastguard Worker #include "base/strlcpy.h"
39*795d594fSAndroid Build Coastguard Worker #include "cmdline_types.h"
40*795d594fSAndroid Build Coastguard Worker #include "jvmti.h"
41*795d594fSAndroid Build Coastguard Worker #include "thread-current-inl.h"
42*795d594fSAndroid Build Coastguard Worker
43*795d594fSAndroid Build Coastguard Worker namespace openjdkjvmti {
44*795d594fSAndroid Build Coastguard Worker
GetLastError(jvmtiEnv * env,char ** data)45*795d594fSAndroid Build Coastguard Worker jvmtiError LogUtil::GetLastError(jvmtiEnv* env, char** data) {
46*795d594fSAndroid Build Coastguard Worker if (env == nullptr || data == nullptr) {
47*795d594fSAndroid Build Coastguard Worker return ERR(INVALID_ENVIRONMENT);
48*795d594fSAndroid Build Coastguard Worker }
49*795d594fSAndroid Build Coastguard Worker ArtJvmTiEnv* tienv = ArtJvmTiEnv::AsArtJvmTiEnv(env);
50*795d594fSAndroid Build Coastguard Worker art::MutexLock mu(art::Thread::Current(), tienv->last_error_mutex_);
51*795d594fSAndroid Build Coastguard Worker if (tienv->last_error_.empty()) {
52*795d594fSAndroid Build Coastguard Worker return ERR(ABSENT_INFORMATION);
53*795d594fSAndroid Build Coastguard Worker }
54*795d594fSAndroid Build Coastguard Worker const size_t size = tienv->last_error_.size() + 1;
55*795d594fSAndroid Build Coastguard Worker char* out;
56*795d594fSAndroid Build Coastguard Worker jvmtiError err = tienv->Allocate(size, reinterpret_cast<unsigned char**>(&out));
57*795d594fSAndroid Build Coastguard Worker if (err != OK) {
58*795d594fSAndroid Build Coastguard Worker return err;
59*795d594fSAndroid Build Coastguard Worker }
60*795d594fSAndroid Build Coastguard Worker strlcpy(out, tienv->last_error_.c_str(), size);
61*795d594fSAndroid Build Coastguard Worker *data = out;
62*795d594fSAndroid Build Coastguard Worker return OK;
63*795d594fSAndroid Build Coastguard Worker }
64*795d594fSAndroid Build Coastguard Worker
ClearLastError(jvmtiEnv * env)65*795d594fSAndroid Build Coastguard Worker jvmtiError LogUtil::ClearLastError(jvmtiEnv* env) {
66*795d594fSAndroid Build Coastguard Worker if (env == nullptr) {
67*795d594fSAndroid Build Coastguard Worker return ERR(INVALID_ENVIRONMENT);
68*795d594fSAndroid Build Coastguard Worker }
69*795d594fSAndroid Build Coastguard Worker ArtJvmTiEnv* tienv = ArtJvmTiEnv::AsArtJvmTiEnv(env);
70*795d594fSAndroid Build Coastguard Worker art::MutexLock mu(art::Thread::Current(), tienv->last_error_mutex_);
71*795d594fSAndroid Build Coastguard Worker tienv->last_error_.clear();
72*795d594fSAndroid Build Coastguard Worker return OK;
73*795d594fSAndroid Build Coastguard Worker }
74*795d594fSAndroid Build Coastguard Worker
SetVerboseFlagExt(jvmtiEnv * env,const char * data,jboolean enable)75*795d594fSAndroid Build Coastguard Worker jvmtiError LogUtil::SetVerboseFlagExt(jvmtiEnv* env, const char* data, jboolean enable) {
76*795d594fSAndroid Build Coastguard Worker if (env == nullptr) {
77*795d594fSAndroid Build Coastguard Worker return ERR(INVALID_ENVIRONMENT);
78*795d594fSAndroid Build Coastguard Worker } else if (data == nullptr) {
79*795d594fSAndroid Build Coastguard Worker return ERR(NULL_POINTER);
80*795d594fSAndroid Build Coastguard Worker }
81*795d594fSAndroid Build Coastguard Worker bool new_value = (enable == JNI_TRUE) ? true : false;
82*795d594fSAndroid Build Coastguard Worker art::CmdlineType<art::LogVerbosity> cmdline_parser;
83*795d594fSAndroid Build Coastguard Worker std::string parse_data(data);
84*795d594fSAndroid Build Coastguard Worker art::CmdlineType<art::LogVerbosity>::Result result = cmdline_parser.Parse(parse_data);
85*795d594fSAndroid Build Coastguard Worker if (result.IsError()) {
86*795d594fSAndroid Build Coastguard Worker JVMTI_LOG(INFO, env) << "Invalid verbose argument: '" << parse_data << "'. Error was "
87*795d594fSAndroid Build Coastguard Worker << result.GetMessage();
88*795d594fSAndroid Build Coastguard Worker return ERR(ILLEGAL_ARGUMENT);
89*795d594fSAndroid Build Coastguard Worker }
90*795d594fSAndroid Build Coastguard Worker
91*795d594fSAndroid Build Coastguard Worker const art::LogVerbosity& input_verbosity = result.GetValue();
92*795d594fSAndroid Build Coastguard Worker const bool* verbosity_arr = reinterpret_cast<const bool*>(&input_verbosity);
93*795d594fSAndroid Build Coastguard Worker bool* g_log_verbosity_arr = reinterpret_cast<bool*>(&art::gLogVerbosity);
94*795d594fSAndroid Build Coastguard Worker // Copy/invert the verbosity byte-by-byte (sizeof(bool) == 1).
95*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i < sizeof(art::LogVerbosity); i++) {
96*795d594fSAndroid Build Coastguard Worker if (verbosity_arr[i]) {
97*795d594fSAndroid Build Coastguard Worker g_log_verbosity_arr[i] = new_value;
98*795d594fSAndroid Build Coastguard Worker }
99*795d594fSAndroid Build Coastguard Worker }
100*795d594fSAndroid Build Coastguard Worker return OK;
101*795d594fSAndroid Build Coastguard Worker }
102*795d594fSAndroid Build Coastguard Worker
SetVerboseFlag(jvmtiEnv * env,jvmtiVerboseFlag flag,jboolean value)103*795d594fSAndroid Build Coastguard Worker jvmtiError LogUtil::SetVerboseFlag([[maybe_unused]] jvmtiEnv* env,
104*795d594fSAndroid Build Coastguard Worker jvmtiVerboseFlag flag,
105*795d594fSAndroid Build Coastguard Worker jboolean value) {
106*795d594fSAndroid Build Coastguard Worker if (flag == jvmtiVerboseFlag::JVMTI_VERBOSE_OTHER) {
107*795d594fSAndroid Build Coastguard Worker // OTHER is special, as it's 0, so can't do a bit check.
108*795d594fSAndroid Build Coastguard Worker bool val = (value == JNI_TRUE) ? true : false;
109*795d594fSAndroid Build Coastguard Worker
110*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.collector = val;
111*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.compiler = val;
112*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.deopt = val;
113*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.heap = val;
114*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.interpreter = val;
115*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.jdwp = val;
116*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.jit = val;
117*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.monitor = val;
118*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.oat = val;
119*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.profiler = val;
120*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.signals = val;
121*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.simulator = val;
122*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.startup = val;
123*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.third_party_jni = val;
124*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.threads = val;
125*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.verifier = val;
126*795d594fSAndroid Build Coastguard Worker // Do not set verifier-debug.
127*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.image = val;
128*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.plugin = val;
129*795d594fSAndroid Build Coastguard Worker
130*795d594fSAndroid Build Coastguard Worker // Note: can't switch systrace_lock_logging. That requires changing entrypoints.
131*795d594fSAndroid Build Coastguard Worker
132*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.agents = val;
133*795d594fSAndroid Build Coastguard Worker } else {
134*795d594fSAndroid Build Coastguard Worker // Spec isn't clear whether "flag" is a mask or supposed to be single. We implement the mask
135*795d594fSAndroid Build Coastguard Worker // semantics.
136*795d594fSAndroid Build Coastguard Worker constexpr std::underlying_type<jvmtiVerboseFlag>::type kMask =
137*795d594fSAndroid Build Coastguard Worker jvmtiVerboseFlag::JVMTI_VERBOSE_GC |
138*795d594fSAndroid Build Coastguard Worker jvmtiVerboseFlag::JVMTI_VERBOSE_CLASS |
139*795d594fSAndroid Build Coastguard Worker jvmtiVerboseFlag::JVMTI_VERBOSE_JNI;
140*795d594fSAndroid Build Coastguard Worker if ((flag & ~kMask) != 0) {
141*795d594fSAndroid Build Coastguard Worker return ERR(ILLEGAL_ARGUMENT);
142*795d594fSAndroid Build Coastguard Worker }
143*795d594fSAndroid Build Coastguard Worker
144*795d594fSAndroid Build Coastguard Worker bool val = (value == JNI_TRUE) ? true : false;
145*795d594fSAndroid Build Coastguard Worker
146*795d594fSAndroid Build Coastguard Worker if ((flag & jvmtiVerboseFlag::JVMTI_VERBOSE_GC) != 0) {
147*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.gc = val;
148*795d594fSAndroid Build Coastguard Worker }
149*795d594fSAndroid Build Coastguard Worker
150*795d594fSAndroid Build Coastguard Worker if ((flag & jvmtiVerboseFlag::JVMTI_VERBOSE_CLASS) != 0) {
151*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.class_linker = val;
152*795d594fSAndroid Build Coastguard Worker }
153*795d594fSAndroid Build Coastguard Worker
154*795d594fSAndroid Build Coastguard Worker if ((flag & jvmtiVerboseFlag::JVMTI_VERBOSE_JNI) != 0) {
155*795d594fSAndroid Build Coastguard Worker art::gLogVerbosity.jni = val;
156*795d594fSAndroid Build Coastguard Worker }
157*795d594fSAndroid Build Coastguard Worker }
158*795d594fSAndroid Build Coastguard Worker
159*795d594fSAndroid Build Coastguard Worker return ERR(NONE);
160*795d594fSAndroid Build Coastguard Worker }
161*795d594fSAndroid Build Coastguard Worker
162*795d594fSAndroid Build Coastguard Worker } // namespace openjdkjvmti
163