The MASM Forum Archive 2004 to 2012

Miscellaneous Forums => 16 bit DOS Programming => Topic started by: MASM on January 02, 2012, 03:05:57 PM

Title: NOOB question
Post by: MASM on January 02, 2012, 03:05:57 PM
Hello everyone!

I used to do some MASM in MS DOS days before I switched to 6502 assembler programming.

Now I would like to brush up my MASM skills.

Therefore, I wrote a little program to put a character into Video Memory in text mode.

Somehow MASM comes up with an error I don't quite understand.

here is my program (video ram starts at B0000H in text mode)

           .MODEL SMALL

            .DATA

VIDEO   EQU B0000H
           
            .CODE
           
            ORG 0100H
           
            MOV AL,41H
            MOV VIDEO,AL
       
            END

here is what MASM says:

A2006: undefined symbol B0000H in line MOVE VIDEO,AL

(I looked up absolute addressing in my ASM86 but it should work)

AND the linker complains about 'unresolved externals'

who can help me? ::)






Title: Re: NOOB question
Post by: Rockphorr on January 02, 2012, 03:08:49 PM
welcome in 16 bit thread and

VIDEO   EQU 0B0000H

B0000H is not digit, it is name
Title: Re: NOOB question
Post by: MASM on January 02, 2012, 03:16:00 PM
oops!

thanks a bunch!

But how does it come that 41H rather than 041H is allowed in MOV AL,41H ?

anyway, now MASM tells me error 2001: immediate operand not allowed in MOV VIDEO,AL

Title: Re: NOOB question
Post by: dedndave on January 02, 2012, 03:59:40 PM
under MASM, if it starts with a numeric (0-9), it is a number
if it starts with a non-numeric, it is a label
so - for hex vales that start with a-f, you have to preceed them with a 0
Title: Re: NOOB question
Post by: MASM on January 02, 2012, 04:27:32 PM
Thank you!

But what about my MOV 0B0000H,AL illegal immediate?

I have to admit according to syntax logic it would look like immediate, but how do I tell MASM I want absolute addressing?

i.e. moving AL into memory location 0B0000H ?

Title: Re: NOOB question
Post by: dedndave on January 02, 2012, 04:43:01 PM
the way you have it written, you are trying to move a byte to a 20-bit immediate - that just doesn't work

first of all, 0B000h is a segment - 16-bit intel processors use segmented addressing
0B0000h is 20 bits  :P

but - try this...

        mov     ax,0B000h
        mov     es,ax

        mov     al,55h
        mov     es:[0],al
Title: Re: NOOB question
Post by: dedndave on January 02, 2012, 04:44:58 PM
the 16-bit segment value is multiplied by 16, then added to the offset to obtain the "physical" address
(well - it was the physical address on an 8088   :wink )

we typically use the DS register to hold the local data segment and the ES register for the video (or other) segment

by the way - this is 16-bit code
we have a special sub-forum for 16-bit stuff, which is quickly becoming obsolete
before you pull out too many hairs figuring out segmented addressing, have a look at 32-bit code   :U
Title: Re: NOOB question
Post by: untio on January 02, 2012, 05:00:10 PM
Hi to all,
After look at the code written by MASM, I have a question about the memory model and the org directive.
I think that with:
.MODEL SMALL
and:
ORG 100H
The program will not work fine.
¿Am I right or do I not? and ¿why?.
Title: Re: NOOB question
Post by: dedndave on January 02, 2012, 05:12:35 PM
you are right - it should be
        .MODEL  Tiny
however - it may work, as long as you don't put any data in a data segment   :P
for a .COM program, the linker wants to see that all code fits in a single segment
so - if no data segment is ever created, Small model may work - never tried it   :bg
Title: Re: NOOB question
Post by: MASM on January 02, 2012, 05:15:51 PM
yep!

segmentation is a good point

so it seems can't express it directly without parting my address into segment : offset

I will try that es:[0] thing.

Also I will change my memory model to .TINY

