From 9de8be0b19b5e0f3ad37e6f56ee241293889f41a Mon Sep 17 00:00:00 2001 From: Tom Tromey Date: Sun, 18 Mar 2001 00:57:13 +0000 Subject: [PATCH] natString.cc (rehash): Don't bother with memset; _Jv_AllocBytes returns zero'd memory. * java/lang/natString.cc (rehash): Don't bother with memset; _Jv_AllocBytes returns zero'd memory. Use _Jv_AllocBytesChecked. Use UNMASK_PTR. (UNMASK_PTR): New macro. (intern): Unmask pointer before returning it. Register finalizer for the string. (unintern): Handle case where (MASK_PTR): New macro. (PTR_MAKSED): Likewise. (_Jv_NewStringUtf8Const): Use UNMASK_PTR. From-SVN: r40593 --- libjava/ChangeLog | 13 ++++++ libjava/java/lang/natString.cc | 76 ++++++++++++++++++++++++++-------- 2 files changed, 71 insertions(+), 18 deletions(-) diff --git a/libjava/ChangeLog b/libjava/ChangeLog index 218639e748f..4ba2d8673b8 100644 --- a/libjava/ChangeLog +++ b/libjava/ChangeLog @@ -1,3 +1,16 @@ +2001-03-17 Tom Tromey + + * java/lang/natString.cc (rehash): Don't bother with memset; + _Jv_AllocBytes returns zero'd memory. Use _Jv_AllocBytesChecked. + Use UNMASK_PTR. + (UNMASK_PTR): New macro. + (intern): Unmask pointer before returning it. Register finalizer + for the string. + (unintern): Handle case where + (MASK_PTR): New macro. + (PTR_MAKSED): Likewise. + (_Jv_NewStringUtf8Const): Use UNMASK_PTR. + 2001-03-01 Andrew Haley * java/lang/natThrowable.cc (printRawStackTrace): Copy the diff --git a/libjava/java/lang/natString.cc b/libjava/java/lang/natString.cc index bb903fbde6f..cc3d8eb46d1 100644 --- a/libjava/java/lang/natString.cc +++ b/libjava/java/lang/natString.cc @@ -1,6 +1,6 @@ // natString.cc - Implementation of java.lang.String native methods. -/* Copyright (C) 1998, 1999, 2000 Free Software Foundation +/* Copyright (C) 1998, 1999, 2000, 2001 Free Software Foundation This file is part of libgcj. @@ -45,6 +45,10 @@ static int strhash_size = 0; /* Number of slots available in strhash. #define DELETED_STRING ((jstring)(~0)) #define SET_STRING_IS_INTERNED(STR) /* nothing */ +#define UNMASK_PTR(Ptr) (((unsigned long) (Ptr)) & ~0x01) +#define MASK_PTR(Ptr) (((unsigned long) (Ptr)) | 0x01) +#define PTR_MASKED(Ptr) (((unsigned long) (Ptr)) & 0x01) + /* Find a slot where the string with elements DATA, length LEN, and hash HASH should go in the strhash table of interned strings. */ jstring* @@ -61,7 +65,8 @@ _Jv_StringFindSlot (jchar* data, jint len, jint hash) for (;;) { jstring* ptr = &strhash[index]; - if (*ptr == NULL) + jstring value = (jstring) UNMASK_PTR (*ptr); + if (value == NULL) { if (deleted_index >= 0) return (&strhash[deleted_index]); @@ -70,8 +75,8 @@ _Jv_StringFindSlot (jchar* data, jint len, jint hash) } else if (*ptr == DELETED_STRING) deleted_index = index; - else if ((*ptr)->length() == len - && memcmp(JvGetStringChars(*ptr), data, 2*len) == 0) + else if (value->length() == len + && memcmp(JvGetStringChars(value), data, 2*len) == 0) return (ptr); index = (index + step) & (strhash_size - 1); JvAssert (index != start_index); @@ -115,16 +120,18 @@ java::lang::String::rehash() if (strhash == NULL) { strhash_size = 1024; - strhash = (jstring *) _Jv_AllocBytes (strhash_size * sizeof (jstring)); + strhash = (jstring *) _Jv_AllocBytesChecked (strhash_size + * sizeof (jstring)); memset (strhash, 0, strhash_size * sizeof (jstring)); } else { int i = strhash_size; jstring* ptr = strhash + i; - strhash_size *= 2; - strhash = (jstring *) _Jv_AllocBytes (strhash_size * sizeof (jstring)); - memset (strhash, 0, strhash_size * sizeof (jstring)); + int nsize = strhash_size * 2; + jstring *next = (jstring *) _Jv_AllocBytesChecked (nsize + * sizeof (jstring)); + memset (next, 0, nsize * sizeof (jstring)); while (--i >= 0) { @@ -134,19 +141,23 @@ java::lang::String::rehash() /* This is faster equivalent of * *__JvGetInternSlot(*ptr) = *ptr; */ - jint hash = (*ptr)->hashCode(); - jint index = hash & (strhash_size - 1); + jstring val = (jstring) UNMASK_PTR (*ptr); + jint hash = val->hashCode(); + jint index = hash & (nsize - 1); jint step = 8 * hash + 7; for (;;) { - if (strhash[index] == NULL) + if (next[index] == NULL) { - strhash[index] = *ptr; + next[index] = *ptr; break; } - index = (index + step) & (strhash_size - 1); + index = (index + step) & (nsize - 1); } } + + strhash_size = nsize; + strhash = next; } } @@ -158,12 +169,16 @@ java::lang::String::intern() rehash(); jstring* ptr = _Jv_StringGetSlot(this); if (*ptr != NULL && *ptr != DELETED_STRING) - return *ptr; + { + // See description in unintern() to understand this. + *ptr = (jstring) MASK_PTR (*ptr); + return (jstring) UNMASK_PTR (*ptr); + } SET_STRING_IS_INTERNED(this); strhash_count++; *ptr = this; // When string is GC'd, clear the slot in the hash table. - // _Jv_RegisterFinalizer ((void *) this, unintern); + _Jv_RegisterFinalizer ((void *) this, unintern); return this; } @@ -176,8 +191,33 @@ java::lang::String::unintern (jobject obj) jstring* ptr = _Jv_StringGetSlot(str); if (*ptr == NULL || *ptr == DELETED_STRING) return; - *ptr = DELETED_STRING; - strhash_count--; + + // We assume the lowest bit of the pointer is free for our nefarious + // manipulations. What we do is set it to `0' (implicitly) when + // interning the String. If we subsequently re-intern the same + // String, then we set the bit. When finalizing, if the bit is set + // then we clear it and re-register the finalizer. We know this is + // a safe approach because both the intern() and unintern() acquire + // the class lock; this bit can't be manipulated when the lock is + // not held. So if we are finalizing and the bit is clear then we + // know all references are gone and we can clear the entry in the + // hash table. The naive approach of simply clearing the pointer + // here fails in the case where a request to intern a new string + // with the same contents is made between the time the intern()d + // string is found to be unreachable and when the finalizer is + // actually run. In this case we could clear a pointer to a valid + // string, and future intern() calls for that particular value would + // spuriously fail. + if (PTR_MASKED (*ptr)) + { + *ptr = (jstring) UNMASK_PTR (*ptr); + _Jv_RegisterFinalizer ((void *) obj, unintern); + } + else + { + *ptr = DELETED_STRING; + strhash_count--; + } } jstring @@ -232,7 +272,7 @@ _Jv_NewStringUtf8Const (Utf8Const* str) int hash = str->hash; jstring* ptr = _Jv_StringFindSlot (chrs, length, hash); if (*ptr != NULL && *ptr != DELETED_STRING) - return *ptr; + return (jstring) UNMASK_PTR (*ptr); strhash_count++; if (jstr == NULL) { -- 2.30.2