Merge
This commit is contained in:
commit
81760af364
@ -5037,6 +5037,7 @@ _JNI_IMPORT_OR_EXPORT_ jint JNICALL JNI_GetDefaultJavaVMInitArgs(void *args_) {
|
|||||||
#include "gc_implementation/g1/heapRegionRemSet.hpp"
|
#include "gc_implementation/g1/heapRegionRemSet.hpp"
|
||||||
#endif
|
#endif
|
||||||
#include "utilities/quickSort.hpp"
|
#include "utilities/quickSort.hpp"
|
||||||
|
#include "utilities/ostream.hpp"
|
||||||
#if INCLUDE_VM_STRUCTS
|
#if INCLUDE_VM_STRUCTS
|
||||||
#include "runtime/vmStructs.hpp"
|
#include "runtime/vmStructs.hpp"
|
||||||
#endif
|
#endif
|
||||||
@ -5060,6 +5061,7 @@ void execute_internal_vm_tests() {
|
|||||||
run_unit_test(CollectedHeap::test_is_in());
|
run_unit_test(CollectedHeap::test_is_in());
|
||||||
run_unit_test(QuickSort::test_quick_sort());
|
run_unit_test(QuickSort::test_quick_sort());
|
||||||
run_unit_test(AltHashing::test_alt_hash());
|
run_unit_test(AltHashing::test_alt_hash());
|
||||||
|
run_unit_test(test_loggc_filename());
|
||||||
#if INCLUDE_VM_STRUCTS
|
#if INCLUDE_VM_STRUCTS
|
||||||
run_unit_test(VMStructs::test());
|
run_unit_test(VMStructs::test());
|
||||||
#endif
|
#endif
|
||||||
|
@ -1839,7 +1839,7 @@ void check_gclog_consistency() {
|
|||||||
(NumberOfGCLogFiles == 0) ||
|
(NumberOfGCLogFiles == 0) ||
|
||||||
(GCLogFileSize == 0)) {
|
(GCLogFileSize == 0)) {
|
||||||
jio_fprintf(defaultStream::output_stream(),
|
jio_fprintf(defaultStream::output_stream(),
|
||||||
"To enable GC log rotation, use -Xloggc:<filename> -XX:+UseGCLogFileRotation -XX:NumberOfGCLogFiles=<num_of_files> -XX:GCLogFileSize=<num_of_size>\n"
|
"To enable GC log rotation, use -Xloggc:<filename> -XX:+UseGCLogFileRotation -XX:NumberOfGCLogFiles=<num_of_files> -XX:GCLogFileSize=<num_of_size>[k|K|m|M|g|G]\n"
|
||||||
"where num_of_file > 0 and num_of_size > 0\n"
|
"where num_of_file > 0 and num_of_size > 0\n"
|
||||||
"GC log rotation is turned off\n");
|
"GC log rotation is turned off\n");
|
||||||
UseGCLogFileRotation = false;
|
UseGCLogFileRotation = false;
|
||||||
@ -1853,6 +1853,51 @@ void check_gclog_consistency() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// This function is called for -Xloggc:<filename>, it can be used
|
||||||
|
// to check if a given file name(or string) conforms to the following
|
||||||
|
// specification:
|
||||||
|
// A valid string only contains "[A-Z][a-z][0-9].-_%[p|t]"
|
||||||
|
// %p and %t only allowed once. We only limit usage of filename not path
|
||||||
|
bool is_filename_valid(const char *file_name) {
|
||||||
|
const char* p = file_name;
|
||||||
|
char file_sep = os::file_separator()[0];
|
||||||
|
const char* cp;
|
||||||
|
// skip prefix path
|
||||||
|
for (cp = file_name; *cp != '\0'; cp++) {
|
||||||
|
if (*cp == '/' || *cp == file_sep) {
|
||||||
|
p = cp + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int count_p = 0;
|
||||||
|
int count_t = 0;
|
||||||
|
while (*p != '\0') {
|
||||||
|
if ((*p >= '0' && *p <= '9') ||
|
||||||
|
(*p >= 'A' && *p <= 'Z') ||
|
||||||
|
(*p >= 'a' && *p <= 'z') ||
|
||||||
|
*p == '-' ||
|
||||||
|
*p == '_' ||
|
||||||
|
*p == '.') {
|
||||||
|
p++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (*p == '%') {
|
||||||
|
if(*(p + 1) == 'p') {
|
||||||
|
p += 2;
|
||||||
|
count_p ++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (*(p + 1) == 't') {
|
||||||
|
p += 2;
|
||||||
|
count_t ++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return count_p < 2 && count_t < 2;
|
||||||
|
}
|
||||||
|
|
||||||
// Check consistency of GC selection
|
// Check consistency of GC selection
|
||||||
bool Arguments::check_gc_consistency() {
|
bool Arguments::check_gc_consistency() {
|
||||||
check_gclog_consistency();
|
check_gclog_consistency();
|
||||||
@ -2806,6 +2851,13 @@ jint Arguments::parse_each_vm_init_arg(const JavaVMInitArgs* args,
|
|||||||
// ostream_init_log(), when called will use this filename
|
// ostream_init_log(), when called will use this filename
|
||||||
// to initialize a fileStream.
|
// to initialize a fileStream.
|
||||||
_gc_log_filename = strdup(tail);
|
_gc_log_filename = strdup(tail);
|
||||||
|
if (!is_filename_valid(_gc_log_filename)) {
|
||||||
|
jio_fprintf(defaultStream::output_stream(),
|
||||||
|
"Invalid file name for use with -Xloggc: Filename can only contain the "
|
||||||
|
"characters [A-Z][a-z][0-9]-_.%%[p|t] but it has been %s\n"
|
||||||
|
"Note %%p or %%t can only be used once\n", _gc_log_filename);
|
||||||
|
return JNI_EINVAL;
|
||||||
|
}
|
||||||
FLAG_SET_CMDLINE(bool, PrintGC, true);
|
FLAG_SET_CMDLINE(bool, PrintGC, true);
|
||||||
FLAG_SET_CMDLINE(bool, PrintGCTimeStamps, true);
|
FLAG_SET_CMDLINE(bool, PrintGCTimeStamps, true);
|
||||||
|
|
||||||
|
@ -342,7 +342,7 @@ void stringStream::write(const char* s, size_t len) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
char* stringStream::as_string() {
|
char* stringStream::as_string() {
|
||||||
char* copy = NEW_RESOURCE_ARRAY(char, buffer_pos+1);
|
char* copy = NEW_RESOURCE_ARRAY(char, buffer_pos + 1);
|
||||||
strncpy(copy, buffer, buffer_pos);
|
strncpy(copy, buffer, buffer_pos);
|
||||||
copy[buffer_pos] = 0; // terminating null
|
copy[buffer_pos] = 0; // terminating null
|
||||||
return copy;
|
return copy;
|
||||||
@ -355,14 +355,190 @@ outputStream* tty;
|
|||||||
outputStream* gclog_or_tty;
|
outputStream* gclog_or_tty;
|
||||||
extern Mutex* tty_lock;
|
extern Mutex* tty_lock;
|
||||||
|
|
||||||
|
#define EXTRACHARLEN 32
|
||||||
|
#define CURRENTAPPX ".current"
|
||||||
|
#define FILENAMEBUFLEN 1024
|
||||||
|
// convert YYYY-MM-DD HH:MM:SS to YYYY-MM-DD_HH-MM-SS
|
||||||
|
char* get_datetime_string(char *buf, size_t len) {
|
||||||
|
os::local_time_string(buf, len);
|
||||||
|
int i = (int)strlen(buf);
|
||||||
|
while (i-- >= 0) {
|
||||||
|
if (buf[i] == ' ') buf[i] = '_';
|
||||||
|
else if (buf[i] == ':') buf[i] = '-';
|
||||||
|
}
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char* make_log_name_internal(const char* log_name, const char* force_directory,
|
||||||
|
int pid, const char* tms) {
|
||||||
|
const char* basename = log_name;
|
||||||
|
char file_sep = os::file_separator()[0];
|
||||||
|
const char* cp;
|
||||||
|
char pid_text[32];
|
||||||
|
|
||||||
|
for (cp = log_name; *cp != '\0'; cp++) {
|
||||||
|
if (*cp == '/' || *cp == file_sep) {
|
||||||
|
basename = cp + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const char* nametail = log_name;
|
||||||
|
// Compute buffer length
|
||||||
|
size_t buffer_length;
|
||||||
|
if (force_directory != NULL) {
|
||||||
|
buffer_length = strlen(force_directory) + strlen(os::file_separator()) +
|
||||||
|
strlen(basename) + 1;
|
||||||
|
} else {
|
||||||
|
buffer_length = strlen(log_name) + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// const char* star = strchr(basename, '*');
|
||||||
|
const char* pts = strstr(basename, "%p");
|
||||||
|
int pid_pos = (pts == NULL) ? -1 : (pts - nametail);
|
||||||
|
|
||||||
|
if (pid_pos >= 0) {
|
||||||
|
jio_snprintf(pid_text, sizeof(pid_text), "pid%u", pid);
|
||||||
|
buffer_length += strlen(pid_text);
|
||||||
|
}
|
||||||
|
|
||||||
|
pts = strstr(basename, "%t");
|
||||||
|
int tms_pos = (pts == NULL) ? -1 : (pts - nametail);
|
||||||
|
if (tms_pos >= 0) {
|
||||||
|
buffer_length += strlen(tms);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create big enough buffer.
|
||||||
|
char *buf = NEW_C_HEAP_ARRAY(char, buffer_length, mtInternal);
|
||||||
|
|
||||||
|
strcpy(buf, "");
|
||||||
|
if (force_directory != NULL) {
|
||||||
|
strcat(buf, force_directory);
|
||||||
|
strcat(buf, os::file_separator());
|
||||||
|
nametail = basename; // completely skip directory prefix
|
||||||
|
}
|
||||||
|
|
||||||
|
// who is first, %p or %t?
|
||||||
|
int first = -1, second = -1;
|
||||||
|
const char *p1st = NULL;
|
||||||
|
const char *p2nd = NULL;
|
||||||
|
|
||||||
|
if (pid_pos >= 0 && tms_pos >= 0) {
|
||||||
|
// contains both %p and %t
|
||||||
|
if (pid_pos < tms_pos) {
|
||||||
|
// case foo%pbar%tmonkey.log
|
||||||
|
first = pid_pos;
|
||||||
|
p1st = pid_text;
|
||||||
|
second = tms_pos;
|
||||||
|
p2nd = tms;
|
||||||
|
} else {
|
||||||
|
// case foo%tbar%pmonkey.log
|
||||||
|
first = tms_pos;
|
||||||
|
p1st = tms;
|
||||||
|
second = pid_pos;
|
||||||
|
p2nd = pid_text;
|
||||||
|
}
|
||||||
|
} else if (pid_pos >= 0) {
|
||||||
|
// contains %p only
|
||||||
|
first = pid_pos;
|
||||||
|
p1st = pid_text;
|
||||||
|
} else if (tms_pos >= 0) {
|
||||||
|
// contains %t only
|
||||||
|
first = tms_pos;
|
||||||
|
p1st = tms;
|
||||||
|
}
|
||||||
|
|
||||||
|
int buf_pos = (int)strlen(buf);
|
||||||
|
const char* tail = nametail;
|
||||||
|
|
||||||
|
if (first >= 0) {
|
||||||
|
tail = nametail + first + 2;
|
||||||
|
strncpy(&buf[buf_pos], nametail, first);
|
||||||
|
strcpy(&buf[buf_pos + first], p1st);
|
||||||
|
buf_pos = (int)strlen(buf);
|
||||||
|
if (second >= 0) {
|
||||||
|
strncpy(&buf[buf_pos], tail, second - first - 2);
|
||||||
|
strcpy(&buf[buf_pos + second - first - 2], p2nd);
|
||||||
|
tail = nametail + second + 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
strcat(buf, tail); // append rest of name, or all of name
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
// log_name comes from -XX:LogFile=log_name or -Xloggc:log_name
|
||||||
|
// in log_name, %p => pipd1234 and
|
||||||
|
// %t => YYYY-MM-DD_HH-MM-SS
|
||||||
|
static const char* make_log_name(const char* log_name, const char* force_directory) {
|
||||||
|
char timestr[32];
|
||||||
|
get_datetime_string(timestr, sizeof(timestr));
|
||||||
|
return make_log_name_internal(log_name, force_directory, os::current_process_id(),
|
||||||
|
timestr);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
void test_loggc_filename() {
|
||||||
|
int pid;
|
||||||
|
char tms[32];
|
||||||
|
char i_result[FILENAMEBUFLEN];
|
||||||
|
const char* o_result;
|
||||||
|
get_datetime_string(tms, sizeof(tms));
|
||||||
|
pid = os::current_process_id();
|
||||||
|
|
||||||
|
// test.log
|
||||||
|
jio_snprintf(i_result, sizeof(char)*FILENAMEBUFLEN, "test.log", tms);
|
||||||
|
o_result = make_log_name_internal("test.log", NULL, pid, tms);
|
||||||
|
assert(strcmp(i_result, o_result) == 0, "failed on testing make_log_name(\"test.log\", NULL)");
|
||||||
|
FREE_C_HEAP_ARRAY(char, o_result, mtInternal);
|
||||||
|
|
||||||
|
// test-%t-%p.log
|
||||||
|
jio_snprintf(i_result, sizeof(char)*FILENAMEBUFLEN, "test-%s-pid%u.log", tms, pid);
|
||||||
|
o_result = make_log_name_internal("test-%t-%p.log", NULL, pid, tms);
|
||||||
|
assert(strcmp(i_result, o_result) == 0, "failed on testing make_log_name(\"test-%%t-%%p.log\", NULL)");
|
||||||
|
FREE_C_HEAP_ARRAY(char, o_result, mtInternal);
|
||||||
|
|
||||||
|
// test-%t%p.log
|
||||||
|
jio_snprintf(i_result, sizeof(char)*FILENAMEBUFLEN, "test-%spid%u.log", tms, pid);
|
||||||
|
o_result = make_log_name_internal("test-%t%p.log", NULL, pid, tms);
|
||||||
|
assert(strcmp(i_result, o_result) == 0, "failed on testing make_log_name(\"test-%%t%%p.log\", NULL)");
|
||||||
|
FREE_C_HEAP_ARRAY(char, o_result, mtInternal);
|
||||||
|
|
||||||
|
// %p%t.log
|
||||||
|
jio_snprintf(i_result, sizeof(char)*FILENAMEBUFLEN, "pid%u%s.log", pid, tms);
|
||||||
|
o_result = make_log_name_internal("%p%t.log", NULL, pid, tms);
|
||||||
|
assert(strcmp(i_result, o_result) == 0, "failed on testing make_log_name(\"%%p%%t.log\", NULL)");
|
||||||
|
FREE_C_HEAP_ARRAY(char, o_result, mtInternal);
|
||||||
|
|
||||||
|
// %p-test.log
|
||||||
|
jio_snprintf(i_result, sizeof(char)*FILENAMEBUFLEN, "pid%u-test.log", pid);
|
||||||
|
o_result = make_log_name_internal("%p-test.log", NULL, pid, tms);
|
||||||
|
assert(strcmp(i_result, o_result) == 0, "failed on testing make_log_name(\"%%p-test.log\", NULL)");
|
||||||
|
FREE_C_HEAP_ARRAY(char, o_result, mtInternal);
|
||||||
|
|
||||||
|
// %t.log
|
||||||
|
jio_snprintf(i_result, sizeof(char)*FILENAMEBUFLEN, "%s.log", tms);
|
||||||
|
o_result = make_log_name_internal("%t.log", NULL, pid, tms);
|
||||||
|
assert(strcmp(i_result, o_result) == 0, "failed on testing make_log_name(\"%%t.log\", NULL)");
|
||||||
|
FREE_C_HEAP_ARRAY(char, o_result, mtInternal);
|
||||||
|
}
|
||||||
|
#endif // PRODUCT
|
||||||
|
|
||||||
fileStream::fileStream(const char* file_name) {
|
fileStream::fileStream(const char* file_name) {
|
||||||
_file = fopen(file_name, "w");
|
_file = fopen(file_name, "w");
|
||||||
_need_close = true;
|
if (_file != NULL) {
|
||||||
|
_need_close = true;
|
||||||
|
} else {
|
||||||
|
warning("Cannot open file %s due to %s\n", file_name, strerror(errno));
|
||||||
|
_need_close = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fileStream::fileStream(const char* file_name, const char* opentype) {
|
fileStream::fileStream(const char* file_name, const char* opentype) {
|
||||||
_file = fopen(file_name, opentype);
|
_file = fopen(file_name, opentype);
|
||||||
_need_close = true;
|
if (_file != NULL) {
|
||||||
|
_need_close = true;
|
||||||
|
} else {
|
||||||
|
warning("Cannot open file %s due to %s\n", file_name, strerror(errno));
|
||||||
|
_need_close = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void fileStream::write(const char* s, size_t len) {
|
void fileStream::write(const char* s, size_t len) {
|
||||||
@ -423,34 +599,51 @@ void fdStream::write(const char* s, size_t len) {
|
|||||||
update_position(s, len);
|
update_position(s, len);
|
||||||
}
|
}
|
||||||
|
|
||||||
rotatingFileStream::~rotatingFileStream() {
|
// dump vm version, os version, platform info, build id,
|
||||||
|
// memory usage and command line flags into header
|
||||||
|
void gcLogFileStream::dump_loggc_header() {
|
||||||
|
if (is_open()) {
|
||||||
|
print_cr(Abstract_VM_Version::internal_vm_info_string());
|
||||||
|
os::print_memory_info(this);
|
||||||
|
print("CommandLine flags: ");
|
||||||
|
CommandLineFlags::printSetFlags(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gcLogFileStream::~gcLogFileStream() {
|
||||||
if (_file != NULL) {
|
if (_file != NULL) {
|
||||||
if (_need_close) fclose(_file);
|
if (_need_close) fclose(_file);
|
||||||
_file = NULL;
|
_file = NULL;
|
||||||
|
}
|
||||||
|
if (_file_name != NULL) {
|
||||||
FREE_C_HEAP_ARRAY(char, _file_name, mtInternal);
|
FREE_C_HEAP_ARRAY(char, _file_name, mtInternal);
|
||||||
_file_name = NULL;
|
_file_name = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
rotatingFileStream::rotatingFileStream(const char* file_name) {
|
gcLogFileStream::gcLogFileStream(const char* file_name) {
|
||||||
_cur_file_num = 0;
|
_cur_file_num = 0;
|
||||||
_bytes_written = 0L;
|
_bytes_written = 0L;
|
||||||
_file_name = NEW_C_HEAP_ARRAY(char, strlen(file_name)+10, mtInternal);
|
_file_name = make_log_name(file_name, NULL);
|
||||||
jio_snprintf(_file_name, strlen(file_name)+10, "%s.%d", file_name, _cur_file_num);
|
|
||||||
_file = fopen(_file_name, "w");
|
// gc log file rotation
|
||||||
_need_close = true;
|
if (UseGCLogFileRotation && NumberOfGCLogFiles > 1) {
|
||||||
|
char tempbuf[FILENAMEBUFLEN];
|
||||||
|
jio_snprintf(tempbuf, sizeof(tempbuf), "%s.%d" CURRENTAPPX, _file_name, _cur_file_num);
|
||||||
|
_file = fopen(tempbuf, "w");
|
||||||
|
} else {
|
||||||
|
_file = fopen(_file_name, "w");
|
||||||
|
}
|
||||||
|
if (_file != NULL) {
|
||||||
|
_need_close = true;
|
||||||
|
dump_loggc_header();
|
||||||
|
} else {
|
||||||
|
warning("Cannot open file %s due to %s\n", _file_name, strerror(errno));
|
||||||
|
_need_close = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
rotatingFileStream::rotatingFileStream(const char* file_name, const char* opentype) {
|
void gcLogFileStream::write(const char* s, size_t len) {
|
||||||
_cur_file_num = 0;
|
|
||||||
_bytes_written = 0L;
|
|
||||||
_file_name = NEW_C_HEAP_ARRAY(char, strlen(file_name)+10, mtInternal);
|
|
||||||
jio_snprintf(_file_name, strlen(file_name)+10, "%s.%d", file_name, _cur_file_num);
|
|
||||||
_file = fopen(_file_name, opentype);
|
|
||||||
_need_close = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void rotatingFileStream::write(const char* s, size_t len) {
|
|
||||||
if (_file != NULL) {
|
if (_file != NULL) {
|
||||||
size_t count = fwrite(s, 1, len, _file);
|
size_t count = fwrite(s, 1, len, _file);
|
||||||
_bytes_written += count;
|
_bytes_written += count;
|
||||||
@ -466,7 +659,12 @@ void rotatingFileStream::write(const char* s, size_t len) {
|
|||||||
// write to gc log file at safepoint. If in future, changes made for mutator threads or
|
// write to gc log file at safepoint. If in future, changes made for mutator threads or
|
||||||
// concurrent GC threads to run parallel with VMThread at safepoint, write and rotate_log
|
// concurrent GC threads to run parallel with VMThread at safepoint, write and rotate_log
|
||||||
// must be synchronized.
|
// must be synchronized.
|
||||||
void rotatingFileStream::rotate_log() {
|
void gcLogFileStream::rotate_log() {
|
||||||
|
char time_msg[FILENAMEBUFLEN];
|
||||||
|
char time_str[EXTRACHARLEN];
|
||||||
|
char current_file_name[FILENAMEBUFLEN];
|
||||||
|
char renamed_file_name[FILENAMEBUFLEN];
|
||||||
|
|
||||||
if (_bytes_written < (jlong)GCLogFileSize) {
|
if (_bytes_written < (jlong)GCLogFileSize) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -481,27 +679,89 @@ void rotatingFileStream::rotate_log() {
|
|||||||
// rotate in same file
|
// rotate in same file
|
||||||
rewind();
|
rewind();
|
||||||
_bytes_written = 0L;
|
_bytes_written = 0L;
|
||||||
|
jio_snprintf(time_msg, sizeof(time_msg), "File %s rotated at %s\n",
|
||||||
|
_file_name, os::local_time_string((char *)time_str, sizeof(time_str)));
|
||||||
|
write(time_msg, strlen(time_msg));
|
||||||
|
dump_loggc_header();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// rotate file in names file.0, file.1, file.2, ..., file.<MaxGCLogFileNumbers-1>
|
#if defined(_WINDOWS)
|
||||||
// close current file, rotate to next file
|
#ifndef F_OK
|
||||||
|
#define F_OK 0
|
||||||
|
#endif
|
||||||
|
#endif // _WINDOWS
|
||||||
|
|
||||||
|
// rotate file in names extended_filename.0, extended_filename.1, ...,
|
||||||
|
// extended_filename.<NumberOfGCLogFiles - 1>. Current rotation file name will
|
||||||
|
// have a form of extended_filename.<i>.current where i is the current rotation
|
||||||
|
// file number. After it reaches max file size, the file will be saved and renamed
|
||||||
|
// with .current removed from its tail.
|
||||||
|
size_t filename_len = strlen(_file_name);
|
||||||
if (_file != NULL) {
|
if (_file != NULL) {
|
||||||
_cur_file_num ++;
|
jio_snprintf(renamed_file_name, filename_len + EXTRACHARLEN, "%s.%d",
|
||||||
if (_cur_file_num >= NumberOfGCLogFiles) _cur_file_num = 0;
|
_file_name, _cur_file_num);
|
||||||
jio_snprintf(_file_name, strlen(Arguments::gc_log_filename()) + 10, "%s.%d",
|
jio_snprintf(current_file_name, filename_len + EXTRACHARLEN, "%s.%d" CURRENTAPPX,
|
||||||
Arguments::gc_log_filename(), _cur_file_num);
|
_file_name, _cur_file_num);
|
||||||
|
jio_snprintf(time_msg, sizeof(time_msg), "%s GC log file has reached the"
|
||||||
|
" maximum size. Saved as %s\n",
|
||||||
|
os::local_time_string((char *)time_str, sizeof(time_str)),
|
||||||
|
renamed_file_name);
|
||||||
|
write(time_msg, strlen(time_msg));
|
||||||
|
|
||||||
fclose(_file);
|
fclose(_file);
|
||||||
_file = NULL;
|
_file = NULL;
|
||||||
|
|
||||||
|
bool can_rename = true;
|
||||||
|
if (access(current_file_name, F_OK) != 0) {
|
||||||
|
// current file does not exist?
|
||||||
|
warning("No source file exists, cannot rename\n");
|
||||||
|
can_rename = false;
|
||||||
|
}
|
||||||
|
if (can_rename) {
|
||||||
|
if (access(renamed_file_name, F_OK) == 0) {
|
||||||
|
if (remove(renamed_file_name) != 0) {
|
||||||
|
warning("Could not delete existing file %s\n", renamed_file_name);
|
||||||
|
can_rename = false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// file does not exist, ok to rename
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (can_rename && rename(current_file_name, renamed_file_name) != 0) {
|
||||||
|
warning("Could not rename %s to %s\n", _file_name, renamed_file_name);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
_file = fopen(_file_name, "w");
|
|
||||||
|
_cur_file_num++;
|
||||||
|
if (_cur_file_num > NumberOfGCLogFiles - 1) _cur_file_num = 0;
|
||||||
|
jio_snprintf(current_file_name, filename_len + EXTRACHARLEN, "%s.%d" CURRENTAPPX,
|
||||||
|
_file_name, _cur_file_num);
|
||||||
|
_file = fopen(current_file_name, "w");
|
||||||
|
|
||||||
if (_file != NULL) {
|
if (_file != NULL) {
|
||||||
_bytes_written = 0L;
|
_bytes_written = 0L;
|
||||||
_need_close = true;
|
_need_close = true;
|
||||||
|
// reuse current_file_name for time_msg
|
||||||
|
jio_snprintf(current_file_name, filename_len + EXTRACHARLEN,
|
||||||
|
"%s.%d", _file_name, _cur_file_num);
|
||||||
|
jio_snprintf(time_msg, sizeof(time_msg), "%s GC log file created %s\n",
|
||||||
|
os::local_time_string((char *)time_str, sizeof(time_str)),
|
||||||
|
current_file_name);
|
||||||
|
write(time_msg, strlen(time_msg));
|
||||||
|
dump_loggc_header();
|
||||||
|
// remove the existing file
|
||||||
|
if (access(current_file_name, F_OK) == 0) {
|
||||||
|
if (remove(current_file_name) != 0) {
|
||||||
|
warning("Could not delete existing file %s\n", current_file_name);
|
||||||
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
tty->print_cr("failed to open rotation log file %s due to %s\n",
|
warning("failed to open rotation log file %s due to %s\n"
|
||||||
|
"Turned off GC log file rotation\n",
|
||||||
_file_name, strerror(errno));
|
_file_name, strerror(errno));
|
||||||
_need_close = false;
|
_need_close = false;
|
||||||
|
FLAG_SET_DEFAULT(UseGCLogFileRotation, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -530,66 +790,6 @@ bool defaultStream::has_log_file() {
|
|||||||
return _log_file != NULL;
|
return _log_file != NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const char* make_log_name(const char* log_name, const char* force_directory) {
|
|
||||||
const char* basename = log_name;
|
|
||||||
char file_sep = os::file_separator()[0];
|
|
||||||
const char* cp;
|
|
||||||
for (cp = log_name; *cp != '\0'; cp++) {
|
|
||||||
if (*cp == '/' || *cp == file_sep) {
|
|
||||||
basename = cp+1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
const char* nametail = log_name;
|
|
||||||
|
|
||||||
// Compute buffer length
|
|
||||||
size_t buffer_length;
|
|
||||||
if (force_directory != NULL) {
|
|
||||||
buffer_length = strlen(force_directory) + strlen(os::file_separator()) +
|
|
||||||
strlen(basename) + 1;
|
|
||||||
} else {
|
|
||||||
buffer_length = strlen(log_name) + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* star = strchr(basename, '*');
|
|
||||||
int star_pos = (star == NULL) ? -1 : (star - nametail);
|
|
||||||
int skip = 1;
|
|
||||||
if (star == NULL) {
|
|
||||||
// Try %p
|
|
||||||
star = strstr(basename, "%p");
|
|
||||||
if (star != NULL) {
|
|
||||||
skip = 2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
star_pos = (star == NULL) ? -1 : (star - nametail);
|
|
||||||
|
|
||||||
char pid[32];
|
|
||||||
if (star_pos >= 0) {
|
|
||||||
jio_snprintf(pid, sizeof(pid), "%u", os::current_process_id());
|
|
||||||
buffer_length += strlen(pid);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create big enough buffer.
|
|
||||||
char *buf = NEW_C_HEAP_ARRAY(char, buffer_length, mtInternal);
|
|
||||||
|
|
||||||
strcpy(buf, "");
|
|
||||||
if (force_directory != NULL) {
|
|
||||||
strcat(buf, force_directory);
|
|
||||||
strcat(buf, os::file_separator());
|
|
||||||
nametail = basename; // completely skip directory prefix
|
|
||||||
}
|
|
||||||
|
|
||||||
if (star_pos >= 0) {
|
|
||||||
// convert foo*bar.log or foo%pbar.log to foo123bar.log
|
|
||||||
int buf_pos = (int) strlen(buf);
|
|
||||||
strncpy(&buf[buf_pos], nametail, star_pos);
|
|
||||||
strcpy(&buf[buf_pos + star_pos], pid);
|
|
||||||
nametail += star_pos + skip; // skip prefix and pid format
|
|
||||||
}
|
|
||||||
|
|
||||||
strcat(buf, nametail); // append rest of name, or all of name
|
|
||||||
return buf;
|
|
||||||
}
|
|
||||||
|
|
||||||
void defaultStream::init_log() {
|
void defaultStream::init_log() {
|
||||||
// %%% Need a MutexLocker?
|
// %%% Need a MutexLocker?
|
||||||
const char* log_name = LogFile != NULL ? LogFile : "hotspot.log";
|
const char* log_name = LogFile != NULL ? LogFile : "hotspot.log";
|
||||||
@ -877,11 +1077,8 @@ void ostream_init_log() {
|
|||||||
|
|
||||||
gclog_or_tty = tty; // default to tty
|
gclog_or_tty = tty; // default to tty
|
||||||
if (Arguments::gc_log_filename() != NULL) {
|
if (Arguments::gc_log_filename() != NULL) {
|
||||||
fileStream * gclog = UseGCLogFileRotation ?
|
fileStream * gclog = new(ResourceObj::C_HEAP, mtInternal)
|
||||||
new(ResourceObj::C_HEAP, mtInternal)
|
gcLogFileStream(Arguments::gc_log_filename());
|
||||||
rotatingFileStream(Arguments::gc_log_filename()) :
|
|
||||||
new(ResourceObj::C_HEAP, mtInternal)
|
|
||||||
fileStream(Arguments::gc_log_filename());
|
|
||||||
if (gclog->is_open()) {
|
if (gclog->is_open()) {
|
||||||
// now we update the time stamp of the GC log to be synced up
|
// now we update the time stamp of the GC log to be synced up
|
||||||
// with tty.
|
// with tty.
|
||||||
|
@ -231,20 +231,24 @@ class fdStream : public outputStream {
|
|||||||
void flush() {};
|
void flush() {};
|
||||||
};
|
};
|
||||||
|
|
||||||
class rotatingFileStream : public fileStream {
|
class gcLogFileStream : public fileStream {
|
||||||
protected:
|
protected:
|
||||||
char* _file_name;
|
const char* _file_name;
|
||||||
jlong _bytes_written;
|
jlong _bytes_written;
|
||||||
uintx _cur_file_num; // current logfile rotation number, from 0 to MaxGCLogFileNumbers-1
|
uintx _cur_file_num; // current logfile rotation number, from 0 to NumberOfGCLogFiles-1
|
||||||
public:
|
public:
|
||||||
rotatingFileStream(const char* file_name);
|
gcLogFileStream(const char* file_name);
|
||||||
rotatingFileStream(const char* file_name, const char* opentype);
|
~gcLogFileStream();
|
||||||
rotatingFileStream(FILE* file) : fileStream(file) {}
|
|
||||||
~rotatingFileStream();
|
|
||||||
virtual void write(const char* c, size_t len);
|
virtual void write(const char* c, size_t len);
|
||||||
virtual void rotate_log();
|
virtual void rotate_log();
|
||||||
|
void dump_loggc_header();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
// unit test for checking -Xloggc:<filename> parsing result
|
||||||
|
void test_loggc_filename();
|
||||||
|
#endif
|
||||||
|
|
||||||
void ostream_init();
|
void ostream_init();
|
||||||
void ostream_init_log();
|
void ostream_init_log();
|
||||||
void ostream_exit();
|
void ostream_exit();
|
||||||
|
Loading…
Reference in New Issue
Block a user