mirror of
https://github.com/corda/corda.git
synced 2025-01-22 12:28:11 +00:00
add proper argument escaping in Runtime.exec for windows
This commit is contained in:
parent
c09f329728
commit
3d361619a3
@ -222,6 +222,68 @@ const char* Locale::DEFAULT_LANGUAGE = "en";
|
|||||||
const char* Locale::DEFAULT_REGION = "";
|
const char* Locale::DEFAULT_REGION = "";
|
||||||
|
|
||||||
#ifdef PLATFORM_WINDOWS
|
#ifdef PLATFORM_WINDOWS
|
||||||
|
|
||||||
|
void appendN(char** dest, char ch, size_t length) {
|
||||||
|
for(size_t i = 0; i < length; i++) {
|
||||||
|
*((*dest)++) = ch;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool needsEscape(const char* src; size_t length) {
|
||||||
|
const char* end = src + length;
|
||||||
|
for(const char* ptr = src; ptr < end; ptr++) {
|
||||||
|
switch(*ptr) {
|
||||||
|
case ' ':
|
||||||
|
case '\t':
|
||||||
|
case '\n':
|
||||||
|
case '\v':
|
||||||
|
case '"':
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void copyAndEscape(char** dest, const char* src, size_t length) {
|
||||||
|
char* destp = *dest;
|
||||||
|
const char* end = src + length;
|
||||||
|
|
||||||
|
if(length != 0 && !needsEscape(src, length)) {
|
||||||
|
for(const char* ptr = src; ptr < end; ptr++) {
|
||||||
|
*(destp++) = *ptr;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
|
||||||
|
*(destp++) = '"';
|
||||||
|
|
||||||
|
for(const char* ptr = src; ; ptr++) {
|
||||||
|
unsigned numBackslashes = 0;
|
||||||
|
|
||||||
|
while(ptr < end && *ptr == '\\') {
|
||||||
|
ptr++;
|
||||||
|
numBackslashes++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(ptr == end) {
|
||||||
|
appendN(&destp, '\\', 2 * numBackslashes);
|
||||||
|
break;
|
||||||
|
} else if(*ptr == '"') {
|
||||||
|
appendN(&destp, '\\', 2 * numBackslashes + 1);
|
||||||
|
*(destp++) = *ptr;
|
||||||
|
} else {
|
||||||
|
appendN(&destp, '\\', numBackslashes);
|
||||||
|
*(destp++) = *ptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
*(destp++) = '"';
|
||||||
|
}
|
||||||
|
|
||||||
|
*dest = destp;
|
||||||
|
}
|
||||||
|
|
||||||
extern "C" JNIEXPORT void JNICALL
|
extern "C" JNIEXPORT void JNICALL
|
||||||
Java_java_lang_Runtime_exec(JNIEnv* e, jclass,
|
Java_java_lang_Runtime_exec(JNIEnv* e, jclass,
|
||||||
jobjectArray command, jlongArray process)
|
jobjectArray command, jlongArray process)
|
||||||
@ -230,7 +292,12 @@ Java_java_lang_Runtime_exec(JNIEnv* e, jclass,
|
|||||||
int size = 0;
|
int size = 0;
|
||||||
for (int i = 0; i < e->GetArrayLength(command); ++i){
|
for (int i = 0; i < e->GetArrayLength(command); ++i){
|
||||||
jstring element = (jstring) e->GetObjectArrayElement(command, i);
|
jstring element = (jstring) e->GetObjectArrayElement(command, i);
|
||||||
size += e->GetStringUTFLength(element) + 1;
|
if(element) {
|
||||||
|
// worst case, assuming every character is '"', and we escape all of them
|
||||||
|
size += 2 * e->GetStringUTFLength(element) + 3;
|
||||||
|
} else {
|
||||||
|
throwNew(e, "java/lang/NullPointerException", strdup("null string array element"));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
RUNTIME_ARRAY(char, line, size);
|
RUNTIME_ARRAY(char, line, size);
|
||||||
@ -239,13 +306,10 @@ Java_java_lang_Runtime_exec(JNIEnv* e, jclass,
|
|||||||
if (i) *(linep++) = _T(' ');
|
if (i) *(linep++) = _T(' ');
|
||||||
jstring element = (jstring) e->GetObjectArrayElement(command, i);
|
jstring element = (jstring) e->GetObjectArrayElement(command, i);
|
||||||
const char* s = e->GetStringUTFChars(element, 0);
|
const char* s = e->GetStringUTFChars(element, 0);
|
||||||
#ifdef _MSC_VER
|
|
||||||
_tcscpy_s(linep, size - (linep - RUNTIME_ARRAY_BODY(line)), s);
|
copyAndEscape(&linep, s, e->GetStringUTFLength(element));
|
||||||
#else
|
|
||||||
_tcscpy(linep, s);
|
|
||||||
#endif
|
|
||||||
e->ReleaseStringUTFChars(element, s);
|
e->ReleaseStringUTFChars(element, s);
|
||||||
linep += e->GetStringUTFLength(element);
|
|
||||||
}
|
}
|
||||||
*(linep++) = _T('\0');
|
*(linep++) = _T('\0');
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user