Skip to content

Commit

Permalink
Add memory debugging
Browse files Browse the repository at this point in the history
This adds a new configure option to GAP which enables an
extremely slow debug mode, which can find problems where
references to memory are incorrectly kept across
garbage collections.
  • Loading branch information
ChrisJefferson authored and fingolfin committed Mar 28, 2018
1 parent 71b6290 commit 562896c
Show file tree
Hide file tree
Showing 8 changed files with 289 additions and 9 deletions.
8 changes: 8 additions & 0 deletions configure.ac
Original file line number Diff line number Diff line change
Expand Up @@ -238,6 +238,14 @@ AC_ARG_ENABLE([debug],
AC_MSG_CHECKING([whether to enable debug mode])
AC_MSG_RESULT([$enable_debug])

AC_ARG_ENABLE([memory-checking],
[AS_HELP_STRING([--enable-memory-checking], [enable memory checking])],
[AC_DEFINE([GAP_MEM_CHECK], [1], [define if building with memory checking])],
[enable_memory_checking=no]
)
AC_MSG_CHECKING([whether to enable memory checking])
AC_MSG_RESULT([$enable_memory_checking])

dnl
dnl User setting: Enable -Werror (off by default)
dnl
Expand Down
3 changes: 3 additions & 0 deletions lib/init.g
Original file line number Diff line number Diff line change
Expand Up @@ -560,6 +560,9 @@ BindGlobal( "ShowKernelInformation", function()
if GAPInfo.KernelInfo.KernelDebug then
Add( config, "KernelDebug" );
fi;
if GAPInfo.KernelInfo.MemCheck then
Add(config, "MemCheck");
fi;
if config <> [] then
print_info( " Configuration: ", config, "\n" );
fi;
Expand Down
3 changes: 2 additions & 1 deletion lib/system.g
Original file line number Diff line number Diff line change
Expand Up @@ -110,7 +110,8 @@ BIND_GLOBAL( "GAPInfo", rec(
help := [ "Run ProfileLineByLine(<filename>) with recordMem := true on GAP start"] ),
rec( long := "cover", default := "", arg := "<file>",
help := [ "Run CoverageLineByLine(<filename>) on GAP start"] ),
],
rec( long := "enableMemCheck", default := false)
],
) );


Expand Down
27 changes: 25 additions & 2 deletions src/gap.c
Original file line number Diff line number Diff line change
Expand Up @@ -1882,6 +1882,18 @@ Obj FuncGASMAN_LIMITS( Obj self )
return list;
}

#ifdef GAP_MEM_CHECK

extern Int EnableMemCheck;

Obj FuncGASMAN_MEM_CHECK(Obj self, Obj newval)
{
EnableMemCheck = INT_INTOBJ(newval);
return 0;
}

#endif

/****************************************************************************
**
*F FuncTotalMemoryAllocated( <self> ) .expert function 'TotalMemoryAllocated'
Expand Down Expand Up @@ -2674,6 +2686,13 @@ Obj FuncKERNEL_INFO(Obj self) {
AssPRec(res, r, False);
#endif

r = RNamName("MemCheck");
#ifdef GAP_MEM_CHECK
AssPRec(res, r, True);
#else
AssPRec(res, r, False);
#endif

MakeImmutable(res);

return res;
Expand Down Expand Up @@ -2769,7 +2788,7 @@ void ThreadedInterpreter(void *funcargs) {
**
*V GVarFuncs . . . . . . . . . . . . . . . . . . list of functions to export
*/
static StructGVarFunc GVarFuncs [] = {
static StructGVarFunc GVarFuncs[] = {

GVAR_FUNC(Runtime, 0, ""),
GVAR_FUNC(RUNTIMES, 0, ""),
Expand All @@ -2790,6 +2809,9 @@ static StructGVarFunc GVarFuncs [] = {
GVAR_FUNC(GASMAN_STATS, 0, ""),
GVAR_FUNC(GASMAN_MESSAGE_STATUS, 0, ""),
GVAR_FUNC(GASMAN_LIMITS, 0, ""),
#ifdef GAP_MEM_CHECK
GVAR_FUNC(GASMAN_MEM_CHECK, 1, "int"),
#endif
GVAR_FUNC(TotalMemoryAllocated, 0, ""),
GVAR_FUNC(SIZE_OBJ, 1, "object"),
GVAR_FUNC(TNUM_OBJ, 1, "object"),
Expand All @@ -2804,7 +2826,8 @@ static StructGVarFunc GVarFuncs [] = {
GVAR_FUNC(QUIT_GAP, -1, "args"),
GVAR_FUNC(FORCE_QUIT_GAP, -1, "args"),
GVAR_FUNC(SHOULD_QUIT_ON_BREAK, 0, ""),
GVAR_FUNC(SHELL, -1, "context, canReturnVoid, canReturnObj, lastDepth, setTime, prompt, promptHook, infile, outfile"),
GVAR_FUNC(SHELL, -1, "context, canReturnVoid, canReturnObj, lastDepth, "
"setTime, prompt, promptHook, infile, outfile"),
GVAR_FUNC(CALL_WITH_CATCH, 2, "func, args"),
GVAR_FUNC(JUMP_TO_CATCH, 1, "payload"),
GVAR_FUNC(KERNEL_INFO, 0, ""),
Expand Down
142 changes: 136 additions & 6 deletions src/gasman.c
Original file line number Diff line number Diff line change
Expand Up @@ -119,6 +119,10 @@

#include <stdio.h>

#ifdef GAP_MEM_CHECK
#include <sys/mman.h>
#endif

/****************************************************************************
**
*F WORDS_BAG( <size> ) . . . . . . . . . . words used by a bag of given size
Expand Down Expand Up @@ -973,6 +977,113 @@ void InitCollectFuncBags (
AfterCollectFuncBags = after_func;
}

/***************************************************************
* GAP_MEM_CHECK
*
* One of the hardest categories of bugs to fix in GAP are where
* a reference to the internals of a GAP object are kept across
* a garbage collection (which moves GAP objects around).
*
* GAP_MEM_CHECK provides a method of detecting such problems, at
* the cost of GREATLY decreased performance (Starting GAP in
* --enableMemCheck mode takes days, rather than seconds).
*
* The fundamental idea behind GAP_MEM_CHECK is, whenever NewBag
* or ResizeBag is called, then the contents of every Bag in
* GAP is moved, and the memory previously being used is marked
* as not readable or writable using 'mprotect'.
*
* Actually copying all GAP's memory space would be extremely
* expensive, so instead we use 'mmap' to set up a set of copies
* of the GAP memory space, which are represented by the same
* underlying physical memory.
*
* The 0th such copy (which we also ensure is the one at the
* lowest memory address) is special -- this is where we
* reference the master pointers (which can't move). We do not
* 'mprotect' any of this memory.
*
* Every time we call 'NewBag' or 'ResizeBag', we change which
* copy of the GASMAN memory space the master pointers point
* to, disabling access to the previous copy, and enabling access
* to the new one.
*
* We never use the master pointers in any copy other than the
* 0th, and we never refer to the Bag area in the 0th copy. However,
* it simplifies things to not try to seperate the master pointer
* and Bag areas, because the master pointer area can grow as GAP
* runs.
*
* Because this code is VERY slow, it can be turned on and off.
* At run time, call GASMAN_MEM_CHECK(1) to enable, and
* GASMAN_MEM_CHECK(0) to disable. Start GAP with --enableMemCheck
* to enable from when GAP starts.
*/

#ifdef GAP_MEM_CHECK

Int EnableMemCheck = 0;

Int enableMemCheck(Char ** argv, void * dummy)
{
SyFputs( "# Warning: --enableMemCheck causes SEVERE slowdowns. Starting GAP may take several days!\n", 3 );
EnableMemCheck = 1;
return 1;
}

static void MoveBagMemory(char * oldbase, char * newbase)
{
Int moveSize = (newbase - oldbase) / sizeof(Bag);
// update the masterpointers
for (Bag * p = MptrBags; p < MptrEndBags; p++) {
if ((Bag)MptrEndBags <= *p)
*p += moveSize;
}

// update 'OldBags', 'YoungBags', 'AllocBags', and 'EndBags'
OldBags += moveSize;
YoungBags += moveSize;
AllocBags += moveSize;
EndBags += moveSize;
}

static void MaybeMoveBags(void)
{
static Int oldBase = 0;

if (!EnableMemCheck)
return;

Int newBase = oldBase + 1;
// Memory buffer 0 is special, as we use that
// copy for the master pointers. Therefore never
// block access to it, and skip it when cycling.
if (newBase >= GetMembufCount())
newBase = 1;

// call the before function (if any)
if (BeforeCollectFuncBags != 0)
(*BeforeCollectFuncBags)();

MoveBagMemory(GetMembuf(oldBase), GetMembuf(newBase));

// Enable access to new memory
mprotect(GetMembuf(newBase), GetMembufSize(), PROT_READ | PROT_WRITE);
// Block access to old memory (except block 0, which will only occur
// on the first call).
if (oldBase != 0) {
mprotect(GetMembuf(oldBase), GetMembufSize(), PROT_NONE);
}

// call the after function (if any)
if (AfterCollectFuncBags != 0)
(*AfterCollectFuncBags)();


oldBase = newBase;
}

#endif

/****************************************************************************
**
Expand Down Expand Up @@ -1032,17 +1143,24 @@ void InitBags (
SyAbortBags("cannot get storage for the initial workspace.");
EndBags = MptrBags + 1024*(initial_size / sizeof(Bag*));

// In GAP_MEM_CHECK we want as few master pointers as possible, as we
// have to loop over them very frequently.
#ifdef GAP_MEM_CHECK
UInt initialBagCount = 100000;
#else
UInt initialBagCount = 1024*initial_size/8/sizeof(Bag*);
#endif
/* 1/8th of the storage goes into the masterpointer area */
FreeMptrBags = (Bag)MptrBags;
for ( p = MptrBags;
p + 2*(SIZE_MPTR_BAGS) <= MptrBags+1024*initial_size/8/sizeof(Bag*);
p + 2*(SIZE_MPTR_BAGS) <= MptrBags+initialBagCount;
p += SIZE_MPTR_BAGS )
{
*p = (Bag)(p + SIZE_MPTR_BAGS);
}

/* the rest is for bags */
MptrEndBags = MptrBags + 1024*initial_size/8/sizeof(Bag*);
MptrEndBags = MptrBags + initialBagCount;
// Add a small gap between the end of the master pointers and OldBags
// This is mainly here to ensure we do not break allowing OldBags and
// MptrEndBags to differ.
Expand Down Expand Up @@ -1104,6 +1222,10 @@ Bag NewBag (
{
Bag bag; /* identifier of the new bag */

#ifdef GAP_MEM_CHECK
MaybeMoveBags();
#endif

#ifdef TREMBLE_HEAP
CollectBags(0,0);
#endif
Expand Down Expand Up @@ -1269,6 +1391,11 @@ UInt ResizeBag (
Bag bag,
UInt new_size )
{

#ifdef GAP_MEM_CHECK
MaybeMoveBags();
#endif

#ifdef TREMBLE_HEAP
CollectBags(0,0);
#endif
Expand Down Expand Up @@ -2064,12 +2191,15 @@ UInt CollectBags (
&& SyAllocBags(-512,0) )
EndBags -= WORDS_BAG(512*1024L);

#ifdef GAP_MEM_CHECK
UInt SpareMasterPointers = 100000;
#else
UInt SpareMasterPointers = SpaceBetweenPointers(EndBags, stopBags)/7;
#endif
/* if we want to increase the masterpointer area */
if ( SizeMptrsArea-NrLiveBags < SpaceBetweenPointers(EndBags,stopBags)/7 ) {

if ( SizeMptrsArea-NrLiveBags < SpareMasterPointers ) {
/* this is how many new masterpointers we want */
i = SpaceBetweenPointers(EndBags,stopBags)/7 - (SizeMptrsArea-NrLiveBags);

i = SpareMasterPointers - (SizeMptrsArea-NrLiveBags);
/* move the bags area */
SyMemmove(OldBags+i, OldBags, SizeAllBagsArea*sizeof(*OldBags));

Expand Down
5 changes: 5 additions & 0 deletions src/gasman.h
Original file line number Diff line number Diff line change
Expand Up @@ -1144,4 +1144,9 @@ extern void CallbackForAllBags( void (*func)(Bag) );
void *AllocateMemoryBlock(UInt size);
#endif


#ifdef GAP_MEM_CHECK
Int enableMemCheck(Char ** argv, void * dummy);
#endif

#endif // GAP_GASMAN_H
Loading

0 comments on commit 562896c

Please sign in to comment.