Actually I was going to produce a MODEL SMALL program and then apply exe2bin, but now I found I can directly produce a .COM file with the .MODEL TINY directive
Title: Re: NOOB question
Post by: MASM on January 02, 2012, 05:18:12 PM
Oh, yes, sorry for posting in the wrong subforum  :lol

Title: Re: NOOB question
Post by: dedndave on January 02, 2012, 05:21:57 PM
no worries - one of the moderators will move it there for you

with 32-bit code, you can directly address 4 gb of space with a single register
with the 16-bit segment:offset, you can indirectly address 1 mb of space
Title: Re: NOOB question
Post by: MASM on January 02, 2012, 05:37:40 PM
it assembles all right now, however the linker complains about several issues now :dance:

warning LNK4078: 'multiple data sections found with different attributes'

(I dropped the .DATA expression from my code and have only section, namely a .CODE section, now)

so what is this warning about, anyway ?

next we have a LNK2001 error: unresolved external symbol _WinMainCRTStartup

AND a fatal LNK1120 error: unresolved external ::)

Title: Re: NOOB question
Post by: dedndave on January 02, 2012, 05:42:05 PM
you will want to use the 16-bit linker - Link16.exe
Title: Re: NOOB question
Post by: jj2007 on January 02, 2012, 05:56:31 PM
Hi,
Welcome to the forum :thumbu
It is natural that you want to continue where you left assembler in the MS DOS days. However, 32-bit coding is a lot easier. Unless you have a really compelling argument to stay with 16-bit code...
include \masm32\include\masm32rt.inc

.code
start: print "Hello World", 13, 10
inkey "that was easy..."
exit

end start


