playing around and trying to write my first operating system I figured out how to close my loop and curious how to write and interrupt to end the loop on left click?
TITLE The New DOS
.model small
.stack 100h
WINDESC STRUCT
upperRow BYTE ?
leftCol BYTE ?
lowerRow BYTE ?
rightCol BYTE ?
foreColor BYTE ?
backColor BYTE ?
WINDESC ENDS
exit MACRO
mov ax, 4C00h
int 21h
ENDM
.data
application WINDESC <05h, 05h, 15h, 45h, 07h, 10h>
.code
curpos PROC
push bp
mov bp, sp
push ax
push bx
push dx
mov ax, 0200h
mov bh, 0
mov dx, [bp+4]
; interrupt
int 10h
pop dx
pop bx
pop ax
pop bp
ret 2
curpos ENDP
putchar PROC
push bp
mov bp, sp
push ax
push bx
push cx
push dx
pop dx
pop cx
pop bx
pop ax
pop bp
ret 2
putchar ENDP
makewin PROC
push bp
mov bp, sp
push ax
push bx
push cx
push dx
push si
mov si, [bp+4]
mov ax, 0600h
mov bh, (WINDESC PTR[si]) .backColor
mov ch, (WINDESC PTR[si]) .upperRow
mov cl, (WINDESC PTR[si]) .leftCol
mov dh, (WINDESC PTR[si]) .lowerRow
mov dl, (WINDESC PTR[si]) .rightCol
;interrupt
int 10h
push cx
call curpos
pop si
pop dx
pop cx
pop bx
pop ax
pop bp
; ret 2
jmp short main
makewin ENDP
main PROC
mov ax, @data
mov ds, ax
mov ax, OFFSET application
push ax
call makewin
exit
main ENDP
end main
Quote from: bcddd214 on December 07, 2011, 09:18:19 PM
makewin proc
;.......
ret 2
;;;jmp short main
makewin ENDP
main PROC
;.....
call makewin
call getchar
exit
main ENDP
end main
getchar PROC
mov ah,00h
int 16h ;this int wait user press a key in keyboard
ret
getchar endp
Not sure if this is what you have question, but have the same effect.
To assemble type:
ml /c /Cp newdos.asm
link16 newdos.obj
or
jwasm -mz newdos.asm
The interrupt is still not working.
I see you close my loop correctly. I am curious how your 'getchar' proc did that? I made a lucky guess on my 'jmp' previous code.
I am very curious how that worked.
and also curious how to get the first interrupt click to work.
PS, I am back on Irvine for now and winblows.
TITLE The New DOS
.model small
.stack 100h
WINDESC STRUCT
upperRow BYTE ?
leftCol BYTE ?
lowerRow BYTE ?
rightCol BYTE ?
foreColor BYTE ?
backColor BYTE ?
WINDESC ENDS
exit MACRO
mov ax, 4C00h
int 21h
ENDM
.data
application WINDESC <05h, 05h, 15h, 45h, 07h, 10h>
.code
curpos PROC
push bp
mov bp, sp
push ax
push bx
push dx
mov ax, 0200h
mov bh, 0
mov dx, [bp+4]
; interrupt
int 10h
pop dx
pop bx
pop ax
pop bp
ret 2
curpos ENDP
putchar PROC
push bp
mov bp, sp
push ax
push bx
push cx
push dx
pop dx
pop cx
pop bx
pop ax
pop bp
ret 2
putchar ENDP
getchar PROC
mov ah,00h
int 16h
ret
getchar ENDP
makewin PROC
push bp
mov bp, sp
push ax
push bx
push cx
push dx
push si
mov si, [bp+4]
mov ax, 0600h
mov bh, (WINDESC PTR[si]) .backColor
mov ch, (WINDESC PTR[si]) .upperRow
mov cl, (WINDESC PTR[si]) .leftCol
mov dh, (WINDESC PTR[si]) .lowerRow
mov dl, (WINDESC PTR[si]) .rightCol
;interrupt
int 10h
push cx
call curpos
pop si
pop dx
pop cx
pop bx
pop ax
pop bp
ret 2
; jmp short main
makewin ENDP
main PROC
mov ax, @data
mov ds, ax
mov ax, OFFSET application
push ax
call makewin
call getchar
exit
main ENDP
end main
When you have inserted that "jump" to a place outside of that procedure, you are creating what we call spaghetti code. A code that is hard to maintain, prone to errors (avalanche), and very hard to find mistakes or bugs (in your 1st example, unbalanced stack). When you get it simple, using call's , you are doing what we call 'modular project', easy to find errors, easy to read and implement the code, ... .
The rules of modular project is; comment your code(what the procedure expect, what is changes, what is returned, what that procedure do), save registers,... .
If the program wait you press a key before ends, so that interruption have worked, but we do not have displayed the returned char to the screen.
TITLE The New DOS
.model small
.stack 100h
WINDESC STRUCT
upperRow BYTE ?
leftCol BYTE ?
lowerRow BYTE ?
rightCol BYTE ?
foreColor BYTE ?
backColor BYTE ?
WINDESC ENDS
exit MACRO
mov ax, 4C00h
int 21h
ENDM
.data
application WINDESC <05h, 05h, 15h, 45h, 07h, 10h>
.code ;here start our program
main PROC ;main function
mov ax, @data
mov ds, ax
call makewin ;draw a box in screen
xor dx,dx ;dh=row,dl=column
call curpos ;put cursor at row,column position
call getchar ;wait and get a key
exit ; give control back to ms-dos
main ENDP
curpos PROC
;input: dh = row (0 based)
; dl = column (0 based)
;return: nothing
push ax
push bx
push dx
mov ax, 0200h
mov bh, 0 ;bh = video page
int 10h
pop dx
pop bx
pop ax
ret
curpos ENDP
getchar PROC
;wait user press a key
;input: nothing
;return: ah, al
;destroy: ax
mov ah,00h
int 16h
ret
getchar endp
makewin PROC
push ax
push bx
push cx
push dx
push si
mov si, OFFSET application
mov ax, 0600h
mov bh, (WINDESC PTR[si]) .backColor
mov ch, (WINDESC PTR[si]) .upperRow
mov cl, (WINDESC PTR[si]) .leftCol
mov dh, (WINDESC PTR[si]) .lowerRow
mov dl, (WINDESC PTR[si]) .rightCol
int 10h
pop si
pop dx
pop cx
pop bx
pop ax
ret
makewin ENDP
end main
You're trying to end the loop when the user clicks the left mouse button?
Quote from: mineiro on December 08, 2011, 04:18:43 PM
When you have inserted that "jump" to a place outside of that procedure, you are creating what we call spaghetti code. A code that is hard to maintain, prone to errors (avalanche), and very hard to find mistakes or bugs (in your 1st example, unbalanced stack). When you
[/quote]
I am afraid that the app does not end (break) on click.
But I also do not understand your response. I am aware that it is spaghetti code but I am trying to understand the mechanics. Before I jump'ed it into an obvious loop but now the app runs to the end of main and I would think that it would quit on it's own because it reaches endmain??
Just looking for some clearity.
mouse click still does not break the loop.
Is good know the mechanics bcddd214.
"end main" tell to masm where is the start point of your program, so when you run your .exe file, the first line of code that will be executed is what is inside "main proc". The flow of this program being executed by the processor is:
main PROC ;main function
mov ax, @data ;;here is the first line that will be executed when your program is loaded by the O.S.
mov ds, ax
call makewin ;this call enter inside makewin procedure, and inside that procedure have a ret, so, after return, the line of code below will be executed
xor dx,dx ;
call curpos ;if have a ret inside this procedure, so next line will be executed,
call getchar ;
exit ;this is the last line of code, program stop here.
main ENDP
Your program will not end if user do not press a key in keyboard (the flow of your program stay freezed in that respective function of "int 16h").
If user press a key, "int 16h" is not more hibernated, and continue the flow of code, and inside that procedure(getchar) have a ret, and the flow of code will return to the next line of code (below the caller, in this example, the macro "exit").
When I have said about spaggethi, I like to talk that is one way to program that you know where your program start's, but do not know where it ends. And believe in me, I have done so many spaghetti code in the past.
In your first example, when you have inserted that jmp, you have done something like:
main PROC ;main function
mov ax, @data ;;here is the first line that will be executed when your program is loaded by the O.S.
mov ds, ax
call makewin ;this call enter inside makewin procedure, and inside that procedure do not have a ret,
;inside makewin proc, have a jump to main, and this is like the line below
jmp main ;so program jump to entry point again, and never reach the end, stay in this loop forever.
xor dx,dx ;so, this line will never be executed, and lines below too, because line above jmp to start point again
call curpos ;never reach here too
call getchar ;
exit ;this program will never stop, because flow of execution never reach here
main ENDP
The int 16h is an interruption that get one key of keyboard, but have many different functions, in that example it waits user press a key (so, it's like a pause), if you don't like this, have another function that uses int 16h that get a key of keyboard without wait.
I do not know if this answer your question, if not, feel free to post. Like you can see, english is not my mother language, and I have difficult in say what I'm thinking.
This is not very good but it does show how to use the mouse driver to get the mouse button status and character coordinates. It works under Windows 2000 and should work under Windows XP, but it may not under more recent version of Windows. To see the character-mode mouse cursor you must run the application full-screen (use Alt+Enter to toggle between windowed and full-screen, or right-click the EXE and set the properties).
;==============================================================================
.model small
.386
;==============================================================================
.stack
.data
msgNoMouseDriver db "no mouse driver",13,10,"$"
msgPressAnyKey db "press any key to exit",13,10,"$"
.code
.startup
;==============================================================================
;-------------------------------------------------------------------
; With MASM the mouse driver functions are called with the function
; number in AX, and depending on the function, the return values may
; be in AX, BX, CX, or DX.
;-------------------------------------------------------------------
;----------------------------------------------------------------------
; Check for a mouse driver. Use the DOS Get Interrupt Vector function
; to get the interrupt 33h vector. If the segment address is zero then
; the mouse driver is not installed.
;----------------------------------------------------------------------
mov ax,3533h
int 21h
mov ax, es
.IF ax == 0
mov ah, 9
mov dx, OFFSET msgNoMouseDriver
int 21h
jmp exit
.ENDIF
;-------------------------------------------------------------------------
; Attempt to reset the mouse driver by calling the Mouse Reset And Status
; function. If the reset fails, indicated by the function returning zero,
; then the mouse driver is not installed.
;-------------------------------------------------------------------------
xor ax, ax
int 33h
.IF ax == 0
mov ah, 9
mov dx, OFFSET msgNoMouseDriver
int 21h
jmp exit
.ENDIF
;------------------------------------------------------------------
; Show the mouse cursor by calling the Mouse Show Cursor function.
;------------------------------------------------------------------
mov ax, 1
int 33h
;-----------------------------------------------------------------------
; Loop, calling the Mouse Get Button Status And Mouse Position function
; until the user presses the left mouse button with the mouse cursor
; at character coordinate 0,0 (the first character at the upper-left
; corner of screen). The function returns the button status in BX, the
; horizontal cursor coordinate in CX, and the vertical cursor coordinate
; in DX. The button status for the left mouse button is returned in
; bit 0 of BX, and the status for the right mouse button in bit 1, with
; the bit set if the button is pressed or cleared if the button is
; released. The cursor coordinates are returned as mouse-driver virtual-
; screen coordinates, where the virtual screen for most of the display
; modes is 640x200. For the 80x25 character (AKA "text") modes you
; convert the virtual-screen coordinates to character coordinates by
; simply dividing the virtual-screen coordinates by 8.
;-----------------------------------------------------------------------
@@:
mov ax, 3
int 33h
.IF bx & 1 ; left mouse button is pressed
shr cx, 3
shr dx, 3
.IF cx || dx ; mouse cursor not over character 0,0 so keep looping
jmp @B
.ENDIF
.ELSE
jmp @B ; left mouse button not pressed so keep looping
.ENDIF
exit:
mov ah, 9
mov dx, OFFSET msgPressAnyKey
int 21h
xor ah, ah
int 16h
.exit
end
Quote from: mineiro on December 09, 2011, 01:51:58 AM
Is good know the mechanics bcddd214.
Before in the makewin proc, when it got to ret 2, it just stopped. Now since the keyboard interrupt was put into place (I realize what you published was for keyboard and not mouse now), the blue box stays in a loop, why?
I can see what closed the loop?
I am working with your mouse software now, thank you. :)
I am not sure why it gives these error?
Assembling: newDOS2.asm
newDOS2.asm(38) : error A2108: use of register assumed to ERROR
newDOS2.asm(47) : error A2108: use of register assumed to ERROR
newDOS2.asm(62) : error A2108: use of register assumed to ERROR
newDOS2.asm(91) : error A2070: invalid instruction operands
newDOS2.asm(92) : error A2070: invalid instruction operands
newDOS2.asm(93) : error A2108: use of register assumed to ERROR
newDOS2.asm(96) : error A2108: use of register assumed to ERROR
newDOS2.asm(98) : error A2108: use of register assumed to ERROR
newDOS2.asm(100) : error A2108: use of register assumed to ERROR
newDOS2.asm(42) : error A2107: cannot have implicit far jump or call to near lab
el
newDOS2.asm(46) : error A2148: invalid symbol type in expression : exit2
newDOS2.asm(57) : error A2107: cannot have implicit far jump or call to near lab
el
newDOS2.asm(61) : error A2148: invalid symbol type in expression : exit2
newDOS2.asm(90) : error A2107: cannot have implicit far jump or call to near lab
el
newDOS2.asm(93) : error A2107: cannot have implicit far jump or call to near lab
el
newDOS2.asm(94) : error A2006: undefined symbol : @@
newDOS2.asm(97) : error A2006: undefined symbol : @@
newDOS2.asm(39) : warning A4012: line number information for segment without cla
ss 'CODE' : _DATA
*********************************************************
TITLE The New DOS
.model small
.stack 100h
WINDESC STRUCT
upperRow BYTE ?
leftCol BYTE ?
lowerRow BYTE ?
rightCol BYTE ?
foreColor BYTE ?
backColor BYTE ?
WINDESC ENDS
exit2 MACRO
mov ax, 4C00h
int 21h
ENDM
.data
application WINDESC <05h, 05h, 15h, 45h, 07h, 10h>
msgNoMouseDriver db "no mouse driver",13,10,"$"
msgPressAnyKey db "press any key to exit",13,10,"$"
;==============================================================================
;-------------------------------------------------------------------
; With MASM the mouse driver functions are called with the function
; number in AX, and depending on the function, the return values may
; be in AX, BX, CX, or DX.
;-------------------------------------------------------------------
;----------------------------------------------------------------------
; Check for a mouse driver. Use the DOS Get Interrupt Vector function
; to get the interrupt 33h vector. If the segment address is zero then
; the mouse driver is not installed.
;----------------------------------------------------------------------
.startup
mov ax,3533h
int 21h
mov ax, es
.IF ax == 0
mov ah, 9
mov dx, OFFSET msgNoMouseDriver
int 21h
jmp exit2
.ENDIF
;-------------------------------------------------------------------------
; Attempt to reset the mouse driver by calling the Mouse Reset And Status
; function. If the reset fails, indicated by the function returning zero,
; then the mouse driver is not installed.
;-------------------------------------------------------------------------
xor ax, ax
int 33h
.IF ax == 0
mov ah, 9
mov dx, OFFSET msgNoMouseDriver
int 21h
jmp exit2
.ENDIF
;------------------------------------------------------------------
; Show the mouse cursor by calling the Mouse Show Cursor function.
;------------------------------------------------------------------
mov ax, 1
int 33h
;-----------------------------------------------------------------------
; Loop, calling the Mouse Get Button Status And Mouse Position function
; until the user presses the left mouse button with the mouse cursor
; at character coordinate 0,0 (the first character at the upper-left
; corner of screen). The function returns the button status in BX, the
; horizontal cursor coordinate in CX, and the vertical cursor coordinate
; in DX. The button status for the left mouse button is returned in
; bit 0 of BX, and the status for the right mouse button in bit 1, with
; the bit set if the button is pressed or cleared if the button is
; released. The cursor coordinates are returned as mouse-driver virtual-
; screen coordinates, where the virtual screen for most of the display
; modes is 640x200. For the 80x25 character (AKA "text") modes you
; convert the virtual-screen coordinates to character coordinates by
; simply dividing the virtual-screen coordinates by 8.
;-----------------------------------------------------------------------
@@:
mov ax, 3
int 33h
.IF bx & 1 ; left mouse button is pressed
shr cx, 3
shr dx, 3
.IF cx || dx ; mouse cursor not over character 0,0 so keep looping
jmp @B
.ENDIF
.ELSE
jmp @B ; left mouse button not pressed so keep looping
.ENDIF
exit:
mov ah, 9
mov dx, OFFSET msgPressAnyKey
int 21h
xor ah, ah
int 16h
.exit
end
.code
curpos PROC
push bp
mov bp, sp
push ax
push bx
push dx
mov ax, 0200h
mov bh, 0
mov dx, [bp+4]
; interrupt
int 10h
pop dx
pop bx
pop ax
pop bp
ret 2
curpos ENDP
putchar PROC
push bp
mov bp, sp
push ax
push bx
push cx
push dx
pop dx
pop cx
pop bx
pop ax
pop bp
ret 2
putchar ENDP
getchar PROC
mov ah,00h
int 16h
ret
getchar ENDP
makewin PROC
push bp
mov bp, sp
push ax
push bx
push cx
push dx
push si
mov si, [bp+4]
mov ax, 0600h
mov bh, (WINDESC PTR[si]) .backColor
mov ch, (WINDESC PTR[si]) .upperRow
mov cl, (WINDESC PTR[si]) .leftCol
mov dh, (WINDESC PTR[si]) .lowerRow
mov dl, (WINDESC PTR[si]) .rightCol
;interrupt
int 10h
push cx
call curpos
pop si
pop dx
pop cx
pop bx
pop ax
pop bp
ret 2
; jmp short main
makewin ENDP
main PROC
mov ax, @data
mov ds, ax
mov ax, OFFSET application
push ax
call makewin
call getchar
exit
main ENDP
end main
surround your code in code tags!
you have code "outside" the code sections.
Right above curpos PROC you have .code and above that you have code. remove that .code and add it right above .startup
Thank you, I didn't see that.
now it is erroring on line 93 and 94 on the shr directive. I have never seen that one before so I would not even know were to begin.
Assembling: newDOS2.asm
newDOS2.asm(93) : error A2070: invalid instruction operands
newDOS2.asm(94) : error A2070: invalid instruction operands
newDOS2.asm(48) : error A2148: invalid symbol type in expression : exit2
newDOS2.asm(63) : error A2148: invalid symbol type in expression : exit2
Press any key to continue . . .
************************************
TITLE The New DOS
.model small
.stack 100h
WINDESC STRUCT
upperRow BYTE ?
leftCol BYTE ?
lowerRow BYTE ?
rightCol BYTE ?
foreColor BYTE ?
backColor BYTE ?
WINDESC ENDS
exit2 MACRO
mov ax, 4C00h
int 21h
ENDM
.data
application WINDESC <05h, 05h, 15h, 45h, 07h, 10h>
msgNoMouseDriver db "no mouse driver",13,10,"$"
msgPressAnyKey db "press any key to exit",13,10,"$"
;==============================================================================
;-------------------------------------------------------------------
; With MASM the mouse driver functions are called with the function
; number in AX, and depending on the function, the return values may
; be in AX, BX, CX, or DX.
;-------------------------------------------------------------------
;----------------------------------------------------------------------
; Check for a mouse driver. Use the DOS Get Interrupt Vector function
; to get the interrupt 33h vector. If the segment address is zero then
; the mouse driver is not installed.
;----------------------------------------------------------------------
.code
.startup
mov ax,3533h
int 21h
mov ax, es
.IF ax == 0
mov ah, 9
mov dx, OFFSET msgNoMouseDriver
int 21h
jmp exit2
.ENDIF
;-------------------------------------------------------------------------
; Attempt to reset the mouse driver by calling the Mouse Reset And Status
; function. If the reset fails, indicated by the function returning zero,
; then the mouse driver is not installed.
;-------------------------------------------------------------------------
xor ax, ax
int 33h
.IF ax == 0
mov ah, 9
mov dx, OFFSET msgNoMouseDriver
int 21h
jmp exit2
.ENDIF
;------------------------------------------------------------------
; Show the mouse cursor by calling the Mouse Show Cursor function.
;------------------------------------------------------------------
mov ax, 1
int 33h
;-----------------------------------------------------------------------
; Loop, calling the Mouse Get Button Status And Mouse Position function
; until the user presses the left mouse button with the mouse cursor
; at character coordinate 0,0 (the first character at the upper-left
; corner of screen). The function returns the button status in BX, the
; horizontal cursor coordinate in CX, and the vertical cursor coordinate
; in DX. The button status for the left mouse button is returned in
; bit 0 of BX, and the status for the right mouse button in bit 1, with
; the bit set if the button is pressed or cleared if the button is
; released. The cursor coordinates are returned as mouse-driver virtual-
; screen coordinates, where the virtual screen for most of the display
; modes is 640x200. For the 80x25 character (AKA "text") modes you
; convert the virtual-screen coordinates to character coordinates by
; simply dividing the virtual-screen coordinates by 8.
;-----------------------------------------------------------------------
@@:
mov ax, 3
int 33h
.IF bx & 1 ; left mouse button is pressed
shr cx, 3
shr dx, 3
.IF cx || dx ; mouse cursor not over character 0,0 so keep looping
jmp @B
.ENDIF
.ELSE
jmp @B ; left mouse button not pressed so keep looping
.ENDIF
exit:
mov ah, 9
mov dx, OFFSET msgPressAnyKey
int 21h
xor ah, ah
int 16h
.exit
end
curpos PROC
push bp
mov bp, sp
push ax
push bx
push dx
mov ax, 0200h
mov bh, 0
mov dx, [bp+4]
; interrupt
int 10h
pop dx
pop bx
pop ax
pop bp
ret 2
curpos ENDP
putchar PROC
push bp
mov bp, sp
push ax
push bx
push cx
push dx
pop dx
pop cx
pop bx
pop ax
pop bp
ret 2
putchar ENDP
getchar PROC
mov ah,00h
int 16h
ret
getchar ENDP
makewin PROC
push bp
mov bp, sp
push ax
push bx
push cx
push dx
push si
mov si, [bp+4]
mov ax, 0600h
mov bh, (WINDESC PTR[si]) .backColor
mov ch, (WINDESC PTR[si]) .upperRow
mov cl, (WINDESC PTR[si]) .leftCol
mov dh, (WINDESC PTR[si]) .lowerRow
mov dl, (WINDESC PTR[si]) .rightCol
;interrupt
int 10h
push cx
call curpos
pop si
pop dx
pop cx
pop bx
pop ax
pop bp
ret 2
; jmp short main
makewin ENDP
main PROC
mov ax, @data
mov ds, ax
mov ax, OFFSET application
push ax
call makewin
call getchar
exit
main ENDP
end main
Quote from: bcddd214 on December 12, 2011, 03:15:27 AM
Thank you, I didn't see that.
now it is erroring on line 93 and 94 on the shr directive. I have never seen that one before so I would not even know were to begin.
One more question. I have a macro called main exit and the mouse driver appears to have a sub procedure called exit also. I don't think my exit macro is doing anything so I just renamed it as a quick fix. I am getting an error there too as you can see.
Obviously you can't have a macro and a routine both called exit, can you? What is the best way to handle this error free?
Quote from: bcddd214 on December 12, 2011, 03:21:43 AM
I have a macro called main exit and the mouse driver appears to have a sub procedure called exit also. I don't think my exit macro is doing anything so I just renamed it as a quick fix. I am getting an error there too as you can see.
Obviously you can't have a macro and a routine both called exit, can you? What is the best way to handle this error free?
Apply "namespace sensitivity". Masm itself, plus Masm32, CRT, MasmBasic and similar libraries occupy a namespace, and you better try to avoid using the most common names for your own stuff:
print
exit
str$
for ... next
repeat ... until
... whatever - the list is long, see e.g. Appendix D, "MASM Reserved Words" of the Masm Programmer's guide.
The easiest rule is call it MyExit, not exit or EXIT or Exit.
still trying to figure out how to get rid of the error at line 93/94 with shr cx, 3
what is shr and how do I use it correctly?
Quote from: bcddd214 on December 12, 2011, 01:10:19 PM
still trying to figure out how to get rid of the error at line 93/94 with shr cx, 3
what is shr and how do I use it correctly?
Hi,
MASM defaults to the 8086/88 instruction set. SHR CX,3 is then
illeagle. You need to put a .386 (or .186 or similar) like MichaelW
did. Or you can use three SHR CX,1 instead. Fun eh? SHR is
the SHift Right instruction.
HTH,
Steve N.
.186 errors really bad but .386 did better
I found another these on the form as possible solutions
.386p
.model small,Flat,StdCALL
.stack 100h
OPTION CaseMap:None
.MMX
.XMM
but now getting really bad 32 bit errors. I am trying to stay in 16 bit for simplicity and can't figure out what is trying to flip me over to 32 bit and making my life even more difficult... :)
Errors
Assembling: newDOS2.asm
newDOS2.asm(48) : error A2199: .STARTUP does not work with 32-bit segments
newDOS2.asm(119) : error A2198: .EXIT does not work with 32-bit segments
newDOS2.asm(54) : error A2022: instruction operands must be the same size
newDOS2.asm(69) : error A2022: instruction operands must be the same size
newDOS2.asm(113) : error A2022: instruction operands must be the same size
*********************************************************************
TITLE The New DOS
.386p
.model small,Flat,StdCALL
.stack 100h
OPTION CaseMap:None
.MMX
.XMM
; Set the window variable framework
WINDESC STRUCT
upperRow BYTE ?
leftCol BYTE ?
lowerRow BYTE ?
rightCol BYTE ?
foreColor BYTE ?
backColor BYTE ?
WINDESC ENDS
; Not really sure about this one
exit MACRO
mov ax, 4C00h
int 21h
ENDM
;set variables
.data
application WINDESC <05h, 05h, 15h, 45h, 07h, 10h>
msgNoMouseDriver db "no mouse driver",13,10,"$"
msgPressAnyKey db "press any key to exit",13,10,"$"
;==============================================================================
;-------------------------------------------------------------------
; With MASM the mouse driver functions are called with the function
; number in AX, and depending on the function, the return values may
; be in AX, BX, CX, or DX.
;-------------------------------------------------------------------
;----------------------------------------------------------------------
; Check for a mouse driver. Use the DOS Get Interrupt Vector function
; to get the interrupt 33h vector. If the segment address is zero then
; the mouse driver is not installed.
;----------------------------------------------------------------------
.code
;mouse driver
.startup
mov ax,3533h
int 21h
mov ax, es
.IF ax == 0
mov ah, 9
mov dx, OFFSET msgNoMouseDriver
int 21h
jmp myExit
.ENDIF
;-------------------------------------------------------------------------
; Attempt to reset the mouse driver by calling the Mouse Reset And Status
; function. If the reset fails, indicated by the function returning zero,
; then the mouse driver is not installed.
;-------------------------------------------------------------------------
xor ax, ax
int 33h
.IF ax == 0
mov ah, 9
mov dx, OFFSET msgNoMouseDriver
int 21h
jmp myExit
.ENDIF
;------------------------------------------------------------------
; Show the mouse cursor by calling the Mouse Show Cursor function.
;------------------------------------------------------------------
mov ax, 1
int 33h
;-----------------------------------------------------------------------
; Loop, calling the Mouse Get Button Status And Mouse Position function
; until the user presses the left mouse button with the mouse cursor
; at character coordinate 0,0 (the first character at the upper-left
; corner of screen). The function returns the button status in BX, the
; horizontal cursor coordinate in CX, and the vertical cursor coordinate
; in DX. The button status for the left mouse button is returned in
; bit 0 of BX, and the status for the right mouse button in bit 1, with
; the bit set if the button is pressed or cleared if the button is
; released. The cursor coordinates are returned as mouse-driver virtual-
; screen coordinates, where the virtual screen for most of the display
; modes is 640x200. For the 80x25 character (AKA "text") modes you
; convert the virtual-screen coordinates to character coordinates by
; simply dividing the virtual-screen coordinates by 8.
;-----------------------------------------------------------------------
@@:
mov ax, 3
int 33h
.IF bx & 1 ; left mouse button is pressed
shr cx, 3
shr dx, 3
.IF cx || dx ; mouse cursor not over character 0,0 so keep looping
jmp @B
.ENDIF
.ELSE
jmp @B ; left mouse button not pressed so keep looping
.ENDIF
myExit:
mov ah, 9
mov dx, OFFSET msgPressAnyKey
int 21h
xor ah, ah
int 16h
.exit
end
;setting the cursor position
curpos PROC
push bp
mov bp, sp
push ax
push bx
push dx
mov ax, 0200h
mov bh, 0
mov dx, [bp+4]
; interrupt
int 10h
pop dx
pop bx
pop ax
pop bp
ret 2
curpos ENDP
;still unused
putchar PROC
push bp
mov bp, sp
push ax
push bx
push cx
push dx
pop dx
pop cx
pop bx
pop ax
pop bp
ret 2
putchar ENDP
getchar PROC
mov ah,00h
int 16h
ret
getchar ENDP
;window construction
makewin PROC
push bp
mov bp, sp
push ax
push bx
push cx
push dx
push si
mov si, [bp+4]
mov ax, 0600h
mov bh, (WINDESC PTR[si]) .backColor
mov ch, (WINDESC PTR[si]) .upperRow
mov cl, (WINDESC PTR[si]) .leftCol
mov dh, (WINDESC PTR[si]) .lowerRow
mov dl, (WINDESC PTR[si]) .rightCol
;interrupt
int 10h
push cx
call curpos
pop si
pop dx
pop cx
pop bx
pop ax
pop bp
ret 2
; jmp short main
makewin ENDP
main PROC
mov ax, @data
mov ds, ax
mov ax, OFFSET application
push ax
call makewin
call getchar
exit
main ENDP
end main
anyone know how to make shr or shift right work?
TITLE The New DOS
.model small
.386
.stack 100h
;mouse driver
mouse proc
mov ax,3533h
int 21h
mov ax, es
.IF ax == 0
mov ah, 9
mov dx, OFFSET msgNoMouseDriver
int 21h
jmp myExit
.ENDIF
;-------------------------------------------------------------------------
; Attempt to reset the mouse driver by calling the Mouse Reset And Status
; function. If the reset fails, indicated by the function returning zero,
; then the mouse driver is not installed.
;-------------------------------------------------------------------------
xor ax, ax
int 33h
.IF ax == 0
mov ah, 9
mov dx, OFFSET msgNoMouseDriver
int 21h
jmp myExit
.ENDIF
;------------------------------------------------------------------
; Show the mouse cursor by calling the Mouse Show Cursor function.
;------------------------------------------------------------------
mov ax, 1
int 33h
;-----------------------------------------------------------------------
; Loop, calling the Mouse Get Button Status And Mouse Position function
; until the user presses the left mouse button with the mouse cursor
; at character coordinate 0,0 (the first character at the upper-left
; corner of screen). The function returns the button status in BX, the
; horizontal cursor coordinate in CX, and the vertical cursor coordinate
; in DX. The button status for the left mouse button is returned in
; bit 0 of BX, and the status for the right mouse button in bit 1, with
; the bit set if the button is pressed or cleared if the button is
; released. The cursor coordinates are returned as mouse-driver virtual-
; screen coordinates, where the virtual screen for most of the display
; modes is 640x200. For the 80x25 character (AKA "text") modes you
; convert the virtual-screen coordinates to character coordinates by
; simply dividing the virtual-screen coordinates by 8.
;-----------------------------------------------------------------------
@@:
mov ax, 3
int 33h
.IF bx & 1 ; left mouse button is pressed
shr cx, 3
shr dx, 3
.IF cx || dx ; mouse cursor not over character 0,0 so keep looping
jmp @B
.ENDIF
.ELSE
jmp @B ; left mouse button not pressed so keep looping
.ENDIF
myExit:
mov ah, 9
mov dx, OFFSET msgPressAnyKey
int 21h
xor ah, ah
int 16h
ret
;.exit
;end
mouse endp
main PROC
mov ax, @data
mov ds, ax
mov ax, OFFSET application
push ax
call makewin
call mouse
; call getchar
exit
main ENDP
end main
Here is my code:
http://pastebin.com/uWuTGrSv
and here is the same errors. I would really appreciate getting this to compile so I can move on... :(
This is compiling in 32bit
Assembling: newDOS2.asm
newDOS2.asm(136) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(183) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(186) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(187) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(188) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(189) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(190) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(56) : error A2022: instruction operands must be the same size
newDOS2.asm(71) : error A2022: instruction operands must be the same size
newDOS2.asm(115) : error A2022: instruction operands must be the same size
newDOS2.asm(211) : error A2022: instruction operands must be the same size
newDOS2.asm(208) : error A2004: symbol type conflict
and this is in 16 bit
Assembling: newDOS2.asm
newDOS2.asm(136) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(183) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(186) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(187) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(188) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(189) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(190) : error A2155: cannot use 16-bit register with a 32-bit address
newDOS2.asm(56) : error A2022: instruction operands must be the same size
newDOS2.asm(71) : error A2022: instruction operands must be the same size
newDOS2.asm(115) : error A2022: instruction operands must be the same size
newDOS2.asm(211) : error A2022: instruction operands must be the same size
newDOS2.asm(208) : error A2178: invalid use of FLAT
Hi,
Look at the first code block in reply #18. Compare to your
posted line 4. .MODEL FLAT is incompatible with 16-bit code
(.MODEL SMALL). Change to the code like that posted by
mineiro. And *.COM programs are normally .TINY though
.SMALL is harmless.
Steve
Ok, major progress. I compile in 16 bit with this:
TITLE The New DOS
.386p
.model small
.stack 100h
OPTION CaseMap:None
.MMX
.XMM
but I am getting mouse driver errors. I do not know this driver code enough to know what it wants re sized. I am guessing it barking at a frame size error? Who's frame is it drawing from?
Error:
C:\Users\wuzamarine\Desktop\asm\asm-progs>make16 newDOS2
Assembling: newDOS2.asm
newDOS2.asm(55) : error A2022: instruction operands must be the same size
newDOS2.asm(70) : error A2022: instruction operands must be the same size
newDOS2.asm(114) : error A2022: instruction operands must be the same size
newDOS2.asm(210) : error A2022: instruction operands must be the same size
Press any key to continue . . .
Read this (http://www.masm32.com/board/index.php?topic=17908.0) topic, one of the error are exactly yours. Follow the answers of Sr MichaelW.
I see half of what you are pointing at. My offset is off. It is looking for a different size. I am just not connecting the two...? :(
the processor ".386" directive should appear after the .MODEL directive
if you specify a 32-bit processor before the .MODEL directive, MASM tries to make a 32-bit program
furthermore - not sure you want .MXX and .XMM :P
at least, not with a 386 processor
error free now but the mouse click still does not interrupt, only keyboard.
http://pastebin.com/SuZ8XzD1
i don't see where the "mouse" proc is ever called :P
lordy, you are correct. :)
that happens :bg
http://pastebin.com/qcN9RxWS
still no mouse click and the program hangs. no errors, just an inescapable hang.
it has been a long time since i played with 16-bit mouse stuff and INT 33h
but, as i remember, it allows you to install your own interrupt handler
let me review the documentation and get back to you...
Thank you!
I got all the pieces parts now, but making them all fit together neatly and functionally is my last leg of this madness.
She is coming together slowly though...
code update.
http://pastebin.com/vm3n1LuP
here is a little test program
see if it works for you...
looking at your code, maybe there is something wrong with the IF/ELSE stuff
as you can see, i don't generally use those :P
Does the replace my text input?
Just dump in main like the code, what do I delete?
hang on - my code seems to have a bug - lol
it worked ok until i put the left-button stuff in there
let me play with it....
strange behaviour
it works ok if i run the program by clicking on it in explorer
but, if i type the name at the prompt, it exits immediately
Is that just using your code exclusively or are you trying to run it from within mine?
Mine is pretty buggy too. That is the stage I am at, final debugging. :(
well - emulated 16-bit is not an ideal place to develop "cool" applications - lol
but - this should work
i made several changes to mine to see if i could isolate it
try it both ways and let me know what happens....
i am using XP pro SP3
you code appears to be work, it compiles fine and runs error free.
I guess I am not following as to what I should replace this code with?
my ascii input or mouse drives?
It appears to be the replacement for ascii and that should fix my hang?
wellllll
i wanted to test our interpretation of INT 33h function 3
your code looks like it ought to work ok, unless i am reading the IF/ELSE/ENDOF stuff wrong :P
so - i wanted to verify that what i think should happen with the function is what is happening
once we verify that - we'll have a look at the IF/ELSE/ENDOF stuff
perhaps with DEBUG, or by looking at the listing
let me get a nap in :bg
i will play with yours afterwards to see what i can see
I am not the debug master of asm yet but I did figure out how to do a memory dump using debug. I am not sure what this all means, but it's a start.
-d 0 8f
17F2:0000 CD 20 FF 9F 00 9A EE FE-1D F0 4F 03 56 12 8A 03 . ........O.V...
17F2:0010 56 12 17 03 56 12 2C 07-01 01 01 00 02 FF FF FF V...V.,.........
17F2:0020 FF FF FF FF FF FF FF FF-FF FF FF FF DA 11 4E 01 ..............N.
17F2:0030 16 17 14 00 18 00 F2 17-FF FF FF FF 00 00 00 00 ................
17F2:0040 05 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................
17F2:0050 CD 21 CB 00 00 00 00 00-00 00 00 00 00 20 20 20 .!...........
17F2:0060 20 20 20 20 20 20 20 20-00 00 00 00 00 20 20 20 .....
17F2:0070 20 20 20 20 20 20 20 20-00 00 00 00 00 00 00 00 ........
17F2:0080 00 0D 6E 65 77 44 4F 53-32 0D 41 32 32 30 20 49 ..newDOS2.A220 I
It is definitely the mouse driver creating the hang, if I rem out the calling in main for the mouse driver, it does not hang anymore.
ok - i had my power-nap :bg
let's have a look
i will use SYMDEB - it's DEBUG on steroids
the biggest improvement is that with SYMDEB - you can set breakpoints and execute code til it hits one
i see Michael is playing with similar code on the other thread :P
he is right
your code does work
thing is - for it to exit via the mouse click, you then have to exit via a key stroke
that is because of the program flow
you call waitkey
then, you call mouse
then, you call getchar
if you look at the loop i wrote, it checks the mouse status, then checks for a key-press, then back to mouse status
it continues to loop until either one of them occurs
you just need to change your pogram flow a little bit
write a routine that checks both mouse and keyboard
are you referring to the myma16.zip code?
I figured it out. wow, how encrypt of a release.
Whats the best way to clean it up?
well - i am not sure what you want to happen
tell us exactly what you expect - exactly what you are trying to do
i do see a problem
the curpos routine expects a parameter be passed on the stack
however, you have not passed one
mov al,application.leftCol
mov ah,application.upperRow
push ax
call curpos
In main put that?
That fixed my cursor position, but still see no text when I type?
that's because you do not make any function calls to get user input
YUP!
and now I remember, that what the unused putchar PROC was put there for. :)
Do you have a template I can put into the proc?
well - i suggest INT 21h, function AH = 0Ah
.DATA
BufLen db 20
RetLen db 0
BufInp db 21 dup(?)
.CODE
mov dx,offset BufLen
mov ah,0Ah
int 21h
the user cannot enter more than the number of characters specified by BufLen
when the function is done, RetLen has the number of characters entered by the user
it is not terminated with a 0 or $ - it is terminated with a 0Dh as i remember - which is a carriage return
notice that we allow one more character in the buffer than BufLen - that is for the carriage return
again - it has been a while since i used with this
my details may be off slightly and you may have to play with it a little to get the hang of it
also - we could make a structure out of the input buffer to make it pretty :P
http://pastebin.com/q9GSQ2ZQ
That had no effect my friend
that's because you have a program flow issue, again
that code is never executed
let me put together a crude logical so you can help me see my error. brb
what is the assignment, EXACTLY
ok, point!! :)
here is a logical overview that should make it easier to point out my error to me...
globals
WINDESC (make window)
.data
user input buffer
mouse db
.code
buffer length
mouse proc
myExit (gather this is where my click out issue lays)
waitkey (more of my clickout issue possibly)
curpos (cursor position)
putchar (unused)
getchar (not sure)
Makewin (make the window)
Filler (new and needs fixing)
border (new and needs fixing)
main PROC
mov ax, @data
mov ds, ax
mov ax, OFFSET application
push ax
call makewin
mov al,application.leftCol
mov ah,application.upperRow
push ax
call curpos
call waitkey
call mouse
call getchar
to produce a text editor that you can type in and then click a button to exit the loop with a pretty border. I am at 'the last screw' :)
can someone kindly point me in the correct direction of my logical error?
characters still do not type on the screen for some reason???