The MASM Forum Archive 2004 to 2012

Miscellaneous Forums => 16 bit DOS Programming => Topic started by: nrdev on May 25, 2009, 04:51:58 PM

Title: drawing
Post by: nrdev on May 25, 2009, 04:51:58 PM
what functions should I use for drawing?
Title: Re: drawing
Post by: FORTRANS on May 25, 2009, 05:31:55 PM
Hi,

   Not sure exactly what you want, but here goes.  the BIOS INT 10H
functions control the video.  Function 0 (zero) sets the video mode.
Function 0CH is the one that sets a pixel to a given value.


   This will start a graphics program.

        MOV     AH,0    ; Set Mode
        MOV     AL,13H  ; 320 x 200 x 256 colors
        INT     10H     ; Video interrupt.

   Or try.

        MOV     AL,12H  ; 640 x 480 x 16 colors

   This will set a pixel.

        MOV     AH,0CH          ; Set pixel function.
        MOV     AL,[Color]      ; Pixel value or color
        MOV     BH,0            ; Page for some modes.
        MOV     CX,[Column]     ; X-coordinate
        MOV     DX,[Row]        ; Y-coordinate
        INT     10H

   This will reset to text at the end.

        MOV     AH,0    ; Set Video mode
        MOV     AL,3    ; = Standard video mode 3
        INT     10H

HTH,

Steve N.
Title: Re: drawing
Post by: nrdev on May 25, 2009, 05:55:40 PM
I meant on what functions, or what should I do to draw point, line, rectangle, and filled rectangle.
By the way third part of code that you have posted needs some more codding.
Title: Re: drawing
Post by: dedndave on May 25, 2009, 06:03:19 PM
lol
FORTRANS wasn't posting entire programs - they are snippets
that is all the code required to set video mode 3
in fact, this works...

        mov     ax,3
        int     10h

as for drawing lines and circles, that is up to you to code
assembler does not give you these functions natively like basic or c might
for circles, you have to write a routine that does math, and plots points
simple verticle and horizontal lines are not too difficult to plot
start there, then work your way up to diagonal lines, that require simple math
Title: Re: drawing
Post by: Jimg on May 25, 2009, 06:05:30 PM
nrdev-

I must have missed your original introduction.  What is you goal here?  Are you indeed trying to write stand-alone dos programs that you won't try to run in windows XP?
Title: Re: drawing
Post by: nrdev on May 25, 2009, 06:45:47 PM
I want to make stand-alone dos program that can be run under win XP and look equally good like programs that look like doom 1 configuration application
Title: Re: drawing
Post by: dedndave on May 25, 2009, 07:06:44 PM
you need to learn VESA i think - or DirectX
also - 32-bit code
you have your work cut out for you
Title: Re: drawing
Post by: Jimg on May 25, 2009, 07:24:29 PM
You want to write a program that can run under the old 16-bit dos operating system, i.e. ms-dos 6.0, and have the same program run under XP?  I don't think that is possible.  The DLL you would need to run directly to the hardware in XP would not run in dos, and vice-versa.  Who do you know that is still running DOS as their operating system?
I'm not familiar with Doom 1, but I doubt the same program runs under both operating systems.  On Windows 98, perhaps, but not any of the NT variants directly.  At the minimum, the XP user would also have to start up some DOS emulator first, if one exists.
Title: Re: drawing
Post by: dedndave on May 25, 2009, 07:28:13 PM
he is refering to 16-bit console mode apps under XP
Title: Re: drawing
Post by: nrdev on May 25, 2009, 07:30:36 PM
for your information it runs on my very new win xp. Programs like that one you have on tones. For example installation program for the game that is made in 1994 like doom. Why it is not possible to do it nowadays. It was possibble then.
Title: Re: drawing
Post by: Jimg on May 25, 2009, 07:40:21 PM
I stand corrected.  (And confused).
Title: Re: drawing
Post by: dedndave on May 25, 2009, 07:43:03 PM
it is possible
but it is not easy
i suspect doom was written in C
if you want to write in assembler, the code can be faster, but you have a learning curve ahead of you
something you might want to look at is code posted by Farabi
he does some interesting game-related graphics - there are others in here, as well
he uses OpenGL for much of his work
you can click on his name to bring up his profile - then look at his posts
you will see pictures and code of stuff he works on
Title: Re: drawing
Post by: Jimg on May 25, 2009, 07:49:02 PM
Where can I download this 1994 version of Doom to try and see what you are talking about?  Is it still copyrighted and sold, or is it availble for free?
Title: Re: drawing
Post by: nrdev on May 25, 2009, 07:55:40 PM
go on google and search for doom or doom ultimate edition, after all this years it is now free and more-less open source.
Title: Re: drawing
Post by: nrdev on May 25, 2009, 08:01:06 PM
When you were doing an 16 bit dos application it is very possible that is has graphics in it, so what have you done to make that.
Title: Re: drawing
Post by: nrdev on May 25, 2009, 08:01:43 PM
When you were doing an 16 bit dos application (other forum members) it is very possible that is has graphics in it, so what have you done to make that.
Title: Re: drawing
Post by: jj2007 on May 25, 2009, 08:28:01 PM
Now I am getting curious :bg

This assembles and links fine (with Link16.exe) but refuses to show me a line. Since I am an absolute noob in DOS, that is probably my fault. But maybe it serves to overcome the first hurdle... see a pixxxxel :wink

.model tiny

.data?
MyX dw ?
MyY dw ?

.code
org 100h
start:

mov AX,13H ; 320 x 200 x 256 colors
; mov AX,12H ; 640 x 480 x 16 colors
INT 10H ; Video interrupt.

and MyX, 0
and MyY, 0

@@: mov AH, 0Ch ; Set pixel function.
mov AL, 1 ; [Color] ; Pixel value or color
mov BH, 0 ; Page for some modes.
mov CX, MyX ; [Column] ; X-coordinate
mov DX, MyY ; [Row] ; Y-coordinate
INT 10H
inc MyX
inc MyY
cmp MyX, 200
jne @B

mov AX, 3 ; = Standard video mode 3, reset to text at the end.
INT 10H

ret
end start


I have a suspicion that JimG and Dave could help out ;-)
Title: Re: drawing
Post by: nrdev on May 25, 2009, 08:38:53 PM
You wrote the basic VGA tutorial done on the hard way

I have found this tutorial that I have modified to work on masm, but it donesn't help me to know how to make line, rectangle, filled rectangle, and a single point.



.model small
.dosseg
.stack 512
.code
.186

; --- main program loop ---
_main   proc    near

           mov  ax,0012h                    ; set mode to 640x480x16
           int  10h

           mov  ax,0A000h
           mov  es,ax
           
           ; start line from (0,0) to (639,479)
           mov  X,0001h                     ; top most pixel (0,0)
           mov  Y,0001h                     ;
           mov  Color,02h                   ; start with color 0
           mov  cx,480                      ; 480 pixels
DrawLine:  call putpixel                    ; put the pixel
           inc  X                           ; move down a row and inc col
           inc  Y                           ;
           ;inc  Color                      ; next color
           ;and  Color,0Fh                  ; 00h - 0Fh only
           loop DrawLine                    ; do it

           
           xor  ah,ah                       ; wait for key press
           int  16h

           mov  ax,3                     ; return to screen 3 (text)
           int  10h

_main endp


; on entry X,Y = location and C = color (0-15)
putpixel   proc near uses ax bx cx dx

; byte offset = Y * (horz_res / 8) + int(X / 8)

           mov  ax,Y                        ; calculate offset
           mov  dx,80                       ;
           mul  dx                          ; ax = y * 80
           mov  bx,X                        ;
           mov  cl,bl                       ; save low byte for below
           shr  bx,03                       ; div by 8
           add  bx,ax                       ; bx = offset this group of 8 pixels

           mov  dx,03CEh                    ; set to video hardware controller

           and  cl,07h                      ; Compute bit mask from X-coordinates
           xor  cl,07h                      ;  and put in ah
           mov  ah,01h                      ;
           shl  ah,cl                       ;
           mov  al,08h                      ; bit mask register
           out  dx,ax                       ;

           mov  ax,0205h                    ; read mode 0, write mode 2
           out  dx,ax                       ;
           
           mov  al,es:[bx]                  ; load to latch register
           mov  al,Color
           mov  es:[bx],al                  ; write to register

           ret
putpixel   endp

X          dw 00h
Y          dw 00h
Color      db 00h

end

Title: Re: drawing
Post by: Jimg on May 25, 2009, 09:20:06 PM
Okay, I see, that's using dos 16m protected mode run-time.

The real question is, why would you want to subject yourself to that much pain?

How many people still have non-windows hardware that would be your target audience?

Anyway, I'm just going to slink out of here now.  Good luck with everything.

Title: Re: drawing
Post by: FORTRANS on May 25, 2009, 09:30:08 PM
Hello jj2007,

   Your program works for me.  Diagonal blue line.
I had to run it in DEBUG to stop it from exiting and
reseting to text mode.

