Commit 9b0d7315 authored by Barry Perlman's avatar Barry Perlman Committed by Yoni Fogel

[t:4182] #4182 Merge tokudb.4182 to main. Add mallocator version string to engine status.

git-svn-id: file:///svn/toku/tokudb@37423 c7de825b-a66e-492c-adef-691d508d4ae1
parent 5a2ae5f9
...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status { ...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status {
uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */ uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */
uint64_t mem_freed; /* number of bytes freed */ uint64_t mem_freed; /* number of bytes freed */
uint64_t max_mem_in_use; /* estimated max value of (used - freed) */ uint64_t max_mem_in_use; /* estimated max value of (used - freed) */
const char * mallocator_version; /* version string from malloc lib */
} ENGINE_STATUS; } ENGINE_STATUS;
typedef enum { typedef enum {
DB_BTREE=1, DB_BTREE=1,
......
...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status { ...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status {
uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */ uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */
uint64_t mem_freed; /* number of bytes freed */ uint64_t mem_freed; /* number of bytes freed */
uint64_t max_mem_in_use; /* estimated max value of (used - freed) */ uint64_t max_mem_in_use; /* estimated max value of (used - freed) */
const char * mallocator_version; /* version string from malloc lib */
} ENGINE_STATUS; } ENGINE_STATUS;
typedef enum { typedef enum {
DB_BTREE=1, DB_BTREE=1,
......
...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status { ...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status {
uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */ uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */
uint64_t mem_freed; /* number of bytes freed */ uint64_t mem_freed; /* number of bytes freed */
uint64_t max_mem_in_use; /* estimated max value of (used - freed) */ uint64_t max_mem_in_use; /* estimated max value of (used - freed) */
const char * mallocator_version; /* version string from malloc lib */
} ENGINE_STATUS; } ENGINE_STATUS;
typedef enum { typedef enum {
DB_BTREE=1, DB_BTREE=1,
......
...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status { ...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status {
uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */ uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */
uint64_t mem_freed; /* number of bytes freed */ uint64_t mem_freed; /* number of bytes freed */
uint64_t max_mem_in_use; /* estimated max value of (used - freed) */ uint64_t max_mem_in_use; /* estimated max value of (used - freed) */
const char * mallocator_version; /* version string from malloc lib */
} ENGINE_STATUS; } ENGINE_STATUS;
typedef enum { typedef enum {
DB_BTREE=1, DB_BTREE=1,
......
...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status { ...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status {
uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */ uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */
uint64_t mem_freed; /* number of bytes freed */ uint64_t mem_freed; /* number of bytes freed */
uint64_t max_mem_in_use; /* estimated max value of (used - freed) */ uint64_t max_mem_in_use; /* estimated max value of (used - freed) */
const char * mallocator_version; /* version string from malloc lib */
} ENGINE_STATUS; } ENGINE_STATUS;
typedef enum { typedef enum {
DB_BTREE=1, DB_BTREE=1,
......
...@@ -664,6 +664,7 @@ int main (int argc __attribute__((__unused__)), char *const argv[] __attribute__ ...@@ -664,6 +664,7 @@ int main (int argc __attribute__((__unused__)), char *const argv[] __attribute__
printf(" uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */ \n"); printf(" uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */ \n");
printf(" uint64_t mem_freed; /* number of bytes freed */ \n"); printf(" uint64_t mem_freed; /* number of bytes freed */ \n");
printf(" uint64_t max_mem_in_use; /* estimated max value of (used - freed) */ \n"); printf(" uint64_t max_mem_in_use; /* estimated max value of (used - freed) */ \n");
printf(" const char * mallocator_version; /* version string from malloc lib */ \n");
printf("} ENGINE_STATUS;\n"); printf("} ENGINE_STATUS;\n");
print_dbtype(); print_dbtype();
......
...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status { ...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status {
uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */ uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */
uint64_t mem_freed; /* number of bytes freed */ uint64_t mem_freed; /* number of bytes freed */
uint64_t max_mem_in_use; /* estimated max value of (used - freed) */ uint64_t max_mem_in_use; /* estimated max value of (used - freed) */
const char * mallocator_version; /* version string from malloc lib */
} ENGINE_STATUS; } ENGINE_STATUS;
typedef enum { typedef enum {
DB_BTREE=1, DB_BTREE=1,
......
...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status { ...@@ -269,6 +269,7 @@ typedef struct __toku_engine_status {
uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */ uint64_t mem_used; /* number of bytes used (obtained from malloc_usable_size()) */
uint64_t mem_freed; /* number of bytes freed */ uint64_t mem_freed; /* number of bytes freed */
uint64_t max_mem_in_use; /* estimated max value of (used - freed) */ uint64_t max_mem_in_use; /* estimated max value of (used - freed) */
const char * mallocator_version; /* version string from malloc lib */
} ENGINE_STATUS; } ENGINE_STATUS;
typedef enum { typedef enum {
DB_BTREE=1, DB_BTREE=1,
......
../windows/memory.c
\ No newline at end of file
/* -*- mode: C; c-basic-offset: 4 -*- */
#ident "Copyright (c) 2007-2011 Tokutek Inc. All rights reserved."
#include <toku_portability.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <malloc.h>
#include <dlfcn.h>
#include "memory.h"
#include "toku_assert.h"
static malloc_fun_t t_malloc = 0;
static malloc_fun_t t_xmalloc = 0;
static free_fun_t t_free = 0;
static realloc_fun_t t_realloc = 0;
static realloc_fun_t t_xrealloc = 0;
static MEMORY_STATUS_S status;
void
toku_memory_get_status(MEMORY_STATUS s) {
if (status.mallocator_version == NULL) {
// mallctl in jemalloc can be used to get the version string
int (*mallctl)(const char *, void *, size_t *, void *, size_t);
mallctl = dlsym(NULL, "mallctl");
if (mallctl) {
size_t version_length = sizeof status.mallocator_version;
int r = mallctl("version", &status.mallocator_version, &version_length, NULL, 0);
assert(r == 0);
} else
status.mallocator_version = "libc";
}
*s = status;
}
// jemalloc's malloc_usable_size does not work with a NULL pointer, so we implement a version that works
static size_t
my_malloc_usable_size(void *p) {
return p == NULL ? 0 : malloc_usable_size(p);
}
// max_in_use may be slightly off because use of max_in_use is not thread-safe.
// It is not worth the overhead to make it completely accurate.
static inline void
set_max(uint64_t sum_used, uint64_t sum_freed) {
uint64_t in_use = (sum_used - sum_freed);
if ((!(in_use & 0x8000000000000000)) // if wrap due to another thread, ignore bogus "negative" value
&& (in_use > status.max_in_use)) {
status.max_in_use = in_use;
}
}
void *toku_malloc(size_t size) {
void *p = t_malloc ? t_malloc(size) : os_malloc(size);
if (p) {
size_t used = my_malloc_usable_size(p);
__sync_add_and_fetch(&status.malloc_count, 1);
__sync_add_and_fetch(&status.requested,size);
__sync_add_and_fetch(&status.used, used);
set_max(status.used, status.freed);
} else {
__sync_add_and_fetch(&status.malloc_fail, 1);
}
return p;
}
void *
toku_calloc(size_t nmemb, size_t size) {
size_t newsize = nmemb * size;
void *p = toku_malloc(newsize);
if (p) memset(p, 0, newsize);
return p;
}
void *
toku_realloc(void *p, size_t size) {
size_t used_orig = p ? my_malloc_usable_size(p) : 0;
void *q = t_realloc ? t_realloc(p, size) : os_realloc(p, size);
if (q) {
size_t used = my_malloc_usable_size(q);
__sync_add_and_fetch(&status.realloc_count, 1);
__sync_add_and_fetch(&status.requested, size);
__sync_add_and_fetch(&status.used, used);
__sync_add_and_fetch(&status.freed, used_orig);
set_max(status.used, status.freed);
} else {
__sync_add_and_fetch(&status.realloc_fail, 1);
}
return q;
}
void *
toku_memdup(const void *v, size_t len) {
void *p = toku_malloc(len);
if (p) memcpy(p, v,len);
return p;
}
char *
toku_strdup(const char *s) {
return toku_memdup(s, strlen(s)+1);
}
void
toku_free(void *p) {
if (p) {
size_t used = my_malloc_usable_size(p);
__sync_add_and_fetch(&status.free_count, 1);
__sync_add_and_fetch(&status.freed, used);
if (t_free)
t_free(p);
else
os_free(p);
}
}
void
toku_free_n(void* p, size_t size __attribute__((unused))) {
toku_free(p);
}
void *
toku_xmalloc(size_t size) {
void *p = t_xmalloc ? t_xmalloc(size) : os_malloc(size);
if (p == NULL) // avoid function call in common case
resource_assert(p);
size_t used = my_malloc_usable_size(p);
__sync_add_and_fetch(&status.malloc_count, 1);
__sync_add_and_fetch(&status.requested, size);
__sync_add_and_fetch(&status.used, used);
set_max(status.used, status.freed);
return p;
}
void *
toku_xcalloc(size_t nmemb, size_t size) {
size_t newsize = nmemb * size;
void *vp = toku_xmalloc(newsize);
if (vp) memset(vp, 0, newsize);
return vp;
}
void *
toku_xrealloc(void *v, size_t size) {
size_t used_orig = v ? my_malloc_usable_size(v) : 0;
void *p = t_xrealloc ? t_xrealloc(v, size) : os_realloc(v, size);
if (p == 0) // avoid function call in common case
resource_assert(p);
size_t used = my_malloc_usable_size(p);
__sync_add_and_fetch(&status.realloc_count, 1);
__sync_add_and_fetch(&status.requested, size);
__sync_add_and_fetch(&status.used, used);
__sync_add_and_fetch(&status.freed, used_orig);
set_max(status.used, status.freed);
return p;
}
size_t
toku_malloc_usable_size(void *p) {
return p == NULL ? 0 : malloc_usable_size(p);
}
void *
toku_xmemdup (const void *v, size_t len) {
void *p = toku_xmalloc(len);
memcpy(p, v, len);
return p;
}
char *
toku_xstrdup (const char *s) {
return toku_xmemdup(s, strlen(s)+1);
}
void
toku_set_func_malloc(malloc_fun_t f) {
t_malloc = f;
t_xmalloc = f;
}
void
toku_set_func_xmalloc_only(malloc_fun_t f) {
t_xmalloc = f;
}
void
toku_set_func_malloc_only(malloc_fun_t f) {
t_malloc = f;
}
void
toku_set_func_realloc(realloc_fun_t f) {
t_realloc = f;
t_xrealloc = f;
}
void
toku_set_func_xrealloc_only(realloc_fun_t f) {
t_xrealloc = f;
}
void
toku_set_func_realloc_only(realloc_fun_t f) {
t_realloc = f;
}
void
toku_set_func_free(free_fun_t f) {
t_free = f;
}
#include <valgrind/drd.h>
void __attribute__((constructor)) toku_memory_drd_ignore(void);
void
toku_memory_drd_ignore(void) {
DRD_IGNORE_VAR(status);
}
...@@ -16,7 +16,7 @@ ifneq ($(GCOV),) ...@@ -16,7 +16,7 @@ ifneq ($(GCOV),)
CFLAGS += -fprofile-arcs -ftest-coverage -DGCOV CFLAGS += -fprofile-arcs -ftest-coverage -DGCOV
endif endif
LDFLAGS = -L../../lib -Wl,-rpath,../../lib LDFLAGS = -L../../lib -Wl,-rpath,../../lib
LDLIBS = -ltokuportability -lpthread LDLIBS = -ltokuportability -lpthread -ldl
SRCS=$(sort $(filter-out dir.%.c,$(wildcard *.c))) SRCS=$(sort $(filter-out dir.%.c,$(wildcard *.c)))
TARGETS = $(patsubst %.c,%,$(SRCS)) TARGETS = $(patsubst %.c,%,$(SRCS))
RUNTARGETS = $(patsubst %,%.tdbrun,$(TARGETS)) RUNTARGETS = $(patsubst %,%.tdbrun,$(TARGETS))
...@@ -53,7 +53,7 @@ test-pwrite4g.tdbrun: TEST_EXTRA_ARGS=. ...@@ -53,7 +53,7 @@ test-pwrite4g.tdbrun: TEST_EXTRA_ARGS=.
ifeq ($(VGRIND),) ifeq ($(VGRIND),)
./$< $(TEST_EXTRA_ARGS) $(SUMMARIZE_CMD) ./$< $(TEST_EXTRA_ARGS) $(SUMMARIZE_CMD)
else else
$(VGRIND) --error-exitcode=1 --quiet --leak-check=full --show-reachable=yes --log-file=$<.check.valgrind ./$< $(TEST_EXTRA_ARGS) >$<.check.output 2>&1; \ $(VGRIND) --error-exitcode=1 --suppressions=../../newbrt/valgrind.suppressions --quiet --leak-check=full --show-reachable=yes --log-file=$<.check.valgrind ./$< $(TEST_EXTRA_ARGS) >$<.check.output 2>&1; \
if [ $$? = 0 ] ; then \ if [ $$? = 0 ] ; then \
lines=`cat $<.check.valgrind | wc -l`; \ lines=`cat $<.check.valgrind | wc -l`; \
if [ $$lines -ne 0 ] ; then cat $<.check.valgrind; test 0 = 1; fi \ if [ $$lines -ne 0 ] ; then cat $<.check.valgrind; test 0 = 1; fi \
...@@ -68,7 +68,7 @@ try-leak-lost.tdbrun: try-leak-lost ...@@ -68,7 +68,7 @@ try-leak-lost.tdbrun: try-leak-lost
ifeq ($(VGRIND),) ifeq ($(VGRIND),)
./$< $(SUMMARIZE_SHOULD_FAIL) ./$< $(SUMMARIZE_SHOULD_FAIL)
else else
$(VGRIND) --error-exitcode=1 --quiet --leak-check=full --show-reachable=yes --log-file=$<.check.valgrind ./$< >$<.check.output 2>&1; \ $(VGRIND) --error-exitcode=1 --suppressions=../../newbrt/valgrind.suppressions --quiet --leak-check=full --show-reachable=yes --log-file=$<.check.valgrind ./$< >$<.check.output 2>&1; \
if [ $$? = 0 ] ; then \ if [ $$? = 0 ] ; then \
lines=`cat $<.check.valgrind | wc -l`; \ lines=`cat $<.check.valgrind | wc -l`; \
if [ $$lines -ne 0 ] ; then cat $<.check.valgrind; test 0 = 1; fi \ if [ $$lines -ne 0 ] ; then cat $<.check.valgrind; test 0 = 1; fi \
...@@ -83,7 +83,7 @@ try-leak-reachable.tdbrun: try-leak-reachable ...@@ -83,7 +83,7 @@ try-leak-reachable.tdbrun: try-leak-reachable
ifeq ($(VGRIND),) ifeq ($(VGRIND),)
./$< $(SUMMARIZE_SHOULD_FAIL) ./$< $(SUMMARIZE_SHOULD_FAIL)
else else
$(VGRIND) --error-exitcode=1 --quiet --leak-check=full --show-reachable=yes --log-file=$<.check.valgrind ./$< >$<.check.output 2>&1; \ $(VGRIND) --error-exitcode=1 --suppressions=../../newbrt/valgrind.suppressions --quiet --leak-check=full --show-reachable=yes --log-file=$<.check.valgrind ./$< >$<.check.output 2>&1; \
if [ $$? = 0 ] ; then \ if [ $$? = 0 ] ; then \
lines=`cat $<.check.valgrind | wc -l`; \ lines=`cat $<.check.valgrind | wc -l`; \
if [ $$lines -ne 0 ] ; then cat $<.check.valgrind; test 0 = 1; fi \ if [ $$lines -ne 0 ] ; then cat $<.check.valgrind; test 0 = 1; fi \
...@@ -98,7 +98,7 @@ try-uninit.tdbrun: try-uninit ...@@ -98,7 +98,7 @@ try-uninit.tdbrun: try-uninit
ifeq ($(VGRIND),) ifeq ($(VGRIND),)
./$< $(SUMMARIZE_SHOULD_FAIL) ./$< $(SUMMARIZE_SHOULD_FAIL)
else else
$(VGRIND) --error-exitcode=1 --quiet --leak-check=full --show-reachable=yes --log-file=$<.check.valgrind ./$< >$<.check.output 2>&1; \ $(VGRIND) --error-exitcode=1 --suppressions=../../newbrt/valgrind.suppressions --quiet --leak-check=full --show-reachable=yes --log-file=$<.check.valgrind ./$< >$<.check.output 2>&1; \
if [ $$? = 0 ] ; then \ if [ $$? = 0 ] ; then \
lines=`cat $<.check.valgrind | wc -l`; \ lines=`cat $<.check.valgrind | wc -l`; \
if [ $$lines -ne 0 ] ; then cat $<.check.valgrind; test 0 = 1; fi \ if [ $$lines -ne 0 ] ; then cat $<.check.valgrind; test 0 = 1; fi \
...@@ -120,6 +120,9 @@ clean: ...@@ -120,6 +120,9 @@ clean:
try-assert0.tdbrun: try-assert0.tdbrun:
check_memory_status: test-memory-status
./test-memory-status
LD_PRELOAD=../../../../mysql/jemalloc-2.2.5/lib/libjemalloc.so ./test-memory-status
foo: foo:
echo $(TARGETS) echo $(TARGETS)
#include <stdio.h>
#include "memory.h"
int main(void) {
MEMORY_STATUS_S s;
toku_memory_get_status(&s);
printf("tokudb mallocator: %s\n", s.mallocator_version);
return 0;
}
...@@ -115,3 +115,23 @@ ...@@ -115,3 +115,23 @@
Memcheck:Value8 Memcheck:Value8
fun:qlz_compress fun:qlz_compress
} }
{
dlsym_on_centos
Memcheck:Leak
fun:calloc
fun:_dlerror_run
fun:dlsym
}
{
dlsym_on_centos
Memcheck:Leak
fun:malloc
fun:_dl_signal_error
fun:_dl_signal_cerror
fun:_dl_lookup_symbol_x
fun:do_sym
fun:dlsym_doit
fun:_dl_catch_error
fun:_dlerror_run
fun:dlsym
}
...@@ -2181,6 +2181,7 @@ env_get_engine_status(DB_ENV * env, ENGINE_STATUS * engstat, char * env_panic_st ...@@ -2181,6 +2181,7 @@ env_get_engine_status(DB_ENV * env, ENGINE_STATUS * engstat, char * env_panic_st
engstat->mem_used = memory_status.used; engstat->mem_used = memory_status.used;
engstat->mem_freed = memory_status.freed; engstat->mem_freed = memory_status.freed;
engstat->max_mem_in_use = memory_status.max_in_use; engstat->max_mem_in_use = memory_status.max_in_use;
engstat->mallocator_version = memory_status.mallocator_version;
} }
} }
return r; return r;
...@@ -2418,6 +2419,7 @@ env_get_engine_status_text(DB_ENV * env, char * buff, int bufsiz) { ...@@ -2418,6 +2419,7 @@ env_get_engine_status_text(DB_ENV * env, char * buff, int bufsiz) {
n += snprintf(buff + n, bufsiz - n, "mem_used %"PRIu64"\n", engstat.mem_used); n += snprintf(buff + n, bufsiz - n, "mem_used %"PRIu64"\n", engstat.mem_used);
n += snprintf(buff + n, bufsiz - n, "mem_freed %"PRIu64"\n", engstat.mem_freed); n += snprintf(buff + n, bufsiz - n, "mem_freed %"PRIu64"\n", engstat.mem_freed);
n += snprintf(buff + n, bufsiz - n, "max_mem_in_use %"PRIu64"\n", engstat.max_mem_in_use); n += snprintf(buff + n, bufsiz - n, "max_mem_in_use %"PRIu64"\n", engstat.max_mem_in_use);
n += snprintf(buff + n, bufsiz - n, "mallocator_version %s\n", engstat.mallocator_version);
} }
if (n > bufsiz) { if (n > bufsiz) {
char * errmsg = "BUFFER TOO SMALL\n"; char * errmsg = "BUFFER TOO SMALL\n";
......
...@@ -167,7 +167,7 @@ LIBPORTABILITY_SO=$(TOKUROOT)lib/libtokuportability.$(SOEXT) ...@@ -167,7 +167,7 @@ LIBPORTABILITY_SO=$(TOKUROOT)lib/libtokuportability.$(SOEXT)
LIBPORTABILITY_A=$(TOKUROOT)lib/libtokuportability.$(AEXT) LIBPORTABILITY_A=$(TOKUROOT)lib/libtokuportability.$(AEXT)
PORTABILITY_HEADERS= $(TOKUROOT)$(SYSTEM) PORTABILITY_HEADERS= $(TOKUROOT)$(SYSTEM)
ALWAYS_LINK= -lz -lpthread ALWAYS_LINK= -lz -lpthread -ldl
ifeq ($(CC),icc) ifeq ($(CC),icc)
# needed for things like intel_fast_memset(), intel_sse2_strlen() # needed for things like intel_fast_memset(), intel_sse2_strlen()
ifeq ($(DEBUG),0) ifeq ($(DEBUG),0)
......
...@@ -95,15 +95,16 @@ void toku_set_func_realloc_only(realloc_fun_t f); ...@@ -95,15 +95,16 @@ void toku_set_func_realloc_only(realloc_fun_t f);
void toku_set_func_free(free_fun_t f); void toku_set_func_free(free_fun_t f);
typedef struct memory_status { typedef struct memory_status {
uint64_t malloc_count; // number of malloc operations uint64_t malloc_count; // number of malloc operations
uint64_t free_count; // number of free operations uint64_t free_count; // number of free operations
uint64_t realloc_count; // number of realloc operations uint64_t realloc_count; // number of realloc operations
uint64_t malloc_fail; // number of malloc operations that failed uint64_t malloc_fail; // number of malloc operations that failed
uint64_t realloc_fail; // number of realloc operations that failed uint64_t realloc_fail; // number of realloc operations that failed
uint64_t requested; // number of bytes requested uint64_t requested; // number of bytes requested
uint64_t used; // number of bytes used (requested + overhead), obtained from malloc_usable_size() uint64_t used; // number of bytes used (requested + overhead), obtained from malloc_usable_size()
uint64_t freed; // number of bytes freed; uint64_t freed; // number of bytes freed;
uint64_t max_in_use; // maximum memory footprint (used - freed), approximate (not worth threadsafety overhead for exact) uint64_t max_in_use; // maximum memory footprint (used - freed), approximate (not worth threadsafety overhead for exact)
const char *mallocator_version;
} MEMORY_STATUS_S, *MEMORY_STATUS; } MEMORY_STATUS_S, *MEMORY_STATUS;
void toku_memory_get_status(MEMORY_STATUS s); void toku_memory_get_status(MEMORY_STATUS s);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment