The MASM Forum

General => The Campus => Topic started by: stefan.velicu on March 30, 2014, 06:34:45 PM

Title: Rotation
Post by: stefan.velicu on March 30, 2014, 06:34:45 PM
    Hi. I have a project where i must simulate somethink like a servomotor. I need your help, because I don't know how to load a image from resources and rotate it accordingly to the angle of rotation given by the user.
    Thank you, cheers!
Title: Re: Rotation
Post by: jj2007 on March 30, 2014, 07:14:14 PM
Hi Stefan,

Can you give us some details?
- what is the context (homework, professional, other)?
- is is 90 degree only, or arbitrary angles?
- what have you tried so far?
- why don't you do it in C/CUDA?
- what do you think of the Arabnia solution?

Welcome to the Forum :icon14:
Title: Re: Rotation
Post by: stefan.velicu on March 30, 2014, 07:44:02 PM
Hi and thank you for your prompt answer. It's a homework, but unfortunately it's mandatory to be in MASM only (not c, c++ or anythnig). So far, I've only managed to open the window  :lol:. About the rotation of the object, it must be a full one, so 360 degrees.
I'll search the Arabnia solution, I don't know it.

Thank you, cheers.
Title: Re: Rotation
Post by: Gunther on March 30, 2014, 07:45:57 PM
Hi Stefan,

you should post your code and we'll see. Welcome to the forum.

Gunther
Title: Re: Rotation
Post by: stefan.velicu on March 30, 2014, 08:10:57 PM
Thank you, Gunther. Here's the code that loads the image that i must rotate.


.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\gdi32.inc
include \masm32\include\masm32.inc
includelib \masm32\lib\user32.lib
includelib \masm32\lib\kernel32.lib
includelib \masm32\lib\gdi32.lib
includelib \masm32\lib\masm32.lib

WinMain proto :DWORD,:DWORD,:DWORD,:DWORD

.DATA
ClassName db "WinClass", 0
AppName db "Simple Window", 0
MouseClick db 0
ImageName1 db "C:\masm32\Images\star.bmp", 0
BigPict dd FALSE

.Data?
hInstance HINSTANCE ?
CommandLine LPSTR ?
maxX DD ?
maxY DD ?
ShapeNumber dd ?
startpoint POINT <>
endpoint POINT <>
hMemDC HDC ?
Bitmap HBITMAP ?
hBitmap HBITMAP ?

.CODE
start:
invoke GetModuleHandle, NULL
mov hInstance,eax
invoke WinMain,hInstance, NULL, NULL, 0
invoke GetCommandLine
mov CommandLine,eax
invoke ExitProcess,eax

invoke LoadImage, hInstance, ADDR ImageName1, IMAGE_BITMAP, 2, 2, LR_LOADFROMFILE
mov hBitmap,eax

WinMain proc hInst:HINSTANCE, hPrevInst:HINSTANCE, CmdLine:LPSTR, CmdShow:DWORD
local wc:WNDCLASSEX
local msg:MSG
local hwnd:HWND
mov wc.cbSize, SIZEOF WNDCLASSEX
mov wc.style, CS_HREDRAW or CS_VREDRAW
mov wc.lpfnWndProc, OFFSET WndProc
mov wc.cbClsExtra, NULL
mov wc.cbWndExtra, NULL
push hInst
pop wc.hInstance
mov wc.lpszClassName, offset ClassName
invoke LoadIcon, NULL, IDI_APPLICATION
mov wc.hIcon, eax
mov wc.hbrBackground,COLOR_WINDOW+1
mov wc.lpszMenuName, NULL
mov wc.lpszClassName, offset ClassName
invoke LoadIcon, NULL, IDI_APPLICATION
mov wc.hIcon, eax
invoke LoadCursor, NULL, IDC_ARROW
mov wc.hCursor, eax
invoke RegisterClassEx, addr wc
invoke CreateWindowEx, 0, addr ClassName, addr AppName, WS_OVERLAPPEDWINDOW or

WS_VISIBLE, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, NULL,
NULL, hInst, NULL
mov hwnd, eax
.while TRUE
invoke GetMessage, addr msg, NULL, 0, 0
.break .if (!eax)
invoke TranslateMessage, addr msg
invoke DispatchMessage, addr msg
.endw
mov eax, msg.wParam
ret
WinMain endp


mov CommandLine,eax

        invoke LoadImage, hInstance, ADDR ImageName1, IMAGE_BITMAP, 100, 100,

LR_LOADFROMFILE
        mov hBitmap,eax

