Skip to content

Commit 5858399

Browse files
committedApr 1, 2021
8264285: Clean the modification of ccstr JVM flags
Reviewed-by: dholmes, coleenp
1 parent 6e0da99 commit 5858399

File tree

9 files changed

+181
-45
lines changed

9 files changed

+181
-45
lines changed
 

‎src/hotspot/share/prims/whitebox.cpp

+4-6
Original file line numberDiff line numberDiff line change
@@ -1372,18 +1372,16 @@ WB_ENTRY(void, WB_SetStringVMFlag(JNIEnv* env, jobject o, jstring name, jstring
13721372
ccstrValue = env->GetStringUTFChars(value, NULL);
13731373
CHECK_JNI_EXCEPTION(env);
13741374
}
1375-
ccstr ccstrResult = ccstrValue;
1376-
bool needFree;
13771375
{
1376+
ccstr param = ccstrValue;
13781377
ThreadInVMfromNative ttvfn(thread); // back to VM
1379-
needFree = SetVMFlag <JVM_FLAG_TYPE(ccstr)> (thread, env, name, &ccstrResult);
1378+
if (SetVMFlag <JVM_FLAG_TYPE(ccstr)> (thread, env, name, &param)) {
1379+
assert(param == NULL, "old value is freed automatically and not returned");
1380+
}
13801381
}
13811382
if (value != NULL) {
13821383
env->ReleaseStringUTFChars(value, ccstrValue);
13831384
}
1384-
if (needFree) {
1385-
FREE_C_HEAP_ARRAY(char, ccstrResult);
1386-
}
13871385
WB_END
13881386

13891387
WB_ENTRY(void, WB_LockCompilation(JNIEnv* env, jobject o, jlong timeout))

‎src/hotspot/share/runtime/flags/allFlags.hpp

+10
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828
#include "compiler/compiler_globals.hpp"
2929
#include "gc/shared/gc_globals.hpp"
3030
#include "gc/shared/tlab_globals.hpp"
31+
#include "runtime/flags/debug_globals.hpp"
3132
#include "runtime/globals.hpp"
3233

3334
// Put LP64/ARCH/JVMCI/COMPILER1/COMPILER2 at the top,
@@ -112,6 +113,15 @@
112113
range, \
113114
constraint) \
114115
\
116+
DEBUG_RUNTIME_FLAGS( \
117+
develop, \
118+
develop_pd, \
119+
product, \
120+
product_pd, \
121+
notproduct, \
122+
range, \
123+
constraint) \
124+
\
115125
GC_FLAGS( \
116126
develop, \
117127
develop_pd, \
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
/*
2+
* Copyright (c) 2021, 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+
*/
24+
25+
#ifndef SHARE_RUNTIME_DEBUG_GLOBALS_HPP
26+
#define SHARE_RUNTIME_DEBUG_GLOBALS_HPP
27+
28+
#include "runtime/globals_shared.hpp"
29+
#include "utilities/macros.hpp"
30+
31+
//
32+
// These flags are needed for testing the implementation of various flag access
33+
// APIs.
34+
//
35+
// For example, DummyManageableStringFlag is needed because we don't
36+
// have any MANAGEABLE flags of the ccstr type, but we really need to
37+
// make sure the implementation is correct (in terms of memory allocation)
38+
// just in case someone may add such a flag in the future.
39+
//
40+
41+
#ifndef ASSERT
42+
43+
#define DEBUG_RUNTIME_FLAGS(develop, \
44+
develop_pd, \
45+
product, \
46+
product_pd, \
47+
notproduct, \
48+
range, \
49+
constraint) \
50+
\
51+
52+
#else
53+
54+
#define DEBUG_RUNTIME_FLAGS(develop, \
55+
develop_pd, \
56+
product, \
57+
product_pd, \
58+
notproduct, \
59+
range, \
60+
constraint) \
61+
\
62+
product(ccstr, DummyManageableStringFlag, NULL, MANAGEABLE, \
63+
"Dummy flag for testing string handling in WriteableFlags") \
64+
\
65+
66+
// end of DEBUG_RUNTIME_FLAGS
67+
68+
#endif // ASSERT
69+
70+
DECLARE_FLAGS(DEBUG_RUNTIME_FLAGS)
71+
72+
#endif // SHARE_RUNTIME_DEBUG_GLOBALS_HPP

