2 Copyright © 2002-2013, The AROS Development Team. All rights reserved.
5 Desc: show deltas of resources usage
9 /******************************************************************************
26 It's like doing avail flush before and after executing a program to detect
27 memleaks, excepted that this program tries to detect much more leaks.
29 Press CRTL-f whenever you want to check for leak.
45 It has to not leak itself, and not even allocate memory between checks.
46 That's why lists and nodes are statically allocated (in a system that could
47 be improved to allow reuse of nodes. (like using an Allocate-based own
48 memory allocator using a huge static array as heap)
52 ******************************************************************************/
54 #include <exec/memory.h>
55 #include <exec/tasks.h>
56 #ifndef EXEC_SEMAPHORES
57 # include <exec/semaphores.h>
59 #include <exec/execbase.h>
60 #include <graphics/gfxbase.h>
61 #include <graphics/text.h>
62 #include <intuition/classes.h>
63 #include <proto/exec.h>
64 #include <proto/graphics.h>
65 #include <dos/dosextens.h>
66 #include <proto/dos.h>
67 #include <clib/alib_protos.h>
72 #include <aros/debug.h>
74 const TEXT version
[] = "$VER: LeakWatch 0.2 (10.03.2013)\n";
76 static struct Library
*GadToolsBase
;
78 struct GadToolsBase_intern
85 Class
* scrollerclass
;
88 Class
* listviewclass
;
89 Class
* checkboxclass
;
94 /* Semaphore to protect the bevel object. */
95 struct SignalSemaphore bevelsema
;
96 /* Actually an Object *. The image used for bevel boxes. */
99 /* RenderHook for GTListView class */
100 struct Hook lv_RenderHook
;
102 /* Seglist pointer */
105 /* Required libraies */
106 APTR gt_IntuitionBase
;
114 /* struct arrays : ORN = OpenedResourceNode, TR = TrackedResources, RD = ResourceDiff
115 MR = ModifiedResource STR=stringbuf
117 #define MAX_ORN 30000
120 #define MAX_STR 300000
123 /* All resources that have an opencount, eg. libs, devs, fonts */
124 struct OpenedResourceNode
{
132 /* All resources that LeakWatch handles. */
133 struct TrackedResources
{
134 struct List opened
; /* List of OpenedResourceNode */
135 IPTR freeMem
; /* total free memory */
138 struct ModifiedResource
{
147 /* store eg. new libraries, or libraries with modified opencount
150 struct ResourceDiff
{
152 struct List modifiedOpened
; /* contains ModifiedResource */
155 /* static storage to avoid interfering with memleaks debugging */
156 static struct OpenedResourceNode _ornbuf
[MAX_ORN
];
157 static struct TrackedResources _trbuf
[MAX_TR
];
158 static struct ResourceDiff _rdbuf
[MAX_RD
];
159 static UBYTE _strbuf
[MAX_STR
];
160 static struct ModifiedResource _mrbuf
[MAX_MR
];
162 static int next_orn
= 0;
163 static int next_tr
= 0;
164 static int next_rd
= 0;
165 static int next_str
= 0;
166 static int next_mr
= 0;
168 static struct OpenedResourceNode
*get_orn()
170 if (next_orn
== MAX_ORN
)
172 return &_ornbuf
[next_orn
++];
175 static void release_orn (struct OpenedResourceNode
*orn
)
179 static struct TrackedResources
*get_tr()
181 if (next_tr
== MAX_TR
)
183 return &_trbuf
[next_tr
++];
186 static void release_tr (struct TrackedResources
*tr
)
190 static struct ResourceDiff
*get_rd()
192 if (next_rd
== MAX_RD
)
194 return &_rdbuf
[next_rd
++];
197 static void release_rd (struct ResourceDiff
*rd
)
201 static struct ModifiedResource
*get_mr()
203 if (next_mr
== MAX_MR
)
205 return &_mrbuf
[next_mr
++];
208 static void release_mr (struct ModifiedResource
*mr
)
212 CONST_STRPTR
StaticStrDup (CONST_STRPTR str
)
214 UBYTE
*start
= &_strbuf
[next_str
];
222 if (len
+ next_str
+ 1 > MAX_STR
)
225 while ((*t
++ = *str
++))
227 next_str
+= t
- start
;
228 return (CONST_STRPTR
)start
;
231 void StrFree (CONST_STRPTR str
)
235 /***********************************************************************/
237 static struct TrackedResources
*NewResourcesState(void);
238 static void DeleteResourcesState(struct TrackedResources
*rs
);
239 static struct ResourceDiff
*NewStateDiff(const struct TrackedResources
*old
,
240 const struct TrackedResources
*new);
241 static void DisplayStateDiff(const struct ResourceDiff
*rd
, int pagelines
);
242 static void DeleteStateDiff(struct ResourceDiff
*rd
);
243 static struct TrackedResources
* CopyResourcesState(const struct TrackedResources
*src
);
245 static BOOL
AddLibs(struct List
*opened
)
250 for(lib
=(struct Library
*)SysBase
->LibList
.lh_Head
;
251 lib
->lib_Node
.ln_Succ
!=NULL
;
252 lib
=(struct Library
*)lib
->lib_Node
.ln_Succ
)
254 struct OpenedResourceNode
*orn
= get_orn();
260 orn
->type
= "Library";
261 orn
->name
= StaticStrDup(lib
->lib_Node
.ln_Name
);
263 orn
->count
= lib
->lib_OpenCnt
;
264 Enqueue(opened
, (struct Node
*)orn
);
270 static BOOL
AddDevs(struct List
*opened
)
275 for(dev
=(struct Device
*)SysBase
->DeviceList
.lh_Head
;
276 dev
->dd_Library
.lib_Node
.ln_Succ
!=NULL
;
277 dev
=(struct Device
*)dev
->dd_Library
.lib_Node
.ln_Succ
)
279 struct OpenedResourceNode
*orn
= get_orn();
285 orn
->type
= "Device";
286 orn
->name
= StaticStrDup(dev
->dd_Library
.lib_Node
.ln_Name
);
288 orn
->count
= dev
->dd_Library
.lib_OpenCnt
;
289 Enqueue(opened
, (struct Node
*)orn
);
295 static BOOL
AddFonts(struct List
*opened
)
300 for(tf
=(struct TextFont
*)GfxBase
->TextFonts
.lh_Head
;
301 tf
->tf_Message
.mn_Node
.ln_Succ
!=NULL
;
302 tf
=(struct TextFont
*)tf
->tf_Message
.mn_Node
.ln_Succ
)
304 struct OpenedResourceNode
*orn
= get_orn();
311 orn
->name
= StaticStrDup(tf
->tf_Message
.mn_Node
.ln_Name
);
313 orn
->count
= tf
->tf_Accessors
;
314 Enqueue(opened
, (struct Node
*)orn
);
320 static BOOL
AddNodeNames(struct List
*opened
, struct List
*list
, CONST_STRPTR type
)
325 for(lib
=(struct Node
*)list
->lh_Head
;
327 lib
=(struct Node
*)lib
->ln_Succ
)
329 struct OpenedResourceNode
*orn
= get_orn();
336 orn
->name
= StaticStrDup(lib
->ln_Name
);
339 Enqueue(opened
, (struct Node
*)orn
);
345 static BOOL
AddASemaphore(struct List
*opened
,
346 struct SignalSemaphore
*ss
,
349 struct OpenedResourceNode
*orn
= get_orn();
357 orn
->type
= "Semaphore";
358 orn
->name
= StaticStrDup(name
);
360 orn
->count
= ss
->ss_NestCount
;
363 Enqueue(opened
, (struct Node
*)orn
);
368 static BOOL
AddAClass(struct List
*opened
,
369 Class
*cl
, CONST_STRPTR name
)
371 struct OpenedResourceNode
*orn
;
386 orn
->name
= StaticStrDup(name
);
388 orn
->count
= cl
->cl_ObjectCount
;
391 Enqueue(opened
, (struct Node
*)orn
);
396 /* Add opencount-based resources to the tracking list. */
397 static BOOL
AddOpenedResources(struct List
*opened
)
399 struct DosInfo
*di
= BADDR(DOSBase
->dl_Root
->rn_Info
);
401 if (!AddLibs(opened
))
403 if (!AddDevs(opened
))
405 if (!AddFonts(opened
))
407 if (!AddNodeNames(opened
, &SysBase
->ResourceList
, "Resource"))
409 if (!AddNodeNames(opened
, &SysBase
->IntrList
, "Interrupt"))
411 if (!AddNodeNames(opened
, &SysBase
->PortList
, "Port"))
413 if (!AddNodeNames(opened
, &SysBase
->SemaphoreList
, "Semaphore"))
415 if (!AddASemaphore(opened
, &di
->di_DevLock
, "di_DevLock"))
417 if (!AddAClass(opened
, ((struct GadToolsBase_intern
*)GadToolsBase
)->buttonclass
, "button"))
419 if (!AddAClass(opened
, ((struct GadToolsBase_intern
*)GadToolsBase
)->textclass
, "text"))
421 if (!AddAClass(opened
, ((struct GadToolsBase_intern
*)GadToolsBase
)->sliderclass
, "slider"))
423 if (!AddAClass(opened
, ((struct GadToolsBase_intern
*)GadToolsBase
)->scrollerclass
, "scroller"))
425 if (!AddAClass(opened
, ((struct GadToolsBase_intern
*)GadToolsBase
)->arrowclass
, "arrow"))
427 if (!AddAClass(opened
, ((struct GadToolsBase_intern
*)GadToolsBase
)->stringclass
, "string"))
429 if (!AddAClass(opened
, ((struct GadToolsBase_intern
*)GadToolsBase
)->listviewclass
, "listview"))
431 if (!AddAClass(opened
, ((struct GadToolsBase_intern
*)GadToolsBase
)->checkboxclass
, "checkbox"))
433 if (!AddAClass(opened
, ((struct GadToolsBase_intern
*)GadToolsBase
)->cycleclass
, "cycle"))
435 if (!AddAClass(opened
, ((struct GadToolsBase_intern
*)GadToolsBase
)->mxclass
, "mx"))
437 if (!AddAClass(opened
, ((struct GadToolsBase_intern
*)GadToolsBase
)->paletteclass
, "palette"))
442 /* Get a snapshot of current resources */
443 static struct TrackedResources
*NewResourcesState(void)
445 struct TrackedResources
*tr
;
452 FreeVec(AllocVec((ULONG
)(~0ul/2), MEMF_ANY
));
454 /* opencount-based stuff */
455 NEWLIST(&tr
->opened
);
456 if (!AddOpenedResources(&tr
->opened
))
460 tr
->freeMem
= AvailMem(MEMF_ANY
);
465 static void DeleteResourceNode(struct OpenedResourceNode
*orn
)
469 StrFree((APTR
)orn
->name
);
474 /* Free snapshot of current resources */
475 static void DeleteResourcesState(struct TrackedResources
*rs
)
477 struct OpenedResourceNode
*orn
;
478 struct OpenedResourceNode
*tmp
;
483 for(orn
=(struct OpenedResourceNode
*)rs
->opened
.lh_Head
;
484 orn
->node
.ln_Succ
!=NULL
;
487 tmp
= (struct OpenedResourceNode
*)orn
->node
.ln_Succ
;
488 Remove((struct Node
*)orn
);
489 DeleteResourceNode(orn
);
494 void DisplayResourcesState(const struct TrackedResources
*rs
, int pagelines
)
496 struct OpenedResourceNode
*orn
;
502 FPuts(Output(), "LeakWatch snapshot:\n");
504 FPrintf(Output(), " Free memory : %id bytes\n", rs
->freeMem
);
506 FPuts(Output(), " Opened resources:\n");
508 for(orn
=(struct OpenedResourceNode
*)rs
->opened
.lh_Head
;
509 orn
->node
.ln_Succ
!=NULL
;
510 orn
=(struct OpenedResourceNode
*)orn
->node
.ln_Succ
)
512 if (currentlines
>= (pagelines
- 2))
516 FPuts(Output(), "--- Press a key to continue ---\n");
518 Read(Input(), &buf
, 1);
521 FPrintf(Output(), " %s: %s (0x%ix) : %lu\n",
522 orn
->type
, orn
->name
, orn
->addr
, orn
->count
);
525 FPuts(Output(), "-- end of state\n");
528 /* Compute the delta between 2 resources snapshots.
529 * the ResourceDiff can have dangling pointers in old and nu, so dont clear
530 * them before being done with rd in the processing loop
532 static struct ResourceDiff
*NewStateDiff(const struct TrackedResources
*old
,
533 const struct TrackedResources
*nu
)
536 struct OpenedResourceNode
*orn
;
537 struct ResourceDiff
*rd
;
543 NEWLIST(&rd
->modifiedOpened
);
545 for(orn
=(struct OpenedResourceNode
*)nu
->opened
.lh_Head
;
546 orn
->node
.ln_Succ
!=NULL
;
547 orn
=(struct OpenedResourceNode
*)orn
->node
.ln_Succ
)
549 struct OpenedResourceNode
*other
;
552 for(other
=(struct OpenedResourceNode
*)old
->opened
.lh_Head
;
553 other
->node
.ln_Succ
!=NULL
;
554 other
=(struct OpenedResourceNode
*)other
->node
.ln_Succ
)
556 if (orn
->addr
== other
->addr
)
558 if (!strcmp(orn
->name
, other
->name
))
561 if (orn
->count
!= other
->count
)
563 struct ModifiedResource
*mr
= get_mr();
566 mr
->type
= other
->type
;
567 mr
->name
= other
->name
;
568 mr
->addr
= other
->addr
;
569 mr
->before_count
= other
->count
;
570 mr
->after_count
= orn
->count
;
571 Enqueue(&rd
->modifiedOpened
, (struct Node
*)mr
);
578 struct ModifiedResource
*mr
= get_mr();
582 mr
->type
= orn
->type
;
583 mr
->name
= orn
->name
;
584 mr
->addr
= orn
->addr
;
585 mr
->before_count
= 0;
586 mr
->after_count
= orn
->count
;
588 Enqueue(&rd
->modifiedOpened
, (struct Node
*)mr
);
593 rd
->memLost
= old
->freeMem
- nu
->freeMem
;
598 static void DisplayStateDiff(const struct ResourceDiff
*rd
, int pagelines
)
600 struct ModifiedResource
*mr
;
603 FPuts(Output(), "LeakWatch report:\n");
604 FPrintf(Output(), " Memory lost : %ld byte%s\n", rd
->memLost
, (rd
->memLost
> 1) ? "s" : "");
606 FPuts(Output(), " Open count:\n");
608 for(mr
=(struct ModifiedResource
*)rd
->modifiedOpened
.lh_Head
;
609 mr
->node
.ln_Succ
!=NULL
;
610 mr
=(struct ModifiedResource
*)mr
->node
.ln_Succ
)
612 if (currentlines
>= (pagelines
- 2))
616 FPuts(Output(), "--- Press a key to continue ---\n");
618 Read(Input(), &buf
, 1);
621 FPrintf(Output(), " %s: %s (0x%ix) : %lu -> %lu\n",
622 mr
->type
, mr
->name
, mr
->addr
, mr
->before_count
,
627 FPuts(Output(), "-- end of diff\n");
631 static void DeleteStateDiff(struct ResourceDiff
*rd
)
634 struct ModifiedResource
*mr
;
635 struct ModifiedResource
*tmpmr
;
640 for(mr
=(struct ModifiedResource
*)rd
->modifiedOpened
.lh_Head
;
641 mr
->node
.ln_Succ
!=NULL
;
644 tmpmr
= (struct ModifiedResource
*)mr
->node
.ln_Succ
;
645 Remove((struct Node
*)mr
);
653 static struct OpenedResourceNode
* CopyResourcesNode(const struct OpenedResourceNode
*src
)
655 struct OpenedResourceNode
*orn
= get_orn();
658 orn
->name
= StaticStrDup(src
->name
);
659 orn
->addr
= src
->addr
;
660 orn
->count
= src
->count
;
664 static struct TrackedResources
* CopyResourcesState(const struct TrackedResources
*src
)
666 struct TrackedResources
*tr
;
667 struct OpenedResourceNode
*orn
;
673 /* opencount-based stuff */
674 NEWLIST(&tr
->opened
);
676 for(orn
=(struct OpenedResourceNode
*)src
->opened
.lh_Head
;
677 orn
->node
.ln_Succ
!=NULL
;
678 orn
=(struct OpenedResourceNode
*)orn
->node
.ln_Succ
)
680 struct OpenedResourceNode
*nc
;
682 nc
= CopyResourcesNode(orn
);
683 Enqueue(&tr
->opened
, (struct Node
*)nc
);
687 tr
->freeMem
= src
->freeMem
;
696 GadToolsBase
= OpenLibrary ( "gadtools.library", 0L );
701 CloseLibrary ( GadToolsBase
);
706 struct TrackedResources
*crs
= NULL
;
707 struct TrackedResources
*start_rs
= NULL
;
711 struct MsgPort
*port
;
713 port
= CreateMsgPort();
716 port
->mp_Node
.ln_Name
= "LeakWatch";
717 port
->mp_Node
.ln_Pri
= 0;
720 portsig
= 1L << port
->mp_SigBit
;
724 FPuts(Output(), "LeakWatch running, CTRL-C to exit, CTRL-E to watch for leaks since beginning, CTRL-F to watch for leaks since last CTRL-F, CTRL-D for an usage snapshot\n");
726 crs
= NewResourcesState();
730 start_rs
= CopyResourcesState(crs
);
736 signals
= Wait(portsig
| SIGBREAKF_CTRL_F
| SIGBREAKF_CTRL_E
| SIGBREAKF_CTRL_D
| SIGBREAKF_CTRL_C
);
738 if (signals
& SIGBREAKF_CTRL_D
)
740 struct TrackedResources
*tr
;
742 tr
= NewResourcesState();
748 DisplayResourcesState(tr
, numlines
);
749 DeleteResourcesState(tr
);
751 if (signals
& SIGBREAKF_CTRL_E
)
753 struct ResourceDiff
*rd
= NULL
;
755 DeleteResourcesState(crs
);
756 crs
= NewResourcesState();
762 /* DisplayResourcesState(crs); */ /* only for debug */
763 rd
= NewStateDiff(start_rs
, crs
);
764 DisplayStateDiff(rd
, numlines
);
767 if (signals
& SIGBREAKF_CTRL_F
)
769 struct TrackedResources
*ors
= crs
;
770 struct ResourceDiff
*rd
= NULL
;
772 crs
= NewResourcesState();
778 rd
= NewStateDiff(ors
, crs
);
779 DisplayStateDiff(rd
, numlines
);
781 DeleteResourcesState(ors
);
783 if (signals
& SIGBREAKF_CTRL_C
)
787 if (signals
& portsig
)
791 while((msg
= (struct Message
*)GetMsg(port
)))
793 D(bug("Received watch message.\n"));
799 } /* while(!quitme) */
801 DeleteResourcesState(crs
);
802 DeleteResourcesState(start_rs
);