The MASM Forum

General => The Workshop => Topic started by: NoCforMe on June 14, 2023, 02:15:24 PM

Title: A way to create "objects" and automatically delete them at program end
Post by: NoCforMe on June 14, 2023, 02:15:24 PM
While we're all mourning Steve's demise, I created this little scheme to deal with the age-old problem of creating "objects" (brushes, pens, etc.) and then having to chase after them and delete them when the program ends. (Or, I'll admit, sometimes just not caring about them; word has it that Windows automagically deletes them anyhow for you, so why bother?)

So here it is, a very simple "objects" manager. Instead of the Win32 functions to create stuff (CreatePen(), CreateSolidBrusH(), etc.) you use the generic function CreateObject(). It creates the thing, stores its handle in a list, then returns with the handle. It knows how to create brushes, pens, fonts and bitmaps. (Other things could be added.)

At program end you just call DestroyObjects(), which goes through the list and destroys all objects that have non-NULL handles. Easy peasy.

Of course you can modify this for your own purposes. I made the handle list 64 items long, which could be easily changed. Minimal error checking (just to see if the handle list is full). If you need to delete just one object before program end, that's on you to keep track of its handle. I didn't want to add that level of complexity (being able to address a particular handle in the list). But you could do it if you wanted to.


;====================================
; Objects.asm
;
; Object-creating and -destroying code
;
; Keeps track of objects (brushes, pens, fonts and bitmaps)
; for you and deletes them at program end (using DestroyObjects() ).
;
; Use CreateObject (typ, parm1, parm2, parm3) to create stuff.
; typ = type of object (from list below)
; parm1-parm3 depend on object type:
; o Brush: parm1 = color
; o Pen: parm1 = style
; parm2 = width
; parm3 = color
; o Font: parm1 = pointer to a LOGFONT structure
; o Bitmap: parm1 = pointer to a BITMAP structure
;====================================

;***** Type codes:  *****
$objBrush EQU 1
$objPen EQU 2
$objFont EQU 3
$objBitmap EQU 4

$maxObjects EQU 64 ;Change if you need more objects

CreateObject PROTO typ:DWORD, parm1:DWORD, parm2:DWORD, parm3:DWORD
DestroyObjects PROTO

.data

ObjStore HANDLE $maxObjects DUP(0) ;All handles initially = NULL.
ObjStoreNextPtr DD ObjStore
NumObjects DD 0

.code
;====================================
; CreateObject (typ, parm1, parm2, parm3)
;
; Returns:
; handle to object created, or
; NULL if object couldn't be created, or
; -1 if max. # objects already created.
;====================================

CreateObject PROC typ:DWORD, parm1:DWORD, parm2:DWORD, parm3:DWORD

; See if we've already created the max. # of objects:
CMP NumObjects, $maxObjects
JNE @F
MOV EAX, -1
RET

@@: MOV EAX, typ
CMP EAX, $objFont
JE do_font
CMP EAX, $objPen
JE do_pen
CMP EAX, $objBitmap
JE do_bitmap

; Must be a brush:
INVOKE CreateSolidBrush, parm1 ;Color

store_handle:
TEST EAX, EAX ;If handle wasn't created because of error,
JZ @F ;  don't store it.
MOV EDX, ObjStoreNextPtr
MOV [EDX], EAX
ADD ObjStoreNextPtr, SIZEOF HANDLE
INC NumObjects
@@: RET

do_font:
INVOKE CreateFontIndirect, parm1 ;Ptr. to LOGFONT struct.
JMP store_handle

do_pen:
INVOKE CreatePen, parm1, parm2, parm3 ;Style, width, color
JMP store_handle

do_bitmap:
INVOKE CreateBitmapIndirect, parm1 ;Ptr. to BITMAP struct.
JMP store_handle

CreateObject ENDP


;====================================
; DestroyObjects()
;
; Destroys any objects that have been created.
;
; Returns nothing.
;====================================

DestroyObjects PROC

PUSH EBX
MOV EBX, OFFSET ObjStore
MOV ECX, $maxObjects
dloop: PUSH ECX
MOV EAX, [EBX]
TEST EAX, EAX ;If handle is 0,
JZ next ;  skip it.
INVOKE DeleteObject, EAX

next: ADD EBX, SIZEOF HANDLE
POP ECX
LOOP dloop

POP EBX
RET

DestroyObjects ENDP
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: jj2007 on June 14, 2023, 02:33:53 PM
Well done, David :thumbsup:

I'm not quite sure if store_handle: should or should not store a zero handle - but that's not a criticism, just food for thought.
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: NoCforMe on June 14, 2023, 02:58:05 PM
Sure, but keep in mind that in this scheme all handles are NULL by default, so nothing to be gained by storing it in that case.
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: jj2007 on June 14, 2023, 06:23:20 PM
Quote from: NoCforMe on June 14, 2023, 02:58:05 PM
Sure, but keep in mind that in this scheme all handles are NULL by default, so nothing to be gained by storing it in that case.

Yes, that's the point ;-)
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: zedd151 on June 14, 2023, 08:03:45 PM
That seems like a nifty idea, David. I would however use the same parameter number for 'color' for instance. Also any structure ought be the same parameter number, etc. It would be easier imo to remember than trying to remember the parameters  for every possible object. There will always be some arguments that are NULL in any case, so instead of using parm1 always if the object needs only one parameter keep some semblance of uniformity maybe... just a thought. Gotta be a little better than parm1=color for one object but parm3=color for another...
So maybe 4 'parms', parm1=style, parm2=color, parm3=width, parm4=structure... (same parm - color for instance - always in same location) or whichever way seems reasonable. Would have to look how many other types of objects can be created and the arguments that might be needed for them. I am on my iPad presently I'll look at this again when I'm at my computer.
~~ later that day ~~
Okay, I'm inside here is what I am talking about...
; o any object:     parm1 = style
;                   parm2 = width
;                   parm3 = color
;                   parm4 = pointer to a ?? structure - structure associated with given object


Unused 'parms' of course will be null
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: jj2007 on June 14, 2023, 10:12:56 PM
A variant (pure Masm32 SDK; 65 lines GUI example attached see reply #11 - check the Gdi handles in Task Manager for leaks):

include \masm32\include\masm32rt.inc
include CreateAndSelect.inc
; coThrowErrors=0 ; default is 1=throw an error if you get a zero handle (inserts extra code)

.data
MyBitmap dd 1, 2, 3
MyLogFont dd 1, 2, 3

.code
MyAlgo proc argDC, arg1, arg2
Local hPen, hBitmap, hFont
  CreateAndSelect(argDC) ; one arg: set the DC and allocate a local handle table
  mov hBrush, CreateAndSelect($objBrush, 0FFDD00h)
  void CreateAndSelect($objBrush, 0FFDD00h) ; no need for a variable if you just select it once
  mov hPen, CreateAndSelect($objPen, PS_SOLID, 3, 0FFDD00h)
  mov hBitmap, CreateAndSelect($objBitmap, addr MyBitmap)
  mov hFont, CreateAndSelect($objFont, addr MyLogFont)
  ; mov hFont, CreateAndSelect($objFont, addr MyLogFont, 123) ; assembly time error
  ReleaseObjects
  ret
MyAlgo endp
start:
  invoke MyAlgo, rv(GetWindowDC, 0), 12, 34
  inkey "done"
  exit

end start


Coding is fun :tongue:
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: jj2007 on June 14, 2023, 11:39:08 PM
Here is the core of the GUI example, called from the WM_PAINT handler:

MyTest proc uses edi hDC
Local hBrush, hPen, rc:RECT, buffer[40]:BYTE
  CreateAndSelect(hDC, objct=4) ; one arg or explicit object count: set the DC
  mov hBrush, CreateAndSelect($objBrush, 0FFDD00h)
  void CreateAndSelect($objBrush, 0FFDD00h) ; no need for a variable if you just select it once
  mov hPen, CreateAndSelect($objPen, PS_SOLID, 3, 0FFDD00h)
  void CreateAndSelect($objFont, addr MyLogFont)
  invoke GetWindowRect, hWin, addr rc
  lea edi, buffer
  invoke crt_sprintf, edi, addr txRcRight, str$(rc.right) ; db "rc.right=%s", 0
  invoke Rectangle, hDC, 9, 9, 200, 64
  invoke TextOut, hDC, 20, 20, edi, len(edi)
  ReleaseObjects
  ret
MyTest endp


As mentioned above, it works also without the local variables hBrush and hPen - just use void CreateAndSelect(...).
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: NoCforMe on June 15, 2023, 01:34:34 PM
Ackshooly, speaking of the clunkiness of the parm1 ... parm3 scheme where you have to figure out what the parms mean, what would be far better would be making the function a vararg function, like wsprintf(), where it takes only the number of arguments needed. So for creating a brush, you'd only need to use


INVOKE CreateObject, $objBrush, brushColor


and not to have to push two useless parms on the stack.

However, I don't know how to code that. Maybe someone can help out here.
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: zedd151 on June 15, 2023, 02:20:43 PM
Quote from: NoCforMe on June 15, 2023, 01:34:34 PM
So for creating a brush, you'd only need to use

   INVOKE   CreateObject, brushColor



I think that you would need at least two arguments. One to specify type of object. The other for parm value(s)
Quote
CreateObject PROC C TypeObj:REQ, Args:VarArg
The arg count could probably be inferred depending on the type of object  :icon_idea: 
Don't forget caller must balance the stack. I had never made a vararg procedure before btw. Would need a little experimentation for sure.



Title: Re: A way to create "objects" and automatically delete them at program end
Post by: NoCforMe on June 15, 2023, 02:37:24 PM
Quote from: zedd151 on June 15, 2023, 02:20:43 PM
Don't forget caller must balance the stack.

Perhaps, but all I know is that I use wsprintf() all the time, with all kinds of different sets of parameters, and I never have to do anything (explicitly) to balance the stack; that's all taken care of for me. So once it's programmed, it's easy peasy to use.
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: zedd151 on June 15, 2023, 03:05:30 PM
Quote from: NoCforMe on June 15, 2023, 02:37:24 PM
Quote from: zedd151 on June 15, 2023, 02:20:43 PM
Don't forget caller must balance the stack.
... I never have to do anything (explicitly) to balance the stack ...


Whoops you're right. Only need to fiddle with the stack if using "push, push, call"  nevermind. It's been a long day.  :tongue:
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: jj2007 on June 15, 2023, 06:41:48 PM
Quote from: NoCforMe on June 15, 2023, 01:34:34 PM
Ackshooly, speaking of the clunkiness of the parm1 ... parm3 scheme where you have to figure out what the parms mean, what would be far better would be making the function a vararg function

Like this? Two brushes, a pen and a font (Times New Roman); check in Task Manager, columns user and Gdi objects, whether there's any leak.

MyTest proc uses edi hDC, pTxt ; pass the DC and one arbitrary argument (in this case, a string showing rc.right)
Local hBrush
  CreateAndSelect(hDC) ; one arg: set the DC
  mov hBrush, CreateAndSelect($objBrush, 0FF0000h)
  invoke Rectangle, hDC, 3, 3, 208, 70
  void CreateAndSelect($objBrush, 0FFh) ; no need for a variable if you just select it once
  void CreateAndSelect($objPen, PS_SOLID, 2, 0FFFFFFh) ; white pen
  void CreateAndSelect($objFont, addr MyLogFont)
  invoke Rectangle, hDC, 9, 9, 204, 64
  invoke TextOut, hDC, 20, 18, pTxt, len(pTxt)
  ReleaseObjects
  ret
MyTest endp


CreateAndSelect will throw an assembly time error if the number of parameters is not correct, and a runtime error if the handle is zero and casThrowErrors=1. Which is the default - it adds some code; once you are sure it works, set casThrowErrors=0 to produce the shortest possible code.

Macros and full test case attached (see WM_PAINT handler; 67 lines of code, no MasmBasic, just plain Masm32 SDK).
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: NoCforMe on June 15, 2023, 06:58:38 PM
Can you show us what that proc (CreateAndSelect) looks like so we can see how the vararg stuff works?
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: jj2007 on June 15, 2023, 07:58:50 PM
CreateAndSelect is a macro, and it's attached above (CreateAndSelect.inc in the archive). I hope it's self-explanatory, but feel free to ask :thup:
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: NoCforMe on June 16, 2023, 09:13:17 AM
Sorry, JJ: instead of tl:dr, this one's tc:dr (too complicated, didn't read).

Remember, I don't like macros ...
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: jj2007 on June 16, 2023, 11:34:15 PM
Quote from: NoCforMe on June 16, 2023, 09:13:17 AM
Sorry, JJ: instead of tl:dr, this one's tc:dr (too complicated, didn't read).