‎src/hotspot/share/runtime/flags/jvmFlagAccess.cpp

+14-25
Original file line numberDiff line numberDiff line change
@@ -317,40 +317,29 @@ JVMFlag::Error JVMFlagAccess::ccstrAtPut(JVMFlag* flag, ccstr* value, JVMFlagOri
317317
new_value = os::strdup_check_oom(*value);
318318
}
319319
flag->set_ccstr(new_value);
320-
if (flag->is_default() && old_value != NULL) {
321-
// Prior value is NOT heap allocated, but was a literal constant.
322-
old_value = os::strdup_check_oom(old_value);
320+
if (!flag->is_default() && old_value != NULL) {
321+
// Old value is heap allocated so free it.
322+
FREE_C_HEAP_ARRAY(char, old_value);
323323
}
324-
*value = old_value;
324+
// Unlike the other APIs, the old vale is NOT returned, so the caller won't need to free it.
325+
// The callers typically don't care what the old value is.
326+
// If the caller really wants to know the old value, read it (and make a copy if necessary)
327+
// before calling this API.
328+
*value = NULL;
325329
flag->set_origin(origin);
326330
return JVMFlag::SUCCESS;
327331
}
328332

329333
// This is called by the FLAG_SET_XXX macros.
330334
JVMFlag::Error JVMFlagAccess::set_impl(JVMFlagsEnum flag_enum, int type_enum, void* value, JVMFlagOrigin origin) {
331-
if (type_enum == JVMFlag::TYPE_ccstr || type_enum == JVMFlag::TYPE_ccstrlist) {
332-
return ccstrAtPut((JVMFlagsEnum)flag_enum, *((ccstr*)value), origin);
333-
}
334-
335335
JVMFlag* flag = JVMFlag::flag_from_enum(flag_enum);
336-
assert(flag->type() == type_enum, "wrong flag type");
337-
return set_impl(flag, type_enum, value, origin);
338-
}
339-
340-
// This is called by the FLAG_SET_XXX macros.
341-
JVMFlag::Error JVMFlagAccess::ccstrAtPut(JVMFlagsEnum flag, ccstr value, JVMFlagOrigin origin) {
342-
JVMFlag* faddr = JVMFlag::flag_from_enum(flag);
343-
assert(faddr->is_ccstr(), "wrong flag type");
344-
ccstr old_value = faddr->get_ccstr();
345-
trace_flag_changed<ccstr, EventStringFlagChanged>(faddr, old_value, value, origin);
346-
char* new_value = os::strdup_check_oom(value);
347-
faddr->set_ccstr(new_value);
348-
if (!faddr->is_default() && old_value != NULL) {
349-
// Prior value is heap allocated so free it.
350-
FREE_C_HEAP_ARRAY(char, old_value);
336+
if (type_enum == JVMFlag::TYPE_ccstr || type_enum == JVMFlag::TYPE_ccstrlist) {
337+
assert(flag->is_ccstr(), "must be");
338+
return ccstrAtPut(flag, (ccstr*)value, origin);
339+
} else {
340+
assert(flag->type() == type_enum, "wrong flag type");
341+
return set_impl(flag, type_enum, value, origin);
351342
}
352-
faddr->set_origin(origin);
353-
return JVMFlag::SUCCESS;
354343
}
355344

356345
JVMFlag::Error JVMFlagAccess::check_range(const JVMFlag* flag, bool verbose) {

‎src/hotspot/share/runtime/flags/jvmFlagAccess.hpp

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2020, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -54,7 +54,6 @@ class JVMFlagAccess : AllStatic {
5454
inline static const FlagAccessImpl* access_impl(const JVMFlag* flag);
5555
static JVMFlag::Error set_impl(JVMFlagsEnum flag_enum, int type_enum, void* value, JVMFlagOrigin origin);
5656
static JVMFlag::Error set_impl(JVMFlag* flag, int type_enum, void* value, JVMFlagOrigin origin);
57-
static JVMFlag::Error ccstrAtPut(JVMFlagsEnum flag, ccstr value, JVMFlagOrigin origin);
5857

5958
public:
6059
static JVMFlag::Error check_range(const JVMFlag* flag, bool verbose);

‎src/hotspot/share/services/writeableFlags.cpp

+6-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -25,6 +25,7 @@
2525
#include "precompiled.hpp"
2626
#include "classfile/javaClasses.hpp"
2727
#include "memory/allocation.inline.hpp"
28+
#include "memory/resourceArea.hpp"
2829
#include "runtime/arguments.hpp"
2930
#include "runtime/flags/jvmFlag.hpp"
3031
#include "runtime/flags/jvmFlagAccess.hpp"
@@ -244,6 +245,9 @@ JVMFlag::Error WriteableFlags::set_double_flag(const char* name, double value, J
244245
JVMFlag::Error WriteableFlags::set_ccstr_flag(const char* name, const char* value, JVMFlagOrigin origin, FormatBuffer<80>& err_msg) {
245246
JVMFlag* flag = JVMFlag::find_flag(name);
246247
JVMFlag::Error err = JVMFlagAccess::ccstrAtPut(flag, &value, origin);
248+
if (err == JVMFlag::SUCCESS) {
249+
assert(value == NULL, "old value is freed automatically and not returned");
250+
}
247251
print_flag_error_message_if_needed(err, flag, err_msg);
248252
return err;
249253
}
@@ -357,11 +361,9 @@ JVMFlag::Error WriteableFlags::set_flag_from_jvalue(JVMFlag* f, const void* valu
357361
err_msg.print("flag value is missing");
358362
return JVMFlag::MISSING_VALUE;
359363
}
364+
ResourceMark rm;
360365
ccstr svalue = java_lang_String::as_utf8_string(str);
361366
JVMFlag::Error ret = WriteableFlags::set_ccstr_flag(f->name(), svalue, origin, err_msg);
362-
if (ret != JVMFlag::SUCCESS) {
363-
FREE_C_HEAP_ARRAY(char, svalue);
364-
}
365367
return ret;
366368
} else {
367369
ShouldNotReachHere();

‎test/hotspot/gtest/runtime/test_globals.cpp

+23
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525
#include "compiler/compiler_globals.hpp"
2626
#include "gc/shared/gc_globals.hpp"
2727
#include "runtime/globals.hpp"
28+
#include "runtime/globals_extension.hpp"
2829
#include "runtime/flags/flagSetting.hpp"
2930
#include "runtime/flags/jvmFlag.hpp"
3031
#include "unittest.hpp"
@@ -71,3 +72,25 @@ TEST_VM(FlagGuard, double_flag) {
7172
TEST_VM(FlagGuard, ccstr_flag) {
7273
TEST_FLAG(PerfDataSaveFile, ccstr, "/a/random/path");
7374
}
75+
76+
77+
// SharedArchiveConfigFile is used only during "java -Xshare:dump", so
78+
// it's safe to modify its value in gtest
79+
80+
TEST_VM(FlagAccess, ccstr_flag) {
81+
FLAG_SET_CMDLINE(SharedArchiveConfigFile, "");
82+
ASSERT_EQ(FLAG_IS_CMDLINE(SharedArchiveConfigFile), true);
83+
ASSERT_EQ(strcmp(SharedArchiveConfigFile, ""), 0);
84+
85+
FLAG_SET_ERGO(SharedArchiveConfigFile, "foobar");
86+
ASSERT_EQ(FLAG_IS_ERGO(SharedArchiveConfigFile), true);
87+
ASSERT_EQ(strcmp(SharedArchiveConfigFile, "foobar") , 0);
88+
89+
FLAG_SET_ERGO(SharedArchiveConfigFile, nullptr);
90+
ASSERT_EQ(FLAG_IS_ERGO(SharedArchiveConfigFile), true);
91+
ASSERT_EQ(SharedArchiveConfigFile, nullptr);
92+
93+
FLAG_SET_ERGO(SharedArchiveConfigFile, "xyz");
94+
ASSERT_EQ(FLAG_IS_ERGO(SharedArchiveConfigFile), true);
95+
ASSERT_EQ(strcmp(SharedArchiveConfigFile, "xyz"), 0);
96+
}

‎test/hotspot/jtreg/serviceability/dcmd/vm/SetVMFlagTest.java

+21-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -21,6 +21,7 @@
2121
* questions.
2222
*/
2323

24+
import jdk.test.lib.Platform;
2425
import jdk.test.lib.process.OutputAnalyzer;
2526
import jdk.test.lib.dcmd.CommandExecutor;
2627
import jdk.test.lib.dcmd.JMXExecutor;
@@ -47,6 +48,7 @@ public void run(CommandExecutor executor) {
4748
setMutableFlagWithInvalidValue(executor);
4849
setImmutableFlag(executor);
4950
setNonExistingFlag(executor);
51+
setStringFlag(executor);
5052
}
5153

5254
@Test
@@ -147,6 +149,24 @@ private void setNonExistingFlag(CommandExecutor executor) {
147149
out.stdoutShouldContain("flag " + unknownFlag + " does not exist");
148150
}
149151

152+
private void setStringFlag(CommandExecutor executor) {
153+
// Today we don't have any manageable flags of the string type in the product build,
154+
// so we can only test DummyManageableStringFlag in the debug build.
155+
if (!Platform.isDebugBuild()) {
156+
return;
157+
}
158+
159+
String flag = "DummyManageableStringFlag";
160+
String toValue = "DummyManageableStringFlag_Is_Set_To_Hello";
161+
162+
System.out.println("### Setting a string flag '" + flag + "'");
163+
OutputAnalyzer out = executor.execute("VM.set_flag " + flag + " " + toValue);
164+
out.stderrShouldBeEmpty();
165+
166+
out = getAllFlags(executor);
167+
out.stdoutShouldContain(toValue);
168+
}
169+
150170
private OutputAnalyzer getAllFlags(CommandExecutor executor) {
151171
return executor.execute("VM.flags -all", true);
152172
}

‎test/jdk/com/sun/management/HotSpotDiagnosticMXBean/SetVMOption.java

+30-7
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2015, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -29,6 +29,7 @@
2929
* @author Mandy Chung
3030
* @author Jaroslav Bachorik
3131
*
32+
* @library /test/lib
3233
* @run main/othervm -XX:+HeapDumpOnOutOfMemoryError SetVMOption
3334
*/
3435

@@ -37,6 +38,7 @@
3738
import com.sun.management.HotSpotDiagnosticMXBean;
3839
import com.sun.management.VMOption;
3940
import com.sun.management.VMOption.Origin;
41+
import jdk.test.lib.Platform;
4042

4143
public class SetVMOption {
4244
private static final String HEAP_DUMP_ON_OOM = "HeapDumpOnOutOfMemoryError";
@@ -94,6 +96,23 @@ public static void main(String[] args) throws Exception {
9496
option.isWriteable() + " expected: " + o.isWriteable());
9597
}
9698

99+
100+
// Today we don't have any manageable flags of the string type in the product build,
101+
// so we can only test DummyManageableStringFlag in the debug build.
102+
if (Platform.isDebugBuild()) {
103+
String optionName = "DummyManageableStringFlag";
104+
String toValue = "DummyManageableStringFlag_Is_Set_To_Hello";
105+
106+
mbean.setVMOption(optionName, toValue);
107+
108+
VMOption stringOption = findOption(optionName);
109+
Object newValue = stringOption.getValue();
110+
if (!toValue.equals(newValue)) {
111+
throw new RuntimeException("Unmatched value: " +
112+
newValue + " expected: " + toValue);
113+
}
114+
}
115+
97116
// check if ManagementServer is not writeable
98117
List<VMOption> options = mbean.getDiagnosticOptions();
99118
VMOption mgmtServerOption = null;
@@ -123,18 +142,22 @@ public static void main(String[] args) throws Exception {
123142
}
124143

125144
public static VMOption findHeapDumpOnOomOption() {
145+
return findOption(HEAP_DUMP_ON_OOM);
146+
}
147+
148+
private static VMOption findOption(String optionName) {
126149
List<VMOption> options = mbean.getDiagnosticOptions();
127-
VMOption gcDetails = null;
150+
VMOption found = null;
128151
for (VMOption o : options) {
129-
if (o.getName().equals(HEAP_DUMP_ON_OOM)) {
130-
gcDetails = o;
152+
if (o.getName().equals(optionName)) {
153+
found = o;
131154
break;
132155
}
133156
}
134-
if (gcDetails == null) {
135-
throw new RuntimeException("VM option " + HEAP_DUMP_ON_OOM +
157+
if (found == null) {
158+
throw new RuntimeException("VM option " + optionName +
136159
" not found");
137160
}
138-
return gcDetails;
161+
return found;
139162
}
140163
}

0 commit comments

Comments
 (0)
Please sign in to comment.