Commit 8ddd6c41 authored by Russ Cox's avatar Russ Cox

runtime: clock garbage collection on bytes allocated, not pages in use

This keeps fragmentation from delaying
garbage collections (and causing more fragmentation).

Cuts fresh godoc (with indexes) from 261M to 166M (120M live).
Cuts toy wc program from 50M to 8M.

Fixes #647.

R=r, cw
CC=golang-dev
https://golang.org/cl/257041
parent 18187e7d
...@@ -78,6 +78,7 @@ type MemStatsType struct { ...@@ -78,6 +78,7 @@ type MemStatsType struct {
Stacks uint64 Stacks uint64
InusePages uint64 InusePages uint64
NextGC uint64 NextGC uint64
HeapAlloc uint64
Lookups uint64 Lookups uint64
Mallocs uint64 Mallocs uint64
PauseNs uint64 PauseNs uint64
......
...@@ -61,7 +61,7 @@ mallocgc(uintptr size, uint32 refflag, int32 dogc, int32 zeroed) ...@@ -61,7 +61,7 @@ mallocgc(uintptr size, uint32 refflag, int32 dogc, int32 zeroed)
npages = size >> PageShift; npages = size >> PageShift;
if((size & PageMask) != 0) if((size & PageMask) != 0)
npages++; npages++;
s = MHeap_Alloc(&mheap, npages, 0); s = MHeap_Alloc(&mheap, npages, 0, 1);
if(s == nil) if(s == nil)
throw("out of memory"); throw("out of memory");
mstats.alloc += npages<<PageShift; mstats.alloc += npages<<PageShift;
...@@ -74,7 +74,7 @@ mallocgc(uintptr size, uint32 refflag, int32 dogc, int32 zeroed) ...@@ -74,7 +74,7 @@ mallocgc(uintptr size, uint32 refflag, int32 dogc, int32 zeroed)
m->mallocing = 0; m->mallocing = 0;
if(dogc && mstats.inuse_pages > mstats.next_gc) if(dogc && mstats.heap_alloc >= mstats.next_gc)
gc(0); gc(0);
return v; return v;
} }
...@@ -113,7 +113,7 @@ free(void *v) ...@@ -113,7 +113,7 @@ free(void *v)
// Large object. // Large object.
mstats.alloc -= s->npages<<PageShift; mstats.alloc -= s->npages<<PageShift;
runtime_memclr(v, s->npages<<PageShift); runtime_memclr(v, s->npages<<PageShift);
MHeap_Free(&mheap, s); MHeap_Free(&mheap, s, 1);
} else { } else {
// Small object. // Small object.
c = m->mcache; c = m->mcache;
......
...@@ -168,12 +168,13 @@ void FixAlloc_Free(FixAlloc *f, void *p); ...@@ -168,12 +168,13 @@ void FixAlloc_Free(FixAlloc *f, void *p);
// Shared with Go: if you edit this structure, also edit extern.go. // Shared with Go: if you edit this structure, also edit extern.go.
struct MStats struct MStats
{ {
uint64 alloc; uint64 alloc; // unprotected (approximate)
uint64 total_alloc; uint64 total_alloc; // unprotected (approximate)
uint64 sys; uint64 sys;
uint64 stacks; uint64 stacks;
uint64 inuse_pages; // protected by mheap.Lock uint64 inuse_pages; // protected by mheap.Lock
uint64 next_gc; // protected by mheap.Lock uint64 next_gc; // protected by mheap.Lock
uint64 heap_alloc; // protected by mheap.Lock
uint64 nlookup; // unprotected (approximate) uint64 nlookup; // unprotected (approximate)
uint64 nmalloc; // unprotected (approximate) uint64 nmalloc; // unprotected (approximate)
uint64 pause_ns; uint64 pause_ns;
...@@ -225,11 +226,12 @@ struct MCache ...@@ -225,11 +226,12 @@ struct MCache
{ {
MCacheList list[NumSizeClasses]; MCacheList list[NumSizeClasses];
uint64 size; uint64 size;
int64 local_alloc; // bytes allocated (or freed) since last lock of heap
}; };
void* MCache_Alloc(MCache *c, int32 sizeclass, uintptr size, int32 zeroed); void* MCache_Alloc(MCache *c, int32 sizeclass, uintptr size, int32 zeroed);
void MCache_Free(MCache *c, void *p, int32 sizeclass, uintptr size); void MCache_Free(MCache *c, void *p, int32 sizeclass, uintptr size);
void MCache_ReleaseAll(MCache *c);
// An MSpan is a run of pages. // An MSpan is a run of pages.
enum enum
...@@ -313,8 +315,8 @@ struct MHeap ...@@ -313,8 +315,8 @@ struct MHeap
extern MHeap mheap; extern MHeap mheap;
void MHeap_Init(MHeap *h, void *(*allocator)(uintptr)); void MHeap_Init(MHeap *h, void *(*allocator)(uintptr));
MSpan* MHeap_Alloc(MHeap *h, uintptr npage, int32 sizeclass); MSpan* MHeap_Alloc(MHeap *h, uintptr npage, int32 sizeclass, int32 acct);
void MHeap_Free(MHeap *h, MSpan *s); void MHeap_Free(MHeap *h, MSpan *s, int32 acct);
MSpan* MHeap_Lookup(MHeap *h, PageID p); MSpan* MHeap_Lookup(MHeap *h, PageID p);
MSpan* MHeap_LookupMaybe(MHeap *h, PageID p); MSpan* MHeap_LookupMaybe(MHeap *h, PageID p);
void MGetSizeClassInfo(int32 sizeclass, int32 *size, int32 *npages, int32 *nobj); void MGetSizeClassInfo(int32 sizeclass, int32 *size, int32 *npages, int32 *nobj);
......
...@@ -46,6 +46,7 @@ MCache_Alloc(MCache *c, int32 sizeclass, uintptr size, int32 zeroed) ...@@ -46,6 +46,7 @@ MCache_Alloc(MCache *c, int32 sizeclass, uintptr size, int32 zeroed)
v->next = nil; v->next = nil;
} }
} }
c->local_alloc += size;
return v; return v;
} }
...@@ -86,6 +87,7 @@ MCache_Free(MCache *c, void *v, int32 sizeclass, uintptr size) ...@@ -86,6 +87,7 @@ MCache_Free(MCache *c, void *v, int32 sizeclass, uintptr size)
l->list = p; l->list = p;
l->nlist++; l->nlist++;
c->size += size; c->size += size;
c->local_alloc -= size;
if(l->nlist >= MaxMCacheListLen) { if(l->nlist >= MaxMCacheListLen) {
// Release a chunk back. // Release a chunk back.
...@@ -113,3 +115,20 @@ MCache_Free(MCache *c, void *v, int32 sizeclass, uintptr size) ...@@ -113,3 +115,20 @@ MCache_Free(MCache *c, void *v, int32 sizeclass, uintptr size)
} }
} }
void
MCache_ReleaseAll(MCache *c)
{
int32 i;
MCacheList *l;
lock(&mheap);
mstats.heap_alloc += c->local_alloc;
c->local_alloc = 0;
unlock(&mheap);
for(i=0; i<NumSizeClasses; i++) {
l = &c->list[i];
ReleaseN(c, l, l->nlist, i);
l->nlistmin = 0;
}
}
...@@ -152,7 +152,7 @@ MCentral_Free(MCentral *c, void *v) ...@@ -152,7 +152,7 @@ MCentral_Free(MCentral *c, void *v)
s->freelist = nil; s->freelist = nil;
c->nfree -= (s->npages << PageShift) / size; c->nfree -= (s->npages << PageShift) / size;
unlock(c); unlock(c);
MHeap_Free(&mheap, s); MHeap_Free(&mheap, s, 0);
lock(c); lock(c);
} }
} }
...@@ -182,7 +182,7 @@ MCentral_Grow(MCentral *c) ...@@ -182,7 +182,7 @@ MCentral_Grow(MCentral *c)
unlock(c); unlock(c);
MGetSizeClassInfo(c->sizeclass, &size, &npages, &n); MGetSizeClassInfo(c->sizeclass, &size, &npages, &n);
s = MHeap_Alloc(&mheap, npages, c->sizeclass); s = MHeap_Alloc(&mheap, npages, c->sizeclass, 0);
if(s == nil) { if(s == nil) {
// TODO(rsc): Log out of memory // TODO(rsc): Log out of memory
lock(c); lock(c);
......
...@@ -194,7 +194,7 @@ sweepspan1(MSpan *s) ...@@ -194,7 +194,7 @@ sweepspan1(MSpan *s)
mstats.alloc -= s->npages<<PageShift; mstats.alloc -= s->npages<<PageShift;
runtime_memclr(p, s->npages<<PageShift); runtime_memclr(p, s->npages<<PageShift);
s->gcref0 = RefFree; s->gcref0 = RefFree;
MHeap_Free(&mheap, s); MHeap_Free(&mheap, s, 1);
break; break;
case RefFinalize: case RefFinalize:
if(pfinq < efinq) { if(pfinq < efinq) {
...@@ -283,6 +283,15 @@ static uint32 gcsema = 1; ...@@ -283,6 +283,15 @@ static uint32 gcsema = 1;
// extra memory used). // extra memory used).
static int32 gcpercent = -2; static int32 gcpercent = -2;
static void
stealcache(void)
{
M *m;
for(m=allm; m; m=m->alllink)
MCache_ReleaseAll(m->mcache);
}
void void
gc(int32 force) gc(int32 force)
{ {
...@@ -313,17 +322,17 @@ gc(int32 force) ...@@ -313,17 +322,17 @@ gc(int32 force)
if(gcpercent < 0) if(gcpercent < 0)
return; return;
//printf("gc...\n");
semacquire(&gcsema); semacquire(&gcsema);
t0 = nanotime(); t0 = nanotime();
m->gcing = 1; m->gcing = 1;
stoptheworld(); stoptheworld();
if(mheap.Lock.key != 0) if(mheap.Lock.key != 0)
throw("mheap locked during gc"); throw("mheap locked during gc");
if(force || mstats.inuse_pages >= mstats.next_gc) { if(force || mstats.heap_alloc >= mstats.next_gc) {
mark(); mark();
sweep(); sweep();
mstats.next_gc = mstats.inuse_pages+mstats.inuse_pages*gcpercent/100; stealcache();
mstats.next_gc = mstats.heap_alloc+mstats.heap_alloc*gcpercent/100;
} }
m->gcing = 0; m->gcing = 0;
......
...@@ -53,14 +53,19 @@ MHeap_Init(MHeap *h, void *(*alloc)(uintptr)) ...@@ -53,14 +53,19 @@ MHeap_Init(MHeap *h, void *(*alloc)(uintptr))
// Allocate a new span of npage pages from the heap // Allocate a new span of npage pages from the heap
// and record its size class in the HeapMap and HeapMapCache. // and record its size class in the HeapMap and HeapMapCache.
MSpan* MSpan*
MHeap_Alloc(MHeap *h, uintptr npage, int32 sizeclass) MHeap_Alloc(MHeap *h, uintptr npage, int32 sizeclass, int32 acct)
{ {
MSpan *s; MSpan *s;
lock(h); lock(h);
mstats.heap_alloc += m->mcache->local_alloc;
m->mcache->local_alloc = 0;
s = MHeap_AllocLocked(h, npage, sizeclass); s = MHeap_AllocLocked(h, npage, sizeclass);
if(s != nil) if(s != nil) {
mstats.inuse_pages += npage; mstats.inuse_pages += npage;
if(acct)
mstats.heap_alloc += npage<<PageShift;
}
unlock(h); unlock(h);
return s; return s;
} }
...@@ -225,10 +230,14 @@ MHeap_LookupMaybe(MHeap *h, PageID p) ...@@ -225,10 +230,14 @@ MHeap_LookupMaybe(MHeap *h, PageID p)
// Free the span back into the heap. // Free the span back into the heap.
void void
MHeap_Free(MHeap *h, MSpan *s) MHeap_Free(MHeap *h, MSpan *s, int32 acct)
{ {
lock(h); lock(h);
mstats.heap_alloc += m->mcache->local_alloc;
m->mcache->local_alloc = 0;
mstats.inuse_pages -= s->npages; mstats.inuse_pages -= s->npages;
if(acct)
mstats.heap_alloc -= s->npages<<PageShift;
MHeap_FreeLocked(h, s); MHeap_FreeLocked(h, s);
unlock(h); unlock(h);
} }
......
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