Probably a simple question, but how do you define a buffer size similar to the C++ equivalent below?
char[1024];
Thanks in advance :bg
buffername db 1024 dup(?) make sure you don't do buffername db 1024 dup(0) as that adds to your pe files actual size, instead of just memory.
can do buffername byte 1024 dup(?) too, is same thing.
Inside procedures, you can declare a local buffer. See \masm32\examples\exampl01\minifile\minifile.asm for an example. Search for PathBuffer inside the file.
Thanks for your replies.
I came across this in the MASM manual and it works ok
szBuffer BYTE 1024 DUP(0), 0
But The following is better is it?
szBuffer 1024 dup(?)
Using that latter, I would have to initialize it to zero wouldn't I?
.data?
buff byte 1024 dup(?)is unitalized data and is automatically empty, doesn't add size to your exe,dll etc on disk...
.data
buff byte 1024 dup(0) is initalized data and will make your exe,dll etc 1024 bytes larger on the disk
if you define it locally in a procedure on the stack like
copystring proc param1:DWORD
LOCAL buff[1024]:BYTE <--this needs to be cleared with 0's as lot of times will contain random garbage
invoke RtlZeroMemory,addr buff,1024 <--zeros the buff
ret
copystring endp
keep in mind you can only define so much data on the stack for a procedure before it'll mess up your program(make it just silently exit) from stack overflow, theres a stackprobe macro on this forum that fixes this issue
;this will stack overflow your program
copystring proc param1:DWORD
LOCAL buff[1024]:BYTE <--this needs to be cleared with 0's as lot of times will contain random garbage
LOCAL buff1[1024]:BYTE
LOCAL buff2[1024]:BYTE
LOCAL buff3[1024]:BYTE
LOCAL buff4[1024]:BYTE
LOCAL buff5[1024]:BYTE
LOCAL buff6[1024]:BYTE
LOCAL buff7[1024]:BYTE
LOCAL buff8[1024]:BYTE
LOCAL buff9[1024]:BYTE
LOCAL buff109[1024]:BYTE
invoke RtlZeroMemory,addr buff,1024 <--zeros the buff
ret
copystring endp
Quote from: E^cube on March 01, 2009, 09:47:19 AM
.data?
buff byte 1024 dup(?)is unitalized data and is automatically empty, doesn't add size to your exe,dll etc on disk...
So, for program compact-ness, this looks like the way to go. And no garbage as a result if I understand correctly.
Thanks for the info! :bg
Hi lonewolf,
The only issue with initializing buffers in the .DATA or .DATA? section is if you're using MASM there is a known bug when declaring very large blocks of data, I prefer the stack for any data that is not persistent or the .CONST section if it is fixed. For strings, since the size would rarely exceed 4166 Bytes (maximum url length in unicode) its not really much of an issue. By the way you can change the size of the stack...
/stacksize xxxx for GoLink
.STACK xxxx for MASM (as a directive after .MODEL)
or
/STACK xxxx for link.exe
Edgar
Quote from: donkey on March 02, 2009, 08:44:16 AM
if you're using MASM there is a known bug when declaring very large blocks of data
That bug is tricky indeed. Allocating reasonable amounts is no problem, but Masm takes a looong time for data like this:
.data?
MyBuf dd 280000 dup (?) ; 2 seconds for 200k, 6 for 250k, 12 for 280k ...
If you go over 300k, masm will appear to hang. The behaviour is exponential, and linked to the dup macro. But there is a simple workaround using the ORG directive. Full example below.
include \masm32\include\masm32rt.inc
crtbuf MACRO NameOfVar, BufLen
LOCAL lbl
.data?
align 4
lbl LABEL byte
ORG $+BufLen-1
db ?
.data
NameOfVar dd lbl ;; define it in the data section
.code
ENDM
.code
start:
crtbuf MyFatBuffer, 10000000 ; 10 MB
invoke lstrcpy, MyFatBuffer, chr$("Howdy") ; test it - copy a string
MsgBox 0, MyFatBuffer, "crtbuf macro:", MB_OK ; and show it
exit
end start
Also note that for constants, many times an EQUate will work instead. Equates are assemble-time only values, and are not stored in the executable for use at run-time. i.e.
equDialogControl EQU 1001
.code
invoke GetDlgItemText,[hMyDialog],equDialogControl,addr myStringResult,255
...
Here, "equDialogControl" is replaced with the value 1001 prior to assembly. Math may also be performed on these values prior to assembly, such as equDialogControl + 3 and similar. Just remember that the math is carried out before the assembly, and the assembled result is then a static value. (This is not the proper method to use if the code is say, putting a variable number of controls on the application at run-time. In that case, an actual memory variable is required to keep track of how many controls are to be displayed.)
Edit: Typo.
looks a bit creepy to me, I've always used memory allocation via OS functions, and then load the data from files.
hi,
Just approximately, how much data can be defined on the stack for a proc before there is a stack overflow ?
t y
pretty sure its 4096 by default unless you change the stack size with the switch mentioned above, I recommend you use this macro as it cures all problems
StackProbe.inc
;:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
;
; MASM Stack Probing PROLOGUE macro
; by chep, 2005/06/22
;
;:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
;
; Allows a procedure to safely use LOCAL variables with a total size of 4kb or more,
; using an unrolled stack probing method.
;
;:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
;
; Usage:
;
; OPTION PROLOGUE:STACKPROBE
; MyProcedure PROC ; ...
; ; ...
; MyProcedure ENDP
; OPTION PROLOGUE:PROLOGUEDEF
;
;:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
;
; Notes:
; - When the total size of the LOCAL variables is less than 4kb, the code generated is
; identical to PROLOGUEDEF, so there is no drawback using this macro
; - See "OPTION PROLOGUE" and "PROC" topics in MASM32.HLP for the macro specifications
;
;:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
;
; Limitations compared to PROLOGUEDEF:
; - Stack probing is relevant only for Windows, ie FLAT model, so it won't accept other models
; - Due to the FLAT model restriction, LOADDS is not supported
; - FORCEFRAME argument doesn't generate a correct epilogue when no LOCAL variables are defined
; So it is not supported for now :(
;
;:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
;
; TODO:
; - Allow looped probing additionally to the current unrolled probing
; This behaviour should be controlled through macro arguments
;
;:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
STACKPROBE MACRO procname, flags, argbytes, localbytes, reglist, userparms:VARARG
LOCAL parm, reg, probe, max_probe, line, page_size
;; Memory page size, can be changed for other architectures
page_size = 4096
;; Throw an error if not in FLAT model, because stack probing is irrelevant otherwise
IF @Model NE 7
line TEXTEQU %@Line
% ECHO @FileCur(line) : STACKPROBE prologue ERROR: stack probing is irrelevant if not using FLAT model
.ERR
EXITM <1>
ENDIF
;; Detect macro arguments to give a warning
IFNB <userparms>
FOR parm,<userparms>
line TEXTEQU %@Line
% ECHO @FileCur(line) : STACKPROBE prologue warning: unknown prologue argument : parm
ENDM
ENDIF
;; Set up stack frame
IF localbytes GT 0
push ebp
mov ebp, esp
IF localbytes LT page_size
;; Normal stack frame, no probing
add esp, (NOT localbytes)+1
ELSE
;; Unrolled stack probing
max_probe = (localbytes+page_size-1) AND (-page_size) ;; round up to next page size
probe = page_size
WHILE probe LE max_probe
;; Probe stack
mov DWORD PTR [ebp-probe], eax
probe = probe + page_size
ENDM
add esp, (-localbytes)
ENDIF
ELSEIF argbytes GT 0
push ebp
mov ebp, esp
ENDIF
;; USES clause
IFNB <reglist>
FOR reg,reglist
push reg
ENDM
ENDIF
EXITM <0>
ENDM
Quote from: Rainstorm on April 01, 2009, 07:16:29 AM
hi,
Just approximately, how much data can be defined on the stack for a proc before there is a stack overflow ?
t y
4096 bytes, the size of a page. You can rely on it - as Raymond Chen puts it (The Old New Thing (http://blogs.msdn.com/oldnewthing/archive/2004/09/08/226797.aspx)), "Note that changing the page size creates all sorts of problems for compatibility. There are large numbers of programs out there that blindly assume that the page size is 4K. Boy are they in for a surprise."
Now the question is whether you can blindly assume that Microsoft will not surprise large numbers of programmers :wink
.data
info SYSTEM_INFO <0>
...
.code
start:
invoke GetSystemInfo, addr info
mov eax, info.dwPageSize
As far as I know you can use stack space in any size chunks until you approach the stack committed size (1MB?). It is when using stack space between the committed size and the reserved size (2MB?) that you must avoid accessing past the guard page.
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
include \masm32\include\masm32rt.inc
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
.data
.code
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
cnt equ 1036213 ;1048576=1MB
biglocals proc
LOCAL smp[cnt]:BYTE
lea edx, smp
mov ecx, cnt
.WHILE ecx
mov [edx+ecx], cl
dec ecx
.ENDW
ret
biglocals endp
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
start:
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
call biglocals
inkey "Press any key to exit..."
exit
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
end start
Quote from: MichaelW on April 01, 2009, 02:53:16 PM
As far as I know you can use stack space in any size chunks until you approach the stack committed size (1MB?). It is when using stack space between the committed size and the reserved size (2MB?) that you must avoid accessing past the guard page.
I had written a similar test some time ago, but it crashes at 4096 bytes with an access violation. Are we talking about the same phenomenon?
EDIT: It seems it works also for 8192 bytes:
Quotemov eax, info.dwPageSize
add eax, eax
mov s4096, eax
But it crashes neatly at
16384 12212 bytes. New code attached shows why: Apparently the OS allocates three pages, 4096*3 = 12288 bytes. A few have been used for other purposes. For 12212 bytes allocated, first access is at 0012D000, i.e. a page boundary; for 12214 bytes it is 2 bytes below and bang there is the access violation.
[attachment deleted by admin]
Reading through the GoLink manual I found these linker options and explanations :
Quote/stackinit xxxx
= initial stack commitment on start-up and on new threads.
Here you can specify the size in bytes (using hex) of the amount of memory which the system should commit for the stack when the application starts or when making a new thread. If you do not use this switch a value of 10000h (64KB) is used by default. Regardless of the value specified using this switch the system always enlarges the amount of committed memory in the stack in 4K chunks as the application needs it.
This switch may be used if your application needs to create space for a large amount of local data. Without this switch an exception could occur if your local data is larger than 4K. The reason for this is that the system uses Structured Exception Handling to manage the enlargement of the stack as and when it is needed, and it does this by establishing guard pages just adjacent to and deeper into the area of stack already committed. If your application attempts to write or read from a place beyond these guard pages, a true exception will occur rather than a signal to the system to create more stack. In experiments the size of the guard pages seems to vary but it is unlikely that it could ever be below 4K which is the normal page size.
Quote/stacksize xxxx
= stack allocation size on start-up or on new threads.
Here you can specify the size in bytes (using hex) of the area of memory which the system should allocate for potential stack use, should the application require it. If you do not use this switch a value of 100000h (1MB) is used by default. At start-up the system merely allocates space in virtual memory and only a small amount is committed at first. The system commits the remainder automatically in 4K chunks if the application needs it.
The value given with this switch applies both to the main thread of the application and also any new threads it makes.
This switch may be used if your application might need more than 1MB of stack space. Specifying more than 1MB does not use memory unnecessarily since the memory is only committed if it is actually used. Normally 1MB is more than sufficient for the stack, but you would need this switch if you were keeping large amounts of local data on the stack in procedures which were highly recursive.
JJ,
I was trying to show that in answer to "how much data can be defined on the stack for a proc before there is a stack overflow", 4096 is not correct, but as it turns out my memory of this was bad. The default commit size for the Microsoft linker appears to be 4096 bytes (and the reserve 1MB), and the (old) code I posted works because it is accessing in byte increments.
It looks to me like your code is trying to access beyond the end of your stack data, a somewhat different problem.
Quote from: MichaelW on April 01, 2009, 07:14:28 PM
JJ,
It looks to me like your code is trying to access beyond the end of your stack data, a somewhat different problem.
Michael,
Thanks. Yes, it had originally another purpose (I simply wanted to see how exactly those LOCALS work...).
Please have a look at the edits I added to my last post, and the revised code. It appears that 3*4096 is the "allowed" size. Don't ask me why :bg
You can define a "large" amount of data on the stack as locals, as long as you do not cross the "guard page" boundaries by more than one page. See the link and the entire thread at:
http://www.masm32.com/board/index.php?topic=2011.msg17056#msg17056
hth,
farrier