Remember, I don't like macros ...

Sorry for that, David. Here is a heavily commented version, I hope it helps.

Minimal use case:
MyTest proc hDC
  CreateAndSelect(hDC) ; set the DC
  void CreateAndSelect($objBrush, 0FF0000h) ; create a blue brush and select it
  invoke Rectangle, hDC, 3, 3, 208, 70 ; use the brush
  ReleaseObjects ; unselect the brush, then delete it
  ret
MyTest endp


The macros:
casThrowErrors=0 ; default is 1 (adds extra code)
casMaxDef=4
CreateAndSelect MACRO casType, arg0, args:VARARG
Local is, tmp$
  ifb <arg0> ; ifblank arg0: User passed CreateAndSelect(hDC) directly under the locals
casDC equ <casType> ; we define the DC
casMax=casMaxDef ; casMaxDef=4: two pens, a brush, a font; if you need more, increase casMaxDef or use CreateAndSelect(hDC:123)
is INSTR <casType>, <:> ; e.g. CreateAndSelect(hDC:8), i.e. user needs 8 Gdi objects
if is
tmp$ SUBSTR <casType>, is+1
casMax=tmp$ ; casMax=8
casDC SUBSTR <casType>, 1, is-1 ; hDC
endif
casCt=0 ; object counter
LOCAL casTable[casMax], casOld[casMax] ; define local tables for Gdi handles and the previously selected objects
EXITM <> ; allow CreateAndSelect(...) instead of mov handle, CreateAndSelect(...)
  else
