The MASM Forum

General => The Campus => Topic started by: wanker742126 on October 30, 2024, 07:30:05 PM

Title: How to Call StringCchCat in Win64 Assembly?
Post by: wanker742126 on October 30, 2024, 07:30:05 PM
Hello everyone,

I'm currently writing a program in Win64 assembly language with the intention of using the StringCchCat function to concatenate two strings. However, I'm encountering the following error during compilation:
tststr.obj : error LNK2019: unresolved external symbol StringCchCat referenced in function main 
tststr.exe : fatal error LNK1120: 1 unresolved externals
It seems like StringCchCat cannot be located. I would like to ask:
    Do I need to manually link to a specific .lib file?
    Are there any particular settings required for using this function in assembly?
Thank you very much for any advice!

My source is tststr.asm:
INCLUDE \masm64\include64\masm64rt.inc

StringCchCat PROTO :QWORD,:DWORD,:QWORD

.DATA
hInst   DQ      ?
szTitle DB      "Test StringCb"
str1    DB      "I learn ",40h DUP (0)
str2    DB      "assembly language.",0

.CODE
main    PROC
        invoke  StringCchCat,ADDR str1,SIZEOF str1,ADDR str2
        invoke  MessageBox,0,ADDR str1,ADDR szTitle,MB_OK
        call    ExitProcess
main    ENDP
END

Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: HSE on October 30, 2024, 08:31:35 PM
You can search in /include64 files.

Probably equate of import is vc_StringCchCat.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: TimoVJL on October 30, 2024, 08:47:36 PM
It's a C macro:
STRSAFEAPI
    StringCchCatA(
            _Inout_updates_(cchDest) _Always_(_Post_z_) STRSAFE_LPSTR pszDest,
            _In_ size_t cchDest,
            _In_ STRSAFE_LPCSTR pszSrc)
{
    HRESULT hr;
    size_t cchDestLength;

    hr = StringValidateDestAndLengthA(pszDest,
            cchDest,
            &cchDestLength,
            STRSAFE_MAX_CCH);

    if (SUCCEEDED(hr))
    {
        hr = StringCopyWorkerA(pszDest + cchDestLength,
                cchDest - cchDestLength,
                NULL,
                pszSrc,
                STRSAFE_MAX_LENGTH);
    }

    return hr;
}
Worker functions are in strsafe.lib
pFile Data Description Value
00000044 00000019 Number of Symbols
00000048 000006B2 Offset to Header StringVPrintf_lWorkerA
0000004C 000006B2 Offset to Header StringVPrintf_lWorkerW
00000050 00003E08 Offset to Header ??_C@_00CNPNBAHC@?$AA@
00000054 00003E08 Offset to Header ??_C@_11LOCGONAA@?$AA?$AA@
00000058 00003E08 Offset to Header StringCopyWorkerA
0000005C 00003E08 Offset to Header StringCopyWorkerW
00000060 00003E08 Offset to Header StringExHandleFillBehindNullA
00000064 00003E08 Offset to Header StringExHandleFillBehindNullW
00000068 00003E08 Offset to Header StringExHandleOtherFlagsA
0000006C 00003E08 Offset to Header StringExHandleOtherFlagsW
00000070 00003E08 Offset to Header StringExValidateDestA
00000074 00003E08 Offset to Header StringExValidateDestAndLengthA
00000078 00003E08 Offset to Header StringExValidateDestAndLengthW
0000007C 00003E08 Offset to Header StringExValidateDestW
00000080 00003E08 Offset to Header StringExValidateSrcA
00000084 00003E08 Offset to Header StringExValidateSrcW
00000088 00003E08 Offset to Header StringLengthWorkerA
0000008C 00003E08 Offset to Header StringLengthWorkerW
00000090 00003E08 Offset to Header StringVPrintfWorkerA
00000094 00003E08 Offset to Header StringVPrintfWorkerW
00000098 00003E08 Offset to Header StringValidateDestA
0000009C 00003E08 Offset to Header StringValidateDestAndLengthA
000000A0 00003E08 Offset to Header StringValidateDestAndLengthW
000000A4 00003E08 Offset to Header StringValidateDestW
000000A8 00003E08 Offset to Header UnalignedStringLengthWorkerW
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: wanker742126 on October 30, 2024, 11:41:12 PM
Thank you, HSE and TimoVJL, for your response.
I couldn't find strsafe.lib in masm64\lib64. Where can I download it?
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: stoo23 on October 31, 2024, 12:20:29 AM
QuoteI couldn't find strsafe.lib in masm64\lib64
No neither could I,.. I have searched our server and We do Not have a copy of it anywhere,... Should We ???  :undecided:
It certainly Isn't in the copies of MASM64 or the SDK ??
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: TimoVJL on October 31, 2024, 12:36:05 AM
Quote from: wanker742126 on October 30, 2024, 11:41:12 PMThank you, HSE and TimoVJL, for your response.
I couldn't find strsafe.lib in masm64\lib64. Where can I download it?
from Windows SDK
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: stoo23 on October 31, 2024, 12:51:39 AM
Quotefrom Windows SDK
Yeah ??? ... Which Windows SDK Timo ???

