The MASM Forum

General => The Campus => Topic started by: ewok on May 01, 2024, 02:15:05 AM

Title: sprintf_s in masm ?
Post by: ewok on May 01, 2024, 02:15:05 AM
Hello, i'm working with dll and i'm trying to recode this c++ to masm:
typedef struct test {
int a;
int b;
int c;
int d;
int e;
char* string;
} hello_struct;

extern "C" DllExport BOOL _cdecl myHelloAPI(hello_struct *hello) {
sprintf_s(hello->string, 12, "Hello World");
return 0;
}

i did my include to msvcrt, and did something like this:
.data
    dataString  db "Hello World", 0
    bufferSize  equ 12
   
    hello_struct     STRUCT
        a1           dd ?
        b2           dd ?
        c3           dd ?
        d4           dd ?
        e5           dd ?
        stringhw     db bufferSize DUP(?)
    hello_struct     ENDS
   
.code
myHelloAPI PROC, dataz:PTR hello_struct
;mov eax, [ebp+dataz]
;push offset dataString ;
;push bufferSize
;push dword ptr [eax+14h] ; Buffer
;call sprintf_s
;add esp, 0Ch
mov eax,0
    ret
myHelloAPI ENDP

I couldn't progress much beside that, forget the code inside the myHelloAPI proc, those are desperate broken attempts with copilot ai, cody ai, and chatgpt. it drive me crazy wasting time on this that looks simple.. .if anyone know how can i reproduce that using  sprintf_s or an equivalent.
Title: Re: sprintf_s in masm ?
Post by: zedd151 on May 01, 2024, 02:34:45 AM
Is myHelloAPI a function that you want to export in a dll?
Could you post the full source of what you have already done?
Include your definition file (xxx.def) as well if that is the case. And batch file to create the dll.
Zip your project and attach it.
Title: Re: sprintf_s in masm ?
Post by: NoCforMe on May 01, 2024, 02:40:26 AM
One thing wrong: in the original structure definition,
    char* string;
is a pointer to a character string, not a string itself as you coded.

You should have this for that member instead:
     stringPtr  dd ?
Title: Re: sprintf_s in masm ?
Post by: HSE on May 01, 2024, 03:02:38 AM
Hi ewok!

I'm not sure sprintf_s function is in Masm32 SDK libraries, but at list will work the more simple "crt_sprintf".

; ¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤
    include \masm32\include\masm32rt.inc
; ¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤

comment * -----------------------------------------------------
                    Build this console app with
                  "MAKEIT.BAT" on the PROJECT menu.
        ----------------------------------------------------- *

    .data
      format1 db " There are %i things", 0
    .code

start:
 
; ¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤

    call main
    inkey
    exit

; ¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤

main proc
    local buffer[248]:byte

    cls
    invoke crt_sprintf, addr buffer, addr format1, 15
    print addr buffer, 13,10

    ret

main endp

; ¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤

end start

Regards, HSE
Title: Re: sprintf_s in masm ?
Post by: ewok on May 01, 2024, 03:07:30 AM
>Is myHelloAPI a function that you want to export in a dll?
Yes!

i couldn't find where to attach a zip so here is it hosted on workupload https://workupload.com/file/G7PsKrNHudD (https://workupload.com/file/G7PsKrNHudD)
Thanks HSE, if i remember i tried also to implement crt_sprintf but it wasn't good.
Title: Re: sprintf_s in masm ?
Post by: NoCforMe on May 01, 2024, 03:11:19 AM
Quote from: ewok on May 01, 2024, 03:07:30 AMi couldn't find where to attach a zip

If you don't see "Click or drag files here to attach them." at the bottom of your post when you're writing it, click "Preview" and it shall magically appear.
Title: Re: sprintf_s in masm ?
Post by: zedd151 on May 01, 2024, 03:15:47 AM
Quote from: ewok on May 01, 2024, 03:07:30 AMi couldn't find where to attach a zip so here is it hosted on workupload
I have attached it to my post here. attachment no longer needed here.
At the bottom of the edit window (when making a post) you should see "Click or drag files here to attach them"