WndProc proc hWnd:HWND, uMsg:UINT, wParam:WPARAM, lParam:LPARAM
                LOCAL hDC:HDC
                LOCAL ps:PAINTSTRUCT
                LOCAL rect:RECT
                LOCAL hInnerDC:HDC
       
        invoke CreateCompatibleDC,hMemDC
        mov hInnerDC,eax

    .IF uMsg==WM_DESTROY
        invoke PostQuitMessage,NULL

     .ELSEIF uMsg==WM_PAINT
      invoke BeginPaint,hWnd, ADDR ps
        mov hDC,eax
        invoke CreateCompatibleDC,hMemDC
        mov hInnerDC,eax
        invoke SelectObject, hInnerDC,hBitmap
        ;.IF BigPict
        ;invoke BitBlt, hMemDC, startpoint.x, startpoint.y, 226, 229, hInnerDC, 0,

0, SRCCOPY ; dimensiunea
        ;.ELSE
        invoke BitBlt, hMemDC, startpoint.x, startpoint.y, 226, 229, hInnerDC, 0,

0, SRCCOPY ;
        ;.ENDIF
        invoke GetClientRect,hWnd,ADDR rect
        invoke SelectObject,hDC, hMemDC
        invoke BitBlt,hDC,0,0,rect.right,rect.bottom,hMemDC,0,0,SRCCOPY
        invoke EndPaint,hWnd,addr ps
        invoke DeleteDC,hInnerDC
        invoke DeleteObject,hBitmap
        ret

       
    .ELSEIF uMsg==WM_CREATE
        invoke GetSystemMetrics, SM_CXSCREEN
        mov maxX, eax
        invoke GetSystemMetrics, SM_CYSCREEN
        mov maxY, eax
        invoke GetDC,hWnd
        mov hDC, eax
        invoke CreateCompatibleDC, hDC
        mov hMemDC, eax
        invoke CreateCompatibleBitmap, hDC, maxX, maxY
        mov Bitmap, eax
        invoke SelectObject, hMemDC, Bitmap
        invoke PatBlt, hMemDC, 0, 0, maxX, maxY, PATCOPY
        mov startpoint.x, 0
        mov startpoint.y, 0
        invoke LoadImage, hInstance,ADDR ImageName1, IMAGE_BITMAP, 100, 100,

LR_LOADFROMFILE
        mov hBitmap,eax
        invoke InvalidateRect,hWnd,NULL,TRUE
        ret
       
     .ELSE
        invoke DefWindowProc,hWnd,uMsg,wParam,lParam
        ret
     .ENDIF
        xor eax,eax
        ret
        WndProc endp

end start


Cheers.
Title: Re: Rotation
Post by: jj2007 on March 30, 2014, 08:36:39 PM
Stefan,

Your code won't run properly, and it's not your fault, it's your teacher who is distributing crappy code. Read this recent thread carefully (http://masm32.com/board/index.php?topic=3049.msg31675#msg31675), adapt your code accordingly, and then come back.