It's NOT anywhere in the copy of MASM64 We have on the server ??
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: zedd151 on October 31, 2024, 01:00:26 AM
Quote from: stoo23 on October 31, 2024, 12:20:29 AM
QuoteI couldn't find strsafe.lib in masm64\lib64
No neither could I,.. I have searched our server and We do Not have a copy of it anywhere,... Should We ??? 
Stoo, the Windows SDK (software development kit) is a Microsoft product, which has a set of its own libraries (*.lib files). Hence are a separate entity from the Masm64 SDK that we offer. I don't think we should duplicate it within the Masm64 SDK.

This is another example of why it is difficult for some users to use Visual Studio with Masm32 SDK and/or Masm64 SDK.
They have the impression that Masm64 SDK has libraries and include files that it simply does not, but are either in the Visual Studio package or in one of the Microsoft Windows SDK's. (Microsoft products)

I do agree however that the Masm64 SDK is missing some include files (*.inc) that it should have - plus some incomplete or erroneous include files, which should not be confused with Microsofts own include files (*.h) which either the Visual Studio package or one of the Windows SDK packages does have.

 :smiley:
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: C3 on October 31, 2024, 01:19:48 AM
Quote from: stoo23 on October 31, 2024, 12:51:39 AM
Quotefrom Windows SDK
Yeah ??? ... Which Windows SDK Timo ???

It's NOT anywhere in the copy of MASM64 We have on the server ??

