The MASM Forum

General => The Campus => Topic started by: Orzel94 on January 21, 2017, 05:54:10 AM

Title: No XMM register changes
Post by: Orzel94 on January 21, 2017, 05:54:10 AM
Hi
I have problem with XMM register. I can't change value inside them even by using MOSD or MOVLDP
I have code in C and in ASM which should do exactly the same (school project).

Could you help mi with it?

http://pastebin.com/q4Z9UzTD
http://pastebin.com/vHVp0RwE
Title: Re: No XMM register changes
Post by: jj2007 on January 21, 2017, 07:12:40 AM
Hi Orzel,
1. your code assembles only if the last line is END aproxASM
2. then, it starts as follows:CPU Disasm
Address         Hex dump            Command                           Comments
<ModuleEntryPoi Ú$  55              push ebp
00401001        ³.  8BEC            mov ebp, esp
00401003        ³.  BB 00000000     mov ebx, 0
00401008        ³.  660F1265 28     movlpd xmm4, [ebp+28]
0040100D        ³.  660F1245 08     movlpd xmm0, [ebp+8]
00401012        ³.  660F124D 10     movlpd xmm1, [ebp+10]

... and all instructions work as expected. My advice: install Olly (http://www.ollydbg.de/version2.html) and debug your executable.

Welcome to the Forum :icon14:
Title: Re: No XMM register changes
Post by: Orzel94 on January 21, 2017, 07:34:32 AM
thanks for answer
I'm using MS VS 2015 for dubbuging (Windows forms C# app with C++ & ASM .dll's)