BTW the best way to get help is to zip the source including necessary image files, preferably with folder names, and to attach the zip file to your post.
Title: Re: Rotation
Post by: stefan.velicu on March 30, 2014, 10:00:28 PM
Ok, I've found out an example in the masm32 directory and managed to load the image that I want to rotate. Now, how could I rotate that image to let's say 90 degrees, or 180?
Title: Re: Rotation
Post by: jj2007 on March 30, 2014, 10:12:28 PM
Good. Now go to forum search (http://masm32.com/board/index.php?action=search;advanced;search=) (more in the old forum (http://www.masmforum.com/board/index.php?action=search;advanced)) and look for GetDIBits
Title: Re: Rotation
Post by: TWell on March 30, 2014, 10:20:18 PM
Quote from: stefan.velicu on March 30, 2014, 10:00:28 PM
Ok, I've found out an example in the masm32 directory and managed to load the image that I want to rotate. Now, how could I rotate that image to let's say 90 degrees, or 180?
PlgBlt (http://msdn.microsoft.com/en-us/library/windows/desktop/dd162804(v=vs.85).aspx)?
Title: Re: Rotation
Post by: stefan.velicu on March 30, 2014, 10:24:31 PM
Thanks, TWell, but I have to do this in assembly.
Title: Re: Rotation
Post by: stefan.velicu on March 30, 2014, 10:25:51 PM
Quote from: jj2007 on March 30, 2014, 10:12:28 PM
Good. Now go to forum search (http://masm32.com/board/index.php?action=search;advanced;search=) (more in the old forum (http://www.masmforum.com/board/index.php?action=search;advanced)) and look for GetDIBits

Thank you, I'll check it out. So far I couldn't find anything to work :D
Title: Re: Rotation
Post by: jj2007 on March 30, 2014, 10:58:01 PM
You need to get your pixels into a memory buffer so that you can work directly on them. That requires GetDiBits with an appropriate buffer created by HeapAlloc. The Masm32 macro alloc() is also fine:
include \masm32\include\masm32rt.inc

.code
start:   mov esi, alloc(1024*1024)
   print LastError$()
   ;... your code ...
   free esi
   print LastError$()
   exit

end start
Title: Re: Rotation
Post by: stefan.velicu on March 30, 2014, 11:47:34 PM
I've found this, but I don't know who are the arguments  :icon_eek:

invoke GetDIBits,hdc,hbmp,0,bmHeight,pBits,OFFSET bi,DIB_RGB_COLORS
Title: Re: Rotation
Post by: stefan.velicu on March 30, 2014, 11:51:07 PM
Quote from: jj2007 on March 30, 2014, 10:58:01 PM
You need to get your pixels into a memory buffer so that you can work directly on them. That requires GetDiBits with an appropriate buffer created by HeapAlloc. The Masm32 macro alloc() is also fine:
include \masm32\include\masm32rt.inc

.code
start:   mov esi, alloc(1024*1024)
   print LastError$()
   ;... your code ...
   free esi
   print LastError$()
   exit

end start
Thank you, but I've got this error:
Quoteundefined symbol : alloc
Title: Re: Rotation
Post by: GoneFishing on March 31, 2014, 12:03:51 AM
Just add one line to your includes:
Quote
include \masm32\macros\macros.asm
Title: Re: Rotation
Post by: stefan.velicu on March 31, 2014, 01:32:11 AM
Thank you very much, vertograd.
Title: Re: Rotation
Post by: stefan.velicu on March 31, 2014, 01:47:11 AM
Now, that I have the .bmp's pixels in esi, how can I rotate them by a specific degree?

P.S.: I've found something like

[xo yo]=[x y]*[cos(p) sin(p); -sin(p) cos(p)]
, for Matlab
Is that ok?
Title: Re: Rotation
Post by: Gunther on March 31, 2014, 02:27:03 AM
Hi Stefan,

Quote from: stefan.velicu on March 31, 2014, 01:47:11 AM
P.S.: I've found something like

[xo yo]=[x y]*[cos(p) sin(p); -sin(p) cos(p)]
, for Matlab
Is that ok?

that's okay. It's the rotation matrix. p is the angle.

Gunther
Title: Re: Rotation
Post by: jj2007 on March 31, 2014, 04:00:48 AM
x' = x*cos(u)-y*sin(u)
y' = x*sin(u)-y*cos(u)
where : x' = is new x position, x - previous x position, u - angle in radians.

(posted yesterday in the Pelles C Forum) (http://forum.pellesc.de/index.php?topic=6129.0)

Note there is an instruction called fsincos. Very useful ;-)
Title: Re: Rotation
Post by: qWord on March 31, 2014, 04:37:49 AM
Quote from: jj2007 on March 31, 2014, 04:00:48 AM
x' = x*cos(u)-y*sin(u)
y' = x*sin(u)-y*cos(u)
where : x' = is new x position, x - previous x position, u - angle in radians.

(posted yesterday in the Pelles C Forum) (http://forum.pellesc.de/index.php?topic=6129.0)
The formula is wrong and that is the reason why that guy get deformed objects :-D
It should be y' = x*sin(u)+y*cos(u).

For the rotation it is important to transform the destination coordinates to source coordinates and not vice versa. A short example:
include \masm32\include\masm32rt.inc
.686
.mmx
.xmm
;DEBUG = 1

WndProc proto hWnd:HWND,uMsg:UINT,wParam:WPARAM,lParam:LPARAM

IDT EQU 100

XMMPS typedef OWORD
XMMLPS typedef QWORD

WND_DATA struct
    hbmpImage   HBITMAP ?
    angle       REAl4 ?
WND_DATA ends

MATRIX2D struct
    m11 REAl4 ?
    m12 REAl4 ?
    m21 REAl4 ?
    m22 REAl4 ?
    m31 REAl4 ?
    m32 REAl4 ?
MATRIX2D ends
PMATRIX2D typedef ptr MATRIX2D

.data?
    g_hInstance HINSTANCE ?
.code
main proc
LOCAL wcex:WNDCLASSEX
LOCAL msg:MSG

    mov g_hInstance,rv(GetModuleHandle,0)
    mov wcex.hInstance,eax
    mov wcex.cbSize,SIZEOF WNDCLASSEX
    mov wcex.style, CS_HREDRAW or CS_VREDRAW
    mov wcex.lpfnWndProc, OFFSET WndProc
    mov wcex.cbClsExtra,NULL
    mov wcex.cbWndExtra,SIZEOF WND_DATA
    mov wcex.hbrBackground,0
    mov wcex.lpszMenuName,NULL
    mov wcex.lpszClassName,chr$("win32class")
    mov wcex.hIcon,rv(LoadIcon,NULL,IDI_APPLICATION)
    mov wcex.hIconSm,eax
    mov wcex.hCursor,rv(LoadCursor,NULL,IDC_ARROW)
    fn RegisterClassEx, &wcex
    fn CreateWindowEx,0,wcex.lpszClassName,"pixrot",WS_VISIBLE or WS_SYSMENU or WS_SIZEBOX or WS_MINIMIZEBOX or WS_MAXIMIZEBOX or WS_CLIPCHILDREN,CW_USEDEFAULT,CW_USEDEFAULT,CW_USEDEFAULT,CW_USEDEFAULT,0,0,g_hInstance,0

    .while 1
        .break .if !rv(GetMessage,&msg,0,0,0)
        fn TranslateMessage, &msg
        fn DispatchMessage, &msg
    .endw

    invoke ExitProcess,msg.wParam

main endp


mul_matrix proc pResult:PMATRIX2D,pA:PMATRIX2D,pB:PMATRIX2D
   
    mov eax,pResult
    mov ecx,pA
    mov edx,pB
   
    movups xmm3,XMMPS ptr [edx]
    movlps xmm7,XMMLPS ptr [edx+16]
    movups xmm0,XMMPS ptr [ecx]
    movlps xmm2,XMMLPS ptr [ecx+16]
    movaps xmm1,xmm0
    shufps xmm3,xmm3,11011000y
    movlhps xmm7,xmm7
    movlhps xmm0,xmm0
    movhlps xmm1,xmm1
    movlhps xmm2,xmm2
    mulps xmm0,xmm3
    mulps xmm1,xmm3
    mulps xmm2,xmm3
    movaps xmm3,xmm0
    movaps xmm4,xmm1
    movaps xmm5,xmm2
    shufps xmm3,xmm3,11110101y
    shufps xmm4,xmm4,11110101y
    shufps xmm5,xmm5,11110101y
    addps xmm0,xmm3
    addps xmm1,xmm4
    addps xmm2,xmm5
    shufps xmm0,xmm1,10001000y
    shufps xmm2,xmm2,11101000y
    addps xmm2,xmm7
    movups XMMPS ptr [eax],xmm0
    movlps XMMLPS ptr [eax+16],xmm2
   
    ret
   
mul_matrix endp


rot_matrix proc pResult:PMATRIX2D,angle:REAL4
LOCAL sincos[2]:REAl4
   
    mov eax,pResult
   
    .const
        align 16
        ps_neg1 REAL4 0.0,80000000r,0.0,0.0
    .code

    fld angle
    fsincos
    fstp sincos[0]
    fstp sincos[4]
    xorps xmm1,xmm1
    movlps xmm0,XMMLPS ptr sincos
    shufps xmm0,xmm0,00010100y
    xorps xmm0,XMMPS ptr ps_neg1
    movups XMMPS ptr [eax],xmm0
    movlps XMMLPS ptr [eax+16],xmm1
   
    ret
   
rot_matrix endp

;/**
; * Rotates the bitmap hBmp around relative point (cx_,cy) and draw
; * to hdc at postion (x,y)=upper left corner of org. bitmap.
; */
draw_image proc uses esi edi ebx hdc:HDC,hBmp:HBITMAP,x:SDWORD,y:SDWORD,cx_:SDWORD,cy:SDWORD,angle:REAl4,bkColor:DWORD
LOCAL matrix1:MATRIX2D
LOCAL matrix2:MATRIX2D
LOCAL matrix3:MATRIX2D
LOCAL hMemDC:HDC
LOCAL bmpinfo:BITMAPINFO
LOCAL bmp:BITMAP,w:SDWORD,h:SDWORD,wm:SDWORD,cr:SDWORD
LOCAL tmpBmp:HBITMAP
IFDEF DEBUG
    LOCAL cb:DWORD
ENDIF
    .const
        align 16
        ps_identity REAL4 1.0,0.0,0.0,1.0
        ps_neg12    REAL4 80000000r,80000000r,0.0,0.0
        ps_mul2     REAL4 2.0,2.0,0.0,0.0
        ps_c0       REAl4 1.0,1.0,1.0,1.0
        ps_msk1     REAl4 0.0,0ffffffffr,0.0,0ffffffffr
        ps_c1       REAl4 0.0,2.0,0.0,2.0
        ps_c2       REAl4 2.0,0.0,2.0,0.0
    .code
   
    ; angle = -( angle % 2pi )
    fld1
    fldpi
    fscale
    fld angle
    fprem
    fchs
    fstp angle
    fstp st
    fstp st
   
    ;/**
    ; * Transformations:
    ; *   - translate (-cx,-cy)
    ; *   - rotate (angle)
    ; *   - translate (cx,cy)
    ; */
    fn rot_matrix,&matrix2,angle
    ;/* setup translation matrices */
    xorps xmm0,xmm0
    movaps xmm1,XMMPS ptr ps_identity
    cvtpi2ps xmm0,QWORD ptr cx_
    mulps xmm0,XMMPS ptr ps_mul2
    movaps xmm2,xmm0
    xorps xmm0,XMMPS ptr ps_neg12
    movups XMMPS ptr matrix1,xmm1
    movlps XMMLPS ptr matrix1.m31,xmm0
    movups XMMPS ptr matrix3,xmm1
    movlps XMMLPS ptr matrix3.m31,xmm2
    ;/* combine transformations */
    fn mul_matrix,&matrix1,&matrix1,&matrix2
    fn mul_matrix,&matrix1,&matrix1,&matrix3
   
    ;/* get bitmap dimensions */
    fn GetObject,hBmp,BITMAP,&bmp
    mov ecx,bmp.bmWidth
    mov edx,bmp.bmHeight
    mov eax,edx
    .if SDWORD ptr eax < 0
        neg eax
    .endif
    mov w,ecx
    mov h,eax
    lea ecx,[ecx*DWORD]
    mov wm,ecx
   
    mov bmpinfo.bmiHeader.biSize,SIZEOF BITMAPINFO.bmiHeader
    mrm bmpinfo.bmiHeader.biWidth,bmp.bmWidth
    ;/* get top-down bitmap */
    .if SDWORD ptr edx > 0
        neg edx
    .endif
    mrm bmpinfo.bmiHeader.biHeight,edx
    mov bmpinfo.bmiHeader.biPlanes,1
    mov bmpinfo.bmiHeader.biBitCount,32
    mov bmpinfo.bmiHeader.biCompression,BI_RGB
    xor eax,eax
    mov bmpinfo.bmiHeader.biSizeImage,eax
    mov bmpinfo.bmiHeader.biXPelsPerMeter,eax
    mov bmpinfo.bmiHeader.biYPelsPerMeter,eax
    mov bmpinfo.bmiHeader.biClrUsed,eax
    mov bmpinfo.bmiHeader.biClrImportant,eax
   
    mov edx,w
    imul edx,h
    lea edx,[edx*4]
    IFDEF DEBUG
        mov cb,edx
    ENDIF
    push edx
    mov edi,alloc(edx) ; destination bitmap
    pop edx
    mov esi,alloc(edx) ; source bitmap
   
    fn GetDIBits,hdc,hBmp,0,h,esi,&bmpinfo,DIB_RGB_COLORS
   
    ;/* prepare matrix for mul. with transposed vector v=(x,y,1) */
    movups xmm6,XMMPS ptr matrix1
    shufps xmm6,xmm6,11011000y
    movlps xmm7,XMMLPS ptr matrix1.m31
    shufps xmm7,xmm7,01010000y
       
    ;/* xmm0 hold the x and y coordinates (mul. by 2) */
    movaps xmm0,XMMPS ptr ps_c0 ; xmm0 = {x=1,y=1,x=1,y=1}
   
    xor eax,eax
    xor ecx,ecx
    mov cr,eax
    .while eax < h
        xor ebx,ebx
        .while ebx < w
           
            ;/**
            ; * get source coordinate from current
            ; * destination coordinate (in xmm0).
            ; * (x,y)=v*matrix1
            ; */
            movaps xmm1,xmm0
            mulps xmm1,xmm6
            movaps xmm2,xmm1
            shufps xmm2,xmm2,11110101y
            addps xmm1,xmm2
            addps xmm1,xmm7 ; xmm1 = {-,ySrc,-,xSrc}
           
            ;/**
            ; * If the source coordinate is in range of the bitmap
            ; * copy the pixel data - otherwise store background color.
            ; *
            ; *   if( 0 <= (xs=RN(xSrc)/2) < w && 0 <= (ys=RN(ySrc)/2) < h)
            ; *      dest[x+y*w] = src[xs+ys*w];
            ; *   else
            ; *      dest[x+y*w] = bkColor;
            ; *
            ; * RN() = round to nearest, ties to even.
            ; */
            movhlps xmm2,xmm1
            cvtss2si eax,xmm1
            cvtss2si edx,xmm2
            .repeat
                shr eax,1
                shr edx,1
                .if SDWORD ptr eax >= 0 && eax < w
                    .if SDWORD ptr edx >= 0 && edx < h
                        lea eax,[eax*4]
                        imul edx,wm
                        lea eax,[eax+edx]
                        IFDEF DEBUG
                            .if eax >= cb
                                int 3
                            .endif
                        ENDIF
                        mov eax,[esi+eax]
                        mov [edi+ecx],eax
                        .break
                    .endif
                .endif
                 mrm DWORD ptr [edi+ecx],bkColor
            .until 1

            addps xmm0,ps_c2    ; xmm0 = {x=x+1,y,x=x+1,y}
            add ecx,DWORD
            add ebx,1
        .endw
        andps xmm0,XMMPS ptr ps_msk1
        addps xmm0,XMMPS ptr ps_c1      ; xmm0 = {x=0,y=y+1,x=0,y=y+1}
        mov eax,cr
        add eax,1
        mov cr,eax
    .endw
   
    free esi
    fnx hMemDC = CreateCompatibleDC,hdc
    fnx tmpBmp = CreateCompatibleBitmap,hdc,w,h
    fn SetDIBits,hdc,tmpBmp,0,h,edi,&bmpinfo,DIB_RGB_COLORS
    free edi
    fnx tmpBmp = SelectObject,hMemDC,tmpBmp
    fn BitBlt,hdc,x,y,w,h,hMemDC,0,0,SRCCOPY
    fnx tmpBmp = SelectObject,hMemDC,tmpBmp
    fn DeleteDC,hMemDC
    fn DeleteObject,tmpBmp
   
    ret
   
draw_image endp

WndProc proc uses ebx esi edi hWnd:HWND,uMsg:UINT,wParam:WPARAM,lParam:LPARAM
LOCAL ps:PAINTSTRUCT
LOCAL bmp:BITMAP
LOCAL angle:REAL4
LOCAL rect:RECT
LOCAL hdc:HDC,hBmp:HBITMAP

    .if uMsg == WM_CLOSE
        invoke DestroyWindow,hWnd
    .elseif uMsg == WM_DESTROY
        invoke DeleteObject,rv(GetWindowLong,hWnd,WND_DATA.hbmpImage)
        invoke PostQuitMessage,NULL
    .elseif uMsg == WM_CREATE
        .if rvx(ebx=LoadImage,0,"image.bmp",IMAGE_BITMAP,0,0,LR_DEFAULTSIZE or LR_LOADFROMFILE)
            invoke SetWindowLong,hWnd,WND_DATA.hbmpImage,ebx
            invoke SetTimer,hWnd,IDT,60,0
            xor eax,eax
        .else
            or eax,-1
        .endif
    .elseif uMsg == WM_TIMER && wParam == IDT
        invoke InvalidateRect,hWnd,0,0
        fnx angle = GetWindowLong,hWnd,WND_DATA.angle
        movss xmm0,angle
        addss xmm0,FP4(0.01)
        movss angle,xmm0
        fn SetWindowLong,hWnd,WND_DATA.angle,angle
        xor eax,eax
    .elseif uMsg == WM_PAINT
        fn GetClientRect,hWnd,&rect
        fn BeginPaint,hWnd,&ps
        fnx ebx = GetWindowLong,hWnd,WND_DATA.hbmpImage
        fnx angle = GetWindowLong,hWnd,WND_DATA.angle
       
        ;/* create offscreen buffer */
        fnx hBmp = CreateCompatibleBitmap,ps.hdc,rect.right,rect.bottom
        fnx hdc = CreateCompatibleDC,ps.hdc
        fnx hBmp = SelectObject,hdc,hBmp
       
        ;/* fill background */
        fn BitBlt,hdc,0,0,rect.right,rect.bottom,ps.hdc,0,0,PATCOPY
       
        fn GetObject,ebx,BITMAP,&bmp
        mov edx,bmp.bmHeight
        .if SDWORD ptr edx < 0
            neg edx
        .endif
        mov ecx,bmp.bmWidth
        shr edx,1
        shr ecx,1
       
        ;/* draw image */
        fn draw_image,hdc,ebx,10,10,ecx,edx,angle,0
       
        ;/* buffer -> screen-DC */
        invoke BitBlt,ps.hdc,0,0,rect.right,rect.bottom,hdc,0,0,SRCCOPY
       
        ;/* delete buffer */
        fnx hBmp = SelectObject,hdc,hBmp
        fn DeleteObject,hBmp
        fn DeleteDC,hdc
       
        fn EndPaint,hWnd,&ps
        xor eax,eax
    .else
        invoke DefWindowProc,hWnd,uMsg,wParam,lParam
    .endif
    ret
WndProc endp
end main


EDIT: fix a +-1 bug and rounding issues.
Title: Re: Rotation
Post by: nevraxSMD on March 31, 2014, 04:52:49 AM
qWord  :t you just  made my life a whole lot easier :D :t
Title: Re: Rotation
Post by: jj2007 on March 31, 2014, 05:02:59 AM
Quote from: qWord on March 31, 2014, 04:37:49 AM
Quote from: jj2007 on March 31, 2014, 04:00:48 AM
x' = x*cos(u)-y*sin(u)
y' = x*sin(u)-y*cos(u)
where : x' = is new x position, x - previous x position, u - angle in radians.

(posted yesterday in the Pelles C Forum) (http://forum.pellesc.de/index.php?topic=6129.0)
The formula is wrong and that is the reason why that guy get deformed objects :-D
It should be y' = x*sin(u)+y*cos(u).

I didn't check that - I was just surprised that exactly the same topic pops up at the same time in two different forums. Maybe half of the class had to do it in C, the other in Asm ;-)
Title: Re: Rotation
Post by: Gunther on March 31, 2014, 07:11:13 PM
Hi Jochen,

Quote from: jj2007 on March 31, 2014, 05:02:59 AM
I didn't check that ...

but qWord did and he's right.

Quote from: jj2007 on March 31, 2014, 05:02:59 AM
I was just surprised that exactly the same topic pops up at the same time in two different forums. Maybe half of the class had to do it in C, the other in Asm ;-)

that could be the reason. By the way, the right formula was already written in post #16 and 17. It's co-ordinate style.

The background of the entire thing is the rotation matrix:



cos (u)  -sin (u)
sin (u)   cos (u)
It lets your point with the co-ordinates (x;y) rotate by the angle u into the counter-clockwise direction. The following matrix:



cos (u)  sin (u)
-sin (u)   cos (u)
lets the same point rotate by the angle u into the clockwise direction. So you could do for the mathematical positive direction:

x' = x*cos(u) - y*sin(u)
y' = x*sin(u) + y*cos(u)

or for the clockwise direction:

x' = x*cos(u) + y*sin(u)
y' = -x*sin(u) + y*cos(u)


Both formulae are sometimes useful.

Gunther
Title: Re: Rotation
Post by: stefan.velicu on April 01, 2014, 05:34:41 AM
Thank you qWord. Sorry, but is a hard time for me to understand the program :D. Please, could you tell me how could I stop the rotation when reaching the proper angle?

Again, thank you very much, cheers!
Title: Re: Rotation
Post by: RuiLoureiro on April 01, 2014, 07:31:57 AM
Hi
we may use The calculator to see the results:

Point= (1,2)
Angle= 90 degrees

vetA=[ 1.0 ; 2.0];
matR=[ cosd(90), -sind(90); sind(90), cosd(90)];
matS=[ cosd(90), sind(90); -sind(90), cosd(90)];

vetB=matR*vetA;
vetC=matS*vetA;

we get:
            vetB=[ -2.0 ;1.0];
            vetC=[2.0   ; -1.0];

We may use complex numbers also.
Title: Re: Rotation
Post by: qWord on April 01, 2014, 08:55:59 AM
If upload new code above that fix rounding issues and a bug that cause read access outside the BMP data.
Quote from: stefan.velicu on April 01, 2014, 05:34:41 AM
Thank you qWord. Sorry, but is a hard time for me to understand the program :D
Actual the algorithm is very simple:
for each row in destination bitmap
{
    for each pixel (x,y) in current row
    {
        transform (x,y) to source coordinate (xs,ys);
        if( 0 <= xs < bmpWidth && 0 <= ys < bmpHeight )
             dest[x+bmpWidth*h] = src[xs+bmpWidth*hs];
        else
             dest[x+bmpWidth*h] = background color;
    }
}

For coordinates the pixel centers are used and not the upper left corner - means e.g. that the upper left pixel in the bitmap has the coordinate (0.5,0.5). Remarks that the new uploaded code does scale all coordinates by 2, to avoid rounding problems due to the limited precision.
For the XMM instruction see Iintel's and/or AMD's documentation.

Quote from: stefan.velicu on April 01, 2014, 05:34:41 AMPlease, could you tell me how could I stop the rotation when reaching the proper angle?
just call the function with the wished angle-parameter.
Title: Re: Rotation
Post by: jj2007 on April 01, 2014, 10:03:39 AM
Quote from: qWord on April 01, 2014, 08:55:59 AM
Quote from: stefan.velicu on April 01, 2014, 05:34:41 AMPlease, could you tell me how could I stop the rotation when reaching the proper angle?
just call the function with the wished angle-parameter.

Search the source for SetTimer, and have a look at line 345:

addss xmm0, FP4(0.01)  ; FP4 is a macro: Add Single precision xmm0, 0.01


Very nicely done, qWord :t
Title: Re: Rotation
Post by: Farabi on April 01, 2014, 04:03:22 PM
Hi, if you wanted to do manual rotation it is simle

x=sin(x)*witdhFromCenter
y=cos(x)*widthFromCenter

But I think you can use GPU for this so the computation will be very very fast.
Title: Re: Rotation
Post by: RuiLoureiro on April 01, 2014, 10:53:41 PM
Quote from: jj2007 on March 31, 2014, 04:00:48 AM
x' = x*cos(u)-y*sin(u)
y' = x*sin(u)-y*cos(u)
where : x' = is new x position, x - previous x position, u - angle in radians.

(posted yesterday in the Pelles C Forum) (http://forum.pellesc.de/index.php?topic=6129.0)

Note there is an instruction called fsincos. Very useful ;-)

:biggrin:
Hi Jochen,

When we want to rotate a complex number z
we need only to multiply it by e^ia
where 'a' is the rotation angle.

One point (x,y) may be seen as z=x+iy.

If 'a' is positive, we rotate it in the positive sense
(to the left); If 'a' is negative, we rotate it
in the negative sense (to the right).

Let be

    1. z=x+i
    2. r=e^ia = cos(a)+i sin(a)
    3. w=x'+iy' is the rotated complex number

To rotate z, we multiply it by r:

       w= (x+iy)*[cos(a)+i sin(a)]                   (i^2=-1)
        =
          [x*cos(a)-y*sin(a)]+i[x*sin(a)+y*cos(a)]

So, we get the following equation system
(the real part of w must be equal to the real part
and the imaginary part of w must be equal to the imaginary part)

     x'= x*cos(a)-y*sin(a)                 ( Equation system A )

     y'= x*sin(a)+y*cos(a)

In matricial form, we have this:

   | x' | = | cos(a)    -sin(a) | * | x |
   |    |    |                           |    |   |
   | y' |    | sin(a)     cos(a) |    | y |
   
To rotate in the negative sense, use '-a':

     x'= x*cos(-a)-y*sin(-a)       (note that cos(-a)=cos(a)
                                                          and sin(-a)=-sin(a))
     y'= x*sin(-a)+y*cos(-a)         

<=>

     x'= x*cos(a)+y*sin(a)                 ( Equation system A' )

     y'=-x*sin(a)+y*cos(a)
:t
Title: Re: Rotation
Post by: RuiLoureiro on April 04, 2014, 02:55:51 AM
Hi
    something more about rotation
   
So, when we want to transform (x,y) into (u,v),
we use

     x*cos(a)-y*sin(a) = u      ( Equation system A )

     x*sin(a)+y*cos(a) = v

Inverting this equation system, we have

     u*cos(a)+v*sin(a) = x      ( Equation system B )

    -u*sin(a)+v*cos(a) = y

which is, of course, «Equation system A'».
(we may get it from Equation system A changing x by u, y by v and a by -a)

It means this:

    if we start from a point (x,y)=(1,2)
    and we want to rotate it 90 degrees (a=90)
    to the left, using «Equation system A»
    we get the point (u,v)=(-2,1).
   
    Now, if we start from this last point (x,y)=(-2,1)
    and we want to rotate it -90 degrees (a=-90)
    to the left (we go to the right ), using
    «Equation system A» we get the first point (x,y)=(1,2).

    In the same way, if we use «Equation system B»
    with the same a=90 and the last point (u,v)=(-2,1)
    we get (x,y)=(1,2).