Google gives this: Microsoft Windows SDK (https://developer.microsoft.com/en-us/windows/downloads/windows-sdk/)

Also included in Microsoft Visual Studio, one reason to use, for me.

Quote from: zedd151 on October 31, 2024, 01:00:26 AM
QuoteI do agree however that the Masm64 SDK is missing some include files (*.inc) that it should have - plus some incomplete or erroneous include files, which should not be confused with Microsofts own include files (*.h) which either the Visual Studio package or one of the Windows SDK packages does have.

Thats what is really needed. I just add stuff from Microsoft headers (*.h) to include files (*.inc) when I need something what is not already there.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: stoo23 on October 31, 2024, 01:36:22 AM
Thanks for the Above link and info' guys  :thumbsup:
I'll try and Place that Windows resource somewhere usefully appropriate  :wink2:  :thumbsup:  Perhaps in the Reference section  :smiley:

QuoteStoo, the Windows SDK (software development kit) is a Microsoft product, which has a set of its own libraries (*.lib files). Hence are a separate entity from the Masm64 SDK that we offer. I don't think we should duplicate it within the Masm64 SDK.
Yep understood, I was just commenting on the fact that We did not have a copy of that file Anywhere on our server, OR in the Files I have locally from Hutch.

As suggested, thanks for the info'  :azn:
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: HSE on October 31, 2024, 03:12:57 AM
Don't pay attention to files names, just search function name.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: BugCatcher on October 31, 2024, 03:56:45 AM
strsafe.h
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: TimoVJL on October 31, 2024, 05:04:40 AM
Quote from: BugCatcher on October 31, 2024, 03:56:45 AMstrsafe.h
MS documents don't always tell whole truth, so we had to give additional info.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: Vortex on October 31, 2024, 05:18:55 AM
Hello wanker742126,

You don't have to download a lot of files to obtain the header file strsafe.h, it comes with Pelles C :

\PellesC\Include\strsafe.h
STRSAFEAPI StringCchCatA(STRSAFE_LPSTR pszDest, size_t cchDest, STRSAFE_LPCSTR pszSrc)
{
    size_t cchDestLength;
    HRESULT hr = _StringValidateDestAndLengthA(pszDest, cchDest, &cchDestLength, STRSAFE_MAX_CCH);
    if (SUCCEEDED(hr)) {
        hr = _StringCopyWorkerA(pszDest + cchDestLength, cchDest - cchDestLength, NULL, pszSrc, STRSAFE_MAX_LENGTH);
    }
    return hr;
}

STRSAFEWORKERAPI _StringCopyWorkerA(STRSAFE_LPSTR pszDest, size_t cchDest, size_t * pcchNewDestLength, STRSAFE_PCNZCH pszSrc, size_t cchToCopy) {
    HRESULT hr = S_OK;
    size_t cchNewDestLength = 0;
    while (cchDest && cchToCopy && (*pszSrc != '\0')) {
        *pszDest++ = *pszSrc++;
        cchDest--;
        cchToCopy--;

        cchNewDestLength++;
    }
    if (cchDest == 0) {
        pszDest--;
        cchNewDestLength--;
        hr = STRSAFE_E_INSUFFICIENT_BUFFER;
    }
    *pszDest = '\0';
    if (pcchNewDestLength) {
        *pcchNewDestLength = cchNewDestLength;
    }
    return hr;
}

Obtaining Pelles C :

http://www.smorgasbordet.com/pellesc/download.htm (http://www.smorgasbordet.com/pellesc/download.htm)
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: wanker742126 on October 31, 2024, 09:12:01 AM
Hi everyone, thanks for your help, I finally solved this problem.

I followed TimoVJL's advice and downloaded the Windows SDK from the link provided by C3, then installed it.

The strsafe.lib file was located in the
C:\Program Files (x86)\Windows Kits\10\Lib\10.0.26100.0\um\x64 folder. However, it's not an import library, and I took quite a detour before realizing this.

Using DUMPBIN /ALL, I discovered that StringCchCatA was not present in the library. After reading the contents of strsafe.h provided by TimoVJL, Vortex and the file located at
C:\Program Files (x86)\Windows Kits\10\Include\10.0.26100.0\shared\strsafe.h, I realized StringCchCatA is not included in strsafe.lib. It's more like a macro, so I wrote one using assembly language syntax. Here's the source file :

INCLUDE \masm64\include64\masm64rt.inc
includelib "c:\Program Files (x86)\Windows Kits\10\Lib\10.0.26100.0\um\x64\strsafe.lib"
STRSAFE_MAX_CCH    EQU 2147483647
STRSAFE_MAX_LENGTH EQU STRSAFE_MAX_CCH-1
StringValidateDestAndLengthA PROTO :QWORD,:DWORD,:QWORD,:DWORD
StringCopyWorkerA            PROTO :QWORD,:DWORD,:DWORD,:QWORD,:DWORD
.DATA
hInst   DQ      ?
szTitle DB      "Test StringCchCat",0
str1    DB      "I learn ",40h DUP (0)
str2    DB      "assembly language.",0

.CODE
StringCchCatA PROC pszDest:QWORD,cchDest:DWORD,pszSrc:QWORD
        LOCAL   cchDestLength:DWORD
        invoke  StringValidateDestAndLengthA,pszDest,cchDest,ADDR cchDestLength,STRSAFE_MAX_CCH
        cmp     eax,0
        jl      exit
        sub     r10,r10
        mov     r10d,cchDestLength
        mov     rcx,pszDest
        add     rcx,r10
        mov     edx,cchDest
        sub     edx,cchDestLength
        invoke  StringCopyWorkerA,rcx,edx,0,pszSrc,STRSAFE_MAX_LENGTH
exit:   ret
StringCchCatA ENDP
main    PROC
        invoke  StringCchCatA,ADDR str1,SIZEOF str1,ADDR str2
        invoke  MessageBox,0,ADDR str1,ADDR szTitle,MB_OK
        call    ExitProcess
main    ENDP
END

Please give me some advice for improvement, thank you for your help.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: ognil on October 31, 2024, 04:04:44 PM
The same but shorter... :smiley: 

include  \masm64\include64\masm64rt.inc

.data
szTitle DB      "Test StringCchCat",0
str1    DB      "I learn ",40h DUP (0)
str2    DB      "assembly language.",0

.code
main    proc
        lea    r8,str1
        lea    r9,str2
@@: 
        cmp    byte ptr[r8],0 
        lea    r8,[r8+1] 
        jne    @b
@@:
        mov    al,byte ptr[r9]
        add    r9,1
        mov    byte ptr[r8-1],al 
        add    r8,1
        test    al,al
        jne    @b
        invoke  MessageBox,0,addr str1,addr szTitle,MB_OK
        call    ExitProcess
main    endp
end
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: wanker742126 on November 01, 2024, 12:47:29 AM
Hi, ognil:
The reason I called StringCchCatA was to avoid buffer overflow security issues, and since strsafe.lib wasn't available, I raised this question. Thank you for your concise and elegant code.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: zedd151 on November 01, 2024, 01:06:05 AM
Quote from: wanker742126 on October 31, 2024, 09:12:01 AMPlease give me some advice for improvement...
* Keep it simple  :biggrin: . Even the legacy API's will work fine for such a simple example.
A large buffer and a simple test of the concatenated string lengths against the buffer size will prevent buffer overflow. Keep in mind that the buffer needs to be at least one byte (assuming ascii strings) longer than the length of the concatenated strings for proper zero termination of the buffer. I should have wrote that into this code, my bad.  :tongue:

include \masm64\include64\masm64rt.inc

.data

    szTitle db "Test lstrcat", 0
    buffer  db 256 dup (0)
   
    str1 db "I learn ", 0
    str2 db "assembly language.", 0
    slen dd 0
   
.code

main    proc
        invoke lstrlen, addr str1  ;; get length of str1
        mov slen, eax
        invoke lstrlen, addr str2  ;;  add length of str2
        add eax, slen
        cmp eax, sizeof buffer      ;; compare length with buffer size
        jg buffererror              ;; if length of strings greater than buffer size, show error message
       
        invoke lstrcpy, addr buffer, addr str1
        invoke lstrcat, addr buffer, addr str2
        invoke MessageBox, 0, addr buffer, addr szTitle, MB_OK
        jmp xit
      buffererror:
        fn MessageBox, 0, "buffer overflow", addr szTitle, 0
      xit:
        invoke ExitProcess, 0
main    endp
end

Just use the included batch file to build this example.  :smiley:
Edit: I just noticed that I used "eax" here, which should be fine in this context... the buffer size is well below the limits of eax.  :mrgreen:  Others may find that objectionable.   :tongue:
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: wanker742126 on November 01, 2024, 01:54:11 AM
zedd151:

Indeed, I got a bit fixated on using StringCchCatA. Your code certainly prevents buffer overflow, and it's simple and clear. However, during my search, I also learned more about the nature of strsafe.lib and how to call StringCchCatA and other StringCch* functions.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: zedd151 on November 01, 2024, 01:57:36 AM
It's all good. More knowledge is always a good thing. But I like the more simple solutions myself.

Happy coding.  :thumbsup:
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: wanker742126 on November 01, 2024, 02:10:42 AM
Who doesn't love simple solutions? It's just that multiple books mention, and even MSDN points out, that functions like lstrlen, lstrcpy, wcscat, etc., have security concerns, recommending the use of StringCch* functions instead. This sparked my curiosity. In fact, I've traced StringCchCatA, and it essentially just checks that the buffer size is greater than the length of the two strings being concatenated.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: zedd151 on November 01, 2024, 02:21:35 AM
I just used lstrlen lstrcpy and lstrcat for demonstration of a simple approach to the problem, using widely available API's.
Normally I would "roll my own" algorithms to do the same functionality, and ognil also showed some of that type of code (replacing functionality of lstrcat). I would put that code into its own procedure though, for reusability in other programs.  :smiley:



Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: wanker742126 on November 01, 2024, 02:27:06 AM
I learned from this experience that "blindly trusting books can be worse than having no books at all." :biggrin:
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: TimoVJL on November 01, 2024, 02:41:10 AM
Have to know context, like using untrusted data.
MSDN don't tell all details of functions.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: zedd151 on November 01, 2024, 03:16:12 AM
Quote from: zedd151 on November 01, 2024, 02:21:35 AMI just used lstrlen lstrcpy and lstrcat for demonstration of a simple approach to the problem, using widely available API's.
Normally I would "roll my own" algorithms to do the same functionality ... I would put that code into its own procedure though, for reusability in other programs.  :smiley:

Here is an example of "self rolled" functions to get string length, concatenate two strings, copy string to a buffer:

stringcat proc dst:qword, src:qword  ;; rcx = dst (destination), first argument  ;; rcx = src  (source), second argument
    dec rcx
  findend:
    inc rcx
    cmp byte ptr [rcx], 0
    jnz findend
    dec rdx
    dec rcx
  copyit:
    inc rdx
    inc rcx
    mov al, [rdx]
    mov [rcx], al
    cmp byte ptr [rdx], 0
    jnz copyit
    mov byte ptr [rcx], 0  ;; ensure zero termination.
    ret
stringcat endp

stringcpy proc dst:qword, src:qword  ;; rcx = dst (destination), first argument  ;; rdx = src  (source), second argument
    dec rdx
    dec rcx
  copyit:
    inc rdx
    inc rcx
    mov al, [rdx]
    mov [rcx], al
    cmp byte ptr [rdx], 0
    jnz copyit
    mov byte ptr [rcx], 0  ;; ensure zero termination.
    ret
stringcpy endp

stringlen proc src:qword  ;; rcx = src (source), first and only argument
    dec rcx
    xor rax, rax
    dec rax
  countbyte:
    inc rax
    inc rcx
    cmp byte ptr [rcx], 0
    jnz countbyte
    ret
stringlen endp

Full source for attached example:

include \masm64\include64\masm64rt.inc

.const
    str1        db "Hello", 0
    strspace    db " ", 0
    str2        db "World!", 0Dh, 0Ah, 0
    str3        db "64 bit assembly example", 0
    strerror    db "Buffer overflow", 0
.data
    str1len    dq 0
    spacelen    dq 0
    str2len    dq 0
    buffer      db 128 dup (0)
   
.code

start proc
    invoke stringlen, addr str1
    mov str1len, rax            ;; length of str1
    invoke stringlen, addr strspace
    mov spacelen, rax            ;; length of strspace
    invoke stringlen, addr str2
    mov str2len, rax            ;; length of str2
    invoke stringlen, addr str3 ;; length of str3 (in rax after call, not in a variable)
   
    add rax, str1len         
    add rax, spacelen         
    add rax, str2len            ;; add all the string lengths
    cmp rax, sizeof buffer      ;; compare to buffer size
    jge buffererr              ;; jump if greater or equal to display error if failed

    invoke stringcpy, addr buffer, addr str1  ;; else display message box with all strings concatenated
    invoke stringcat, addr buffer, addr strspace
    invoke stringcat, addr buffer, addr str2
    invoke stringcat, addr buffer, addr str3
    fn MessageBox, 0, addr buffer, 0, 0
   
    jmp xit                      ;; jump over error message if succesful
  buffererr:
    invoke MessageBox, 0, addr strerror, 0, 0


  xit:
    invoke ExitProcess, 0
start endp

stringcat proc dst:qword, src:qword  ;; rcx = dst (destination), first argument  ;; rcx = src  (source), second argument
    dec rcx
  findend:
    inc rcx
    cmp byte ptr [rcx], 0
    jnz findend
    dec rdx
    dec rcx
  copyit:
    inc rdx
    inc rcx
    mov al, [rdx]
    mov [rcx], al
    cmp byte ptr [rdx], 0
    jnz copyit
    mov byte ptr [rcx], 0  ;; ensure zero termination.
    ret
stringcat endp

stringcpy proc dst:qword, src:qword  ;; rcx = dst (destination), first argument  ;; rdx = src  (source), second argument
    dec rdx
    dec rcx
  copyit:
    inc rdx
    inc rcx
    mov al, [rdx]
    mov [rcx], al
    cmp byte ptr [rdx], 0
    jnz copyit
    mov byte ptr [rcx], 0  ;; ensure zero termination.
    ret
stringcpy endp

stringlen proc src:qword  ;; rcx = src (source), first and only argument
    dec rcx
    xor rax, rax
    dec rax
  countbyte:
    inc rax
    inc rcx
    cmp byte ptr [rcx], 0
    jnz countbyte
    ret
stringlen endp

end

edit: added code to ensure zero termination in the 'stringcpy' and 'stringcat ' functions
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: sinsi on November 01, 2024, 04:38:34 AM
Slightly optimised...just as an exercise :biggrin:
stringcat proc dst:qword, src:qword  ;; rcx = dst (destination), first argument  ;; rcx = src  (source), second argument
@0: inc rcx
cmp byte ptr [rcx-1],0
jnz @0
@1: mov al,[rdx]
        mov [rcx-1],al
inc rdx
inc rcx
test al,al
jnz @1
        ret
stringcat endp

stringcpy proc dst:qword, src:qword  ;; rcx = dst (destination), first argument  ;; rdx = src  (source), second argument
@0:     mov al,[rdx]
        mov [rcx],al
inc rdx
inc rcx
        test al,al
jnz @0
        ret
stringcpy endp

stringlen proc src:qword  ;; rcx = src (source), first and only argument
or rax,-1
@@: inc rax
cmp byte ptr [rcx+rax],0
jnz @b
@@: ret
stringlen endp
Since the procs don't call any other proc you could turn off frame generation, but I'm not sure how to disable then enable it with the MASM64 SDK  :sad:
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: zedd151 on November 01, 2024, 04:46:16 AM
Hi sinsi, yes the algos I presented could be optimized, but that would not be Campus material. Better suited for the Laboratory imo.  :tongue:

Also you should ensure zero termination in both stringcat and stringcpy (by placing a zero byte when finished there) since the buffer may already have data in it from a possible previous use of the buffer. correction, my mistake.  :tongue:

:biggrin:
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: sinsi on November 01, 2024, 04:50:02 AM
Quote from: zedd151 on November 01, 2024, 04:46:16 AMHi sinsi, yes the algos I presented could be optimized, but that would not be Campus material. Better suited for the Laboratory.  :tongue:
The Campus is for learning, maybe someone can learn from this?

Quote from: zedd151 on November 01, 2024, 04:46:16 AMAlso you should ensure zero termination in both stringcat and stringcpy (by placing a zero byte when finished there) since the buffer may already have data in it from a possible previous use of the buffer.  :icon_idea:
All characters are copied, even the terminating NULL, then end-of-string is tested for.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: zedd151 on November 01, 2024, 04:51:19 AM
Oh, ok. I only did a quick perusal at the code. Apologies.  :cool:
Quote from: sinsi on November 01, 2024, 04:50:02 AMAll characters are copied, even the terminating NULL, then end-of-string is tested for.
Clever indeed.  I just tested your algos in the program using a "full" buffer...  :biggrin:  Works 100% as it should.  :thumbsup:
Yes, someone could learn from your example.  :smiley:  My critique was only a suggestion.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: ognil on November 01, 2024, 06:02:47 AM
One more.. :smiley:

include \masm64\include64\masm64rt.inc
extern  malloc:proc
extern  free:proc

.data
hMemo   DQ      0 
szTitle DB      "Test StringCchCat",0
str1    DB      "I learn ",0
nLen1   equ     $-str1-1
str2    DB      "assembly language.",0
nLen2   equ     $-str2-1

.code
main    proc
        mov     eax,nLen1
        mov     ecx,nLen2
        lea     rcx,[rcx+rax+32]
        call    malloc
        mov     rcx,rax
        jrcxz   @ErrMemory
        mov     r8d,nLen1 
        lea     rdx,str1
        mov     hMemo, rax
        lea     r9,[rcx+r8]
        call    memcpyA   
        mov     rcx, r9
        mov     r8d,nLen2 
        lea     rdx,str2
        call    memcpyA   
        invoke  MessageBox,0,hMemo,addr szTitle,MB_OK
        ;mov    rcx,hMemo
        ;call   free
@ErrMemory:
        call    ExitProcess
main    endp
;*******************************************************************;
memcpyA proc  
push rsi
push rdi
cld
mov rdi,rcx
mov rcx,r8
mov rsi,rdx
shr rcx, 3
rep movsq
mov rcx,r8
and rcx, 7
rep movsb
pop rdi
pop rsi
ret
memcpyA endp
end
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: sinsi on November 01, 2024, 02:36:27 PM
Quote from: zedd151 on November 01, 2024, 04:51:19 AMMy critique was only a suggestion.
Hey, I didn't even consider it a critique, just a statement :biggrin:
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: NoCforMe on November 01, 2024, 05:05:59 PM
In my not-so-humble opinion, nearly all of the string functions ought to be written by the programmer as learning exercises, rather than relying on canned library functions. At least that's what I've done (in 32-bit code):
This covers probably 95% of what actually gets used, and these are not at all difficult to write.

If you need fast routines, you can either optimize these yourself or search this site for the mo-fasta code that people have written. But usually string manipulation is not something where speed really matters.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: wanker742126 on November 01, 2024, 06:02:03 PM
Returning to my original question, I'd like to ask about the code for StringCchCatA in strsafe.h:

STRSAFEAPI
    StringCchCatA(
            _Inout_updates_(cchDest) _Always_(_Post_z_) STRSAFE_LPSTR pszDest,
            _In_ size_t cchDest,
            _In_ STRSAFE_LPCSTR pszSrc)
{
    HRESULT hr;
    size_t cchDestLength;

    hr = StringValidateDestAndLengthA(pszDest,
            cchDest,
            &cchDestLength,
            STRSAFE_MAX_CCH);

    if (SUCCEEDED(hr))
    {
        hr = StringCopyWorkerA(pszDest + cchDestLength,
                cchDest - cchDestLength,
                NULL,
                pszSrc,
                STRSAFE_MAX_LENGTH);
    }

    return hr;
}

If I were to rewrite this in assembly language, would it look something like this?

StringCchCatA PROC pszDest:QWORD,cchDest:DWORD,pszSrc:QWORD
        LOCAL   cchDestLength:DWORD
        invoke  StringValidateDestAndLengthA,pszDest,cchDest,ADDR cchDestLength,STRSAFE_MAX_CCH
        cmp     eax,0
        jl      exit
        sub     r10,r10
        mov     r10d,cchDestLength
        mov     rcx,pszDest
        add     rcx,r10
        mov     edx,cchDest
        sub     edx,cchDestLength
        invoke  StringCopyWorkerA,rcx,edx,0,pszSrc,STRSAFE_MAX_LENGTH
exit:   ret
StringCchCatA ENDP

If there are any mistakes, please let me know.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: NoCforMe on November 01, 2024, 07:06:43 PM
        invoke  StringValidateDestAndLengthA,pszDest,cchDest,ADDR cchDestLength,STRSAFE_MAX_CCH
        cmp    eax,0
        jl      exit

Pretty sure the return value from a Windows function will never be negative. All unsigned values.
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: sinsi on November 01, 2024, 08:12:27 PM
Quote from: NoCforMe on November 01, 2024, 07:06:43 PMPretty sure the return value from a Windows function will never be negative. All unsigned values.
The function (like a lot of others) returns a HRESULT.

Quote from: WikipediaHRESULT is defined in a system header file as a 32-bit, signed integer
Usually the high bit set signifies an error.
QuoteBit 31: S - Severity - indicates success (0) or failure (1)
Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: TimoVJL on November 01, 2024, 08:35:42 PM
#if !defined(_HRESULT_DEFINED) && !defined(__midl)
#define _HRESULT_DEFINED
typedef long HRESULT;
#endif /* _HRESULT_DEFINED */

FORCEINLINE HRESULT HRESULT_FROM_WIN32(unsigned long x) {
    return (HRESULT)(x) <= 0 ? (HRESULT)(x) : (HRESULT)(((x) & 0x0000FFFF) | (FACILITY_WIN32 << 16) | 0x80000000);
}
in strsafe.h#ifndef _HRESULT_DEFINED
#define _HRESULT_DEFINED
typedef long HRESULT;
#endif /* _HRESULT_DEFINED */

typedef unsigned long DWORD;

#define SUCCEEDED(hr)  (((HRESULT)(hr)) >= 0)
#define FAILED(hr)     (((HRESULT)(hr)) < 0)

#define S_OK  ((HRESULT)0L)

Title: Re: How to Call StringCchCat in Win64 Assembly?
Post by: NoCforMe on November 02, 2024, 07:57:26 AM
Quote from: sinsi on November 01, 2024, 08:12:27 PMUsually the high bit set signifies an error.
QuoteBit 31: S - Severity - indicates success (0) or failure (1)

Ah so. Thanks. I stand (sit) corrected.