mirror of
https://github.com/openjdk/jdk.git
synced 2026-05-23 11:58:03 +00:00
8016110: Japanese char (MS932) 0x5C cannot be used as an argument when quoted
Reviewed-by: ksrini, akhil
This commit is contained in:
parent
e9806ae426
commit
3debf156b1
@ -53,6 +53,16 @@ typedef struct {
|
||||
static StdArg *stdargs;
|
||||
static int stdargc;
|
||||
|
||||
static int copyCh(USHORT ch, char* dest) {
|
||||
if (HIBYTE(ch) == 0) {
|
||||
*dest = (char)ch;
|
||||
return 1;
|
||||
} else {
|
||||
*((USHORT *)dest) = ch;
|
||||
return 2;
|
||||
}
|
||||
}
|
||||
|
||||
static char* next_arg(char* cmdline, char* arg, jboolean* wildcard) {
|
||||
|
||||
char* src = cmdline;
|
||||
@ -61,31 +71,43 @@ static char* next_arg(char* cmdline, char* arg, jboolean* wildcard) {
|
||||
int quotes = 0;
|
||||
int slashes = 0;
|
||||
|
||||
char prev = 0;
|
||||
char ch = 0;
|
||||
// "prev"/"ch" may contain either a single byte, or a double byte
|
||||
// character encoded in CP_ACP.
|
||||
USHORT prev = 0;
|
||||
USHORT ch = 0;
|
||||
int i;
|
||||
jboolean done = JNI_FALSE;
|
||||
int charLength;
|
||||
|
||||
*wildcard = JNI_FALSE;
|
||||
while ((ch = *src) != 0 && !done) {
|
||||
while (!done) {
|
||||
charLength = CharNextExA(CP_ACP, src, 0) - src;
|
||||
if (charLength == 0) {
|
||||
break;
|
||||
} else if (charLength == 1) {
|
||||
ch = (USHORT)(UCHAR)src[0];
|
||||
} else {
|
||||
ch = ((USHORT *)src)[0];
|
||||
}
|
||||
|
||||
switch (ch) {
|
||||
case '"':
|
||||
case L'"':
|
||||
if (separator) {
|
||||
done = JNI_TRUE;
|
||||
break;
|
||||
}
|
||||
if (prev == '\\') {
|
||||
if (prev == L'\\') {
|
||||
for (i = 1; i < slashes; i += 2) {
|
||||
*dest++ = prev;
|
||||
dest += copyCh(prev, dest);
|
||||
}
|
||||
if (slashes % 2 == 1) {
|
||||
*dest++ = ch;
|
||||
dest += copyCh(ch, dest);
|
||||
} else {
|
||||
quotes++;
|
||||
}
|
||||
} else if (prev == '"' && quotes % 2 == 0) {
|
||||
} else if (prev == L'"' && quotes % 2 == 0) {
|
||||
quotes++;
|
||||
*dest++ = ch; // emit every other consecutive quote
|
||||
dest += copyCh(ch, dest); // emit every other consecutive quote
|
||||
} else if (quotes == 0) {
|
||||
quotes++; // starting quote
|
||||
} else {
|
||||
@ -94,7 +116,7 @@ static char* next_arg(char* cmdline, char* arg, jboolean* wildcard) {
|
||||
slashes = 0;
|
||||
break;
|
||||
|
||||
case '\\':
|
||||
case L'\\':
|
||||
slashes++;
|
||||
if (separator) {
|
||||
done = JNI_TRUE;
|
||||
@ -102,23 +124,23 @@ static char* next_arg(char* cmdline, char* arg, jboolean* wildcard) {
|
||||
}
|
||||
break;
|
||||
|
||||
case ' ':
|
||||
case '\t':
|
||||
if (prev == '\\') {
|
||||
case L' ':
|
||||
case L'\t':
|
||||
if (prev == L'\\') {
|
||||
for (i = 0 ; i < slashes; i++) {
|
||||
*dest++ = prev;
|
||||
dest += copyCh(prev, dest);
|
||||
}
|
||||
}
|
||||
if (quotes % 2 == 1) {
|
||||
*dest++ = ch;
|
||||
dest += copyCh(ch, dest);
|
||||
} else {
|
||||
separator = JNI_TRUE;
|
||||
}
|
||||
slashes = 0;
|
||||
break;
|
||||
|
||||
case '*':
|
||||
case '?':
|
||||
case L'*':
|
||||
case L'?':
|
||||
if (separator) {
|
||||
done = JNI_TRUE;
|
||||
separator = JNI_FALSE;
|
||||
@ -127,36 +149,36 @@ static char* next_arg(char* cmdline, char* arg, jboolean* wildcard) {
|
||||
if (quotes % 2 == 0) {
|
||||
*wildcard = JNI_TRUE;
|
||||
}
|
||||
if (prev == '\\') {
|
||||
if (prev == L'\\') {
|
||||
for (i = 0 ; i < slashes ; i++) {
|
||||
*dest++ = prev;
|
||||
dest += copyCh(prev, dest);
|
||||
}
|
||||
}
|
||||
*dest++ = ch;
|
||||
dest += copyCh(ch, dest);
|
||||
break;
|
||||
|
||||
default:
|
||||
if (prev == '\\') {
|
||||
if (prev == L'\\') {
|
||||
for (i = 0 ; i < slashes ; i++) {
|
||||
*dest++ = prev;
|
||||
dest += copyCh(prev, dest);
|
||||
}
|
||||
*dest++ = ch;
|
||||
dest += copyCh(ch, dest);
|
||||
} else if (separator) {
|
||||
done = JNI_TRUE;
|
||||
} else {
|
||||
*dest++ = ch;
|
||||
dest += copyCh(ch, dest);
|
||||
}
|
||||
slashes = 0;
|
||||
}
|
||||
|
||||
if (!done) {
|
||||
prev = ch;
|
||||
src++;
|
||||
src += charLength;
|
||||
}
|
||||
}
|
||||
if (prev == '\\') {
|
||||
if (prev == L'\\') {
|
||||
for (i = 0; i < slashes; i++) {
|
||||
*dest++ = prev;
|
||||
dest += copyCh(prev, dest);
|
||||
}
|
||||
}
|
||||
*dest = 0;
|
||||
|
||||
111
jdk/test/tools/launcher/I18NArgTest.java
Normal file
111
jdk/test/tools/launcher/I18NArgTest.java
Normal file
@ -0,0 +1,111 @@
|
||||
/*
|
||||
* Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved.
|
||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||
*
|
||||
* This code is free software; you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License version 2 only, as
|
||||
* published by the Free Software Foundation.
|
||||
*
|
||||
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||
* version 2 for more details (a copy is included in the LICENSE file that
|
||||
* accompanied this code).
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License version
|
||||
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||
*
|
||||
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||
* or visit www.oracle.com if you need additional information or have any
|
||||
* questions.
|
||||
*/
|
||||
|
||||
/*
|
||||
* @test
|
||||
* @bug 8016110
|
||||
* @summary verify Japanese character in an argument are treated correctly
|
||||
* @compile -XDignore.symbol.file I18NArgTest.java
|
||||
* @run main I18NArgTest
|
||||
*/
|
||||
import java.io.IOException;
|
||||
|
||||
public class I18NArgTest extends TestHelper {
|
||||
public static void main(String... args) throws IOException {
|
||||
if (!isWindows) {
|
||||
return;
|
||||
}
|
||||
if (!"MS932".equals(System.getProperty("sun.jnu.encoding"))) {
|
||||
System.err.println("MS932 encoding not set, test skipped");
|
||||
return;
|
||||
}
|
||||
if (args.length == 0) {
|
||||
execTest(0x30bd); // MS932 Katakana SO, 0x835C
|
||||
} else {
|
||||
testCharacters(args);
|
||||
}
|
||||
}
|
||||
static void execTest(int unicodeValue) {
|
||||
String hexValue = Integer.toHexString(unicodeValue);
|
||||
String unicodeStr = Character.toString((char)unicodeValue);
|
||||
execTest("\"" + unicodeStr + "\"", hexValue);
|
||||
execTest("\\" + unicodeStr + "\\", hexValue);
|
||||
execTest(" " + unicodeStr + " ", hexValue);
|
||||
execTest("'" + unicodeStr + "'", hexValue);
|
||||
execTest("\t" + unicodeStr + "\t", hexValue);
|
||||
execTest("*" + unicodeStr + "*", hexValue);
|
||||
execTest("?" + unicodeStr + "?", hexValue);
|
||||
|
||||
execTest("\"" + unicodeStr + unicodeStr + "\"", hexValue + hexValue);
|
||||
execTest("\\" + unicodeStr + unicodeStr + "\\", hexValue + hexValue);
|
||||
execTest(" " + unicodeStr + unicodeStr + " ", hexValue + hexValue);
|
||||
execTest("'" + unicodeStr + unicodeStr + "'", hexValue + hexValue);
|
||||
execTest("\t" + unicodeStr + unicodeStr + "\t", hexValue + hexValue);
|
||||
execTest("*" + unicodeStr + unicodeStr + "*", hexValue + hexValue);
|
||||
execTest("?" + unicodeStr + unicodeStr + "?", hexValue + hexValue);
|
||||
|
||||
execTest("\"" + unicodeStr + "a" + unicodeStr + "\"", hexValue + "61" + hexValue);
|
||||
execTest("\\" + unicodeStr + "a" + unicodeStr + "\\", hexValue + "61" + hexValue);
|
||||
execTest(" " + unicodeStr + "a" + unicodeStr + " ", hexValue + "61"+ hexValue);
|
||||
execTest("'" + unicodeStr + "a" + unicodeStr + "'", hexValue + "61"+ hexValue);
|
||||
execTest("\t" + unicodeStr + "a" + unicodeStr + "\t", hexValue + "61"+ hexValue);
|
||||
execTest("*" + unicodeStr + "a" + unicodeStr + "*", hexValue + "61"+ hexValue);
|
||||
execTest("?" + unicodeStr + "a" + unicodeStr + "?", hexValue + "61"+ hexValue);
|
||||
|
||||
execTest("\"" + unicodeStr + "\u00b1" + unicodeStr + "\"", hexValue + "b1" + hexValue);
|
||||
execTest("\\" + unicodeStr + "\u00b1" + unicodeStr + "\\", hexValue + "b1" + hexValue);
|
||||
execTest(" " + unicodeStr + "\u00b1" + unicodeStr + " ", hexValue + "b1"+ hexValue);
|
||||
execTest("'" + unicodeStr + "\u00b1" + unicodeStr + "'", hexValue + "b1"+ hexValue);
|
||||
execTest("\t" + unicodeStr + "\u00b1" + unicodeStr + "\t", hexValue + "b1"+ hexValue);
|
||||
execTest("*" + unicodeStr + "\u00b1" + unicodeStr + "*", hexValue + "b1"+ hexValue);
|
||||
execTest("?" + unicodeStr + "\u00b1" + unicodeStr + "?", hexValue + "b1"+ hexValue);
|
||||
}
|
||||
static void execTest(String unicodeStr, String hexValue) {
|
||||
TestResult tr = doExec(javaCmd,
|
||||
"-Dtest.src=" + TEST_SOURCES_DIR.getAbsolutePath(),
|
||||
"-Dtest.classes=" + TEST_CLASSES_DIR.getAbsolutePath(),
|
||||
"-cp", TEST_CLASSES_DIR.getAbsolutePath(),
|
||||
"I18NArgTest", unicodeStr, hexValue);
|
||||
System.out.println(tr.testOutput);
|
||||
if (!tr.isOK()) {
|
||||
System.err.println(tr);
|
||||
throw new RuntimeException("test fails");
|
||||
}
|
||||
}
|
||||
static void testCharacters(String... args) {
|
||||
String input = args[0];
|
||||
String expected = args[1];
|
||||
String hexValue = "";
|
||||
for (int i = 0; i < input.length(); i++) {
|
||||
hexValue = hexValue.concat(Integer.toHexString((int)input.charAt(i)));
|
||||
}
|
||||
System.out.println("input:" + input);
|
||||
System.out.println("expected:" + expected);
|
||||
System.out.println("obtained:" + hexValue);
|
||||
if (!hexValue.contains(expected)) {
|
||||
String message = "Error: output does not contain expected value" +
|
||||
"expected:" + expected + " obtained:" + hexValue;
|
||||
throw new RuntimeException(message);
|
||||
}
|
||||
}
|
||||
}
|
||||
Loading…
x
Reference in New Issue
Block a user