memory-manager.hpp /usr/include/gecode/kernel.hh Gecode::MemoryChunk Gecode::HeapChunk Gecode::SharedMemory Gecode::FreeList Gecode::MemoryManager Gecode /*-*-mode:C++;c-basic-offset:2;indent-tabs-mode:nil-*-*/ /* *Mainauthors: *ChristianSchulte<schulte@gecode.org> * *Contributingauthors: *GuidoTack<tack@gecode.org> * *Bugfixesprovidedby: *ZandraNorman * *Copyright: *ChristianSchulte,2002 *GuidoTack,2004 * *Lastmodified: *$Date:2013-07-1112:30:18+0200(Thu,11Jul2013)$by$Author:schulte$ *$Revision:13840$ * *ThisfileispartofGecode,thegenericconstraint *developmentenvironment: *http://www.gecode.org * *Permissionisherebygranted,freeofcharge,toanypersonobtaining *acopyofthissoftwareandassociateddocumentationfiles(the *"Software"),todealintheSoftwarewithoutrestriction,including *withoutlimitationtherightstouse,copy,modify,merge,publish, *distribute,sublicense,and/orsellcopiesoftheSoftware,andto *permitpersonstowhomtheSoftwareisfurnishedtodoso,subjectto *thefollowingconditions: * *Theabovecopyrightnoticeandthispermissionnoticeshallbe *includedinallcopiesorsubstantialportionsoftheSoftware. * *THESOFTWAREISPROVIDED"ASIS",WITHOUTWARRANTYOFANYKIND, *EXPRESSORIMPLIED,INCLUDINGBUTNOTLIMITEDTOTHEWARRANTIESOF *MERCHANTABILITY,FITNESSFORAPARTICULARPURPOSEAND *NONINFRINGEMENT.INNOEVENTSHALLTHEAUTHORSORCOPYRIGHTHOLDERSBE *LIABLEFORANYCLAIM,DAMAGESOROTHERLIABILITY,WHETHERINANACTION *OFCONTRACT,TORTOROTHERWISE,ARISINGFROM,OUTOFORINCONNECTION *WITHTHESOFTWAREORTHEUSEOROTHERDEALINGSINTHESOFTWARE. * */ namespaceGecode{ classMemoryChunk{ public: MemoryChunk*next; size_tsize; }; classHeapChunk:publicMemoryChunk{ public: doublearea[1]; }; classRegion; classSharedMemory{ friendclassRegion; private: unsignedintuse_cnt; struct{ size_tfree; doublearea[MemoryConfig::region_area_size/sizeof(double)]; }region; struct{ unsignedintn_hc; HeapChunk*hc; }heap; public: SharedMemory(void); voidflush(void); ~SharedMemory(void); //@ boolregion_alloc(size_ts,void*&p); //@ HeapChunk*heap_alloc(size_ts,size_tl); voidheap_free(HeapChunk*hc); SharedMemory*copy(boolshare); boolrelease(void); staticvoid*operatornew(size_ts); staticvoidoperatordelete(void*p); }; classFreeList{ protected: FreeList*_next; public: FreeList(void); FreeList(FreeList*n); FreeList*next(void)const; FreeList**nextRef(void); voidnext(FreeList*n); }; classMemoryManager{ public: MemoryManager(SharedMemory*sm); MemoryManager(SharedMemory*sm,MemoryManager&mm,size_ts_sub); voidrelease(SharedMemory*sm); private: size_tcur_hcsz; HeapChunk*cur_hc; size_trequested; char*start; size_tlsz; GECODE_KERNEL_EXPORT voidalloc_refill(SharedMemory*sm,size_ts); voidalloc_fill(SharedMemory*sm,size_ts,boolfirst); public: void*alloc(SharedMemory*sm,size_ts); void*subscriptions(void)const; private: FreeList*fl[MemoryConfig::fl_size_max-MemoryConfig::fl_size_min+1]; template<size_t>voidfl_refill(SharedMemory*sm); staticsize_tsz2i(size_t); staticsize_ti2sz(size_t); public: template<size_ts> void*fl_alloc(SharedMemory*sm); template<size_t>voidfl_dispose(FreeList*f,FreeList*l); private: MemoryChunk*slack; public: voidreuse(void*p,size_ts); }; /* *Sharedmemoryarea * */ forceinlinevoid* SharedMemory::operatornew(size_ts){ returnGecode::heap.ralloc(s); } forceinlinevoid SharedMemory::operatordelete(void*p){ Gecode::heap.rfree(p); } forceinline SharedMemory::SharedMemory(void) :use_cnt(1){ region.free=MemoryConfig::region_area_size; heap.n_hc=0; heap.hc=NULL; } forceinlinevoid SharedMemory::flush(void){ heap.n_hc=0; while(heap.hc!=NULL){ HeapChunk*hc=heap.hc; heap.hc=static_cast<HeapChunk*>(hc->next); Gecode::heap.rfree(hc); } } forceinline SharedMemory::~SharedMemory(void){ flush(); } forceinlineSharedMemory* SharedMemory::copy(boolshare){ if(share){ use_cnt++; returnthis; }else{ returnnewSharedMemory(); } } forceinlinebool SharedMemory::release(void){ return--use_cnt==0; } forceinlinebool SharedMemory::region_alloc(size_ts,void*&p){ MemoryConfig::align(s); if(s>region.free) returnfalse; region.free-=s; p=ptr_cast<char*>(&region.area[0])+region.free; returntrue; } forceinlineHeapChunk* SharedMemory::heap_alloc(size_ts,size_tl){ while((heap.hc!=NULL)&&(heap.hc->size<l)){ heap.n_hc--; HeapChunk*hc=heap.hc; heap.hc=static_cast<HeapChunk*>(hc->next); Gecode::heap.rfree(hc); } if(heap.hc==NULL){ assert(heap.n_hc==0); HeapChunk*hc=static_cast<HeapChunk*>(Gecode::heap.ralloc(s)); hc->size=s; returnhc; }else{ heap.n_hc--; HeapChunk*hc=heap.hc; heap.hc=static_cast<HeapChunk*>(hc->next); returnhc; } } forceinlinevoid SharedMemory::heap_free(HeapChunk*hc){ if(heap.n_hc==MemoryConfig::n_hc_cache){ Gecode::heap.rfree(hc); }else{ heap.n_hc++; hc->next=heap.hc;heap.hc=hc; } } /* *Freelists * */ forceinline FreeList::FreeList(void){} forceinline FreeList::FreeList(FreeList*n) :_next(n){} forceinlineFreeList* FreeList::next(void)const{ return_next; } forceinlineFreeList** FreeList::nextRef(void){ return&_next; } forceinlinevoid FreeList::next(FreeList*n){ _next=n; } forceinlinesize_t MemoryManager::sz2i(size_ts){ assert(s>=(MemoryConfig::fl_size_min<<MemoryConfig::fl_unit_size)); assert(s<=(MemoryConfig::fl_size_max<<MemoryConfig::fl_unit_size)); return(s>>MemoryConfig::fl_unit_size)-MemoryConfig::fl_size_min; } forceinlinesize_t MemoryManager::i2sz(size_ti){ return(i+MemoryConfig::fl_size_min)<<MemoryConfig::fl_unit_size; } /* *Theactivememorymanager * */ forceinlinevoid* MemoryManager::alloc(SharedMemory*sm,size_tsz){ assert(sz>0); //Performalignment MemoryConfig::align(sz); //Checkwhethersufficientmemoryleft if(sz>lsz) alloc_refill(sm,sz); lsz-=sz; returnstart+lsz; } forceinlinevoid* MemoryManager::subscriptions(void)const{ return&cur_hc->area[0]; } forceinlinevoid MemoryManager::alloc_fill(SharedMemory*sm,size_tsz,boolfirst){ //Adjustcurrentheapchunksize if(((requested>MemoryConfig::hcsz_inc_ratio*cur_hcsz)|| (sz>cur_hcsz))&& (cur_hcsz<MemoryConfig::hcsz_max)&& !first){ cur_hcsz<<=1; } //Incrementthesizethatitcatersfortheinitialoverhead size_toverhead=sizeof(HeapChunk)-sizeof(double); sz+=overhead; //Roundsizetonextmultipleofcurrentheapchunksize size_tallocate=((sz>cur_hcsz)? (((size_t)(sz/cur_hcsz))+1)*cur_hcsz:cur_hcsz); //Requestachunkofpreferablysizeallocate,butatleastsizesz HeapChunk*hc=sm->heap_alloc(allocate,sz); start=ptr_cast<char*>(&hc->area[0]); lsz=hc->size-overhead; //Linkheapchunk,wherethefirstheapchunkiskeptinplace if(first){ requested=hc->size; hc->next=NULL;cur_hc=hc; }else{ requested+=hc->size; hc->next=cur_hc->next;cur_hc->next=hc; } #ifdefGECODE_MEMORY_CHECK for(char*c=start;c<(start+lsz);c++) *c=0; #endif } forceinline MemoryManager::MemoryManager(SharedMemory*sm) :cur_hcsz(MemoryConfig::hcsz_min),requested(0),slack(NULL){ alloc_fill(sm,cur_hcsz,true); for(size_ti=MemoryConfig::fl_size_max-MemoryConfig::fl_size_min+1; i--;) fl[i]=NULL; } forceinline MemoryManager::MemoryManager(SharedMemory*sm,MemoryManager&mm, size_ts_sub) :cur_hcsz(mm.cur_hcsz),requested(0),slack(NULL){ MemoryConfig::align(s_sub); if((mm.requested<MemoryConfig::hcsz_dec_ratio*mm.cur_hcsz)&& (cur_hcsz>MemoryConfig::hcsz_min)&& (s_sub*2<cur_hcsz)) cur_hcsz>>=1; alloc_fill(sm,cur_hcsz+s_sub,true); //Skipthememoryareaatthebeginningforsubscriptions lsz-=s_sub; start+=s_sub; for(size_ti=MemoryConfig::fl_size_max-MemoryConfig::fl_size_min+1; i--;) fl[i]=NULL; } forceinlinevoid MemoryManager::release(SharedMemory*sm){ //Releaseallallocatedheapchunks HeapChunk*hc=cur_hc; do{ HeapChunk*t=hc;hc=static_cast<HeapChunk*>(hc->next); sm->heap_free(t); }while(hc!=NULL); } /* *Slackmemorymanagement * */ forceinlinevoid MemoryManager::reuse(void*p,size_ts){ #ifdefGECODE_MEMORY_CHECK { char*c=static_cast<char*>(p); char*e=c+s; while(c<e){ *c=0;c++; } } #endif if(s<(MemoryConfig::fl_size_min<<MemoryConfig::fl_unit_size)) return; if(s>(MemoryConfig::fl_size_max<<MemoryConfig::fl_unit_size)){ MemoryChunk*rc=static_cast<MemoryChunk*>(p); rc->next=slack; rc->size=s; slack=rc; }else{ size_ti=sz2i(s); FreeList*f=static_cast<FreeList*>(p); f->next(fl[i]);fl[i]=f; } } /* *Freelistmanagement * */ template<size_ts> forceinlinevoid* MemoryManager::fl_alloc(SharedMemory*sm){ size_ti=sz2i(s); FreeList*f=fl[i]; if(f==NULL){ fl_refill<s>(sm);f=fl[i]; } FreeList*n=f->next(); fl[i]=n; returnf; } template<size_ts> forceinlinevoid MemoryManager::fl_dispose(FreeList*f,FreeList*l){ size_ti=sz2i(s); #ifdefGECODE_AUDIT FreeList*cur=f; while(cur!=l){ assert(cur->next()); cur=cur->next(); } #endif l->next(fl[i]);fl[i]=f; } template<size_tsz> void MemoryManager::fl_refill(SharedMemory*sm){ //Trytoacquirememoryfromslack if(slack!=NULL){ MemoryChunk*m=slack; slack=NULL; do{ char*block=ptr_cast<char*>(m); size_ts=m->size; assert(s>=sz); m=m->next; fl[sz2i(sz)]=ptr_cast<FreeList*>(block); while(s>=2*sz){ ptr_cast<FreeList*>(block)->next(ptr_cast<FreeList*>(block+sz)); block+=sz; s-=sz; } ptr_cast<FreeList*>(block)->next(NULL); }while(m!=NULL); }else{ char*block=static_cast<char*>(alloc(sm,MemoryConfig::fl_refill*sz)); fl[sz2i(sz)]=ptr_cast<FreeList*>(block); inti=MemoryConfig::fl_refill-2; do{ ptr_cast<FreeList*>(block+i*sz)->next(ptr_cast<FreeList*>(block+(i+1)*sz)); }while(--i>=0); ptr_cast<FreeList*>(block+ (MemoryConfig::fl_refill-1)*sz)->next (ptr_cast<FreeList*>(NULL)); } } } //STATISTICS:kernel-memory