Steve N.
Title: Re: drawing
Post by: FORTRANS on May 25, 2009, 09:53:30 PM
Quote from: nrdev on May 25, 2009, 08:38:53 PM
I have found this tutorial that I have modified to work on masm, but it donesn't help me to know how to make line, rectangle, filled rectangle, and a single point.

Hi,

   Probably the best way to learn all that is the book "Programmer's
Guide to PC Video Systems", second edition, by Richard Wilton, ISBN
1-55615-641-3.  Very good code explained well.  I found a ZIP file
of the first edition somewhere.  But the figures were missing.

Regards,

Steve N.
Title: Re: drawing
Post by: MichaelW on May 25, 2009, 09:56:19 PM
Ron Thomas has a nice tutorial that covers the basics.

http://www.ronthomas.plus.com/

Navigate to the downloads page and click the Gbook.zip link.

And in case there is a problem, here is a direct link:

http://www.ronthomas.plus.com/Downloads.html
Title: Re: drawing
Post by: dedndave on May 25, 2009, 10:42:43 PM
with mode 13h, it is easier and much faster to update the video buffer directly

the buffer for 13h (the one i used) is A000:0000-F9FF
one byte = one pixel
there are 256 palette registers - 3 bytes each (r,g,b) - each byte is 0-3Fh (only 6 bits used for each color)
the function to set the palette registers is int 10h, function 1002h
so - set up the palette as you like - then the pixel byte values are indexes into the palette

Addr = 320 * Y + X
(x,y 0,0 = upper left corner of screen)
set the ES register to A000h and away you go
Title: Re: drawing
Post by: Jimg on May 26, 2009, 01:02:01 AM
I really had to dig into some old archives for this, but it looks like what you were asking for, circles, filled recs, etc.

Heres a sample out of of an example program-


;****************** Main Loops

loop1:      call do_rand            ;Pixels
            lcall putpix ax,bx
            lnk loop1
            lcall cls

loop2:      call do_rand            ;Lines
            lcall line ax,bx,cx,dx
            lnk loop2
            lcall cls

loop3:      call do_rand            ;Rectangles
            lcall rect ax,bx,cx,dx
            lnk loop3
            lcall cls

loop4:      call do_rand            ;Filled Rectangles
            lcall frect ax,bx,cx,dx
            lnk loop4
            lcall cls

loop5:      call do_rand            ;Circles
            lcall circle ax,bx,si
            lnk loop5
            lcall cls

loop6:      call do_rand            ;Ellipses
            lcall ellipse ax,bx,si,di
            lnk loop6
            lcall cls

loop7:      call do_rand            ;Filled Circles
            lcall fcircle ax,bx,si
            lnk loop7
            lcall cls

loop8:      call do_rand            ;Filled Ellipses
            lcall fellipse ax,bx,si,di
            lnk loop8
            lcall cls

loop9:      call do_rand            ;Triangles
            push ax bx cx dx
            call do_rand
            lcall triangle ax,bx
            lnk loop9
            lcall cls


Comes with full source code for the library.  Perhaps you can get some hints from this.


[attachment deleted by admin]
Title: Re: drawing
Post by: sinsi on May 26, 2009, 02:44:28 AM
jj, you need an 'inkey' before you switch back to text mode otherwise you won't see anything -

;wait for a keypress
  sub ah,ah
  int 16h
;back to text mode
  mov ax,3
  int 10h
;quit to DOS - this is better than 'ret'
  mov ah,4ch
  int 21h

Title: Re: drawing
Post by: dedndave on May 26, 2009, 02:59:28 AM
wow Jim - that is an interesting library
they do trig fuctions in fixed-point
i have one i wrote years ago
it uses floating point - but the format does not match that of intel float
about that time, 486's came out and everyone had fpu's built in so i abandoned the project
if i were to continue it, i would want to go through the entire library and convert it to intel float
Title: Re: drawing
Post by: jj2007 on May 26, 2009, 05:59:28 AM
Quote from: sinsi on May 26, 2009, 02:44:28 AM

jj, you need an 'inkey' before you switch back to text mode otherwise you won't see anything -


Yep, that's it - I knew you are my friend!! :U
Title: Re: drawing
Post by: sinsi on May 26, 2009, 06:11:12 AM
Quote from: jj2007 on May 26, 2009, 05:59:28 AMYep, that's it - I knew you are my friend!! :U
heh heh
Title: Re: drawing
Post by: nrdev on May 26, 2009, 10:34:54 AM
QuotePerhaps you can get some hints from this.

