Skip to content

Commit 6428165

Browse files
committedFeb 11, 2025
8349254: Disable "best-fit" mapping on Windows environment variables
Reviewed-by: jlu, jpai
1 parent 32dc41c commit 6428165

File tree

3 files changed

+95
-64
lines changed

3 files changed

+95
-64
lines changed
 

‎src/java.base/share/native/libjli/args.c

+54-7
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2024, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2025, 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
@@ -83,6 +83,9 @@ static jboolean relaunch = JNI_FALSE;
8383
* Prototypes for internal functions.
8484
*/
8585
static jboolean expand(JLI_List args, const char *str, const char *var_name);
86+
#ifdef _WIN32
87+
static char * winGetEnv(const char * var_name);
88+
#endif
8689

8790
JNIEXPORT void JNICALL
8891
JLI_InitArgProcessing(jboolean hasJavaArgs, jboolean disableArgFile) {
@@ -465,8 +468,6 @@ int isTerminalOpt(char *arg) {
465468

466469
JNIEXPORT jboolean JNICALL
467470
JLI_AddArgsFromEnvVar(JLI_List args, const char *var_name) {
468-
char *env = getenv(var_name);
469-
470471
if (firstAppArgIndex == 0) {
471472
// Not 'java', return
472473
return JNI_FALSE;
@@ -476,12 +477,25 @@ JLI_AddArgsFromEnvVar(JLI_List args, const char *var_name) {
476477
return JNI_FALSE;
477478
}
478479

479-
if (NULL == env) {
480-
return JNI_FALSE;
480+
#ifdef _WIN32
481+
char *env = winGetEnv(var_name);
482+
#else
483+
char *env = getenv(var_name);
484+
#endif
485+
486+
jboolean ret = JNI_FALSE;
487+
488+
if (NULL != env) {
489+
JLI_ReportMessage(ARG_INFO_ENVVAR, var_name, env);
490+
ret = expand(args, env, var_name);
481491
}
482492

483-
JLI_ReportMessage(ARG_INFO_ENVVAR, var_name, env);
484-
return expand(args, env, var_name);
493+
#ifdef _WIN32
494+
if (NULL != env) {
495+
JLI_MemFree(env);
496+
}
497+
#endif
498+
return ret;
485499
}
486500

487501
/*
@@ -583,6 +597,39 @@ static jboolean expand(JLI_List args, const char *str, const char *var_name) {
583597
return JNI_TRUE;
584598
}
585599

600+
#ifdef _WIN32
601+
/*
602+
* getenv() without best-fit mapping. The return value is constructed by converting
603+
* _wgetenv()'s return encoded in wide char to ANSI code page without best-fit map.
604+
*/
605+
static char * winGetEnv(const char * var_name) {
606+
char * mbEnvVar = NULL;
607+
608+
int wcCount = MultiByteToWideChar(CP_ACP, 0, var_name, -1, NULL, 0);
609+
if (wcCount > 0) {
610+
LPWSTR wcVarName = JLI_MemAlloc(wcCount * sizeof(wchar_t));
611+
if (MultiByteToWideChar(CP_ACP, 0, var_name, -1, wcVarName, wcCount) != 0) {
612+
LPWSTR wcEnvVar = _wgetenv(wcVarName);
613+
if (wcEnvVar != NULL) {
614+
int mbSize = WideCharToMultiByte(CP_ACP,
615+
WC_NO_BEST_FIT_CHARS | WC_COMPOSITECHECK | WC_DEFAULTCHAR,
616+
wcEnvVar, -1, NULL, 0, NULL, NULL);
617+
if (mbSize > 0) {
618+
mbEnvVar = JLI_MemAlloc(mbSize);
619+
if (WideCharToMultiByte(CP_ACP, WC_NO_BEST_FIT_CHARS | WC_COMPOSITECHECK | WC_DEFAULTCHAR,
620+
wcEnvVar, -1, mbEnvVar, mbSize, NULL, NULL) == 0) {
621+
JLI_MemFree(mbEnvVar);
622+
mbEnvVar = NULL;
623+
}
624+
}
625+
}
626+
}
627+
JLI_MemFree(wcVarName);
628+
}
629+
return mbEnvVar;
630+
}
631+
#endif
632+
586633
#ifdef DEBUG_ARGFILE
587634
/*
588635
* Stand-alone sanity test, build with following command line

‎src/java.base/windows/native/libjava/ProcessEnvironment_md.c

+6-43
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2003, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2003, 2025, 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
@@ -28,55 +28,18 @@
2828
#include "jni_util.h"
2929
#include <windows.h>
3030

31-
static jstring
32-
environmentBlock9x(JNIEnv *env)
33-
{
34-
int i;
35-
jmethodID String_init_ID;
36-
jbyteArray bytes;
37-
jbyte *blockA;
38-
jclass string_class;
39-
40-
string_class = JNU_ClassString(env);
41-
CHECK_NULL_RETURN(string_class, NULL);
42-
43-
String_init_ID =
44-
(*env)->GetMethodID(env, string_class, "<init>", "([B)V");
45-
CHECK_NULL_RETURN(String_init_ID, NULL);
46-
47-
blockA = (jbyte *) GetEnvironmentStringsA();
48-
if (blockA == NULL) {
49-
/* Both GetEnvironmentStringsW and GetEnvironmentStringsA
50-
* failed. Out of memory is our best guess. */
51-
JNU_ThrowOutOfMemoryError(env, "GetEnvironmentStrings failed");
52-
return NULL;
53-
}
54-
55-
/* Don't search for "\0\0", since an empty environment block may
56-
legitimately consist of a single "\0". */
57-
for (i = 0; blockA[i];)
58-
while (blockA[i++])
59-
;
60-
61-
if ((bytes = (*env)->NewByteArray(env, i)) == NULL) {
62-
FreeEnvironmentStringsA(blockA);
63-
return NULL;
64-
}
65-
(*env)->SetByteArrayRegion(env, bytes, 0, i, blockA);
66-
FreeEnvironmentStringsA(blockA);
67-
return (*env)->NewObject(env, string_class,
68-
String_init_ID, bytes);
69-
}
70-
7131
/* Returns a Windows style environment block, discarding final trailing NUL */
7232
JNIEXPORT jstring JNICALL
7333
Java_java_lang_ProcessEnvironment_environmentBlock(JNIEnv *env, jclass klass)
7434
{
7535
int i;
7636
jstring envblock;
7737
jchar *blockW = (jchar *) GetEnvironmentStringsW();
78-
if (blockW == NULL)
79-
return environmentBlock9x(env);
38+
if (blockW == NULL) {
39+
/* Out of memory is our best guess. */
40+
JNU_ThrowOutOfMemoryError(env, "GetEnvironmentStrings failed");
41+
return NULL;
42+
}
8043

8144
/* Don't search for "\u0000\u0000", since an empty environment
8245
block may legitimately consist of a single "\u0000". */

‎test/jdk/tools/launcher/DisableBestFitMappingTest.java

+35-14
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2024, 2025, 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
@@ -23,9 +23,10 @@
2323

2424
/*
2525
* @test
26-
* @bug 8337506
27-
* @summary Verify Command Line arguments are not mapped with
28-
* "best-fit" mappings on Windows
26+
* @bug 8337506 8349254
27+
* @summary Verify command line arguments, including ones from
28+
* "JDK_JAVA_OPTIONS" environment variables are not mapped
29+
* with "best-fit" mappings on Windows
2930
* @requires (os.family == "windows")
3031
* @library /test/lib
3132
* @run junit DisableBestFitMappingTest
@@ -34,9 +35,9 @@
3435
import java.nio.charset.Charset;
3536
import java.nio.charset.CharsetEncoder;
3637
import java.util.stream.Stream;
38+
3739
import jdk.test.lib.process.ProcessTools;
3840

39-
import org.junit.jupiter.api.Test;
4041
import org.junit.jupiter.params.ParameterizedTest;
4142
import org.junit.jupiter.params.provider.Arguments;
4243
import org.junit.jupiter.params.provider.MethodSource;
@@ -49,10 +50,12 @@ public class DisableBestFitMappingTest {
4950
Charset.forName(System.getProperty("native.encoding")).newEncoder();
5051
private static final String REPLACEMENT =
5152
NATIVE_ENC.charset().decode(ByteBuffer.wrap(NATIVE_ENC.replacement())).toString();
53+
private static final String TEST_ENV_VAR = "JDK_JAVA_OPTIONS";
54+
private static final String TEST_PROP_KEY = "testProp";
5255
private static final int EXIT_SUCCESS = 0;
5356
private static final int EXIT_FAILURE = -1;
5457

55-
static Stream<Arguments> CMD_ARGS() {
58+
static Stream<Arguments> TEST_ARGS() {
5659
return Stream.of(
5760
Arguments.of("aa\uff02 \uff02bb", "aa" + REPLACEMENT + " " + REPLACEMENT + "bb"),
5861
Arguments.of("aa\uff01bb", "aa" + REPLACEMENT + "bb"),
@@ -61,23 +64,41 @@ static Stream<Arguments> CMD_ARGS() {
6164
}
6265

6366
@ParameterizedTest
64-
@MethodSource("CMD_ARGS")
65-
void testDisableBestFitMapping(String arg, String expected) throws Exception {
67+
@MethodSource("TEST_ARGS")
68+
void testCommandLineArgument(String arg, String expected) throws Exception {
6669
// Only execute if the arg cannot be encoded
6770
assumeFalse(NATIVE_ENC.canEncode(arg),
6871
"native.encoding (%s) can encode the argument '%s'. Test ignored."
6972
.formatted(NATIVE_ENC.charset(), arg));
7073

71-
var result= ProcessTools.executeTestJava(
72-
DisableBestFitMappingTest.class.getSimpleName(), arg, expected);
74+
var result = ProcessTools.executeTestJava(
75+
DisableBestFitMappingTest.class.getSimpleName(), expected, arg);
76+
result.asLines().forEach(System.out::println);
77+
assertEquals(EXIT_SUCCESS, result.getExitValue(),
78+
"Command line argument mapping failed");
79+
}
80+
81+
@ParameterizedTest
82+
@MethodSource("TEST_ARGS")
83+
void testEnvironmentVariable(String propVal, String expected) throws Exception {
84+
// Only execute if the arg from the environment variable cannot be encoded
85+
assumeFalse(NATIVE_ENC.canEncode(propVal),
86+
"native.encoding (%s) can encode the argument '%s'. Test ignored."
87+
.formatted(NATIVE_ENC.charset(), propVal));
88+
89+
var pb = ProcessTools.createTestJavaProcessBuilder(
90+
DisableBestFitMappingTest.class.getSimpleName(), expected);
91+
pb.environment().put(TEST_ENV_VAR, "-D" + TEST_PROP_KEY + "=\"" + propVal + "\"");
92+
var result = ProcessTools.executeProcess(pb);
7393
result.asLines().forEach(System.out::println);
7494
assertEquals(EXIT_SUCCESS, result.getExitValue(),
75-
"Disabling best-fit mapping failed");
95+
"Argument from JDK_JAVA_OPTIONS mapping failed");
7696
}
7797

7898
public static void main(String... args) {
79-
System.out.println(args[0]);
80-
System.out.println(args[1]);
81-
System.exit(args[0].equals(args[1]) ? EXIT_SUCCESS : EXIT_FAILURE);
99+
var expected = args[0];
100+
var actual = args.length > 1 ? args[1] : System.getProperty(TEST_PROP_KEY);
101+
System.out.printf("expected: %s, actual: %s%n", expected, actual);
102+
System.exit(expected.equals(actual) ? EXIT_SUCCESS : EXIT_FAILURE);
82103
}
83104
}

0 commit comments

Comments
 (0)
Please sign in to comment.