$objBrush=1 ; define object identifiers
$objPen=2
$objFont=3
$objBitmap=4
if casType eq $objBrush
tmp$ equ <brush>
invoke CreateSolidBrush, arg0
ifnb <args> ; args not blank: user passed more than one argument, error
tmp$ CATSTR <### too many arguments for >, tmp$, < in line >, %@Line, < ###>
% echo tmp$
.err
endif
elseif casType eq $objPen
tmp$ equ <pen>
invoke CreatePen, arg0, args
elseif casType eq $objFont
tmp$ equ <font>
invoke CreateFontIndirect, arg0
ifnb <args> ; args not blank: user passed more than one argument, error
tmp$ CATSTR <### too many arguments for >, tmp$, < in line >, %@Line, < ###>
% echo tmp$
.err
endif
elseif casType eq $objBitmap
tmp$ equ <bitmap>
invoke CreateBitmapIndirect, arg0
ifnb <args> ; args not blank: user passed more than one argument, error
tmp$ CATSTR <### too many arguments for >, tmp$, < in line >, %@Line, < ###>
% echo tmp$
.err
endif
else
tmp$ CATSTR <### unknown object type >, <casType>, < ###>
% echo tmp$
.err
EXITM <>
endif
if casThrowErrors ; default: casThrowErrors=1 (inserts additional code for runtime error)
.if !eax
tmp$ CATSTR <print "*** line >, %@Line, < (>, tmp$, <): ">
% tmp$
print LastError$()
xor eax, eax
.endif
endif
push eax
mov casTable[casCt*4], eax ; store Gdi handle
invoke SelectObject, casDC, eax ; use it
mov casOld[casCt*4], eax ; store the previous handle
pop eax
casCt=casCt+1 ; advance counter for the two tables
EXITM <eax>
  endif