Your source code is below... see my comment there...
Also msvcrt.inc should be before msvcrt.lib, I changed it in the source below
The include (.inc) file should always be before the library (.lib) file

.486
option    casemap :none ; case sensitive


    include \masm32\include\masm32rt.inc
    include \masm32\include\windows.inc
    include \masm32\include\msvcrt.inc  ; Include the C runtime library
    includelib msvcrt.lib  ; Include the C runtime library

_DllMainCRTStartup PROTO :DWORD,:DWORD,:DWORD
myHelloAPI PROTO :DWORD,:DWORD

.data
    dataString  db "Hello World", 0
    bufferSize  equ 12
 
    hello_struct    STRUCT
        a1          dd ?
        b2          dd ?
        c3          dd ?
        d4          dd ?
        e5          dd ?
        stringhw    db bufferSize DUP(?)  ;;; should be a pointer to the string, as NoCforme mentioned above
    hello_struct    ENDS

.code

_DllMainCRTStartup proc instance:DWORD,reason:DWORD,unused:DWORD
    mov    eax,1
    ret
_DllMainCRTStartup endp

myHelloAPI proc q:DWORD,w:DWORD
        ;mov eax, [ebp+dataz]
        ;push offset dataString ;
        ;push bufferSize
        ;push dword ptr [eax+14h] ; Buffer
        ;call sprintf_s
        ;add esp, 0Ch
    mov    eax,0
    ret
myHelloAPI endp

end
Your definition file looks ok.