When I add aproxASM at the end there is no changes, but when I replace my code with yours (don't know if I should actually) Ive got some errors

screenshots: https://1drv.ms/f/s!AhFXsdGfJyB21F2tZSEVSKwjDf_n

by the way: you mean
MOV eax,ebx
ret
aproxASM endp
END aproxASM


or just

MOV eax,ebx

END aproxASM
Title: Re: No XMM register changes
Post by: jj2007 on January 21, 2017, 08:26:29 AM
What I showed you is the disassembly, not the source. I have used your source with one single change: the last line, which is END in your code, becomes END aproxASM

Thus, the assembler (MASM = ML.exe) knows that aproxASM is the entry point. I do not use Visual S. for assembly, it's an overkill, so I don't know what VS actually does under the hood. Try \masm32\qEditor.exe, it's a better tool.
Title: Re: No XMM register changes
Post by: Orzel94 on January 21, 2017, 07:31:21 PM
When I've tried to assembly my code in QEditor I get "syntax error : xmm"

When I open .dll in Ollydbg the code is completly different...   https://1drv.ms/i/s!AhFXsdGfJyB21GAiSPipxuAKmU1o
Title: Re: No XMM register changes
Post by: jj2007 on January 21, 2017, 07:54:47 PM
Mateusz,
You get the "syntax error : xmm" because Masm32 ships with a very old version of MASM. RichMasm (http://masm32.com/board/index.php?topic=5314.0) would detect this and install automatically a new one when you hit F6, but with qEditor you will have to do it manually:
- download HJWasm32 (http://www.terraspace.co.uk/hjwasm218_x86.zip)
- extract HJWasm32.exe to \Masm32\bin
- rename \Masm32\bin\ml.exe to mlv614.exe
- rename \Masm32\bin\HJWasm32.exe to ml.exe
- retry console assembly and link in qEditor

Btw you don't need screenshots to show disassemblies. Just select the lines in Olly that are interesting in Olly and hit CtrlC; then paste them here inside code tags (the # icon).

Btw in your screenshot, you show the point after LoadLibrary. The interesting part is when you call the address received by GetProcAddress.
Title: Re: No XMM register changes
Post by: TWell on January 21, 2017, 07:58:25 PM
tested with ml.exe 10

DllEntry proc
MOV EAX, 1
RET
DllEntry endp
...
...
END DllEntry
or_DllMainCRTStartup proc
MOV EAX, 1
RET
_DllMainCRTStartup endp
...
...
END _DllMainCRTStartup
and use def-file for dlls.
Title: Re: No XMM register changes
Post by: hutch-- on January 21, 2017, 08:30:45 PM
Just get a later version of ML.EXE.
Title: Re: No XMM register changes
Post by: Orzel94 on January 21, 2017, 08:42:20 PM
debuging is ok now and it compiles to .exe but after "create new .dll" in qEditor a pasting my code (by replacing all) I get sth like that:
ASM.asm: 50 lines, 2 passes, 69 ms, 0 warnings, 0 errors
Microsoft (R) Incremental Linker Version 5.12.8078
Copyright (C) Microsoft Corp 1992-1998. All rights reserved.
ASM.def : error LNK2001: unresolved external symbol aproxASM
ASM.lib : fatal error LNK1120: 1 unresolved externals
LINK : fatal error LNK1141: failure during build of exports file
Could Not Find C:\Users\Mateusz\Documents\Visual Studio 2015\Projects\Generator fraktali 2D\ASM\masmdll\ASM.exp
Volume in drive C is OS
Volume Serial Number is 8482-C98C

Directory of C:\Users\Mateusz\Documents\Visual Studio 2015\Projects\Generator fraktali 2D\ASM\masmdll

21.01.2017  10:34             1 662 ASM.asm
21.01.2017  10:36                31 ASM.def
               2 File(s)          1 693 bytes
               0 Dir(s)  35 056 488 448 bytes free


ASM.def is:
LIBRARY ASM
EXPORTS aproxASM
Title: Re: No XMM register changes
Post by: TWell on January 21, 2017, 08:45:55 PM
LIBRARY ASM
EXPORTS aproxASM=_aproxASM@40
Title: Re: No XMM register changes
Post by: Orzel94 on January 21, 2017, 08:51:30 PM
Creating library ASM.lib and object ASM.exp
ASM.exp : error LNK2001: unresolved external symbol _aproxASM@40
ASM.dll : fatal error LNK1120: 1 unresolved externals
Title: Re: No XMM register changes
Post by: jj2007 on January 21, 2017, 09:09:55 PM
Your ASM.def file is OK. I would choose another file name, though ;-)

The normal structure of a DLL source is like this:

.code
LibMain proc instance:DWORD, reason:DWORD, unused:DWORD
  mov eax, 1
  ret
LibMain endp

aproxASM proc p_real: REAL8, p_imaginary: REAL8, c_real:REAL8, c_imaginary:REAL8, four:REAL8
MOV ebx,0
...
ret
aproxASM endp

end LibMain
Title: Re: No XMM register changes
Post by: Orzel94 on January 21, 2017, 09:27:02 PM
its ok know, I have some errors (exception at the exit of procedure) but I hope (or not XD) it's logic issue (I don't know yet :/ )

last thing to make sure
1) to return integer value i must put it into EAX? Is this enought or shall I do sth more with it?
2)to load library from C# app I need
[DllImport("ASM.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.Cdecl)]
        static extern int aproxASM(double p_real, double p_imaginary, double c_real, double c_imaginary, double four);

or just
[DllImport("ASM.dll")]
        static extern int aproxASM(double p_real, double p_imaginary, double c_real, double c_imaginary, double four);
Title: Re: No XMM register changes
Post by: TWell on January 22, 2017, 01:15:44 AM
A small test:
using System;
using System.Runtime.InteropServices;
// c:\Windows\Microsoft.NET\Framework\v2.0.50727\csc.exe aproxTestCc.cs /platform:x86
class aproxTestCs
{
[DllImport("aproxC.dll")]
static extern int aproxC(double p_real, double p_imaginary, double c_real, double c_imaginary);
[DllImport("aproxASM.dll")]
static extern int aproxASM(double p_real, double p_imaginary, double c_real, double c_imaginary, double four);

static void Main()
{
double p_real = -1.5;
double p_imaginary = -1.5;
double c_real = 0.65;
double c_imaginary = -0.03;
double four = 4.0;

int ac = aproxC(p_real, p_imaginary, c_real, c_imaginary);
Console.WriteLine("ac={0}", ac);
int aa = aproxASM(p_real, p_imaginary, c_real, c_imaginary, four);
Console.WriteLine("aa={0}", aa);
}
}
C version works, but ASM version crash.

But this outputs
ac=1
aa=360#include <stdio.h>

int __stdcall aproxC(double p_real, double p_imaginary, double c_real, double c_imaginary);
#pragma comment(lib, "aproxC.lib")
int __stdcall aproxASM(double p_real, double p_imaginary, double c_real, double c_imaginary, double four);
#pragma comment(lib, "aproxASM.lib")

int __cdecl main(void)
{
double p_real = -1.5;
double p_imaginary = -1.5;
double c_real = 0.65;
double c_imaginary = -0.03;
double four = 4.0;

int ac = aproxC(p_real, p_imaginary, c_real, c_imaginary);
printf("ac=%d\n", ac);
int aa = aproxASM(p_real, p_imaginary, c_real, c_imaginary, four);
printf("aa=%d\n", aa);

return 0;
}
BTW ebx is reserved register, use ecx instead or push/pop ebx
Title: Re: No XMM register changes
Post by: Orzel94 on January 22, 2017, 03:00:41 AM
I changed to ECX, create project with your code and add .lib files by properties -> linker -> Additional Library Directories
and I got:
LNK2028 unresolved token (0A00003C) "int __stdcall aproxC(double,double,double,double)" (?aproxC@@$$FYGHNNNN@Z) referenced in function "int __cdecl main(void)" (?main@@$$HYAHXZ) test 2 C:\Users\Mateusz\documents\visual studio 2015\Projects\test 2\test 2\Source.obj 1
LNK2028 unresolved token (0A000045) "int __stdcall aproxASM(double,double,double,double,double)" (?aproxASM@@$$FYGHNNNNN@Z) referenced in function "int __cdecl main(void)" (?main@@$$HYAHXZ) test 2 C:\Users\Mateusz\documents\visual studio 2015\Projects\test 2\test 2\Source.obj 1
LNK2019 unresolved external symbol "int __stdcall aproxC(double,double,double,double)" (?aproxC@@$$FYGHNNNN@Z) referenced in function "int __cdecl main(void)" (?main@@$$HYAHXZ) test 2 C:\Users\Mateusz\documents\visual studio 2015\Projects\test 2\test 2\Source.obj 1
LNK2019 unresolved external symbol "int __stdcall aproxASM(double,double,double,double,double)" (?aproxASM@@$$FYGHNNNNN@Z) referenced in function "int __cdecl main(void)" (?main@@$$HYAHXZ) test 2 C:\Users\Mateusz\documents\visual studio 2015\Projects\test 2\test 2\Source.obj 1
LNK1120 4 unresolved externals test 2 C:\Users\Mateusz\documents\visual studio 2015\Projects\test 2\Debug\test 2.exe 1


edit: sory for such trivial (I think) things but it's my first touch to DLLs and Assembler (semester must be passed XD but its quite interesting)
Title: Re: No XMM register changes
Post by: TWell on January 22, 2017, 03:20:07 AM
My second code example is C code, not C++. Compile it as C source (*.c).
Title: Re: No XMM register changes
Post by: Orzel94 on January 22, 2017, 07:53:59 AM
so again i've got

Error LNK2019 unresolved external symbol _aproxC@32 referenced in function _main test 2 C:\Users\Mateusz\documents\visual studio 2015\Projects\test 2\test 2\Source.obj 1
Error LNK2019 unresolved external symbol _aproxASM@40 referenced in function _main test 2 C:\Users\Mateusz\documents\visual studio 2015\Projects\test 2\test 2\Source.obj 1

first time it was caused by lack of libMain part but now I have it in ASM (but in C++ dll is error too)
Title: Re: No XMM register changes
Post by: jj2007 on January 22, 2017, 08:58:16 AM
Try building it without a def file but with aproxASM proc syscall export p_real: REAL8, ...
Title: Re: No XMM register changes
Post by: hutch-- on January 22, 2017, 01:41:07 PM
Orzel94,

Just make sure you write the correct prototypes for your C code calling either object modules written in assembler OR a DLL written in assembler. 32 bit MASM supports both STDCALL and C calling conventions, make sure you get this right and look up your compiler reference for "extern c" for the prototypes. In 64 bit the calling convention is FASTCALL and you have to get this right.
Title: Re: No XMM register changes
Post by: coder on January 22, 2017, 03:56:43 PM
I am confused. I thought C returns to ST(0) for 32-bit floaty things.
Title: Re: No XMM register changes
Post by: TWell on January 22, 2017, 07:55:49 PM
I am a bit confused :(

ml.exe -FeaproxASM.dll aproxASM.asm -link -dll -export:aproxASM=_aproxASM@40
cl.exe -Gz -GS- -MD -FeaproxC.dll aproxC.c -link -dll -export:aproxC=_aproxC@32
cl.exe -GS- -MD aprox_test.c


aprox_test.obj : error LNK2019: unresolved external symbol _aproxC@32 referenced in function _main
aprox_test.obj : error LNK2019: unresolved external symbol _aproxASM@40 referenced in function _main
aprox_test.exe : fatal error LNK1120: 2 unresolved externals


Even after reading these:
EXPORTS
https://msdn.microsoft.com/en-us/library/aa278935(v=vs.60).aspx
-export
https://msdn.microsoft.com/en-us/library/aa235424(v=vs.60).aspx

polink.exe works as documented.

EDIT:
EXPORTS
https://msdn.microsoft.com/en-us/library/hyx1zcd3(v=vs.140).aspx

This syntax should be still validentryname[=internalname] [@ordinal [NONAME]] [[PRIVATE] | [DATA]]

EDIT:
With polib.exe or def2lib.exe we can have correct libs.LIBRARY aproxC.dll
EXPORTS
;aproxC=_aproxC@32
"_aproxC@32"
LIBRARY aproxASM.dll
EXPORTS
;aproxASM=_aproxASM@40
"_aproxASM@40"

Title: Re: No XMM register changes
Post by: jj2007 on January 22, 2017, 08:19:45 PM
Quote from: coder on January 22, 2017, 03:56:43 PM
I am confused. I thought C returns to ST(0) for 32-bit floaty things.

That's correct. In assembly, you must do it by hand. Here are two options:
push eax ; return int(eax) in ST(0)
fild dword ptr [esp]
pop eax
ret

sub esp, 8 ; create a REAL8 slot
movlps qword ptr [esp], xmm0   ; move the value to be returned from xmm0 to the stack
fld REAL8 ptr [esp]
add esp, 8
ret


As to building Mateusz' dll code, see reply #17. It works.
Title: Re: No XMM register changes
Post by: Orzel94 on January 22, 2017, 09:53:14 PM
I have now completely mess in mind...
Without .def file i can only compile it to .exe file not .dll

even when I compile asm code which do simply nothing but returnig integer value

;-------------------------------------------------------------------------
.686
.MMX
.XMM
.MODEL FLAT, STDCALL
OPTION CASEMAP:NONE
.DATA
circle DD 360
.CODE
LibMain proc instance:DWORD, reason:DWORD, unused:DWORD
  mov eax, 1
  ret
LibMain endp
aproxASM proc p_real: REAL8, p_imaginary: REAL8, c_real:REAL8, c_imaginary:REAL8, four:REAL8
MOV eax,12
ret
aproxASM endp

end LibMain

or like jj2007 mentioned
;-------------------------------------------------------------------------
.686
.MMX
.XMM
.MODEL FLAT, STDCALL
OPTION CASEMAP:NONE
.DATA
circle DD 360
.CODE
LibMain proc instance:DWORD, reason:DWORD, unused:DWORD
  mov eax, 1
  ret
LibMain endp
aproxASM proc p_real: REAL8, p_imaginary: REAL8, c_real:REAL8, c_imaginary:REAL8, four:REAL8
MOV eax,12
push eax ; return int(eax) in ST(0)
fild dword ptr [esp]
pop eax
ret
aproxASM endp

end LibMain


I've got stack balance error
Title: Re: No XMM register changes
Post by: TWell on January 22, 2017, 10:13:29 PM
You function return int.
jj2007 comment was for float, so not for your code, if that return int.

BTW
use eax in your code as it is free register, so return value is already in right place.

That syscall tip works to decoration issue, but read this:
http://masm32.com/board/index.php?topic=3640.msg38361#msg38361

Otherwise you need some export manipulation for undecorated name.

Title: Re: No XMM register changes
Post by: jj2007 on January 22, 2017, 10:51:07 PM
It seems I got a solution (no def file needed):

aproxASM proc c export p_real: REAL8, p_imaginary: REAL8, c_real:REAL8, c_imaginary:REAL8, four:REAL8
  ...
  ret

Attached testbed needs MasmBasic (http://masm32.com/board/index.php?topic=94.0) to build, but you can test the DLL with your C compiler as well.

@OP: It would help if you provided a usage example, i.e. five float arguments, and the expected result.

CheckMateusz.exe loads the DLL and executes aproxASM with 5 dummy args, it returns 123456789 and checks if the stack is balanced.

In case you want to test the build:
- create an empty TmpDllEx.def in the source folder, and write-protect it (sorry, this is a hack, I know)
- open Mateusz.asc in RichMasm (http://masm32.com/board/index.php?topic=5314.0)
- hit F6 to a) build the DLL and b) run CheckMateusz.exe to test it

It works also with stdcall, see second attachment.
Title: Re: No XMM register changes
Post by: TWell on January 23, 2017, 01:04:39 AM
test program in C for dll#define WIN32_LEAN_AND_MEAN
#include <windows.h>
#include <stdio.h>

//int aproxASM(double p_real, double p_imaginary, double c_real, double c_imaginary, double four);
typedef int (APROXASM)(double p_real, double p_imaginary, double c_real, double c_imaginary, double four);
typedef APROXASM*PAPROXASM;

int __cdecl main(void)
{
double p_real = -1.5;
double p_imaginary = -1.5;
double c_real = 0.65;
double c_imaginary = -0.03;
double four = 4.0;
HANDLE hLib;
static PAPROXASM aproxASM;
hLib = LoadLibrary("Mateusz.dll");
aproxASM = (PAPROXASM)GetProcAddress(hLib, "aproxASM");

int aa = aproxASM(p_real, p_imaginary, c_real, c_imaginary, four);
printf("aa=%d\n", aa);

return 0;
}
Title: Re: No XMM register changes
Post by: Orzel94 on January 23, 2017, 01:12:34 AM
YES!
it's working! (jj2007 solution) it's returning values! It's returning 360 instead of 1 but It does not matter it's working afer 3 days of fighting with it!

So now it's look like this
when I compile it with VS it's working and returning value to C#  and I know that VS dubbuger has SH** value because it show no changes in XMM registersand it shows that in EAX is 168 but it returns 360 (when I put only one instruction MOV eax,12 it returns 12 so I thing its debugger showing sth wrong)
OllyDGB some assembler code but it completly don't look like mine (matter if I compile it in debug mode?)

when I compile the same code by QEditor using Makeit.bat  it don't work with C# (exception is thrown) but in OllyDGB I see my code but I don't know how tu put parameters tu the procedure when using Call DLL export (it says only integer values can be put)

So I have pretty working code but I have no idea how tu debug it XD
Title: Re: No XMM register changes
Post by: jj2007 on January 23, 2017, 01:50:36 AM
Quote from: Orzel94 on January 23, 2017, 01:12:34 AMin EAX is 168 but it returns 360

168 is the hex representation of 360 decimal. Olly shows hex most of the time, except if the values have a dot, like 123.

If you want symbols in the debugger:
; OPT_Symbols   1 ; for RichMasm, somewhere in the source

Otherwise, add
/Zi /Zf to the assembler,
/debug to the linker commandline

There are not many symbols to see in the debugger, but if the symbols work properly, you will see for example:72E81095          660F2FE2          comisd xmm4, xmm2
72E81099         77 C7             ja short 72E81062
72E8109B          B8 68010000       mov eax, 168
72E810A0          391D 0040E872     cmp [circle], ebx   ; <<<<<<<<<<<<<<<<<<<<<<<<
72E810A6         7F BA             jg short 72E81062
72E810A8          8BC3              mov eax, ebx
72E810AA          5B                pop ebx
72E810AB          5D                pop ebp
72E810AC          C3                retn


Note the mov eax, 168 - it's 360, of course. Now that the building and testing process is OK, maybe it's time to look at the code again? ;-)
aproxASM_stdcall proc stdcall export uses ebx p_real: REAL8, p_imaginary: REAL8, c_real:REAL8, c_imaginary:REAL8, four:REAL8
  xor eax, eax ; zero
  movlpd xmm4, four
  movlpd xmm0, p_real
  movlpd xmm1, p_imaginary

doloop: ; real_part
  inc eax ; loop indexer
  vmulsd xmm2, xmm0, xmm0
  vmulsd xmm3, xmm1, xmm1
  subsd xmm2, xmm3
  addsd xmm2, c_real

  ; imaginary_part
  vmulsd xmm3, xmm0, xmm1
  addsd xmm3, xmm3
  addsd xmm3, c_imaginary

  movsd xmm0, xmm2
  movsd xmm1, xmm3


  ; complexMod
  vmulsd xmm2, xmm0, xmm0
  vmulsd xmm3, xmm1, xmm1
  addsd xmm2, xmm3

  ; while (modulo < 4 && i < 360);
  comisd xmm4, xmm2
  ja doloop
  cmp circle, 360 ; makes no sense at all, circle is DD 360!
  jg doloop
  ret
aproxASM_stdcall endp
Title: Re: No XMM register changes
Post by: Orzel94 on January 23, 2017, 03:38:52 AM
here it's how the code looks now
;-------------------------------------------------------------------------
.686
.MMX
.XMM

.MODEL FLAT, STDCALL
OPTION CASEMAP:NONE


.DATA
circle DD 360
.CODE
LibMain proc  instance:DWORD, reason:DWORD, unused:DWORD
  mov eax, 1
  ret
LibMain endp
;-------------------------------------------------------------------------
aproxASM proc c export p_real: REAL8, p_imaginary: REAL8, c_real:REAL8, c_imaginary:REAL8, four:REAL8
MOV ecx,0
MOVLPD XMM4,[four]
MOVLPD XMM0,[p_real]
MOVLPD XMM1,[p_imaginary]
do:

;real_part
VMULSD XMM2,XMM0,XMM0
VMULSD XMM3,XMM1,XMM1
SUBSD XMM2,XMM3
ADDSD XMM2,c_real

;imaginary_part
VMULSD XMM3,XMM0,XMM1
ADDSD XMM3,XMM3
ADDSD XMM3,[c_imaginary]

MOVSD XMM0,XMM2
MOVSD XMM1,XMM3

INC ecx ;loop indexer

;complexMod
VMULSD XMM2,XMM0,XMM0
VMULSD XMM3,XMM1,XMM1
ADDSD XMM2,XMM3

;while (modulo < 4 && i < 360);
COMISD XMM4,XMM2
JG second
JMP done
;MOV eax,360
second:
CMP circle,ecx
JG do
done:
MOV eax,ecx
;MOV eax,12


ret
aproxASM endp

end LibMain

this cmp ccircle,360 disapeared earlier
I fix issue with "while" implementation and now its working excatly as code in C (only difference is result i ASM == result in C + ~10 or + ~11 (I thing)  I don't know why yet but the only thing it changes it move a little bit color in HSV (instead of red is more orange (i wrote 2D fractal generator)) so I can say to teacher that I have planned it XD )

So thank you very much ;)