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? ::)
welcome in 16 bit thread and
VIDEO EQU 0B0000H
B0000H is not digit, it is name
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
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
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 ?
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
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
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?.
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
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
Oh, yes, sorry for posting in the wrong subforum :lol
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
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 ::)
you will want to use the 16-bit linker - Link16.exe
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
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
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
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?
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
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
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 ?
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
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.
@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?
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.
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.
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
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.
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.
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)
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????
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)
masm manual in PDF form...
http://www.4shared.com/zip/39MdNf_v/MASMProgGuide.html
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
version 6.15 is available in this package...
http://download.microsoft.com/download/vb60ent/Update/6/W9X2KXP/EN-US/vcpp5.exe
thanks for the link!
that's what I have been looking for
I downloaded version 6.1 which is closest to 6.0
..
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
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
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
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
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)
maybe something like this would work ? MOV [B8000H],41H
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
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).
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
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.
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
@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