Sorry if that doesn't have the old assembler look & feel - but you can use all combinations of "true" assembler and Masm. The first letter of your nick stands for MACRO, and that's what makes your life easier.
\masm32\macros\macros.asm is a good lecture in this respect :wink
Title: Re: NOOB question
Post by: MASM on January 02, 2012, 06:28:36 PM
link16 sounds a good idea (I can't remember using link16 though)

@jj2007: I would like to start where I left with 16 bits and then move on to 32 bits

(I found a junk of code which switches my 80386 from real mode to protected mode AND I would like to implement it)

my currect program is to  show the concept of text mode

store an ascii value into video memory and BANG it appears on the screen

Title: Re: NOOB question
Post by: dedndave on January 02, 2012, 07:05:58 PM
well - in the old days, it was just called link
we call it link16 or link563 (it's version 5.63) so that it can co-exist with our 32-bit linker
the newer versions of link do not support 16-bit code
Title: Re: NOOB question
Post by: MASM on January 02, 2012, 07:15:40 PM
I see.

I remember having used the /c (no linking) option with ML but not having used link16

I succeeded in producing a .COM file by now.

there is only one more strange thing about it

I included an ORG 0100H instruction to let the code start at 0100H

still my linker says: L4055: start address not equal to 0x100h for /TINY

what now?
Title: Re: NOOB question
Post by: dedndave on January 02, 2012, 07:34:27 PM
i am playing with this....
        .MODEL  Tiny
        OPTION  CaseMap:None

;******************************************************************************

        .CODE

;------------------------------------------------------------------------------

        ORG     100h

_main   PROC

        mov     ax,0B000h
        mov     es,ax

        mov     al,41h
        mov     es:[0],al

        mov     ah,1
        int     16h
        mov     ah,0
        int     16h

        mov     ax,4C00h
        int     21h

_main   ENDP

;******************************************************************************

        END     _main


to create a .COM program....
ml /c MyProg.asm
Link16 /TINY MyProg.obj

however, i am seeing a strange issue
if i build it as an EXE, it works fine - the "A" character is displayed
if i build it as a COM, the "A" character is not displayed   :eek

it may have to do with how NTVDM emulates 16-bit mode - i dunno
Title: Re: NOOB question
Post by: dedndave on January 02, 2012, 07:43:42 PM
ok - it is not the COM/EXE thing that makes a difference
i have to create a new console window each time i run it
then - it works as a COM or as an EXE

still - this seems strange - something is amiss   :P
Title: Re: NOOB question
Post by: MASM on January 02, 2012, 07:52:04 PM
yes we have the same assembler/ linker setting

Doesn't your linker complain about the starting address not being 0100H for a .COM file ?

Anyway, thanks, I will apopt your code AND then slowly, step by step, migrate to my code.

thus, I will see where I get stuck.

one more thing

your approach to use ES as a pointer looks very elegant

still it is a lot of programming to do to load ES:offset each time for an absolute addressing

isn't there any other way to tell the assembler an absoulte addressing ?
Title: Re: NOOB question
Post by: dedndave on January 02, 2012, 07:59:38 PM
well - no matter what method you use, you will have to have a segment register pointing to the video buffer
of course, you can set ES to 0B000h and leave it - it won't be altered for most DOS/BIOS calls

i typically use ES because it is fast to use REP MOVSB, REP MOVSW, REP STOSB, or REP STOSW to fill the screen
also - that frees up the DS register for local data

however, if you want to use some other addressing mode (like the one we are playing with), then using ES is not all that elegant because there is a segment override code byte each time
in such cases, it may be more appropriate to use DS, so that no segment override is required

but - if you want to specify an absolute address, the assembler still expects the syntax...
        mov     ds:[0],al
even though no segment override code byte is required

you can use BX, SI, or DI to address the buffer - no override required
Title: Re: NOOB question
Post by: Rockphorr on January 02, 2012, 08:08:19 PM
Quote from: dedndave on January 02, 2012, 05:21:57 PM
no worries - one of the moderators will move it there for you

with 32-bit code, you can directly address 4 gb of space with a single register
with the 16-bit segment:offset, you can indirectly address 1 mb of space


no worries about 16 bit when you write by asm. For 10kb of pure code you work about 2 years.
Title: Re: NOOB question
Post by: MASM on January 02, 2012, 08:33:14 PM
@DEDNDAVE strange - your code does not produce ANY output in my system

@ROCKPHORR it took me 2 months to produce 4.4 kB of pure code in 6502 machine language

here is something from the 2010 paperback edition:

this book was written on an Archives III microcomputer AND sent from Colombo to New York on one five inch diskette

July 1981

if  300 pages are no more than 360 kB (including formatting) can please anybody tell me why people need GIGABYTES of memory these days?




Title: Re: NOOB question
Post by: MASM on January 02, 2012, 09:48:05 PM
I found it!

For some reason my video memory starts at B8000 rather than B0000.

Both your and mine program work now.

Still, for SOME reason I still get a warning from my linker, according to which my segment starts NOT at 100h even though I put an ORG 100h instruction in.
Title: Re: NOOB question
Post by: MichaelW on January 02, 2012, 10:10:11 PM
Segment address 0b000h is the base of the display memory for the monochrome alphanumeric modes. For the color alphanumeric modes it should be 0b800h. It was done this way so a monochrome display subsystem  (MDA/HGC) could coexist with the EGA/VGA. This is also the reason that the two systems use different I/O port address ranges.
Title: Re: NOOB question
Post by: dedndave on January 03, 2012, 12:26:23 AM
many of the graphics modes start at 0B000h, as well
as they have paged buffers

at any rate - i got an "A" on the screen with a base of 0B000h - i immediately tried 0B800h when the COM didn't work
basically because i wasn't sure which was correct and i was too lazy to look it up   :bg
it's easier to just try it both ways - lol
Title: Re: NOOB question
Post by: MichaelW on January 03, 2012, 01:17:43 AM
For the VGA only one of the alphanumeric modes, 7, uses B0000h as the starting host-memory address. The default mode is 3, and I can't recall ever seeing a VGA system that deviated from this, although I have never tested a very recent adapter. Here is the functionality/state information returned by Interrupt 10h, function 1Bh on my P3 system running Windows 2000:

Active video mode            : 3
Character columns            : 80
Bytes per page               : 4096
Current page offset          : 0
Cursor start and end         : 2707h
Current display page         : 0
Base I/O port address        : 3D4h
Character rows               : 25
Bytes per character          : 16
Active display code          : VGA with analog color
Current number of colors     : 16
Current number of pages      : 8
Current number of scan lines : 400
Primary character set        : 0
Secondary character set      : 0
Size of active character set : 256
Character width in pixels    : 9


As to why B000h would work in the default mode, perhaps (some) newer adapters/systems have dropped all support for the monochrome adapters (apart from there having been no systems with ISA slots for > 10 years), and are mapping the display memory into the host address space differently.
Title: Re: NOOB question
Post by: MASM on January 05, 2012, 06:04:44 PM
one last question about this issue:

lets say I switch to protected mode and want to do the same thing in protected mode.

my memory model is flat with one data segment ranging from 0 to 800000H

ES is tied to address 0h where the segment starts.

is the following instruction legal?

MOV ES:[68000h],41h

sorry for my question BUT I dont have any manual for my assembler.
Title: Re: NOOB question
Post by: dedndave on January 05, 2012, 06:13:00 PM
well - you can get a manual for newer versions, at least - let us know what version of MASM you are using

        MOV     ES:[68000h],41h

the address exceeds 64kb
the largest offset you can use is...
        MOV     ES:[0FFFFh],41h

if you want to address that specific location, you'll have to change the segment register
if you add 6000h to the segment value, it adds 60000h to the calculated physical address
not sure what you are addressing, though   :P

XXXX     segment (16-bit hex)
 XXXX    offset (16-bit hex)
-----    +
XXXXX    address (20-bit hex)
Title: Re: NOOB question
Post by: MASM on January 05, 2012, 06:32:33 PM
my MASM version is 6.0 AND I think I left my manual at home when I moved out.

I am messing around with a piece of code which puts the 80386 into protected mode.

After switching to protected mode I want to do the same thing as before, outputting a character on the display

oops - now I see, it should read B8000 rather than 68000.

so you are saying my segment must not exceed 64kB in protected mode????

Title: Re: NOOB question
Post by: dedndave on January 05, 2012, 06:35:38 PM
the segment register is only 16 bits, so it can hold a max value of 0FFFFh
the offset - same thing
so, to obtain a 20-bit address....

look carefully at this little math ditty...
XXXX.    segment (16-bit hex)
.XXXX    offset (16-bit hex)
-----    +
XXXXX    address (20-bit hex)
Title: Re: NOOB question
Post by: dedndave on January 05, 2012, 06:37:36 PM
masm manual in PDF form...

http://www.4shared.com/zip/39MdNf_v/MASMProgGuide.html
Title: Re: NOOB question
Post by: dedndave on January 05, 2012, 06:59:45 PM
i think that manual is for version 6.14, which is signifigantly different from version 6.10
you can obtain version 6.14 here
ftp://ftp.microsoft.com/softlib/mslfiles/ML614.EXE (http://ftp://ftp.microsoft.com/softlib/mslfiles/ML614.EXE)
or download and install the MASM32 package
Title: Re: NOOB question
Post by: dedndave on January 05, 2012, 07:04:42 PM
version 6.15 is available in this package...
http://download.microsoft.com/download/vb60ent/Update/6/W9X2KXP/EN-US/vcpp5.exe
Title: Re: NOOB question
Post by: MASM on January 06, 2012, 09:20:41 PM
thanks for the link!

that's what I have been looking for

I downloaded version 6.1 which is closest to 6.0

..
Title: Re: NOOB question
Post by: MASM on January 06, 2012, 09:25:32 PM
about the math

I know the math for real mode.

now actually I thought the MOV ES:[..] would work with an 32 bit offset rather than a 16 bit offset in protected mode.

now that I have a prog guide I can look these things up
Title: Re: NOOB question
Post by: dedndave on January 06, 2012, 10:58:09 PM
in 32-bit code, we rarely use the segment registers (which are then called "selectors", as they look up info in a table)
you can directly address 4 Gb of space with a single 32 bit register   :P
Title: Re: NOOB question
Post by: MASM on January 08, 2012, 02:45:00 PM
Please forgive me one last question:

(I looked it up in the MASM manual but what I came up with in the end looked to me like some pointer)

how do I say in 32 mode MOVE 41H into memory location B8000H ?

assuming a FLAT memory model with the segment base address tied to 0H
Title: Re: NOOB question
Post by: dedndave on January 08, 2012, 03:30:22 PM
ok - the flat memory model is for 32-bit code
in which case, you can directly address up to 4 gb with a register
however, in 32-bit you are in protected mode - you cannot access the video buffer directly

for 16-bit code (real mode), you load a segment register (probably DS or ES) with B800h
the segment register value is multiplied by 16 to provide the segment base address
then, you can access up to 64 kb of the video buffer with a register, starting at 0

the video buffer is only 32 kb in size   :P
Title: Re: NOOB question
Post by: MASM on January 08, 2012, 09:12:50 PM
why shouldn't I be able to access the video ram in protected mode?

I got a piece of code which switches my system to protected mode

my program runs on privilege level 0

I come out in text mode (NO WINDOWS)



Title: Re: NOOB question
Post by: MASM on January 08, 2012, 09:17:30 PM
maybe something like this would work ? MOV [B8000H],41H
Title: Re: NOOB question
Post by: redskull on January 08, 2012, 09:49:38 PM
Since you have switched into protected mode yourself, what selectors in the GDT/LDT did you make?  If you don't know (or just copy and pasted what someone else did), you have lots of studying of the Intel manuals ahead of you.  Like dave said, access is through selectors, so it the end result depends on how you configured the GDT/LDT.

-r
Title: Re: NOOB question
Post by: MichaelW on January 09, 2012, 01:37:56 AM
Quote from: dedndave on January 08, 2012, 03:30:22 PM
the video buffer is only 32 kb in size   :P

It is 32KB for the alphanumeric modes, but 64KB for the graphics modes (segment address A000h).
Title: Re: NOOB question
Post by: MichaelW on January 09, 2012, 02:23:00 AM
Quote from: MASM on January 08, 2012, 09:12:50 PM
why shouldn't I be able to access the video ram in protected mode?

You can. At one time I had a memory test application that ran in PM and that displayed by directly manipulating the display buffer (and the VGA hardware via I/O ports for cursor control). To do this I included a descriptor for the display (alphanumeric) buffer in my GDT, and at runtime loaded the selector for it into ES. The essential parts of the code are below. Note that you should probably ignore the IDT stuff and just keep interrupts disabled, and that I set everything up so the application actually executed in the graphics display buffer so it could test the memory as one contiguous chunk, instead of having to dance around the application code, data, and stack. So the BASE...MUST BE ADJUSTED comments are included because the application code, data, and stack get moved at runtime.

; Structure for segment descriptors
; Access rights byte:
;       P:1             Present (in memory)
;       DPL:2           Descriptor privilege level (0-3)
;       S:1             Segment descriptor (code/data/stack)
;       E:1             Executable (code)
;       C/ED:1          Conforming (code)/Expand down (stack)
;       R/W:1           Readable/Writeable (code/data)
; Extra byte:
;       G:1             Granularity (page)
;       D:1             Default operand size (32 bits)
;       Z:1             Must be 0
;       AVL:1           Available
;       limit_high:4    Segment limit 19-16
SEGMENT_DESCRIPTOR struc
seg_limit_low           dw      0
seg_base_low            dw      0
seg_base_mid            db      0
seg_access_rights       db      0
seg_extra_byte          db      0
seg_base_high           db      0
SEGMENT_DESCRIPTOR ends

; Structure for gate descriptors
; Extra byte:
;       Z:3             Must be 0
;       WORD_COUNT:5    Call gates only
; Access rights byte:
;       P:1             Present (in memory)
;       DPL:2           Descriptor privilege level (0-3)
;       Z:1             Must be 0
;       TYPE:4          Should be 06h for 286 interrupt gate
GATE_DESCRIPTOR struc
gate_offset_low         dw      0
gate_selector           dw      0
gate_extra_byte         db      0
gate_access_rights      db      0
gate_offset_high        dw      0
GATE_DESCRIPTOR ends

; Structure for table descriptors
TABLE_DESCRIPTOR struc
table_limit             dw      0
table_base              dd      0
TABLE_DESCRIPTOR ends

CODE_SELECTOR   equ     8       ; Code segment
DATA_SELECTOR   equ     16      ; Data/stack segment
VIDEO_SELECTOR  equ     24      ; Video buffer segment (for display from PM)
TEST_SELECTOR   equ     32      ; Test segment (entire address space)

R_MULT          equ     16      ; Refresh period multiplier

.model small,c
.stack                          ; Defaults to 1024 bytes

.data

; Create and initialize IDT
;       Offset low w:   Offset of handler
;       Selector w:     CODE_SELECTOR
;       Extra b:        0 (default)
;       AR b:           P = 1, DPL = 00, 0, TYPE = 0006h
;       Offset high w:  0 (default)
; Note that the code segment descriptor is recognized as a
; 286 descriptor because the most significant word is 0
; NMI is Int 02, others for debugging
idt GATE_DESCRIPTOR     <offset Isr00, CODE_SELECTOR, , 86h>, \
<offset Isr01, CODE_SELECTOR, , 86h>, \
<offset Isr02, CODE_SELECTOR, , 86h>, \
<offset Isr03, CODE_SELECTOR, , 86h>, \
<offset Isr04, CODE_SELECTOR, , 86h>, \
<offset Isr05, CODE_SELECTOR, , 86h>, \
<offset Isr06, CODE_SELECTOR, , 86h>, \
<offset Isr07, CODE_SELECTOR, , 86h>
    GATE_DESCRIPTOR     <offset Isr08, CODE_SELECTOR, , 86h>, \
<offset Isr09, CODE_SELECTOR, , 86h>, \
<offset Isr0A, CODE_SELECTOR, , 86h>, \
<offset Isr0B, CODE_SELECTOR, , 86h>, \
<offset Isr0C, CODE_SELECTOR, , 86h>, \
<offset Isr0D, CODE_SELECTOR, , 86h>, \
<offset Isr0E, CODE_SELECTOR, , 86h>, \
<offset Isr0F, CODE_SELECTOR, , 86h>
    GATE_DESCRIPTOR     <offset Isr10, CODE_SELECTOR, , 86h>, \
<offset Isr11, CODE_SELECTOR, , 86h>, \
<offset Isr12, CODE_SELECTOR, , 86h>, \
<offset Isr13, CODE_SELECTOR, , 86h>, \
<offset Isr14, CODE_SELECTOR, , 86h>, \
<offset Isr15, CODE_SELECTOR, , 86h>, \
<offset Isr16, CODE_SELECTOR, , 86h>, \
<offset Isr17, CODE_SELECTOR, , 86h>
    GATE_DESCRIPTOR     <offset Isr18, CODE_SELECTOR, , 86h>, \
<offset Isr19, CODE_SELECTOR, , 86h>, \
<offset Isr1A, CODE_SELECTOR, , 86h>, \
<offset Isr1B, CODE_SELECTOR, , 86h>, \
<offset Isr1C, CODE_SELECTOR, , 86h>, \
<offset Isr1D, CODE_SELECTOR, , 86h>, \
<offset Isr1E, CODE_SELECTOR, , 86h>, \
<offset Isr1F, CODE_SELECTOR, , 86h>

; Create and initialize GDT
; BASE FOR DATA SEGMENT MUST BE ADJUSTED AT RUN TIME
; Null selector first
; Code segment initialized for operation from VGA display buffer
;       Limit low w:    0ffffh
;       Base low w:     9000h
;       Base mid b:     0bh
;       AR b:           P = 1, DPL = 00, S = 1, E = 1, C = 0, R = 1, A = 0
;       Extra b:        G = 0, 0, AVL = 0, D = 0, limit_high = 0000 (default)
; Data/stack segment initialized for operation from VGA display buffer
;       Limit low w:    0ffffh
;       Base low w:     9000h
;       Base mid b:     0bh
;       AR b:           P = 1, DPL = 00, S = 1, E = 0, ED = 0, W = 1, A = 0
;       Extra b:        G = 0, 0, AVL = 0, D = 0, limit_high = 0000 (default)
; Video buffer segment initialized for buffer access from PM
;       Limit low w:    0ffffh
;       Base low w:     8000h
;       Base mid b:     0bh
;       AR b:           P = 1, DPL = 00, S = 1, E = 0, ED = 0, W = 1, A = 0
;       Extra b:        G = 0, 0, AVL = 0, D = 0, limit_high = 0000 (default)
; Test segment initialized for maximum segment limit and page granularity
;       Limit low w:    0ffffh
;       Base low w:     0 (default)
;       Base mid b:     0 (default)
;       AR b:           P = 1, DPL = 00, S = 1, E = 0, ED = 0, W = 1, A = 0
;       Extra b:        G = 1, 0, AVL = 0, D = 0, limit_high = 0fh
gdt SEGMENT_DESCRIPTOR  <>, \
< 0ffffh, 9000h, 0bh, 9ah >, \
< 0ffffh, 9000h, 0bh, 92h >, \
< 0ffffh, 8000h, 0bh, 92h >, \
< 0ffffh, , , 92h, 8fh >

; Create IDT descriptor and initialize base and limit
; BASE MUST BE ADJUSTED AT RUN TIME
idt_descriptor TABLE_DESCRIPTOR  <32 * 8 - 1, 0b9000h>

; Create GDT descriptor and initialize base and limit
; BASE MUST BE ADJUSTED AT RUN TIME
gdt_descriptor TABLE_DESCRIPTOR  <5 * 8 - 1, 0b9000h>

. . .

; Load descriptor table registers
lidt    fword ptr [idt_descriptor]
lgdt    fword ptr [gdt_descriptor]

; Initialize far pointer for mode change
mov     word ptr jmp_ptr,offset reentry
mov     word ptr jmp_ptr[2],CODE_SELECTOR

; Go to PM
mov     eax,cr0
or      al,01h
mov     cr0,eax

; Do intersegment jump to set cs and flush instruction queue
jmp     dword ptr jmp_ptr
reentry:

; Load descriptor caches
push    DATA_SELECTOR
pop     ds
push    DATA_SELECTOR
pop     ss
push    VIDEO_SELECTOR
pop     es

Title: Re: NOOB question
Post by: sinsi on January 09, 2012, 10:50:34 AM
If you mean by "flat" that your ES selector has base=0 and limit=4G then the video memory in mode 3 starts at linear address 000B8000h.

  mov byte ptr es:[0b8000h],'A'
;or
  mov edi,0b8000h
  mov byte ptr es:[edi],'A'

Not sure how masm handles the size overrides if you use this code in a 16-bit segment.
Title: Re: NOOB question
Post by: MASM on January 10, 2012, 05:55:12 PM
Thank you sinsi that's what I have been looking for

MOV BYTE PTR ES:[0B8000H], 'A'

yes, it does make make sense to install a descriptor pointing to 0B8000H

first, though, I needed to know how to address memory in protected mode
Title: Re: NOOB question
Post by: MASM on January 10, 2012, 06:10:17 PM
@redskull & MichealW

I am adapting a piece of code from the 80386 System Software Writer's Guide.

my Data Descriptor is as follows

DATA DW 0FFFFH       ; segment limit
         DB  0,0,0          ; segment base 0-23
         DB  10010011B ; access right byte
         DB  10001111B ; G D/B 0 AVL limit 16-19
         DB  0               ; segment base 24-31