What are the arguments  "q:DWORD,w:DWORD" for?
At any rate, the dll builds with the batch file in the zip archive. Fix the structure member and put the .inc declaration before the .lib declaration for msvcrt to see if those are the only issues that you have.
Title: Re: sprintf_s in masm ?
Post by: zedd151 on May 01, 2024, 04:01:56 AM
Quote from: HSE on May 01, 2024, 03:02:38 AMI'm not sure sprintf_s function is in Masm32 SDK libraries
Definitely not... sprintf_s (https://learn.microsoft.com/en-us/cpp/c-runtime-library/reference/sprintf-s-sprintf-s-l-swprintf-s-swprintf-s-l?view=msvc-170)  :cool:
Title: Re: sprintf_s in masm ?
Post by: zedd151 on May 01, 2024, 04:18:41 AM
main.asm(38) : error A2006: undefined symbol : sprintf_sseems that sprintf_s is not in msvcrt.dll, at least on Windows 7. There are other alternatives as HSE mentioned above.
Title: Re: sprintf_s in masm ?
Post by: HSE on May 01, 2024, 04:20:29 AM
Quote from: sudoku on May 01, 2024, 04:01:56 AMDefinitely not... sprintf_s (https://learn.microsoft.com/en-us/cpp/c-runtime-library/reference/sprintf-s-sprintf-s-l-swprintf-s-swprintf-s-l?view=msvc-170)  :cool:

Is working here with:
    externdef _imp__sprintf_s:PTR c_msvcrt
    crt_sprintf_s equ <_imp__sprintf>

but I think the linker load that function from a newer msvcrt.lib (version 12), but that have to work in Win7
Title: Re: sprintf_s in masm ?
Post by: zedd151 on May 01, 2024, 04:27:19 AM
Quote from: HSE on May 01, 2024, 04:20:29 AMIs working here with:
    externdef _imp__sprintf_s:PTR c_msvcrt
    crt_sprintf_s equ <_imp__sprintf>

Okay, it does work here with that, on Windows 7. .
My mistake. It does not run in Windows 7 even with the exterdef, I was still using "printf" in the test code, forgot to change it back to sprintf_s.   :rolleyes:

So apparently sprintf_s is only on later versions of msvcrt than what is in Windows 7.

What version of Windows are you running, ewok?
Title: Re: sprintf_s in masm ?
Post by: NoCforMe on May 01, 2024, 05:03:58 AM
Hey, here's an idea: forget about sprintf(). Just use wsprintf() instead, which is a standard Win32 function, that (probably) does what you need it to do (i.e, format a string to a memory buffer).

Documentation here (https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-wsprintfa?redirectedfrom=MSDN).
Title: Re: sprintf_s in masm ?
Post by: Vortex on May 01, 2024, 05:25:52 AM
Hello,

sprintf_s works on Windows 7. Attached is a quick example. The problem is that the import library msvcrt.lib built by the Masm32 package does not export this function :

\PellesC\bin\podump.exe /SYMBOLS G:\masm32\lib\msvcrt.lib | findstr sprintf
0003 00000000 SECT1  notype ()   external     | _sprintf
0006 00000000 SECT2  notype      external     | __imp__sprintf
0003 00000000 SECT1  notype ()   external     | _vsprintf
0006 00000000 SECT2  notype      external     | __imp__vsprintf

An easy solution is to rebuild the library.

.386
.model flat, stdcall
option casemap :none

include     \masm32\include\windows.inc
include     \masm32\include\user32.inc
include     \masm32\include\kernel32.inc
include     \masm32\include\masm32.inc

includelib  \masm32\lib\user32.lib
includelib  \masm32\lib\kernel32.lib
includelib  \masm32\lib\masm32.lib
includelib  msvcrt.lib

sprintf_s PROTO C :DWORD,:VARARG

SIZE_OF_BUFFER equ 64

.data

a1  db 'This is',0
a2  db 'a test.',0
a3  db '%s %s',0

.data?

buffer  db SIZE_OF_BUFFER dup(?)

.code

start:

    invoke  sprintf_s,ADDR buffer,\
            SIZE_OF_BUFFER,\
            ADDR a3,ADDR a1,ADDR a2

    invoke  StdOut,ADDR buffer

    invoke  ExitProcess,0

END start
Title: Re: sprintf_s in masm ?
Post by: zedd151 on May 01, 2024, 05:28:39 AM
Oh, Vortex.  :thumbsup:
I was correct in my initial assumption that hutch missed that one in the include file for msvcrt. Bravo.
I had not thought about rebuilding the library, leading me to believe it did not exist in Windows 7 msvcrt.dll.

Under the hood, msvcrt calls "vsprintf_s" in case anyone is interested

(https://i.postimg.cc/wRM7NH3p/untitled.png) (https://postimg.cc/wRM7NH3p)
Title: Re: sprintf_s in masm ?
Post by: jj2007 on May 01, 2024, 06:11:10 AM
Works like a charm on Windows 10:
include \masm32\MasmBasic\MasmBasic.inc
.DATA?
buffer db 10 dup(?)
  Init
  Dll "msvcrt"
  Declare sprintf_s, C:?
  Print Str$("%i characters copied", sprintf_s(addr buffer, sizeof buffer, "%i", 123456789))
  MsgBox 0, offset buffer, "Result:", MB_OK
EndOfCode
Title: Re: sprintf_s in masm ?
Post by: ewok on May 02, 2024, 01:31:47 AM
Quote from: sudoku on May 01, 2024, 03:15:47 AMWhat are the arguments  "q:DWORD,w:DWORD" for?
That was part of desperate coding, at a point i started doing non-sense code/test

Quote from: NoCforMe on May 01, 2024, 02:40:26 AMOne thing wrong: in the original structure definition,
    char* string;
is a pointer to a character string, not a string itself as you coded.

You should have this for that member instead:
     stringPtr  dd ?
I thought of a struct was like an array, so it's just pointers inside a struct? i'll have to fix that.

Quote from: sudoku on May 01, 2024, 04:27:19 AMWhat version of Windows are you running, ewok?
win 10

Quote from: Vortex on May 01, 2024, 05:25:52 AMThe problem is that the import library msvcrt.lib built by the Masm32 package does not export this function
If one day there would be a new release of masm that could be a good addition!
Thanks also for your custom lib. I'm gonna fix my struct and try that lib
Title: Re: sprintf_s in masm ?
Post by: NoCforMe on May 02, 2024, 02:40:39 AM
Quote from: ewok on May 02, 2024, 01:31:47 AM
Quote from: NoCforMe on May 01, 2024, 02:40:26 AMOne thing wrong: in the original structure definition,
    char* string;
is a pointer to a character string, not a string itself as you coded.

You should have this for that member instead:
    stringPtr  dd ?
I thought of a struct was like an array, so it's just pointers inside a struct? i'll have to fix that.
A structure (C or assembly language, doesn't matter) is a collection which can contain anything (including other structures).

That field could be an actual string, as you had originally coded it, but in this case it's a pointer to a string instead (because the C definition was char *string). You can have strings--otherwise known as arrays of characters--in a structure; in fact, several Win32 structures, like LOGFONT (https://learn.microsoft.com/en-us/windows/win32/api/Wingdi/ns-wingdi-logfonta), have them.

If the C definition had been char string[some_size], then you'd want to put a string (array) there instead.
Title: Re: sprintf_s in masm ?
Post by: Vortex on May 02, 2024, 05:17:04 AM
Quote from: sudoku on May 01, 2024, 05:28:39 AMUnder the hood, msvcrt calls "vsprintf_s" in case anyone is interested

Hi sudoku,

msvcrt.inc from the Masm32 package defines only this version :

externdef _imp__vsprintf:PTR c_msvcrt
    crt_vsprintf equ <_imp__vsprintf>
Title: Re: sprintf_s in masm ?
Post by: zedd151 on May 02, 2024, 05:19:52 AM
I have rebuilt my msvcrt.lib here, after adjusting the include file. Thanks for that advice.

From that quote in your post, it should have been
QuoteUnder the hood, msvcrt calls "vsprintf_s" when the user calls "sprintf_s" in case anyone is interested
as illustrated by the image in that post. I thought it was obvious in the context of the thread, so I felt no need to elaborate. :smiley:
Title: Re: sprintf_s in masm ?
Post by: alCoPaUL on May 16, 2024, 12:48:12 PM
sounds like sprintf_s is for sanitizing db/characters/strings..

https://pastebin.com/m54wj4QL (https://pastebin.com/m54wj4QL)

replace the ' in the last line with ; to assemble and link...
Title: Re: sprintf_s in masm ?
Post by: alCoPaUL on May 16, 2024, 05:45:13 PM
you don't need to specify the size of the data/string/character to be processed.

and this is good for dynamic data/string/character processing. like you have a fixed buffer of zeroes in your code and you can just use that with the sprintf_s sanitizer and it will cut on the first ,0 that it encounters, whatever size is that provided that it won't exceed the buffer size..
Title: Re: sprintf_s in masm ?
Post by: zedd151 on May 17, 2024, 02:40:01 AM
Quote from: alCoPaUL on May 16, 2024, 05:45:13 PMyou don't need to specify the size of the data/string/character to be processed.

and this is good for dynamic data/string/character processing. like you have a fixed buffer of zeroes in your code and you can just use that with the sprintf_s sanitizer and it will cut on the first ,0 that it encounters, whatever size is that provided that it won't exceed the buffer size..
Now suppose you have a case where the string is NOT zero terminated??? It would be good to specify the string size in that instance, no? Else as you say 'it will cut on the first ",0" that it encounters' ( the quotes inferred by your post but not explicitly stated there). Where would that be (the first ",0"), in the case of a string that is not zero terminated?? I see a possible buffer overflow (and program crash) in that case.
Title: Re: sprintf_s in masm ?
Post by: alCoPaUL on May 17, 2024, 03:32:17 AM
Quote from: sudoku on May 17, 2024, 02:40:01 AM
Quote from: alCoPaUL on May 16, 2024, 05:45:13 PMyou don't need to specify the size of the data/string/character to be processed.

and this is good for dynamic data/string/character processing. like you have a fixed buffer of zeroes in your code and you can just use that with the sprintf_s sanitizer and it will cut on the first ,0 that it encounters, whatever size is that provided that it won't exceed the buffer size..
Now suppose you have a case where the string is NOT zero terminated??? It would be good to specify the string size in that instance, no? Else as you say 'it will cut on the first ",0" that it encounters' ( the quotes inferred by your post but not explicitly stated there). Where would that be (the first ",0"), in the case of a string that is not zero terminated?? I see a possible buffer overflow (and program crash) in that case.

but my example has a 999426 zeroes as a buffer..

you put 76 bytes on a 999426 zeroes as a buffer, printf will just take the 76 bytes and the first zero (thanks to sprintf_s processing the buffer as "%s")
and say your code still have to process 6675 bytes before printing it, printf will just take 6675 bytes and the first zero, again thanks to sprintf_s.

dynamic data processing will be stifled by doing strlen() everytime you pass a buffer..

and it's allowed, as you can see (if you can assemble and link my source)..
Title: Re: sprintf_s in masm ?
Post by: zedd151 on May 17, 2024, 03:47:23 AM
Quote from: alCoPaUL on May 17, 2024, 03:32:17 AMand it's allowed, as you can see (if you can assemble and link my source)..

b db 999426 DUP(0)Nah, I'll pass. One megabyte-ish in .data ??
If you have a smaller example, I will consider it.

You can post your source here, did you know? It would make it much easier if anyone is interested in it, rather than having to copy and paste your pastebin link (its not clickable - btw) and go to a third party site to obtain your code.
Title: Re: sprintf_s in masm ?
Post by: alCoPaUL on May 17, 2024, 04:19:03 AM
Quote from: sudoku on May 17, 2024, 03:47:23 AM
Quote from: alCoPaUL on May 17, 2024, 03:32:17 AMand it's allowed, as you can see (if you can assemble and link my source)..

b db 999426 DUP(0)Nah, I'll pass. One megabyte-ish in .data ??
If you have a smaller example, I will consider it.

You can post your source here, did you know? It would make it much easier if anyone is interested in it, rather than having to copy and paste your link (its not clickable - btw) and go to a third party site to obtain your code.

it's the era of datasscience..

i was supposed to post that on this forum the time i finished that but the forum was down..

again, sprintf_s formats that to %s and cuts the rest of the zeroes and finally you pass the buffer (the string data and ,0) to printf, printing that to %s.

so what's happening is double sanitization to %s..

clear from meme buffer overflow..

and when the code is using the 999426 zeroes as a buffer and your code is manipulating data to information and the maxed string that it manipulates is 9999 bytes, it's always underflowing...

but when it assembles the final string by combining 9999 and 9999 and 9999, it still underflowing the buffer coz that's only 29997 bytes inside a 999426 zero buffer...

and sprintf_s only takes the 29997 bytes & ,0 from that humongous buffer to pass to printf....

Title: Re: sprintf_s in masm ?
Post by: zedd151 on May 17, 2024, 04:21:17 AM
Quote from: alCoPaUL on May 17, 2024, 04:19:03 AMi was supposed to post that on this forum the time i finished that but the forum was down..
Ah, okay.
Title: Re: sprintf_s in masm ?
Post by: alCoPaUL on May 17, 2024, 05:20:25 AM
source code & essential files in the zip..
Title: Re: sprintf_s in masm ?
Post by: alCoPaUL on May 17, 2024, 09:46:53 AM
masm32 version

; sprintf_s + printf
; by alCoPaUL [GIMO]
; 5/16/2024 NYC
;
; ml.exe printf_s32.asm /link /subsystem:console /defaultlib:32msvcrt.lib /entry:start [or /entry:main (will assemble and link with warning)]
;
.686
.model flat,stdcall
extern printf:proc
extern sprintf_s:proc
.code
start:
main proc
enter 0,0
push offset x
push offset i
push f
push offset b
call sprintf_s
push eax
push offset b
push offset i
call printf
leave
ret
main endp
.data
f equ 10               ; Scale This Length
x db 'Thereupon',0     ; Manually Or Use StrLen()
b db 999426 DUP(0)     ; As Usual, Humongous Buffer
i db 25h,73h,0
end start
Title: Re: sprintf_s in masm ?
Post by: zedd151 on May 17, 2024, 12:19:08 PM
Tried to run your executable
(https://i.postimg.cc/63YS30B5/untitled.png)
I am quite sure that the "32msvcrt.lib" that you are using is not for msvcrt.dll, but rather a dll from the vcruntime dll shown in the message box. A non-starter for me.
Title: Re: sprintf_s in masm ?
Post by: alCoPaUL on May 17, 2024, 12:30:20 PM
Quote from: sudoku on May 17, 2024, 12:19:08 PMTried to run your executable
(https://i.postimg.cc/63YS30B5/untitled.png)

the library (.lib) is from visual studio 2010, needs the appropriate redistributable runtimes..
Title: Re: sprintf_s in masm ?
Post by: jj2007 on May 17, 2024, 03:09:59 PM
Dll hell in the 21st Century, isn't it lovely?
C:\Windows\System32\msvcrt.dll
C:\Windows\System32\msvcr100_clr0400.dll
C:\Windows\System32\msvcr120_clr0400.dll
C:\Windows\System32\msvcrt20.dll
C:\Windows\System32\msvcrt40.dll
C:\Windows\System32\msvcr71.dll
C:\Windows\System32\msvcr80.dll
C:\Windows\System32\msvcr100.dll

There is a simple solution:
- find out what's the latest msvcr*.dll
- use LoadLibrary and GetProcAddress
Title: Re: sprintf_s in masm ?
Post by: alCoPaUL on May 17, 2024, 04:37:40 PM
Quote from: jj2007 on May 17, 2024, 03:09:59 PMDll hell in the 21st Century, isn't it lovely?
C:\Windows\System32\msvcrt.dll
C:\Windows\System32\msvcr100_clr0400.dll
C:\Windows\System32\msvcr120_clr0400.dll
C:\Windows\System32\msvcrt20.dll
C:\Windows\System32\msvcrt40.dll
C:\Windows\System32\msvcr71.dll
C:\Windows\System32\msvcr80.dll
C:\Windows\System32\msvcr100.dll

There is a simple solution:
- find out what's the latest msvcr*.dll
- use LoadLibrary and GetProcAddress

m$ tactics - either for the $$ or from old school m$ guises who don't want all the os versions to be swiped by malware so they grouped the default development software for the os via different redistributable runtime files..

anyway, attached is similar code that i previously posted with 3 C Standard lib functions but it's written in another assembler. you can easily format this to masm32 assembly src file using the previous code that i shared as template..

splitted the zip (file size restrictions) so just rename faxx.jpg to faxx.z01...
Title: Re: sprintf_s in masm ?
Post by: alCoPaUL on May 18, 2024, 06:13:42 AM
StrLen() must be sanitized by checking if it's FFFFFFFEh to get the Maxed Capacity.

But the Primacy of the 999426 DUP(0) buffer is upheld so StrLen(data), by practice, is wayyy smaller than the function's Maxed Capacity.

And since I posted the 64-bit version, now it's clear why the size of the buffer in 64-bit sprintf_s can be ommited coz the maxed capacity of StrLen(data), to suffice the length of the inputted data to sprintf_s, Ideally, is effing FFFFFFFFFFFFFFFEh bytes.