It is what I want but the code doesn't work, I see that it is slightly different from other source codes that I was able to see. What I need to do to make this code work
Title: Re: drawing
Post by: dedndave on May 26, 2009, 10:51:48 AM
QuoteWhat I need to do to make this code work

we will help you in many ways
we can point you in the right direction to get going
but, we aren't going to write the code for you
and - you aren't going to become an expert programmer in 3 days
i started writing assembly language programs before IBM made PCs
when it comes to 32-bit code and the windows API, i am a newbie
although, i will learn faster than most because of experience

start out, as most of us did, by reading tutorials and making simple programs
then, expand on your knowledge by trying different things
we have given you a number of resources from which to learn - use them
Title: Re: drawing
Post by: nrdev on May 26, 2009, 11:03:17 AM
it is ok I don't want anyone to write entire program for me, I need this code working so I can get started.
Title: Re: drawing
Post by: nrdev on May 26, 2009, 11:51:08 AM
Quote
LINE
       Function         Draw a straight line between two points

       Entry            SI      Address of LINE parameter block
                        [SI]     X1
                        [SI+2]   Y1
                        [SI+4]   X2
                        [SI+6]   Y2
                        [SI+8]   BYTE,   Colour
                        [SI+14]   Address of dot plotting routine
                        [SI+16]   Number of pixels/line
                        [SI+18]   Number of bytes/line
                        [SI+20]   Number of lines/screen
                        [SI+22] 0 = pixel replace; 2 = XOR pixel
                        [SI+24] DWORD Address of the memory bank select routine
                         ES     Address of start of video memory

This is what is written in GLIB library that tried to use but I'am very confused by these definitions, how should I use this command for example with masm
Title: Re: drawing
Post by: dedndave on May 26, 2009, 12:32:36 PM
honestly, i don't think i would try to use that library directly
you might, however, look at the asx files and see how some of the routines work
having looked at a few of them - not sure what you'd gain from it
drawing a line is a fun project

the first thing a line-draw routine should do is to test for special cases
if X length is 0, it is a straight up and down line - very simple
if Y length is 0, it is a straight left and right line - even simpler

next, see whether the X axis is longer than the Y axis
if X is longer, calculate a Y pixel address for each X in the line
if Y is longer, calculate a X pixel address for each Y in the line
if X length is the same as the Y length (45, 135, 225, 315 degrees), you can use either loop to draw the line
you could even make a special loop for these lines if you want it to be faster

to calculate points, use the formulas

Y = mX+b
X = (Y-b)/m

then - calculate the pixel address as i outlined in the earlier post


http://www.wtamu.edu/academic/anns/mps/math/mathlab/int_algebra/int_alg_tut15_slope.htm
Title: Re: drawing
Post by: Jimg on May 26, 2009, 01:40:18 PM
Quote from: nrdev on May 26, 2009, 10:34:54 AM
QuotePerhaps you can get some hints from this.

It is what I want but the code doesn't work, I see that it is slightly different from other source codes that I was able to see. What I need to do to make this code work

Taking a quick look, I think it's in Tasm.  You'd have to convert it to Masm if that's what you are using.  I assumed if you were going to try something this complicated, you would be more advanced (than me even, I wouldn't try it).  So I think this is all way beyond your skill level at this point.  I still don't understand why you are doing this, but good luck.
Title: Re: drawing
Post by: FORTRANS on May 26, 2009, 02:47:08 PM
Hi,

   Well, for a simple example, here is a program to draw circles.  Just
to show why people use Bresenham's algorithm, I guess.

Steve N.


        PAGE ,132
TITLE Circle16
NAME  Circle16
; - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
; Example of drawing a circle, SRN 26 May 2009.
; Draw a circle using the equation:  R*R =  X*X + Y*Y.  No trig.

; MASM Circle16;
; LINK Circle16;
; EXE2BIN Circle16.EXE Circle16.COM

CODE    SEGMENT
        ASSUME  CS:CODE,DS:CODE
        ORG     100H ; COM file opening
START:
        MOV     AX,CS   ;\ If run as an EXE...
        MOV     DS,AX   ;/
        JMP     Setup

; - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
; Data for main program and circle routine.

CenterX DW      319     ; Center of circle
CenterY DW      239
Radius  DW      200     ; Initial size of circle.
Radius2 DW      40000, 0

MinX    DW      119     ; Area to scan.
MaxX    DW      519
MinY    DW      39
MaxY    DW      439

ColorIn DW      2       ; Choose your colors.
ColorOn DW      15      ; Try 15 and 3, to go with 1 and 2.
ColorOut DW     1       ; 15 seems fun, 3 looks smoother.

