8224624: Inefficiencies in CodeStrings::add_comment cause timeouts

Changing CodeStrings to a doubly-linked-list and searching for the comment with the right offset in reverse.

Reviewed-by: kvn
This commit is contained in:
Tobias Hartmann 2019-08-22 12:24:02 +02:00
parent 0941bf8b5e
commit 7cff981f5a
2 changed files with 29 additions and 9 deletions

View File

@ -1050,10 +1050,11 @@ class CodeString: public CHeapObj<mtCode> {
friend class CodeStrings; friend class CodeStrings;
const char * _string; const char * _string;
CodeString* _next; CodeString* _next;
CodeString* _prev;
intptr_t _offset; intptr_t _offset;
~CodeString() { ~CodeString() {
assert(_next == NULL, "wrong interface for freeing list"); assert(_next == NULL && _prev == NULL, "wrong interface for freeing list");
os::free((void*)_string); os::free((void*)_string);
} }
@ -1061,7 +1062,7 @@ class CodeString: public CHeapObj<mtCode> {
public: public:
CodeString(const char * string, intptr_t offset = -1) CodeString(const char * string, intptr_t offset = -1)
: _next(NULL), _offset(offset) { : _next(NULL), _prev(NULL), _offset(offset) {
_string = os::strdup(string, mtCode); _string = os::strdup(string, mtCode);
} }
@ -1069,7 +1070,12 @@ class CodeString: public CHeapObj<mtCode> {
intptr_t offset() const { assert(_offset >= 0, "offset for non comment?"); return _offset; } intptr_t offset() const { assert(_offset >= 0, "offset for non comment?"); return _offset; }
CodeString* next() const { return _next; } CodeString* next() const { return _next; }
void set_next(CodeString* next) { _next = next; } void set_next(CodeString* next) {
_next = next;
if (next != NULL) {
next->_prev = this;
}
}
CodeString* first_comment() { CodeString* first_comment() {
if (is_comment()) { if (is_comment()) {
@ -1097,12 +1103,9 @@ CodeString* CodeStrings::find(intptr_t offset) const {
// Convenience for add_comment. // Convenience for add_comment.
CodeString* CodeStrings::find_last(intptr_t offset) const { CodeString* CodeStrings::find_last(intptr_t offset) const {
CodeString* a = find(offset); CodeString* a = _strings_last;
if (a != NULL) { while (a != NULL && !a->is_comment() && a->offset() > offset) {
CodeString* c = NULL; a = a->_prev;
while (((c = a->next_comment()) != NULL) && (c->offset() == offset)) {
a = c;
}
} }
return a; return a;
} }
@ -1121,12 +1124,16 @@ void CodeStrings::add_comment(intptr_t offset, const char * comment) {
c->set_next(_strings); c->set_next(_strings);
_strings = c; _strings = c;
} }
if (c->next() == NULL) {
_strings_last = c;
}
} }
void CodeStrings::assign(CodeStrings& other) { void CodeStrings::assign(CodeStrings& other) {
other.check_valid(); other.check_valid();
assert(is_null(), "Cannot assign onto non-empty CodeStrings"); assert(is_null(), "Cannot assign onto non-empty CodeStrings");
_strings = other._strings; _strings = other._strings;
_strings_last = other._strings_last;
#ifdef ASSERT #ifdef ASSERT
_defunct = false; _defunct = false;
#endif #endif
@ -1142,8 +1149,11 @@ void CodeStrings::copy(CodeStrings& other) {
assert(is_null(), "Cannot copy onto non-empty CodeStrings"); assert(is_null(), "Cannot copy onto non-empty CodeStrings");
CodeString* n = other._strings; CodeString* n = other._strings;
CodeString** ps = &_strings; CodeString** ps = &_strings;
CodeString* prev = NULL;
while (n != NULL) { while (n != NULL) {
*ps = new CodeString(n->string(),n->offset()); *ps = new CodeString(n->string(),n->offset());
(*ps)->_prev = prev;
prev = *ps;
ps = &((*ps)->_next); ps = &((*ps)->_next);
n = n->next(); n = n->next();
} }
@ -1180,6 +1190,10 @@ void CodeStrings::free() {
// unlink the node from the list saving a pointer to the next // unlink the node from the list saving a pointer to the next
CodeString* p = n->next(); CodeString* p = n->next();
n->set_next(NULL); n->set_next(NULL);
if (p != NULL) {
assert(p->_prev == n, "missing prev link");
p->_prev = NULL;
}
delete n; delete n;
n = p; n = p;
} }
@ -1190,6 +1204,9 @@ const char* CodeStrings::add_string(const char * string) {
check_valid(); check_valid();
CodeString* s = new CodeString(string); CodeString* s = new CodeString(string);
s->set_next(_strings); s->set_next(_strings);
if (_strings == NULL) {
_strings_last = s;
}
_strings = s; _strings = s;
assert(s->string() != NULL, "should have a string"); assert(s->string() != NULL, "should have a string");
return s->string(); return s->string();

View File

@ -249,6 +249,7 @@ class CodeStrings {
private: private:
#ifndef PRODUCT #ifndef PRODUCT
CodeString* _strings; CodeString* _strings;
CodeString* _strings_last;
#ifdef ASSERT #ifdef ASSERT
// Becomes true after copy-out, forbids further use. // Becomes true after copy-out, forbids further use.
bool _defunct; // Zero bit pattern is "valid", see memset call in decode_env::decode_env bool _defunct; // Zero bit pattern is "valid", see memset call in decode_env::decode_env
@ -262,6 +263,7 @@ private:
void set_null_and_invalidate() { void set_null_and_invalidate() {
#ifndef PRODUCT #ifndef PRODUCT
_strings = NULL; _strings = NULL;
_strings_last = NULL;
#ifdef ASSERT #ifdef ASSERT
_defunct = true; _defunct = true;
#endif #endif
@ -272,6 +274,7 @@ public:
CodeStrings() { CodeStrings() {
#ifndef PRODUCT #ifndef PRODUCT
_strings = NULL; _strings = NULL;
_strings_last = NULL;
#ifdef ASSERT #ifdef ASSERT
_defunct = false; _defunct = false;
#endif #endif