8297766: Remove UseMallocOnly development option
Reviewed-by: coleenp, stuefe, dholmes
This commit is contained in:
parent
b9eec96889
commit
bd381886e0
@ -273,10 +273,6 @@ Arena::~Arena() {
|
|||||||
|
|
||||||
// Destroy this arenas contents and reset to empty
|
// Destroy this arenas contents and reset to empty
|
||||||
void Arena::destruct_contents() {
|
void Arena::destruct_contents() {
|
||||||
if (UseMallocOnly && _first != NULL) {
|
|
||||||
char* end = _first->next() ? _first->top() : _hwm;
|
|
||||||
free_malloced_objects(_first, _first->bottom(), end, _hwm);
|
|
||||||
}
|
|
||||||
// reset size before chop to avoid a rare racing condition
|
// reset size before chop to avoid a rare racing condition
|
||||||
// that can have total arena memory exceed total chunk memory
|
// that can have total arena memory exceed total chunk memory
|
||||||
set_size_in_bytes(0);
|
set_size_in_bytes(0);
|
||||||
@ -342,19 +338,6 @@ void *Arena::Arealloc(void* old_ptr, size_t old_size, size_t new_size, AllocFail
|
|||||||
assert(old_size == 0, "sanity");
|
assert(old_size == 0, "sanity");
|
||||||
return Amalloc(new_size, alloc_failmode); // as with realloc(3), a NULL old ptr is equivalent to malloc(3)
|
return Amalloc(new_size, alloc_failmode); // as with realloc(3), a NULL old ptr is equivalent to malloc(3)
|
||||||
}
|
}
|
||||||
#ifdef ASSERT
|
|
||||||
if (UseMallocOnly) {
|
|
||||||
// always allocate a new object (otherwise we'll free this one twice)
|
|
||||||
char* copy = (char*)Amalloc(new_size, alloc_failmode);
|
|
||||||
if (copy == NULL) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
size_t n = MIN2(old_size, new_size);
|
|
||||||
if (n > 0) memcpy(copy, old_ptr, n);
|
|
||||||
Afree(old_ptr,old_size); // Mostly done to keep stats accurate
|
|
||||||
return copy;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
char *c_old = (char*)old_ptr; // Handy name
|
char *c_old = (char*)old_ptr; // Handy name
|
||||||
// Stupid fast special case
|
// Stupid fast special case
|
||||||
if( new_size <= old_size ) { // Shrink in-place
|
if( new_size <= old_size ) { // Shrink in-place
|
||||||
@ -386,24 +369,6 @@ void *Arena::Arealloc(void* old_ptr, size_t old_size, size_t new_size, AllocFail
|
|||||||
|
|
||||||
// Determine if pointer belongs to this Arena or not.
|
// Determine if pointer belongs to this Arena or not.
|
||||||
bool Arena::contains( const void *ptr ) const {
|
bool Arena::contains( const void *ptr ) const {
|
||||||
#ifdef ASSERT
|
|
||||||
if (UseMallocOnly) {
|
|
||||||
// really slow, but not easy to make fast
|
|
||||||
if (_chunk == NULL) return false;
|
|
||||||
char** bottom = (char**)_chunk->bottom();
|
|
||||||
for (char** p = (char**)_hwm - 1; p >= bottom; p--) {
|
|
||||||
if (*p == ptr) return true;
|
|
||||||
}
|
|
||||||
for (Chunk *c = _first; c != NULL; c = c->next()) {
|
|
||||||
if (c == _chunk) continue; // current chunk has been processed
|
|
||||||
char** bottom = (char**)c->bottom();
|
|
||||||
for (char** p = (char**)c->top() - 1; p >= bottom; p--) {
|
|
||||||
if (*p == ptr) return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
if( (void*)_chunk->bottom() <= ptr && ptr < (void*)_hwm )
|
if( (void*)_chunk->bottom() <= ptr && ptr < (void*)_hwm )
|
||||||
return true; // Check for in this chunk
|
return true; // Check for in this chunk
|
||||||
for (Chunk *c = _first; c; c = c->next()) {
|
for (Chunk *c = _first; c; c = c->next()) {
|
||||||
@ -414,51 +379,3 @@ bool Arena::contains( const void *ptr ) const {
|
|||||||
}
|
}
|
||||||
return false; // Not in any Chunk, so not in Arena
|
return false; // Not in any Chunk, so not in Arena
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#ifdef ASSERT
|
|
||||||
void* Arena::malloc(size_t size) {
|
|
||||||
assert(UseMallocOnly, "shouldn't call");
|
|
||||||
// use malloc, but save pointer in res. area for later freeing
|
|
||||||
char** save = (char**)internal_amalloc(sizeof(char*));
|
|
||||||
return (*save = (char*)os::malloc(size, mtChunk));
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
//--------------------------------------------------------------------------------------
|
|
||||||
// Non-product code
|
|
||||||
|
|
||||||
#ifndef PRODUCT
|
|
||||||
|
|
||||||
// debugging code
|
|
||||||
inline void Arena::free_all(char** start, char** end) {
|
|
||||||
for (char** p = start; p < end; p++) if (*p) os::free(*p);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Arena::free_malloced_objects(Chunk* chunk, char* hwm, char* max, char* hwm2) {
|
|
||||||
assert(UseMallocOnly, "should not call");
|
|
||||||
// free all objects malloced since resource mark was created; resource area
|
|
||||||
// contains their addresses
|
|
||||||
if (chunk->next()) {
|
|
||||||
// this chunk is full, and some others too
|
|
||||||
for (Chunk* c = chunk->next(); c != NULL; c = c->next()) {
|
|
||||||
char* top = c->top();
|
|
||||||
if (c->next() == NULL) {
|
|
||||||
top = hwm2; // last junk is only used up to hwm2
|
|
||||||
assert(c->contains(hwm2), "bad hwm2");
|
|
||||||
}
|
|
||||||
free_all((char**)c->bottom(), (char**)top);
|
|
||||||
}
|
|
||||||
assert(chunk->contains(hwm), "bad hwm");
|
|
||||||
assert(chunk->contains(max), "bad max");
|
|
||||||
free_all((char**)hwm, (char**)max);
|
|
||||||
} else {
|
|
||||||
// this chunk was partially used
|
|
||||||
assert(chunk->contains(hwm), "bad hwm");
|
|
||||||
assert(chunk->contains(hwm2), "bad hwm2");
|
|
||||||
free_all((char**)hwm, (char**)hwm2);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // Non-product
|
|
||||||
|
@ -125,7 +125,6 @@ protected:
|
|||||||
// on both 32 and 64 bit platforms. Required for atomic jlong operations on 32 bits.
|
// on both 32 and 64 bit platforms. Required for atomic jlong operations on 32 bits.
|
||||||
void* Amalloc(size_t x, AllocFailType alloc_failmode = AllocFailStrategy::EXIT_OOM) {
|
void* Amalloc(size_t x, AllocFailType alloc_failmode = AllocFailStrategy::EXIT_OOM) {
|
||||||
x = ARENA_ALIGN(x); // note for 32 bits this should align _hwm as well.
|
x = ARENA_ALIGN(x); // note for 32 bits this should align _hwm as well.
|
||||||
debug_only(if (UseMallocOnly) return malloc(x);)
|
|
||||||
// Amalloc guarantees 64-bit alignment and we need to ensure that in case the preceding
|
// Amalloc guarantees 64-bit alignment and we need to ensure that in case the preceding
|
||||||
// allocation was AmallocWords. Only needed on 32-bit - on 64-bit Amalloc and AmallocWords are
|
// allocation was AmallocWords. Only needed on 32-bit - on 64-bit Amalloc and AmallocWords are
|
||||||
// identical.
|
// identical.
|
||||||
@ -138,7 +137,6 @@ protected:
|
|||||||
// is 4 bytes on 32 bits, hence the name.
|
// is 4 bytes on 32 bits, hence the name.
|
||||||
void* AmallocWords(size_t x, AllocFailType alloc_failmode = AllocFailStrategy::EXIT_OOM) {
|
void* AmallocWords(size_t x, AllocFailType alloc_failmode = AllocFailStrategy::EXIT_OOM) {
|
||||||
assert(is_aligned(x, BytesPerWord), "misaligned size");
|
assert(is_aligned(x, BytesPerWord), "misaligned size");
|
||||||
debug_only(if (UseMallocOnly) return malloc(x);)
|
|
||||||
return internal_amalloc(x, alloc_failmode);
|
return internal_amalloc(x, alloc_failmode);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -149,7 +147,6 @@ protected:
|
|||||||
}
|
}
|
||||||
#ifdef ASSERT
|
#ifdef ASSERT
|
||||||
if (ZapResourceArea) memset(ptr, badResourceValue, size); // zap freed memory
|
if (ZapResourceArea) memset(ptr, badResourceValue, size); // zap freed memory
|
||||||
if (UseMallocOnly) return true;
|
|
||||||
#endif
|
#endif
|
||||||
if (((char*)ptr) + size == _hwm) {
|
if (((char*)ptr) + size == _hwm) {
|
||||||
_hwm = (char*)ptr;
|
_hwm = (char*)ptr;
|
||||||
|
@ -105,10 +105,6 @@ public:
|
|||||||
assert(_nesting > state._nesting, "rollback to inactive mark");
|
assert(_nesting > state._nesting, "rollback to inactive mark");
|
||||||
assert((_nesting - state._nesting) == 1, "rollback across another mark");
|
assert((_nesting - state._nesting) == 1, "rollback across another mark");
|
||||||
|
|
||||||
if (UseMallocOnly) {
|
|
||||||
free_malloced_objects(state._chunk, state._hwm, state._max, _hwm);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state._chunk->next() != nullptr) { // Delete later chunks.
|
if (state._chunk->next() != nullptr) { // Delete later chunks.
|
||||||
// Reset size before deleting chunks. Otherwise, the total
|
// Reset size before deleting chunks. Otherwise, the total
|
||||||
// size could exceed the total chunk size.
|
// size could exceed the total chunk size.
|
||||||
|
@ -32,11 +32,6 @@
|
|||||||
inline char* ResourceArea::allocate_bytes(size_t size, AllocFailType alloc_failmode) {
|
inline char* ResourceArea::allocate_bytes(size_t size, AllocFailType alloc_failmode) {
|
||||||
#ifdef ASSERT
|
#ifdef ASSERT
|
||||||
verify_has_resource_mark();
|
verify_has_resource_mark();
|
||||||
if (UseMallocOnly) {
|
|
||||||
// use malloc, but save pointer in res. area for later freeing
|
|
||||||
char** save = (char**)internal_amalloc(sizeof(char*));
|
|
||||||
return (*save = (char*)os::malloc(size, mtThread, CURRENT_PC));
|
|
||||||
}
|
|
||||||
#endif // ASSERT
|
#endif // ASSERT
|
||||||
return (char*)Amalloc(size, alloc_failmode);
|
return (char*)Amalloc(size, alloc_failmode);
|
||||||
}
|
}
|
||||||
|
@ -455,10 +455,6 @@ const int ObjectAlignmentInBytes = 8;
|
|||||||
notproduct(bool, VerifyCodeCache, false, \
|
notproduct(bool, VerifyCodeCache, false, \
|
||||||
"Verify code cache on memory allocation/deallocation") \
|
"Verify code cache on memory allocation/deallocation") \
|
||||||
\
|
\
|
||||||
develop(bool, UseMallocOnly, false, \
|
|
||||||
"Use only malloc/free for allocation (no resource area/arena). " \
|
|
||||||
"Used to help diagnose memory stomping bugs.") \
|
|
||||||
\
|
|
||||||
develop(bool, ZapResourceArea, trueInDebug, \
|
develop(bool, ZapResourceArea, trueInDebug, \
|
||||||
"Zap freed resource/arena space") \
|
"Zap freed resource/arena space") \
|
||||||
\
|
\
|
||||||
|
@ -196,7 +196,6 @@ class HandleArea: public Arena {
|
|||||||
// Handle allocation
|
// Handle allocation
|
||||||
private:
|
private:
|
||||||
oop* real_allocate_handle(oop obj) {
|
oop* real_allocate_handle(oop obj) {
|
||||||
// Ignore UseMallocOnly by allocating only in arena.
|
|
||||||
oop* handle = (oop*)internal_amalloc(oopSize);
|
oop* handle = (oop*)internal_amalloc(oopSize);
|
||||||
*handle = obj;
|
*handle = obj;
|
||||||
return handle;
|
return handle;
|
||||||
|
@ -57,17 +57,13 @@ TEST_VM(Arena, alloc_size_0) {
|
|||||||
void* p = ar.Amalloc(0);
|
void* p = ar.Amalloc(0);
|
||||||
ASSERT_NOT_NULL(p);
|
ASSERT_NOT_NULL(p);
|
||||||
ASSERT_ALIGN_AMALLOC(p);
|
ASSERT_ALIGN_AMALLOC(p);
|
||||||
if (!UseMallocOnly) {
|
|
||||||
// contains works differently for malloced mode (and there its broken anyway)
|
ASSERT_FALSE(ar.contains(p));
|
||||||
ASSERT_FALSE(ar.contains(p));
|
|
||||||
}
|
|
||||||
// Allocate again. The new allocations should have the same position as the 0-sized
|
// Allocate again. The new allocations should have the same position as the 0-sized
|
||||||
// first one.
|
// first one.
|
||||||
if (!UseMallocOnly) {
|
void* p2 = ar.Amalloc(1);
|
||||||
void* p2 = ar.Amalloc(1);
|
ASSERT_AMALLOC(ar, p2);
|
||||||
ASSERT_AMALLOC(ar, p2);
|
ASSERT_EQ(p2, p);
|
||||||
ASSERT_EQ(p2, p);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Test behavior for Arealloc(p, 0)
|
// Test behavior for Arealloc(p, 0)
|
||||||
@ -81,10 +77,8 @@ TEST_VM(Arena, realloc_size_0) {
|
|||||||
ASSERT_NULL(p2);
|
ASSERT_NULL(p2);
|
||||||
|
|
||||||
// a subsequent allocation should get the same pointer
|
// a subsequent allocation should get the same pointer
|
||||||
if (!UseMallocOnly) {
|
void* p3 = ar.Amalloc(0x20);
|
||||||
void* p3 = ar.Amalloc(0x20);
|
ASSERT_EQ(p3, p1);
|
||||||
ASSERT_EQ(p3, p1);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Realloc equal sizes is a noop
|
// Realloc equal sizes is a noop
|
||||||
@ -96,9 +90,7 @@ TEST_VM(Arena, realloc_same_size) {
|
|||||||
|
|
||||||
void* p2 = ar.Arealloc(p1, 0x200, 0x200);
|
void* p2 = ar.Arealloc(p1, 0x200, 0x200);
|
||||||
|
|
||||||
if (!UseMallocOnly) {
|
ASSERT_EQ(p2, p1);
|
||||||
ASSERT_EQ(p2, p1);
|
|
||||||
}
|
|
||||||
ASSERT_RANGE_IS_MARKED(p2, 0x200);
|
ASSERT_RANGE_IS_MARKED(p2, 0x200);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -157,29 +149,26 @@ TEST_VM(Arena, free_top) {
|
|||||||
DEBUG_ONLY(ASSERT_RANGE_IS_MARKED_WITH(p, 0x10, badResourceValue);)
|
DEBUG_ONLY(ASSERT_RANGE_IS_MARKED_WITH(p, 0x10, badResourceValue);)
|
||||||
|
|
||||||
// a subsequent allocation should get the same pointer
|
// a subsequent allocation should get the same pointer
|
||||||
if (!UseMallocOnly) {
|
void* p2 = ar.Amalloc(0x20);
|
||||||
void* p2 = ar.Amalloc(0x20);
|
ASSERT_EQ(p2, p);
|
||||||
ASSERT_EQ(p2, p);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// In-place shrinking.
|
// In-place shrinking.
|
||||||
TEST_VM(Arena, realloc_top_shrink) {
|
TEST_VM(Arena, realloc_top_shrink) {
|
||||||
if (!UseMallocOnly) {
|
Arena ar(mtTest);
|
||||||
Arena ar(mtTest);
|
|
||||||
|
|
||||||
void* p1 = ar.Amalloc(0x200);
|
void* p1 = ar.Amalloc(0x200);
|
||||||
ASSERT_AMALLOC(ar, p1);
|
ASSERT_AMALLOC(ar, p1);
|
||||||
GtestUtils::mark_range(p1, 0x200);
|
GtestUtils::mark_range(p1, 0x200);
|
||||||
|
|
||||||
void* p2 = ar.Arealloc(p1, 0x200, 0x100);
|
void* p2 = ar.Arealloc(p1, 0x200, 0x100);
|
||||||
ASSERT_EQ(p1, p2);
|
ASSERT_EQ(p1, p2);
|
||||||
ASSERT_RANGE_IS_MARKED(p2, 0x100); // realloc should preserve old content
|
ASSERT_RANGE_IS_MARKED(p2, 0x100); // realloc should preserve old content
|
||||||
|
|
||||||
// A subsequent allocation should be placed right after the end of the first, shrunk, allocation
|
// A subsequent allocation should be placed right after the end of the first, shrunk, allocation
|
||||||
void* p3 = ar.Amalloc(1);
|
void* p3 = ar.Amalloc(1);
|
||||||
ASSERT_EQ(p3, ((char*)p1) + 0x100);
|
ASSERT_EQ(p3, ((char*)p1) + 0x100);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// not-in-place shrinking.
|
// not-in-place shrinking.
|
||||||
@ -193,9 +182,7 @@ TEST_VM(Arena, realloc_nontop_shrink) {
|
|||||||
void* p_other = ar.Amalloc(20); // new top, p1 not top anymore
|
void* p_other = ar.Amalloc(20); // new top, p1 not top anymore
|
||||||
|
|
||||||
void* p2 = ar.Arealloc(p1, 200, 100);
|
void* p2 = ar.Arealloc(p1, 200, 100);
|
||||||
if (!UseMallocOnly) {
|
ASSERT_EQ(p1, p2); // should still shrink in place
|
||||||
ASSERT_EQ(p1, p2); // should still shrink in place
|
|
||||||
}
|
|
||||||
ASSERT_RANGE_IS_MARKED(p2, 100); // realloc should preserve old content
|
ASSERT_RANGE_IS_MARKED(p2, 100); // realloc should preserve old content
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -208,9 +195,7 @@ TEST_VM(Arena, realloc_top_grow) {
|
|||||||
GtestUtils::mark_range(p1, 0x10);
|
GtestUtils::mark_range(p1, 0x10);
|
||||||
|
|
||||||
void* p2 = ar.Arealloc(p1, 0x10, 0x20);
|
void* p2 = ar.Arealloc(p1, 0x10, 0x20);
|
||||||
if (!UseMallocOnly) {
|
ASSERT_EQ(p1, p2);
|
||||||
ASSERT_EQ(p1, p2);
|
|
||||||
}
|
|
||||||
ASSERT_RANGE_IS_MARKED(p2, 0x10); // realloc should preserve old content
|
ASSERT_RANGE_IS_MARKED(p2, 0x10); // realloc should preserve old content
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,40 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2020, 2021, Oracle and/or its affiliates. All rights reserved.
|
|
||||||
* Copyright (c) 2020 SAP SE. 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.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Note: This runs the Arena portion of the gtests with UseMallocOnly
|
|
||||||
* (restricted to debug since UseMallocOnly is debug-only)
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* @test
|
|
||||||
* @bug 8271242
|
|
||||||
* @summary Run arena tests with UseMallocOnly
|
|
||||||
* @requires vm.debug
|
|
||||||
* @library /test/lib
|
|
||||||
* @modules java.base/jdk.internal.misc
|
|
||||||
* java.xml
|
|
||||||
* @requires vm.flagless
|
|
||||||
* @run main/native GTestWrapper --gtest_filter=Arena* -XX:+UseMallocOnly
|
|
||||||
*/
|
|
@ -1,41 +0,0 @@
|
|||||||
/*
|
|
||||||
* 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 8007475
|
|
||||||
* @summary Test memory stomp in stack map test
|
|
||||||
* @run main/othervm -XX:+IgnoreUnrecognizedVMOptions -XX:+UseMallocOnly StackMapFrameTest
|
|
||||||
*/
|
|
||||||
public class StackMapFrameTest {
|
|
||||||
|
|
||||||
public static void foo() {
|
|
||||||
Object o = new Object();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void main(String args[]) {
|
|
||||||
for (int i = 0; i < 25000; i++) {
|
|
||||||
foo();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user