usX     DW      ?       ; Work temporaries
usY     DW      ?
Hello   DB      ' Hit a key. $'

; - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
; Main program, setup and scan over area calling the plot circle routine.
Setup:
        MOV     AH,0    ; BIOS Set Video Mode function
        MOV     AL,12H  ; 640x480x16 video mode
        INT     10H     ; Video BIOS interrupt

; Wait for things to settle down, get user to hit a key.
        MOV     DX,OFFSET Hello
        MOV     AH,9    ; DOS Out String Fn
        INT     21H

        MOV     AH,10H  ; Read Extended Keyboard Input.
        INT     16H     ; {Just used for a pause.}

; - - - MAIN LOOP OF PROGRAM - - -
; Could make a loop to reduce radius.
Main1:

; - - - Y / Row loop - - -
        MOV     CX,[MaxY]
        SUB     CX,[MinY]
        INC     CX      ; Count of rows.
        MOV     AX,[MinY]
        MOV     [usY],AX
Main2:
        PUSH    CX

; - - - X / Column loop - - -
        MOV     CX,[MaxX]
        SUB     CX,[MinX]
        INC     CX      ; Count of columns.
        MOV     AX,[MinX]
        MOV     [usX],AX
Main3:
        PUSH    CX
        CALL    CirclePl

        INC     [usX]
        POP     CX
        LOOP    Main3   ; X loop

        INC     [usY]
        POP     CX
        LOOP    Main2   ; Y loop

; Wait for user to hit a key.
        MOV     AH,10H  ; Read Extended Keyboard Input.
        INT     16H     ; AL = ASCII, AH = Scan code

        CMP     AL,27   ; An attempt to leave early?
        JE      Reset_Vid ; Escape!

; - - - What the heck... - - -
; Reduce radius, update Radius2, test, and loop to Main1.
        MOV     AX,[Radius]
        DEC     AX
        JL      Reset_Vid
        MOV     [Radius],AX
        MUL     AX
        MOV     [Radius2],AX
        MOV     [Radius2+2],DX
        JMP     Main1

; - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
; Exit processing, DON'T leave in graphics mode under Win2k or XP.
RESET_VID:
        MOV     AH,0
        MOV     AL,3    ; = Standard video mode 3
        INT     10H ; Set Video mode

        MOV     AH,4CH
        MOV     AL,0
        INT     21H     ; Exit to DOS

; - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
; Circle plotting the _SLOW_ way.
CirclePl:
        MOV     AX,[usX]
        SUB     AX,[CenterX]    ; Delta X
        IMUL    AX              ; dX * dX
        MOV     BX,AX
        MOV     CX,DX

        MOV     AX,[usY]
        SUB     AX,[CenterY]    ; Delta Y
        IMUL    AX              ; dY * dY

        ADD     BX,AX           ; Form R * R in CX:BX.
        ADC     CX,DX
; - - -
        MOV     AX,[ColorOut]
        CMP     CX,[Radius2+2]  ; Compare R*R to Radius2.
        JE      Circ1           ; Always for initial version of code.
        JA      Circ3           ; Confirm out.
        MOV     AX,[ColorIn]    ; It's inside.
        JMP     Circ3

; - - -
Circ1:  ; High word equal, test low word.
        CMP     BX,[Radius2]
        JA      Circ3           ; Confirm out.
        JE      Circ2           ; It's on the circle?
        MOV     AX,[ColorIn]    ; Else it is inside.
        JMP     Circ3
Circ2:
        MOV     AX,[ColorOn]

; - - - And plot the pixel - - -
Circ3:
        AND     AL,0FH  ; Ensure valid color.
        MOV     DX,[usY]
        MOV     CX,[usX]
        MOV     BX,0    ; Page #
        MOV     AH,0CH  ; Write pixel BIOS function
        INT     10H     ; Video BIOS interrupt

        RET     ; End CirclePl

; - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
CODE    ENDS
        END START
Title: Re: drawing
Post by: nrdev on May 26, 2009, 04:06:11 PM
 :(
I have tried to do something about this but unfortinetually I give up, this is too hard for me to do, and understand.
I could make much better success in 32-bit programming then in 16-bit.
Title: Re: drawing
Post by: Jimg on May 26, 2009, 04:18:56 PM
You'll never regret it. :U
Title: Re: drawing
Post by: PBrennick on May 29, 2009, 01:03:04 PM
nrdev,

Actually, I think you are confused as to what is the difference between a 16-bit dos program that will run in a dos window and a 32-bit console program. They are apples and oranges.  Both run in a console box, only one of those console boxes is a dos box.

Paul