ENDM
ReleaseObjects MACRO
Local tmp$
  if casCt gt casMax ; throw assembly time error, give hint how to solve it
tmp$ CATSTR <## use 'CreateAndSelect(>, casDC, <:>, %casCt, <)' at start of proc ##>
% echo tmp$
.err
  endif
  push esi
  push ebx
  lea esi, casOld
  m2m ebx, casCt
  .While 1
dec ebx
.Break .if Sign?
invoke SelectObject, casDC, [esi+4*ebx] ; select old handles
invoke DeleteObject, [esi+4*ebx+4*casMax] ; delete Gdi objects created above
  .Endw
  pop ebx
  pop esi
ENDM
void MACRO args ; allow the void CreateAndSelect($objBrush, 0FF0000h) syntax
  @CatStr(<;>, <args>)
ENDM
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: NoCforMe on June 17, 2023, 07:11:21 AM
First of all, I appreciate the effort that went into this, as goes into all that you do (even if you didn't do it just for me!).

My purpose for this thing is different from yours; your examples show you using it in your WM_PAINT handler, where you create things to select into the DC.

I work a little differently. Instead of creating brushes, pens, etc., "on the fly" when handling WM_PAINT, then destroying them each time, I create them once only, somewhere near the top of my code, make them global variables, use them in the paint routine over and over, then destroy them at program end. I'm not suggesting this is a better way than yours; it's just the way I like to roll. (There are times I have to create them in the paint handler, say if I need a brush of a certain color that the user has selected.)

I see why you not only create objects but automatically select them into your DC, then select the original object when you destroy them. Makes a lot of sense for the way you work.

This is at best a marginal time-saver, but I guess every little bit helps, right?

P.S.: This makes me wonder: is it really necessary to save the "original" object when you select something into a DC, then restore it before you exit your paint handler? I often don't bother to do this, and it doesn't seem to affect program operation at all. I think this may be one of those things that was needed in a long-ago version of Windows but is no longer necessary. Probably our friend Ray Chen has had a word or three to say about this. I actually don't know; does anyone know for sure whether this is really necessary? (Similar to how Windows automatically gets rid of most things you created, fonts, bitmaps, etc., at program end if you fail to do so, so that this kind of memory leak due to stranded handles isn't really a thing anymore.)
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: jj2007 on June 17, 2023, 07:30:01 AM
Quote from: NoCforMe on June 17, 2023, 07:11:21 AM
First of all, I appreciate the effort that went into this, as goes into all that you do (even if you didn't do it just for me!).

I simply found it an interesting idea ;-)

QuoteI work a little differently. Instead of creating brushes, pens, etc., "on the fly" when handling WM_PAINT, then destroying them each time, I create them once only, somewhere near the top of my code, make them global variables, use them in the paint routine over and over, then destroy them at program end.

That's an absolutely valid way of coding, and indeed, I often do it exactly like that. Raymond Chen might tell you that destroying the objects at program end is not necessary, though. The create-select-restore old-destroy sequence is orthodox Windows doctrine. With permanent handles, select-restore old looks like the minimum required. The restore old part may not be required in modern Windows versions, but test that carefully with a loop to avoid ugly surprises. Then there is the issue of private DCs...
Title: Re: A way to create "objects" and automatically delete them at program end
Post by: NoCforMe on June 17, 2023, 12:23:11 PM
Regarding the whole DC selection-resetting thing, see this new thread (http://masm32.com/board/index.php?topic=10808.0).